]> git.openstreetmap.org Git - rails.git/blobdiff - vendor/assets/iD/iD.js
Update to iD v2.20.0
[rails.git] / vendor / assets / iD / iD.js
index 24cb83d9fde13bd007f17358c2acda98dc09a2d5..6ac452788fbd27e1c5715ac6b4b86aff4aeb0d37 100644 (file)
@@ -2,18 +2,9 @@
 
        var commonjsGlobal = typeof globalThis !== 'undefined' ? globalThis : typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : {};
 
-       function createCommonjsModule(fn, basedir, module) {
-               return module = {
-                       path: basedir,
-                       exports: {},
-                       require: function (path, base) {
-                               return commonjsRequire(path, (base === undefined || base === null) ? module.path : base);
-                       }
-               }, fn(module, module.exports), module.exports;
-       }
-
-       function commonjsRequire () {
-               throw new Error('Dynamic requires are not currently supported by @rollup/plugin-commonjs');
+       function createCommonjsModule(fn) {
+         var module = { exports: {} };
+               return fn(module, module.exports), module.exports;
        }
 
        var check = function (it) {
        };
 
        // https://github.com/zloirock/core-js/issues/86#issuecomment-115759028
-       var global_1 =
-         /* global globalThis -- safe */
+       var global$2 =
+         // eslint-disable-next-line es/no-global-this -- safe
          check(typeof globalThis == 'object' && globalThis) ||
          check(typeof window == 'object' && window) ||
+         // eslint-disable-next-line no-restricted-globals -- safe
          check(typeof self == 'object' && self) ||
          check(typeof commonjsGlobal == 'object' && commonjsGlobal) ||
          // eslint-disable-next-line no-new-func -- fallback
 
        // Detect IE8's incomplete defineProperty implementation
        var descriptors = !fails(function () {
+         // eslint-disable-next-line es/no-object-defineproperty -- required for testing
          return Object.defineProperty({}, 1, { get: function () { return 7; } })[1] != 7;
        });
 
-       var nativePropertyIsEnumerable = {}.propertyIsEnumerable;
-       var getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor;
+       var $propertyIsEnumerable$1 = {}.propertyIsEnumerable;
+       // eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe
+       var getOwnPropertyDescriptor$5 = Object.getOwnPropertyDescriptor;
 
        // Nashorn ~ JDK8 bug
-       var NASHORN_BUG = getOwnPropertyDescriptor && !nativePropertyIsEnumerable.call({ 1: 2 }, 1);
+       var NASHORN_BUG = getOwnPropertyDescriptor$5 && !$propertyIsEnumerable$1.call({ 1: 2 }, 1);
 
        // `Object.prototype.propertyIsEnumerable` method implementation
        // https://tc39.es/ecma262/#sec-object.prototype.propertyisenumerable
-       var f = NASHORN_BUG ? function propertyIsEnumerable(V) {
-         var descriptor = getOwnPropertyDescriptor(this, V);
+       var f$7 = NASHORN_BUG ? function propertyIsEnumerable(V) {
+         var descriptor = getOwnPropertyDescriptor$5(this, V);
          return !!descriptor && descriptor.enumerable;
-       } : nativePropertyIsEnumerable;
+       } : $propertyIsEnumerable$1;
 
        var objectPropertyIsEnumerable = {
-               f: f
+               f: f$7
        };
 
        var createPropertyDescriptor = function (bitmap, value) {
          };
        };
 
-       var toString = {}.toString;
+       var toString$1 = {}.toString;
 
        var classofRaw = function (it) {
-         return toString.call(it).slice(8, -1);
+         return toString$1.call(it).slice(8, -1);
        };
 
-       var split = ''.split;
+       var split$2 = ''.split;
 
        // fallback for non-array-like ES3 and non-enumerable old V8 strings
        var indexedObject = fails(function () {
@@ -83,7 +77,7 @@
          // eslint-disable-next-line no-prototype-builtins -- safe
          return !Object('z').propertyIsEnumerable(0);
        }) ? function (it) {
-         return classofRaw(it) == 'String' ? split.call(it, '') : Object(it);
+         return classofRaw(it) == 'String' ? split$2.call(it, '') : Object(it);
        } : Object;
 
        // `RequireObjectCoercible` abstract operation
          return indexedObject(requireObjectCoercible(it));
        };
 
-       var isObject = function (it) {
+       var isObject$4 = function (it) {
          return typeof it === 'object' ? it !== null : typeof it === 'function';
        };
 
        // instead of the ES6 spec version, we didn't implement @@toPrimitive case
        // and the second argument - flag - preferred type is a string
        var toPrimitive = function (input, PREFERRED_STRING) {
-         if (!isObject(input)) return input;
+         if (!isObject$4(input)) return input;
          var fn, val;
-         if (PREFERRED_STRING && typeof (fn = input.toString) == 'function' && !isObject(val = fn.call(input))) return val;
-         if (typeof (fn = input.valueOf) == 'function' && !isObject(val = fn.call(input))) return val;
-         if (!PREFERRED_STRING && typeof (fn = input.toString) == 'function' && !isObject(val = fn.call(input))) return val;
+         if (PREFERRED_STRING && typeof (fn = input.toString) == 'function' && !isObject$4(val = fn.call(input))) return val;
+         if (typeof (fn = input.valueOf) == 'function' && !isObject$4(val = fn.call(input))) return val;
+         if (!PREFERRED_STRING && typeof (fn = input.toString) == 'function' && !isObject$4(val = fn.call(input))) return val;
          throw TypeError("Can't convert object to primitive value");
        };
 
-       var hasOwnProperty = {}.hasOwnProperty;
+       // `ToObject` abstract operation
+       // https://tc39.es/ecma262/#sec-toobject
+       var toObject = function (argument) {
+         return Object(requireObjectCoercible(argument));
+       };
+
+       var hasOwnProperty$3 = {}.hasOwnProperty;
 
-       var has = function (it, key) {
-         return hasOwnProperty.call(it, key);
+       var has$1 = Object.hasOwn || function hasOwn(it, key) {
+         return hasOwnProperty$3.call(toObject(it), key);
        };
 
-       var document$1 = global_1.document;
+       var document$3 = global$2.document;
        // typeof document.createElement is 'object' in old IE
-       var EXISTS = isObject(document$1) && isObject(document$1.createElement);
+       var EXISTS = isObject$4(document$3) && isObject$4(document$3.createElement);
 
        var documentCreateElement = function (it) {
-         return EXISTS ? document$1.createElement(it) : {};
+         return EXISTS ? document$3.createElement(it) : {};
        };
 
        // Thank's IE8 for his funny defineProperty
        var ie8DomDefine = !descriptors && !fails(function () {
+         // eslint-disable-next-line es/no-object-defineproperty -- requied for testing
          return Object.defineProperty(documentCreateElement('div'), 'a', {
            get: function () { return 7; }
          }).a != 7;
        });
 
-       var nativeGetOwnPropertyDescriptor = Object.getOwnPropertyDescriptor;
+       // eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe
+       var $getOwnPropertyDescriptor$1 = Object.getOwnPropertyDescriptor;
 
        // `Object.getOwnPropertyDescriptor` method
        // https://tc39.es/ecma262/#sec-object.getownpropertydescriptor
-       var f$1 = descriptors ? nativeGetOwnPropertyDescriptor : function getOwnPropertyDescriptor(O, P) {
+       var f$6 = descriptors ? $getOwnPropertyDescriptor$1 : function getOwnPropertyDescriptor(O, P) {
          O = toIndexedObject(O);
          P = toPrimitive(P, true);
          if (ie8DomDefine) try {
-           return nativeGetOwnPropertyDescriptor(O, P);
+           return $getOwnPropertyDescriptor$1(O, P);
          } catch (error) { /* empty */ }
-         if (has(O, P)) return createPropertyDescriptor(!objectPropertyIsEnumerable.f.call(O, P), O[P]);
+         if (has$1(O, P)) return createPropertyDescriptor(!objectPropertyIsEnumerable.f.call(O, P), O[P]);
        };
 
        var objectGetOwnPropertyDescriptor = {
-               f: f$1
+               f: f$6
        };
 
        var anObject = function (it) {
-         if (!isObject(it)) {
+         if (!isObject$4(it)) {
            throw TypeError(String(it) + ' is not an object');
          } return it;
        };
 
-       var nativeDefineProperty = Object.defineProperty;
+       // eslint-disable-next-line es/no-object-defineproperty -- safe
+       var $defineProperty$1 = Object.defineProperty;
 
        // `Object.defineProperty` method
        // https://tc39.es/ecma262/#sec-object.defineproperty
-       var f$2 = descriptors ? nativeDefineProperty : function defineProperty(O, P, Attributes) {
+       var f$5 = descriptors ? $defineProperty$1 : function defineProperty(O, P, Attributes) {
          anObject(O);
          P = toPrimitive(P, true);
          anObject(Attributes);
          if (ie8DomDefine) try {
-           return nativeDefineProperty(O, P, Attributes);
+           return $defineProperty$1(O, P, Attributes);
          } catch (error) { /* empty */ }
          if ('get' in Attributes || 'set' in Attributes) throw TypeError('Accessors not supported');
          if ('value' in Attributes) O[P] = Attributes.value;
        };
 
        var objectDefineProperty = {
-               f: f$2
+               f: f$5
        };
 
        var createNonEnumerableProperty = descriptors ? function (object, key, value) {
 
        var setGlobal = function (key, value) {
          try {
-           createNonEnumerableProperty(global_1, key, value);
+           createNonEnumerableProperty(global$2, key, value);
          } catch (error) {
-           global_1[key] = value;
+           global$2[key] = value;
          } return value;
        };
 
        var SHARED = '__core-js_shared__';
-       var store = global_1[SHARED] || setGlobal(SHARED, {});
+       var store$1 = global$2[SHARED] || setGlobal(SHARED, {});
 
-       var sharedStore = store;
+       var sharedStore = store$1;
 
        var functionToString = Function.toString;
 
-       // this helper broken in `3.4.1-3.4.4`, so we can't use `shared` helper
+       // this helper broken in `core-js@3.4.1-3.4.4`, so we can't use `shared` helper
        if (typeof sharedStore.inspectSource != 'function') {
          sharedStore.inspectSource = function (it) {
            return functionToString.call(it);
 
        var inspectSource = sharedStore.inspectSource;
 
-       var WeakMap = global_1.WeakMap;
+       var WeakMap$1 = global$2.WeakMap;
 
-       var nativeWeakMap = typeof WeakMap === 'function' && /native code/.test(inspectSource(WeakMap));
+       var nativeWeakMap = typeof WeakMap$1 === 'function' && /native code/.test(inspectSource(WeakMap$1));
 
        var isPure = false;
 
        (module.exports = function (key, value) {
          return sharedStore[key] || (sharedStore[key] = value !== undefined ? value : {});
        })('versions', []).push({
-         version: '3.9.1',
-         mode:  'global',
+         version: '3.15.2',
+         mode: 'global',
          copyright: '© 2021 Denis Pushkarev (zloirock.ru)'
        });
        });
 
-       var id = 0;
+       var id$1 = 0;
        var postfix = Math.random();
 
        var uid = function (key) {
-         return 'Symbol(' + String(key === undefined ? '' : key) + ')_' + (++id + postfix).toString(36);
+         return 'Symbol(' + String(key === undefined ? '' : key) + ')_' + (++id$1 + postfix).toString(36);
        };
 
-       var keys = shared('keys');
+       var keys$3 = shared('keys');
 
        var sharedKey = function (key) {
-         return keys[key] || (keys[key] = uid(key));
+         return keys$3[key] || (keys$3[key] = uid(key));
        };
 
-       var hiddenKeys = {};
+       var hiddenKeys$1 = {};
 
-       var WeakMap$1 = global_1.WeakMap;
-       var set, get, has$1;
+       var OBJECT_ALREADY_INITIALIZED = 'Object already initialized';
+       var WeakMap = global$2.WeakMap;
+       var set$4, get$5, has;
 
        var enforce = function (it) {
-         return has$1(it) ? get(it) : set(it, {});
+         return has(it) ? get$5(it) : set$4(it, {});
        };
 
        var getterFor = function (TYPE) {
          return function (it) {
            var state;
-           if (!isObject(it) || (state = get(it)).type !== TYPE) {
+           if (!isObject$4(it) || (state = get$5(it)).type !== TYPE) {
              throw TypeError('Incompatible receiver, ' + TYPE + ' required');
            } return state;
          };
        };
 
-       if (nativeWeakMap) {
-         var store$1 = sharedStore.state || (sharedStore.state = new WeakMap$1());
-         var wmget = store$1.get;
-         var wmhas = store$1.has;
-         var wmset = store$1.set;
-         set = function (it, metadata) {
+       if (nativeWeakMap || sharedStore.state) {
+         var store = sharedStore.state || (sharedStore.state = new WeakMap());
+         var wmget = store.get;
+         var wmhas = store.has;
+         var wmset = store.set;
+         set$4 = function (it, metadata) {
+           if (wmhas.call(store, it)) throw new TypeError(OBJECT_ALREADY_INITIALIZED);
            metadata.facade = it;
-           wmset.call(store$1, it, metadata);
+           wmset.call(store, it, metadata);
            return metadata;
          };
-         get = function (it) {
-           return wmget.call(store$1, it) || {};
+         get$5 = function (it) {
+           return wmget.call(store, it) || {};
          };
-         has$1 = function (it) {
-           return wmhas.call(store$1, it);
+         has = function (it) {
+           return wmhas.call(store, it);
          };
        } else {
          var STATE = sharedKey('state');
-         hiddenKeys[STATE] = true;
-         set = function (it, metadata) {
+         hiddenKeys$1[STATE] = true;
+         set$4 = function (it, metadata) {
+           if (has$1(it, STATE)) throw new TypeError(OBJECT_ALREADY_INITIALIZED);
            metadata.facade = it;
            createNonEnumerableProperty(it, STATE, metadata);
            return metadata;
          };
-         get = function (it) {
-           return has(it, STATE) ? it[STATE] : {};
+         get$5 = function (it) {
+           return has$1(it, STATE) ? it[STATE] : {};
          };
-         has$1 = function (it) {
-           return has(it, STATE);
+         has = function (it) {
+           return has$1(it, STATE);
          };
        }
 
        var internalState = {
-         set: set,
-         get: get,
-         has: has$1,
+         set: set$4,
+         get: get$5,
+         has: has,
          enforce: enforce,
          getterFor: getterFor
        };
          var noTargetGet = options ? !!options.noTargetGet : false;
          var state;
          if (typeof value == 'function') {
-           if (typeof key == 'string' && !has(value, 'name')) {
+           if (typeof key == 'string' && !has$1(value, 'name')) {
              createNonEnumerableProperty(value, 'name', key);
            }
            state = enforceInternalState(value);
              state.source = TEMPLATE.join(typeof key == 'string' ? key : '');
            }
          }
-         if (O === global_1) {
+         if (O === global$2) {
            if (simple) O[key] = value;
            else setGlobal(key, value);
            return;
        });
        });
 
-       var path = global_1;
+       var path = global$2;
 
-       var aFunction = function (variable) {
+       var aFunction$1 = function (variable) {
          return typeof variable == 'function' ? variable : undefined;
        };
 
        var getBuiltIn = function (namespace, method) {
-         return arguments.length < 2 ? aFunction(path[namespace]) || aFunction(global_1[namespace])
-           : path[namespace] && path[namespace][method] || global_1[namespace] && global_1[namespace][method];
+         return arguments.length < 2 ? aFunction$1(path[namespace]) || aFunction$1(global$2[namespace])
+           : path[namespace] && path[namespace][method] || global$2[namespace] && global$2[namespace][method];
        };
 
-       var ceil = Math.ceil;
-       var floor = Math.floor;
+       var ceil$1 = Math.ceil;
+       var floor$7 = Math.floor;
 
        // `ToInteger` abstract operation
        // https://tc39.es/ecma262/#sec-tointeger
        var toInteger = function (argument) {
-         return isNaN(argument = +argument) ? 0 : (argument > 0 ? floor : ceil)(argument);
+         return isNaN(argument = +argument) ? 0 : (argument > 0 ? floor$7 : ceil$1)(argument);
        };
 
-       var min = Math.min;
+       var min$9 = Math.min;
 
        // `ToLength` abstract operation
        // https://tc39.es/ecma262/#sec-tolength
        var toLength = function (argument) {
-         return argument > 0 ? min(toInteger(argument), 0x1FFFFFFFFFFFFF) : 0; // 2 ** 53 - 1 == 9007199254740991
+         return argument > 0 ? min$9(toInteger(argument), 0x1FFFFFFFFFFFFF) : 0; // 2 ** 53 - 1 == 9007199254740991
        };
 
-       var max = Math.max;
-       var min$1 = Math.min;
+       var max$4 = Math.max;
+       var min$8 = Math.min;
 
        // Helper for a popular repeating case of the spec:
        // Let integer be ? ToInteger(index).
        // If integer < 0, let result be max((length + integer), 0); else let result be min(integer, length).
        var toAbsoluteIndex = function (index, length) {
          var integer = toInteger(index);
-         return integer < 0 ? max(integer + length, 0) : min$1(integer, length);
+         return integer < 0 ? max$4(integer + length, 0) : min$8(integer, length);
        };
 
        // `Array.prototype.{ indexOf, includes }` methods implementation
-       var createMethod = function (IS_INCLUDES) {
+       var createMethod$6 = function (IS_INCLUDES) {
          return function ($this, el, fromIndex) {
            var O = toIndexedObject($this);
            var length = toLength(O.length);
        var arrayIncludes = {
          // `Array.prototype.includes` method
          // https://tc39.es/ecma262/#sec-array.prototype.includes
-         includes: createMethod(true),
+         includes: createMethod$6(true),
          // `Array.prototype.indexOf` method
          // https://tc39.es/ecma262/#sec-array.prototype.indexof
-         indexOf: createMethod(false)
+         indexOf: createMethod$6(false)
        };
 
        var indexOf = arrayIncludes.indexOf;
          var i = 0;
          var result = [];
          var key;
-         for (key in O) !has(hiddenKeys, key) && has(O, key) && result.push(key);
+         for (key in O) !has$1(hiddenKeys$1, key) && has$1(O, key) && result.push(key);
          // Don't enum bug & hidden keys
-         while (names.length > i) if (has(O, key = names[i++])) {
+         while (names.length > i) if (has$1(O, key = names[i++])) {
            ~indexOf(result, key) || result.push(key);
          }
          return result;
          'valueOf'
        ];
 
-       var hiddenKeys$1 = enumBugKeys.concat('length', 'prototype');
+       var hiddenKeys = enumBugKeys.concat('length', 'prototype');
 
        // `Object.getOwnPropertyNames` method
        // https://tc39.es/ecma262/#sec-object.getownpropertynames
-       var f$3 = Object.getOwnPropertyNames || function getOwnPropertyNames(O) {
-         return objectKeysInternal(O, hiddenKeys$1);
+       // eslint-disable-next-line es/no-object-getownpropertynames -- safe
+       var f$4 = Object.getOwnPropertyNames || function getOwnPropertyNames(O) {
+         return objectKeysInternal(O, hiddenKeys);
        };
 
        var objectGetOwnPropertyNames = {
-               f: f$3
+               f: f$4
        };
 
-       var f$4 = Object.getOwnPropertySymbols;
+       // eslint-disable-next-line es/no-object-getownpropertysymbols -- safe
+       var f$3 = Object.getOwnPropertySymbols;
 
        var objectGetOwnPropertySymbols = {
-               f: f$4
+               f: f$3
        };
 
        // all object keys, includes non-enumerable and symbols
          var getOwnPropertyDescriptor = objectGetOwnPropertyDescriptor.f;
          for (var i = 0; i < keys.length; i++) {
            var key = keys[i];
-           if (!has(target, key)) defineProperty(target, key, getOwnPropertyDescriptor(source, key));
+           if (!has$1(target, key)) defineProperty(target, key, getOwnPropertyDescriptor(source, key));
          }
        };
 
        var replacement = /#|\.prototype\./;
 
        var isForced = function (feature, detection) {
-         var value = data[normalize(feature)];
+         var value = data[normalize$1(feature)];
          return value == POLYFILL ? true
            : value == NATIVE ? false
            : typeof detection == 'function' ? fails(detection)
            : !!detection;
        };
 
-       var normalize = isForced.normalize = function (string) {
+       var normalize$1 = isForced.normalize = function (string) {
          return String(string).replace(replacement, '.').toLowerCase();
        };
 
 
        var isForced_1 = isForced;
 
-       var getOwnPropertyDescriptor$1 = objectGetOwnPropertyDescriptor.f;
+       var getOwnPropertyDescriptor$4 = objectGetOwnPropertyDescriptor.f;
 
 
 
          var STATIC = options.stat;
          var FORCED, target, key, targetProperty, sourceProperty, descriptor;
          if (GLOBAL) {
-           target = global_1;
+           target = global$2;
          } else if (STATIC) {
-           target = global_1[TARGET] || setGlobal(TARGET, {});
+           target = global$2[TARGET] || setGlobal(TARGET, {});
          } else {
-           target = (global_1[TARGET] || {}).prototype;
+           target = (global$2[TARGET] || {}).prototype;
          }
          if (target) for (key in source) {
            sourceProperty = source[key];
            if (options.noTargetGet) {
-             descriptor = getOwnPropertyDescriptor$1(target, key);
+             descriptor = getOwnPropertyDescriptor$4(target, key);
              targetProperty = descriptor && descriptor.value;
            } else targetProperty = target[key];
            FORCED = isForced_1(GLOBAL ? key : TARGET + (STATIC ? '.' : '#') + key, options.forced);
          }
        });
 
-       var DatePrototype = Date.prototype;
+       var DatePrototype$1 = Date.prototype;
        var INVALID_DATE = 'Invalid Date';
-       var TO_STRING = 'toString';
-       var nativeDateToString = DatePrototype[TO_STRING];
-       var getTime = DatePrototype.getTime;
+       var TO_STRING$1 = 'toString';
+       var nativeDateToString = DatePrototype$1[TO_STRING$1];
+       var getTime$1 = DatePrototype$1.getTime;
 
        // `Date.prototype.toString` method
        // https://tc39.es/ecma262/#sec-date.prototype.tostring
        if (new Date(NaN) + '' != INVALID_DATE) {
-         redefine(DatePrototype, TO_STRING, function toString() {
-           var value = getTime.call(this);
+         redefine(DatePrototype$1, TO_STRING$1, function toString() {
+           var value = getTime$1.call(this);
            // eslint-disable-next-line no-self-compare -- NaN check
            return value === value ? nativeDateToString.call(this) : INVALID_DATE;
          });
          return _typeof(obj);
        }
 
-       function _classCallCheck(instance, Constructor) {
+       function _classCallCheck$1(instance, Constructor) {
          if (!(instance instanceof Constructor)) {
            throw new TypeError("Cannot call a class as a function");
          }
        }
 
-       function _defineProperties(target, props) {
+       function _defineProperties$1(target, props) {
          for (var i = 0; i < props.length; i++) {
            var descriptor = props[i];
            descriptor.enumerable = descriptor.enumerable || false;
          }
        }
 
-       function _createClass(Constructor, protoProps, staticProps) {
-         if (protoProps) _defineProperties(Constructor.prototype, protoProps);
-         if (staticProps) _defineProperties(Constructor, staticProps);
+       function _createClass$1(Constructor, protoProps, staticProps) {
+         if (protoProps) _defineProperties$1(Constructor.prototype, protoProps);
+         if (staticProps) _defineProperties$1(Constructor, staticProps);
          return Constructor;
        }
 
        }
 
        function _iterableToArray(iter) {
-         if (typeof Symbol !== "undefined" && Symbol.iterator in Object(iter)) return Array.from(iter);
+         if (typeof Symbol !== "undefined" && iter[Symbol.iterator] != null || iter["@@iterator"] != null) return Array.from(iter);
        }
 
        function _iterableToArrayLimit(arr, i) {
-         if (typeof Symbol === "undefined" || !(Symbol.iterator in Object(arr))) return;
+         var _i = arr == null ? null : typeof Symbol !== "undefined" && arr[Symbol.iterator] || arr["@@iterator"];
+
+         if (_i == null) return;
          var _arr = [];
          var _n = true;
          var _d = false;
-         var _e = undefined;
+
+         var _s, _e;
 
          try {
-           for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) {
+           for (_i = _i.call(arr); !(_n = (_s = _i.next()).done); _n = true) {
              _arr.push(_s.value);
 
              if (i && _arr.length === i) break;
        }
 
        function _createForOfIteratorHelper(o, allowArrayLike) {
-         var it;
+         var it = typeof Symbol !== "undefined" && o[Symbol.iterator] || o["@@iterator"];
 
-         if (typeof Symbol === "undefined" || o[Symbol.iterator] == null) {
+         if (!it) {
            if (Array.isArray(o) || (it = _unsupportedIterableToArray(o)) || allowArrayLike && o && typeof o.length === "number") {
              if (it) o = it;
              var i = 0;
              err;
          return {
            s: function () {
-             it = o[Symbol.iterator]();
+             it = it.call(o);
            },
            n: function () {
              var step = it.next();
          };
        }
 
-       var engineIsNode = classofRaw(global_1.process) == 'process';
-
        var engineUserAgent = getBuiltIn('navigator', 'userAgent') || '';
 
-       var process$1 = global_1.process;
-       var versions = process$1 && process$1.versions;
+       var process$4 = global$2.process;
+       var versions = process$4 && process$4.versions;
        var v8 = versions && versions.v8;
-       var match, version;
+       var match, version$1;
 
        if (v8) {
          match = v8.split('.');
-         version = match[0] + match[1];
+         version$1 = match[0] < 4 ? 1 : match[0] + match[1];
        } else if (engineUserAgent) {
          match = engineUserAgent.match(/Edge\/(\d+)/);
          if (!match || match[1] >= 74) {
            match = engineUserAgent.match(/Chrome\/(\d+)/);
-           if (match) version = match[1];
+           if (match) version$1 = match[1];
          }
        }
 
-       var engineV8Version = version && +version;
+       var engineV8Version = version$1 && +version$1;
+
+       /* eslint-disable es/no-symbol -- required for testing */
 
+       // eslint-disable-next-line es/no-object-getownpropertysymbols -- required for testing
        var nativeSymbol = !!Object.getOwnPropertySymbols && !fails(function () {
-         /* global Symbol -- required for testing */
-         return !Symbol.sham &&
-           // Chrome 38 Symbol has incorrect toString conversion
+         var symbol = Symbol();
+         // Chrome 38 Symbol has incorrect toString conversion
+         // `get-own-property-symbols` polyfill symbols converted to object are not Symbol instances
+         return !String(symbol) || !(Object(symbol) instanceof Symbol) ||
            // Chrome 38-40 symbols are not inherited from DOM collections prototypes to instances
-           (engineIsNode ? engineV8Version === 38 : engineV8Version > 37 && engineV8Version < 41);
+           !Symbol.sham && engineV8Version && engineV8Version < 41;
        });
 
+       /* eslint-disable es/no-symbol -- required for testing */
+
        var useSymbolAsUid = nativeSymbol
-         /* global Symbol -- safe */
          && !Symbol.sham
          && typeof Symbol.iterator == 'symbol';
 
-       var WellKnownSymbolsStore = shared('wks');
-       var Symbol$1 = global_1.Symbol;
+       var WellKnownSymbolsStore$1 = shared('wks');
+       var Symbol$1 = global$2.Symbol;
        var createWellKnownSymbol = useSymbolAsUid ? Symbol$1 : Symbol$1 && Symbol$1.withoutSetter || uid;
 
        var wellKnownSymbol = function (name) {
-         if (!has(WellKnownSymbolsStore, name) || !(nativeSymbol || typeof WellKnownSymbolsStore[name] == 'string')) {
-           if (nativeSymbol && has(Symbol$1, name)) {
-             WellKnownSymbolsStore[name] = Symbol$1[name];
+         if (!has$1(WellKnownSymbolsStore$1, name) || !(nativeSymbol || typeof WellKnownSymbolsStore$1[name] == 'string')) {
+           if (nativeSymbol && has$1(Symbol$1, name)) {
+             WellKnownSymbolsStore$1[name] = Symbol$1[name];
            } else {
-             WellKnownSymbolsStore[name] = createWellKnownSymbol('Symbol.' + name);
+             WellKnownSymbolsStore$1[name] = createWellKnownSymbol('Symbol.' + name);
            }
-         } return WellKnownSymbolsStore[name];
+         } return WellKnownSymbolsStore$1[name];
        };
 
-       var f$5 = wellKnownSymbol;
+       var f$2 = wellKnownSymbol;
 
        var wellKnownSymbolWrapped = {
-               f: f$5
+               f: f$2
        };
 
-       var defineProperty = objectDefineProperty.f;
+       var defineProperty$9 = objectDefineProperty.f;
 
        var defineWellKnownSymbol = function (NAME) {
          var Symbol = path.Symbol || (path.Symbol = {});
-         if (!has(Symbol, NAME)) defineProperty(Symbol, NAME, {
+         if (!has$1(Symbol, NAME)) defineProperty$9(Symbol, NAME, {
            value: wellKnownSymbolWrapped.f(NAME)
          });
        };
        // https://tc39.es/ecma262/#sec-symbol.iterator
        defineWellKnownSymbol('iterator');
 
-       var TO_STRING_TAG = wellKnownSymbol('toStringTag');
-       var test = {};
-
-       test[TO_STRING_TAG] = 'z';
-
-       var toStringTagSupport = String(test) === '[object z]';
-
-       var TO_STRING_TAG$1 = wellKnownSymbol('toStringTag');
-       // ES3 wrong here
-       var CORRECT_ARGUMENTS = classofRaw(function () { return arguments; }()) == 'Arguments';
-
-       // fallback for IE11 Script Access Denied error
-       var tryGet = function (it, key) {
-         try {
-           return it[key];
-         } catch (error) { /* empty */ }
-       };
-
-       // getting tag from ES6+ `Object.prototype.toString`
-       var classof = toStringTagSupport ? classofRaw : function (it) {
-         var O, tag, result;
-         return it === undefined ? 'Undefined' : it === null ? 'Null'
-           // @@toStringTag case
-           : typeof (tag = tryGet(O = Object(it), TO_STRING_TAG$1)) == 'string' ? tag
-           // builtinTag case
-           : CORRECT_ARGUMENTS ? classofRaw(O)
-           // ES3 arguments fallback
-           : (result = classofRaw(O)) == 'Object' && typeof O.callee == 'function' ? 'Arguments' : result;
-       };
-
-       // `Object.prototype.toString` method implementation
-       // https://tc39.es/ecma262/#sec-object.prototype.tostring
-       var objectToString = toStringTagSupport ? {}.toString : function toString() {
-         return '[object ' + classof(this) + ']';
-       };
-
-       // `Object.prototype.toString` method
-       // https://tc39.es/ecma262/#sec-object.prototype.tostring
-       if (!toStringTagSupport) {
-         redefine(Object.prototype, 'toString', objectToString, { unsafe: true });
-       }
-
-       // `String.prototype.{ codePointAt, at }` methods implementation
-       var createMethod$1 = function (CONVERT_TO_STRING) {
-         return function ($this, pos) {
-           var S = String(requireObjectCoercible($this));
-           var position = toInteger(pos);
-           var size = S.length;
-           var first, second;
-           if (position < 0 || position >= size) return CONVERT_TO_STRING ? '' : undefined;
-           first = S.charCodeAt(position);
-           return first < 0xD800 || first > 0xDBFF || position + 1 === size
-             || (second = S.charCodeAt(position + 1)) < 0xDC00 || second > 0xDFFF
-               ? CONVERT_TO_STRING ? S.charAt(position) : first
-               : CONVERT_TO_STRING ? S.slice(position, position + 2) : (first - 0xD800 << 10) + (second - 0xDC00) + 0x10000;
-         };
-       };
-
-       var stringMultibyte = {
-         // `String.prototype.codePointAt` method
-         // https://tc39.es/ecma262/#sec-string.prototype.codepointat
-         codeAt: createMethod$1(false),
-         // `String.prototype.at` method
-         // https://github.com/mathiasbynens/String.prototype.at
-         charAt: createMethod$1(true)
-       };
-
-       // `ToObject` abstract operation
-       // https://tc39.es/ecma262/#sec-toobject
-       var toObject = function (argument) {
-         return Object(requireObjectCoercible(argument));
-       };
-
-       var correctPrototypeGetter = !fails(function () {
-         function F() { /* empty */ }
-         F.prototype.constructor = null;
-         return Object.getPrototypeOf(new F()) !== F.prototype;
-       });
-
-       var IE_PROTO = sharedKey('IE_PROTO');
-       var ObjectPrototype = Object.prototype;
-
-       // `Object.getPrototypeOf` method
-       // https://tc39.es/ecma262/#sec-object.getprototypeof
-       var objectGetPrototypeOf = correctPrototypeGetter ? Object.getPrototypeOf : function (O) {
-         O = toObject(O);
-         if (has(O, IE_PROTO)) return O[IE_PROTO];
-         if (typeof O.constructor == 'function' && O instanceof O.constructor) {
-           return O.constructor.prototype;
-         } return O instanceof Object ? ObjectPrototype : null;
-       };
-
-       var ITERATOR = wellKnownSymbol('iterator');
-       var BUGGY_SAFARI_ITERATORS = false;
-
-       var returnThis = function () { return this; };
-
-       // `%IteratorPrototype%` object
-       // https://tc39.es/ecma262/#sec-%iteratorprototype%-object
-       var IteratorPrototype, PrototypeOfArrayIteratorPrototype, arrayIterator;
-
-       if ([].keys) {
-         arrayIterator = [].keys();
-         // Safari 8 has buggy iterators w/o `next`
-         if (!('next' in arrayIterator)) BUGGY_SAFARI_ITERATORS = true;
-         else {
-           PrototypeOfArrayIteratorPrototype = objectGetPrototypeOf(objectGetPrototypeOf(arrayIterator));
-           if (PrototypeOfArrayIteratorPrototype !== Object.prototype) IteratorPrototype = PrototypeOfArrayIteratorPrototype;
-         }
-       }
-
-       var NEW_ITERATOR_PROTOTYPE = IteratorPrototype == undefined || fails(function () {
-         var test = {};
-         // FF44- legacy iterators case
-         return IteratorPrototype[ITERATOR].call(test) !== test;
-       });
-
-       if (NEW_ITERATOR_PROTOTYPE) IteratorPrototype = {};
-
-       // 25.1.2.1.1 %IteratorPrototype%[@@iterator]()
-       if ( !has(IteratorPrototype, ITERATOR)) {
-         createNonEnumerableProperty(IteratorPrototype, ITERATOR, returnThis);
-       }
-
-       var iteratorsCore = {
-         IteratorPrototype: IteratorPrototype,
-         BUGGY_SAFARI_ITERATORS: BUGGY_SAFARI_ITERATORS
-       };
-
        // `Object.keys` method
        // https://tc39.es/ecma262/#sec-object.keys
+       // eslint-disable-next-line es/no-object-keys -- safe
        var objectKeys = Object.keys || function keys(O) {
          return objectKeysInternal(O, enumBugKeys);
        };
 
        // `Object.defineProperties` method
        // https://tc39.es/ecma262/#sec-object.defineproperties
+       // eslint-disable-next-line es/no-object-defineproperties -- safe
        var objectDefineProperties = descriptors ? Object.defineProperties : function defineProperties(O, Properties) {
          anObject(O);
          var keys = objectKeys(Properties);
 
        var GT = '>';
        var LT = '<';
-       var PROTOTYPE = 'prototype';
+       var PROTOTYPE$2 = 'prototype';
        var SCRIPT = 'script';
        var IE_PROTO$1 = sharedKey('IE_PROTO');
 
          } catch (error) { /* ignore */ }
          NullProtoObject = activeXDocument ? NullProtoObjectViaActiveX(activeXDocument) : NullProtoObjectViaIFrame();
          var length = enumBugKeys.length;
-         while (length--) delete NullProtoObject[PROTOTYPE][enumBugKeys[length]];
+         while (length--) delete NullProtoObject[PROTOTYPE$2][enumBugKeys[length]];
          return NullProtoObject();
        };
 
-       hiddenKeys[IE_PROTO$1] = true;
+       hiddenKeys$1[IE_PROTO$1] = true;
 
        // `Object.create` method
        // https://tc39.es/ecma262/#sec-object.create
        var objectCreate = Object.create || function create(O, Properties) {
          var result;
          if (O !== null) {
-           EmptyConstructor[PROTOTYPE] = anObject(O);
+           EmptyConstructor[PROTOTYPE$2] = anObject(O);
            result = new EmptyConstructor();
-           EmptyConstructor[PROTOTYPE] = null;
+           EmptyConstructor[PROTOTYPE$2] = null;
            // add "__proto__" for Object.getPrototypeOf polyfill
            result[IE_PROTO$1] = O;
          } else result = NullProtoObject();
          return Properties === undefined ? result : objectDefineProperties(result, Properties);
        };
 
-       var defineProperty$1 = objectDefineProperty.f;
+       var UNSCOPABLES = wellKnownSymbol('unscopables');
+       var ArrayPrototype$1 = Array.prototype;
+
+       // Array.prototype[@@unscopables]
+       // https://tc39.es/ecma262/#sec-array.prototype-@@unscopables
+       if (ArrayPrototype$1[UNSCOPABLES] == undefined) {
+         objectDefineProperty.f(ArrayPrototype$1, UNSCOPABLES, {
+           configurable: true,
+           value: objectCreate(null)
+         });
+       }
 
+       // add a key to Array.prototype[@@unscopables]
+       var addToUnscopables = function (key) {
+         ArrayPrototype$1[UNSCOPABLES][key] = true;
+       };
 
+       var iterators = {};
 
-       var TO_STRING_TAG$2 = wellKnownSymbol('toStringTag');
+       var correctPrototypeGetter = !fails(function () {
+         function F() { /* empty */ }
+         F.prototype.constructor = null;
+         // eslint-disable-next-line es/no-object-getprototypeof -- required for testing
+         return Object.getPrototypeOf(new F()) !== F.prototype;
+       });
 
-       var setToStringTag = function (it, TAG, STATIC) {
-         if (it && !has(it = STATIC ? it : it.prototype, TO_STRING_TAG$2)) {
-           defineProperty$1(it, TO_STRING_TAG$2, { configurable: true, value: TAG });
+       var IE_PROTO = sharedKey('IE_PROTO');
+       var ObjectPrototype$3 = Object.prototype;
+
+       // `Object.getPrototypeOf` method
+       // https://tc39.es/ecma262/#sec-object.getprototypeof
+       // eslint-disable-next-line es/no-object-getprototypeof -- safe
+       var objectGetPrototypeOf = correctPrototypeGetter ? Object.getPrototypeOf : function (O) {
+         O = toObject(O);
+         if (has$1(O, IE_PROTO)) return O[IE_PROTO];
+         if (typeof O.constructor == 'function' && O instanceof O.constructor) {
+           return O.constructor.prototype;
+         } return O instanceof Object ? ObjectPrototype$3 : null;
+       };
+
+       var ITERATOR$8 = wellKnownSymbol('iterator');
+       var BUGGY_SAFARI_ITERATORS$1 = false;
+
+       var returnThis$2 = function () { return this; };
+
+       // `%IteratorPrototype%` object
+       // https://tc39.es/ecma262/#sec-%iteratorprototype%-object
+       var IteratorPrototype$2, PrototypeOfArrayIteratorPrototype, arrayIterator;
+
+       /* eslint-disable es/no-array-prototype-keys -- safe */
+       if ([].keys) {
+         arrayIterator = [].keys();
+         // Safari 8 has buggy iterators w/o `next`
+         if (!('next' in arrayIterator)) BUGGY_SAFARI_ITERATORS$1 = true;
+         else {
+           PrototypeOfArrayIteratorPrototype = objectGetPrototypeOf(objectGetPrototypeOf(arrayIterator));
+           if (PrototypeOfArrayIteratorPrototype !== Object.prototype) IteratorPrototype$2 = PrototypeOfArrayIteratorPrototype;
          }
+       }
+
+       var NEW_ITERATOR_PROTOTYPE = IteratorPrototype$2 == undefined || fails(function () {
+         var test = {};
+         // FF44- legacy iterators case
+         return IteratorPrototype$2[ITERATOR$8].call(test) !== test;
+       });
+
+       if (NEW_ITERATOR_PROTOTYPE) IteratorPrototype$2 = {};
+
+       // `%IteratorPrototype%[@@iterator]()` method
+       // https://tc39.es/ecma262/#sec-%iteratorprototype%-@@iterator
+       if (!has$1(IteratorPrototype$2, ITERATOR$8)) {
+         createNonEnumerableProperty(IteratorPrototype$2, ITERATOR$8, returnThis$2);
+       }
+
+       var iteratorsCore = {
+         IteratorPrototype: IteratorPrototype$2,
+         BUGGY_SAFARI_ITERATORS: BUGGY_SAFARI_ITERATORS$1
        };
 
-       var iterators = {};
+       var defineProperty$8 = objectDefineProperty.f;
+
+
+
+       var TO_STRING_TAG$4 = wellKnownSymbol('toStringTag');
+
+       var setToStringTag = function (it, TAG, STATIC) {
+         if (it && !has$1(it = STATIC ? it : it.prototype, TO_STRING_TAG$4)) {
+           defineProperty$8(it, TO_STRING_TAG$4, { configurable: true, value: TAG });
+         }
+       };
 
        var IteratorPrototype$1 = iteratorsCore.IteratorPrototype;
 
        };
 
        var aPossiblePrototype = function (it) {
-         if (!isObject(it) && it !== null) {
+         if (!isObject$4(it) && it !== null) {
            throw TypeError("Can't set " + String(it) + ' as a prototype');
          } return it;
        };
 
        /* eslint-disable no-proto -- safe */
 
-
-
        // `Object.setPrototypeOf` method
        // https://tc39.es/ecma262/#sec-object.setprototypeof
        // Works with __proto__ only. Old v8 can't work with null proto objects.
+       // eslint-disable-next-line es/no-object-setprototypeof -- safe
        var objectSetPrototypeOf = Object.setPrototypeOf || ('__proto__' in {} ? function () {
          var CORRECT_SETTER = false;
          var test = {};
          var setter;
          try {
+           // eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe
            setter = Object.getOwnPropertyDescriptor(Object.prototype, '__proto__').set;
            setter.call(test, []);
            CORRECT_SETTER = test instanceof Array;
          };
        }() : undefined);
 
-       var IteratorPrototype$2 = iteratorsCore.IteratorPrototype;
-       var BUGGY_SAFARI_ITERATORS$1 = iteratorsCore.BUGGY_SAFARI_ITERATORS;
-       var ITERATOR$1 = wellKnownSymbol('iterator');
+       var IteratorPrototype = iteratorsCore.IteratorPrototype;
+       var BUGGY_SAFARI_ITERATORS = iteratorsCore.BUGGY_SAFARI_ITERATORS;
+       var ITERATOR$7 = wellKnownSymbol('iterator');
        var KEYS = 'keys';
        var VALUES = 'values';
        var ENTRIES = 'entries';
 
-       var returnThis$2 = function () { return this; };
+       var returnThis = function () { return this; };
 
        var defineIterator = function (Iterable, NAME, IteratorConstructor, next, DEFAULT, IS_SET, FORCED) {
          createIteratorConstructor(IteratorConstructor, NAME, next);
 
          var getIterationMethod = function (KIND) {
            if (KIND === DEFAULT && defaultIterator) return defaultIterator;
-           if (!BUGGY_SAFARI_ITERATORS$1 && KIND in IterablePrototype) return IterablePrototype[KIND];
+           if (!BUGGY_SAFARI_ITERATORS && KIND in IterablePrototype) return IterablePrototype[KIND];
            switch (KIND) {
              case KEYS: return function keys() { return new IteratorConstructor(this, KIND); };
              case VALUES: return function values() { return new IteratorConstructor(this, KIND); };
          var TO_STRING_TAG = NAME + ' Iterator';
          var INCORRECT_VALUES_NAME = false;
          var IterablePrototype = Iterable.prototype;
-         var nativeIterator = IterablePrototype[ITERATOR$1]
+         var nativeIterator = IterablePrototype[ITERATOR$7]
            || IterablePrototype['@@iterator']
            || DEFAULT && IterablePrototype[DEFAULT];
-         var defaultIterator = !BUGGY_SAFARI_ITERATORS$1 && nativeIterator || getIterationMethod(DEFAULT);
+         var defaultIterator = !BUGGY_SAFARI_ITERATORS && nativeIterator || getIterationMethod(DEFAULT);
          var anyNativeIterator = NAME == 'Array' ? IterablePrototype.entries || nativeIterator : nativeIterator;
          var CurrentIteratorPrototype, methods, KEY;
 
          // fix native
          if (anyNativeIterator) {
            CurrentIteratorPrototype = objectGetPrototypeOf(anyNativeIterator.call(new Iterable()));
-           if (IteratorPrototype$2 !== Object.prototype && CurrentIteratorPrototype.next) {
-             if ( objectGetPrototypeOf(CurrentIteratorPrototype) !== IteratorPrototype$2) {
+           if (IteratorPrototype !== Object.prototype && CurrentIteratorPrototype.next) {
+             if (objectGetPrototypeOf(CurrentIteratorPrototype) !== IteratorPrototype) {
                if (objectSetPrototypeOf) {
-                 objectSetPrototypeOf(CurrentIteratorPrototype, IteratorPrototype$2);
-               } else if (typeof CurrentIteratorPrototype[ITERATOR$1] != 'function') {
-                 createNonEnumerableProperty(CurrentIteratorPrototype, ITERATOR$1, returnThis$2);
+                 objectSetPrototypeOf(CurrentIteratorPrototype, IteratorPrototype);
+               } else if (typeof CurrentIteratorPrototype[ITERATOR$7] != 'function') {
+                 createNonEnumerableProperty(CurrentIteratorPrototype, ITERATOR$7, returnThis);
                }
              }
              // Set @@toStringTag to native iterators
            }
          }
 
-         // fix Array#{values, @@iterator}.name in V8 / FF
+         // fix Array.prototype.{ values, @@iterator }.name in V8 / FF
          if (DEFAULT == VALUES && nativeIterator && nativeIterator.name !== VALUES) {
            INCORRECT_VALUES_NAME = true;
            defaultIterator = function values() { return nativeIterator.call(this); };
          }
 
          // define iterator
-         if ( IterablePrototype[ITERATOR$1] !== defaultIterator) {
-           createNonEnumerableProperty(IterablePrototype, ITERATOR$1, defaultIterator);
+         if (IterablePrototype[ITERATOR$7] !== defaultIterator) {
+           createNonEnumerableProperty(IterablePrototype, ITERATOR$7, defaultIterator);
          }
          iterators[NAME] = defaultIterator;
 
              entries: getIterationMethod(ENTRIES)
            };
            if (FORCED) for (KEY in methods) {
-             if (BUGGY_SAFARI_ITERATORS$1 || INCORRECT_VALUES_NAME || !(KEY in IterablePrototype)) {
+             if (BUGGY_SAFARI_ITERATORS || INCORRECT_VALUES_NAME || !(KEY in IterablePrototype)) {
                redefine(IterablePrototype, KEY, methods[KEY]);
              }
-           } else _export({ target: NAME, proto: true, forced: BUGGY_SAFARI_ITERATORS$1 || INCORRECT_VALUES_NAME }, methods);
+           } else _export({ target: NAME, proto: true, forced: BUGGY_SAFARI_ITERATORS || INCORRECT_VALUES_NAME }, methods);
          }
 
          return methods;
        };
 
-       var charAt = stringMultibyte.charAt;
-
-
-
-       var STRING_ITERATOR = 'String Iterator';
-       var setInternalState = internalState.set;
-       var getInternalState = internalState.getterFor(STRING_ITERATOR);
-
-       // `String.prototype[@@iterator]` method
-       // https://tc39.es/ecma262/#sec-string.prototype-@@iterator
-       defineIterator(String, 'String', function (iterated) {
-         setInternalState(this, {
-           type: STRING_ITERATOR,
-           string: String(iterated),
-           index: 0
-         });
-       // `%StringIteratorPrototype%.next` method
-       // https://tc39.es/ecma262/#sec-%stringiteratorprototype%.next
-       }, function next() {
-         var state = getInternalState(this);
-         var string = state.string;
-         var index = state.index;
-         var point;
-         if (index >= string.length) return { value: undefined, done: true };
-         point = charAt(string, index);
-         state.index += point.length;
-         return { value: point, done: false };
-       });
-
-       var UNSCOPABLES = wellKnownSymbol('unscopables');
-       var ArrayPrototype = Array.prototype;
-
-       // Array.prototype[@@unscopables]
-       // https://tc39.es/ecma262/#sec-array.prototype-@@unscopables
-       if (ArrayPrototype[UNSCOPABLES] == undefined) {
-         objectDefineProperty.f(ArrayPrototype, UNSCOPABLES, {
-           configurable: true,
-           value: objectCreate(null)
-         });
-       }
-
-       // add a key to Array.prototype[@@unscopables]
-       var addToUnscopables = function (key) {
-         ArrayPrototype[UNSCOPABLES][key] = true;
-       };
-
        var ARRAY_ITERATOR = 'Array Iterator';
-       var setInternalState$1 = internalState.set;
-       var getInternalState$1 = internalState.getterFor(ARRAY_ITERATOR);
+       var setInternalState$7 = internalState.set;
+       var getInternalState$5 = internalState.getterFor(ARRAY_ITERATOR);
 
        // `Array.prototype.entries` method
        // https://tc39.es/ecma262/#sec-array.prototype.entries
        // `CreateArrayIterator` internal method
        // https://tc39.es/ecma262/#sec-createarrayiterator
        var es_array_iterator = defineIterator(Array, 'Array', function (iterated, kind) {
-         setInternalState$1(this, {
+         setInternalState$7(this, {
            type: ARRAY_ITERATOR,
            target: toIndexedObject(iterated), // target
            index: 0,                          // next index
        // `%ArrayIteratorPrototype%.next` method
        // https://tc39.es/ecma262/#sec-%arrayiteratorprototype%.next
        }, function () {
-         var state = getInternalState$1(this);
+         var state = getInternalState$5(this);
          var target = state.target;
          var kind = state.kind;
          var index = state.index++;
        addToUnscopables('values');
        addToUnscopables('entries');
 
+       var TO_STRING_TAG$3 = wellKnownSymbol('toStringTag');
+       var test$2 = {};
+
+       test$2[TO_STRING_TAG$3] = 'z';
+
+       var toStringTagSupport = String(test$2) === '[object z]';
+
+       var TO_STRING_TAG$2 = wellKnownSymbol('toStringTag');
+       // ES3 wrong here
+       var CORRECT_ARGUMENTS = classofRaw(function () { return arguments; }()) == 'Arguments';
+
+       // fallback for IE11 Script Access Denied error
+       var tryGet = function (it, key) {
+         try {
+           return it[key];
+         } catch (error) { /* empty */ }
+       };
+
+       // getting tag from ES6+ `Object.prototype.toString`
+       var classof = toStringTagSupport ? classofRaw : function (it) {
+         var O, tag, result;
+         return it === undefined ? 'Undefined' : it === null ? 'Null'
+           // @@toStringTag case
+           : typeof (tag = tryGet(O = Object(it), TO_STRING_TAG$2)) == 'string' ? tag
+           // builtinTag case
+           : CORRECT_ARGUMENTS ? classofRaw(O)
+           // ES3 arguments fallback
+           : (result = classofRaw(O)) == 'Object' && typeof O.callee == 'function' ? 'Arguments' : result;
+       };
+
+       // `Object.prototype.toString` method implementation
+       // https://tc39.es/ecma262/#sec-object.prototype.tostring
+       var objectToString$1 = toStringTagSupport ? {}.toString : function toString() {
+         return '[object ' + classof(this) + ']';
+       };
+
+       // `Object.prototype.toString` method
+       // https://tc39.es/ecma262/#sec-object.prototype.tostring
+       if (!toStringTagSupport) {
+         redefine(Object.prototype, 'toString', objectToString$1, { unsafe: true });
+       }
+
+       // `String.prototype.{ codePointAt, at }` methods implementation
+       var createMethod$5 = function (CONVERT_TO_STRING) {
+         return function ($this, pos) {
+           var S = String(requireObjectCoercible($this));
+           var position = toInteger(pos);
+           var size = S.length;
+           var first, second;
+           if (position < 0 || position >= size) return CONVERT_TO_STRING ? '' : undefined;
+           first = S.charCodeAt(position);
+           return first < 0xD800 || first > 0xDBFF || position + 1 === size
+             || (second = S.charCodeAt(position + 1)) < 0xDC00 || second > 0xDFFF
+               ? CONVERT_TO_STRING ? S.charAt(position) : first
+               : CONVERT_TO_STRING ? S.slice(position, position + 2) : (first - 0xD800 << 10) + (second - 0xDC00) + 0x10000;
+         };
+       };
+
+       var stringMultibyte = {
+         // `String.prototype.codePointAt` method
+         // https://tc39.es/ecma262/#sec-string.prototype.codepointat
+         codeAt: createMethod$5(false),
+         // `String.prototype.at` method
+         // https://github.com/mathiasbynens/String.prototype.at
+         charAt: createMethod$5(true)
+       };
+
+       var charAt$1 = stringMultibyte.charAt;
+
+
+
+       var STRING_ITERATOR = 'String Iterator';
+       var setInternalState$6 = internalState.set;
+       var getInternalState$4 = internalState.getterFor(STRING_ITERATOR);
+
+       // `String.prototype[@@iterator]` method
+       // https://tc39.es/ecma262/#sec-string.prototype-@@iterator
+       defineIterator(String, 'String', function (iterated) {
+         setInternalState$6(this, {
+           type: STRING_ITERATOR,
+           string: String(iterated),
+           index: 0
+         });
+       // `%StringIteratorPrototype%.next` method
+       // https://tc39.es/ecma262/#sec-%stringiteratorprototype%.next
+       }, function next() {
+         var state = getInternalState$4(this);
+         var string = state.string;
+         var index = state.index;
+         var point;
+         if (index >= string.length) return { value: undefined, done: true };
+         point = charAt$1(string, index);
+         state.index += point.length;
+         return { value: point, done: false };
+       });
+
        // iterable DOM collections
        // flag - `iterable` interface - 'entries', 'keys', 'values', 'forEach' methods
        var domIterables = {
          TouchList: 0
        };
 
-       var ITERATOR$2 = wellKnownSymbol('iterator');
-       var TO_STRING_TAG$3 = wellKnownSymbol('toStringTag');
+       var ITERATOR$6 = wellKnownSymbol('iterator');
+       var TO_STRING_TAG$1 = wellKnownSymbol('toStringTag');
        var ArrayValues = es_array_iterator.values;
 
-       for (var COLLECTION_NAME in domIterables) {
-         var Collection = global_1[COLLECTION_NAME];
-         var CollectionPrototype = Collection && Collection.prototype;
-         if (CollectionPrototype) {
+       for (var COLLECTION_NAME$1 in domIterables) {
+         var Collection$1 = global$2[COLLECTION_NAME$1];
+         var CollectionPrototype$1 = Collection$1 && Collection$1.prototype;
+         if (CollectionPrototype$1) {
            // some Chrome versions have non-configurable methods on DOMTokenList
-           if (CollectionPrototype[ITERATOR$2] !== ArrayValues) try {
-             createNonEnumerableProperty(CollectionPrototype, ITERATOR$2, ArrayValues);
+           if (CollectionPrototype$1[ITERATOR$6] !== ArrayValues) try {
+             createNonEnumerableProperty(CollectionPrototype$1, ITERATOR$6, ArrayValues);
            } catch (error) {
-             CollectionPrototype[ITERATOR$2] = ArrayValues;
+             CollectionPrototype$1[ITERATOR$6] = ArrayValues;
            }
-           if (!CollectionPrototype[TO_STRING_TAG$3]) {
-             createNonEnumerableProperty(CollectionPrototype, TO_STRING_TAG$3, COLLECTION_NAME);
+           if (!CollectionPrototype$1[TO_STRING_TAG$1]) {
+             createNonEnumerableProperty(CollectionPrototype$1, TO_STRING_TAG$1, COLLECTION_NAME$1);
            }
-           if (domIterables[COLLECTION_NAME]) for (var METHOD_NAME in es_array_iterator) {
+           if (domIterables[COLLECTION_NAME$1]) for (var METHOD_NAME in es_array_iterator) {
              // some Chrome versions have non-configurable methods on DOMTokenList
-             if (CollectionPrototype[METHOD_NAME] !== es_array_iterator[METHOD_NAME]) try {
-               createNonEnumerableProperty(CollectionPrototype, METHOD_NAME, es_array_iterator[METHOD_NAME]);
+             if (CollectionPrototype$1[METHOD_NAME] !== es_array_iterator[METHOD_NAME]) try {
+               createNonEnumerableProperty(CollectionPrototype$1, METHOD_NAME, es_array_iterator[METHOD_NAME]);
              } catch (error) {
-               CollectionPrototype[METHOD_NAME] = es_array_iterator[METHOD_NAME];
+               CollectionPrototype$1[METHOD_NAME] = es_array_iterator[METHOD_NAME];
              }
            }
          }
 
        // `IsArray` abstract operation
        // https://tc39.es/ecma262/#sec-isarray
+       // eslint-disable-next-line es/no-array-isarray -- safe
        var isArray = Array.isArray || function isArray(arg) {
          return classofRaw(arg) == 'Array';
        };
 
-       var nativeGetOwnPropertyNames = objectGetOwnPropertyNames.f;
+       /* eslint-disable es/no-object-getownpropertynames -- safe */
 
-       var toString$1 = {}.toString;
+       var $getOwnPropertyNames$1 = objectGetOwnPropertyNames.f;
+
+       var toString = {}.toString;
 
        var windowNames = typeof window == 'object' && window && Object.getOwnPropertyNames
          ? Object.getOwnPropertyNames(window) : [];
 
        var getWindowNames = function (it) {
          try {
-           return nativeGetOwnPropertyNames(it);
+           return $getOwnPropertyNames$1(it);
          } catch (error) {
            return windowNames.slice();
          }
        };
 
        // fallback for IE11 buggy Object.getOwnPropertyNames with iframe and window
-       var f$6 = function getOwnPropertyNames(it) {
-         return windowNames && toString$1.call(it) == '[object Window]'
+       var f$1 = function getOwnPropertyNames(it) {
+         return windowNames && toString.call(it) == '[object Window]'
            ? getWindowNames(it)
-           : nativeGetOwnPropertyNames(toIndexedObject(it));
+           : $getOwnPropertyNames$1(toIndexedObject(it));
        };
 
        var objectGetOwnPropertyNamesExternal = {
-               f: f$6
+               f: f$1
        };
 
-       var aFunction$1 = function (it) {
+       var aFunction = function (it) {
          if (typeof it != 'function') {
            throw TypeError(String(it) + ' is not a function');
          } return it;
 
        // optional / simple context binding
        var functionBindContext = function (fn, that, length) {
-         aFunction$1(fn);
+         aFunction(fn);
          if (that === undefined) return fn;
          switch (length) {
            case 0: return function () {
          };
        };
 
-       var SPECIES = wellKnownSymbol('species');
+       var SPECIES$6 = wellKnownSymbol('species');
 
        // `ArraySpeciesCreate` abstract operation
        // https://tc39.es/ecma262/#sec-arrayspeciescreate
            C = originalArray.constructor;
            // cross-realm fallback
            if (typeof C == 'function' && (C === Array || isArray(C.prototype))) C = undefined;
-           else if (isObject(C)) {
-             C = C[SPECIES];
+           else if (isObject$4(C)) {
+             C = C[SPECIES$6];
              if (C === null) C = undefined;
            }
          } return new (C === undefined ? Array : C)(length === 0 ? 0 : length);
        var push = [].push;
 
        // `Array.prototype.{ forEach, map, filter, some, every, find, findIndex, filterOut }` methods implementation
-       var createMethod$2 = function (TYPE) {
+       var createMethod$4 = function (TYPE) {
          var IS_MAP = TYPE == 1;
          var IS_FILTER = TYPE == 2;
          var IS_SOME = TYPE == 3;
        var arrayIteration = {
          // `Array.prototype.forEach` method
          // https://tc39.es/ecma262/#sec-array.prototype.foreach
-         forEach: createMethod$2(0),
+         forEach: createMethod$4(0),
          // `Array.prototype.map` method
          // https://tc39.es/ecma262/#sec-array.prototype.map
-         map: createMethod$2(1),
+         map: createMethod$4(1),
          // `Array.prototype.filter` method
          // https://tc39.es/ecma262/#sec-array.prototype.filter
-         filter: createMethod$2(2),
+         filter: createMethod$4(2),
          // `Array.prototype.some` method
          // https://tc39.es/ecma262/#sec-array.prototype.some
-         some: createMethod$2(3),
+         some: createMethod$4(3),
          // `Array.prototype.every` method
          // https://tc39.es/ecma262/#sec-array.prototype.every
-         every: createMethod$2(4),
+         every: createMethod$4(4),
          // `Array.prototype.find` method
          // https://tc39.es/ecma262/#sec-array.prototype.find
-         find: createMethod$2(5),
+         find: createMethod$4(5),
          // `Array.prototype.findIndex` method
          // https://tc39.es/ecma262/#sec-array.prototype.findIndex
-         findIndex: createMethod$2(6),
+         findIndex: createMethod$4(6),
          // `Array.prototype.filterOut` method
          // https://github.com/tc39/proposal-array-filtering
-         filterOut: createMethod$2(7)
+         filterOut: createMethod$4(7)
        };
 
-       var $forEach = arrayIteration.forEach;
+       var $forEach$2 = arrayIteration.forEach;
 
        var HIDDEN = sharedKey('hidden');
        var SYMBOL = 'Symbol';
        var PROTOTYPE$1 = 'prototype';
        var TO_PRIMITIVE = wellKnownSymbol('toPrimitive');
-       var setInternalState$2 = internalState.set;
-       var getInternalState$2 = internalState.getterFor(SYMBOL);
-       var ObjectPrototype$1 = Object[PROTOTYPE$1];
-       var $Symbol = global_1.Symbol;
+       var setInternalState$5 = internalState.set;
+       var getInternalState$3 = internalState.getterFor(SYMBOL);
+       var ObjectPrototype$2 = Object[PROTOTYPE$1];
+       var $Symbol = global$2.Symbol;
        var $stringify = getBuiltIn('JSON', 'stringify');
        var nativeGetOwnPropertyDescriptor$1 = objectGetOwnPropertyDescriptor.f;
-       var nativeDefineProperty$1 = objectDefineProperty.f;
-       var nativeGetOwnPropertyNames$1 = objectGetOwnPropertyNamesExternal.f;
-       var nativePropertyIsEnumerable$1 = objectPropertyIsEnumerable.f;
+       var nativeDefineProperty = objectDefineProperty.f;
+       var nativeGetOwnPropertyNames = objectGetOwnPropertyNamesExternal.f;
+       var nativePropertyIsEnumerable = objectPropertyIsEnumerable.f;
        var AllSymbols = shared('symbols');
        var ObjectPrototypeSymbols = shared('op-symbols');
        var StringToSymbolRegistry = shared('string-to-symbol-registry');
        var SymbolToStringRegistry = shared('symbol-to-string-registry');
-       var WellKnownSymbolsStore$1 = shared('wks');
-       var QObject = global_1.QObject;
+       var WellKnownSymbolsStore = shared('wks');
+       var QObject = global$2.QObject;
        // Don't use setters in Qt Script, https://github.com/zloirock/core-js/issues/173
        var USE_SETTER = !QObject || !QObject[PROTOTYPE$1] || !QObject[PROTOTYPE$1].findChild;
 
        // fallback for old Android, https://code.google.com/p/v8/issues/detail?id=687
        var setSymbolDescriptor = descriptors && fails(function () {
-         return objectCreate(nativeDefineProperty$1({}, 'a', {
-           get: function () { return nativeDefineProperty$1(this, 'a', { value: 7 }).a; }
+         return objectCreate(nativeDefineProperty({}, 'a', {
+           get: function () { return nativeDefineProperty(this, 'a', { value: 7 }).a; }
          })).a != 7;
        }) ? function (O, P, Attributes) {
-         var ObjectPrototypeDescriptor = nativeGetOwnPropertyDescriptor$1(ObjectPrototype$1, P);
-         if (ObjectPrototypeDescriptor) delete ObjectPrototype$1[P];
-         nativeDefineProperty$1(O, P, Attributes);
-         if (ObjectPrototypeDescriptor && O !== ObjectPrototype$1) {
-           nativeDefineProperty$1(ObjectPrototype$1, P, ObjectPrototypeDescriptor);
+         var ObjectPrototypeDescriptor = nativeGetOwnPropertyDescriptor$1(ObjectPrototype$2, P);
+         if (ObjectPrototypeDescriptor) delete ObjectPrototype$2[P];
+         nativeDefineProperty(O, P, Attributes);
+         if (ObjectPrototypeDescriptor && O !== ObjectPrototype$2) {
+           nativeDefineProperty(ObjectPrototype$2, P, ObjectPrototypeDescriptor);
          }
-       } : nativeDefineProperty$1;
+       } : nativeDefineProperty;
 
-       var wrap = function (tag, description) {
+       var wrap$2 = function (tag, description) {
          var symbol = AllSymbols[tag] = objectCreate($Symbol[PROTOTYPE$1]);
-         setInternalState$2(symbol, {
+         setInternalState$5(symbol, {
            type: SYMBOL,
            tag: tag,
            description: description
          return symbol;
        };
 
-       var isSymbol = useSymbolAsUid ? function (it) {
+       var isSymbol$1 = useSymbolAsUid ? function (it) {
          return typeof it == 'symbol';
        } : function (it) {
          return Object(it) instanceof $Symbol;
        };
 
        var $defineProperty = function defineProperty(O, P, Attributes) {
-         if (O === ObjectPrototype$1) $defineProperty(ObjectPrototypeSymbols, P, Attributes);
+         if (O === ObjectPrototype$2) $defineProperty(ObjectPrototypeSymbols, P, Attributes);
          anObject(O);
          var key = toPrimitive(P, true);
          anObject(Attributes);
-         if (has(AllSymbols, key)) {
+         if (has$1(AllSymbols, key)) {
            if (!Attributes.enumerable) {
-             if (!has(O, HIDDEN)) nativeDefineProperty$1(O, HIDDEN, createPropertyDescriptor(1, {}));
+             if (!has$1(O, HIDDEN)) nativeDefineProperty(O, HIDDEN, createPropertyDescriptor(1, {}));
              O[HIDDEN][key] = true;
            } else {
-             if (has(O, HIDDEN) && O[HIDDEN][key]) O[HIDDEN][key] = false;
+             if (has$1(O, HIDDEN) && O[HIDDEN][key]) O[HIDDEN][key] = false;
              Attributes = objectCreate(Attributes, { enumerable: createPropertyDescriptor(0, false) });
            } return setSymbolDescriptor(O, key, Attributes);
-         } return nativeDefineProperty$1(O, key, Attributes);
+         } return nativeDefineProperty(O, key, Attributes);
        };
 
        var $defineProperties = function defineProperties(O, Properties) {
          anObject(O);
          var properties = toIndexedObject(Properties);
          var keys = objectKeys(properties).concat($getOwnPropertySymbols(properties));
-         $forEach(keys, function (key) {
+         $forEach$2(keys, function (key) {
            if (!descriptors || $propertyIsEnumerable.call(properties, key)) $defineProperty(O, key, properties[key]);
          });
          return O;
 
        var $propertyIsEnumerable = function propertyIsEnumerable(V) {
          var P = toPrimitive(V, true);
-         var enumerable = nativePropertyIsEnumerable$1.call(this, P);
-         if (this === ObjectPrototype$1 && has(AllSymbols, P) && !has(ObjectPrototypeSymbols, P)) return false;
-         return enumerable || !has(this, P) || !has(AllSymbols, P) || has(this, HIDDEN) && this[HIDDEN][P] ? enumerable : true;
+         var enumerable = nativePropertyIsEnumerable.call(this, P);
+         if (this === ObjectPrototype$2 && has$1(AllSymbols, P) && !has$1(ObjectPrototypeSymbols, P)) return false;
+         return enumerable || !has$1(this, P) || !has$1(AllSymbols, P) || has$1(this, HIDDEN) && this[HIDDEN][P] ? enumerable : true;
        };
 
        var $getOwnPropertyDescriptor = function getOwnPropertyDescriptor(O, P) {
          var it = toIndexedObject(O);
          var key = toPrimitive(P, true);
-         if (it === ObjectPrototype$1 && has(AllSymbols, key) && !has(ObjectPrototypeSymbols, key)) return;
+         if (it === ObjectPrototype$2 && has$1(AllSymbols, key) && !has$1(ObjectPrototypeSymbols, key)) return;
          var descriptor = nativeGetOwnPropertyDescriptor$1(it, key);
-         if (descriptor && has(AllSymbols, key) && !(has(it, HIDDEN) && it[HIDDEN][key])) {
+         if (descriptor && has$1(AllSymbols, key) && !(has$1(it, HIDDEN) && it[HIDDEN][key])) {
            descriptor.enumerable = true;
          }
          return descriptor;
        };
 
        var $getOwnPropertyNames = function getOwnPropertyNames(O) {
-         var names = nativeGetOwnPropertyNames$1(toIndexedObject(O));
+         var names = nativeGetOwnPropertyNames(toIndexedObject(O));
          var result = [];
-         $forEach(names, function (key) {
-           if (!has(AllSymbols, key) && !has(hiddenKeys, key)) result.push(key);
+         $forEach$2(names, function (key) {
+           if (!has$1(AllSymbols, key) && !has$1(hiddenKeys$1, key)) result.push(key);
          });
          return result;
        };
 
        var $getOwnPropertySymbols = function getOwnPropertySymbols(O) {
-         var IS_OBJECT_PROTOTYPE = O === ObjectPrototype$1;
-         var names = nativeGetOwnPropertyNames$1(IS_OBJECT_PROTOTYPE ? ObjectPrototypeSymbols : toIndexedObject(O));
+         var IS_OBJECT_PROTOTYPE = O === ObjectPrototype$2;
+         var names = nativeGetOwnPropertyNames(IS_OBJECT_PROTOTYPE ? ObjectPrototypeSymbols : toIndexedObject(O));
          var result = [];
-         $forEach(names, function (key) {
-           if (has(AllSymbols, key) && (!IS_OBJECT_PROTOTYPE || has(ObjectPrototype$1, key))) {
+         $forEach$2(names, function (key) {
+           if (has$1(AllSymbols, key) && (!IS_OBJECT_PROTOTYPE || has$1(ObjectPrototype$2, key))) {
              result.push(AllSymbols[key]);
            }
          });
            var description = !arguments.length || arguments[0] === undefined ? undefined : String(arguments[0]);
            var tag = uid(description);
            var setter = function (value) {
-             if (this === ObjectPrototype$1) setter.call(ObjectPrototypeSymbols, value);
-             if (has(this, HIDDEN) && has(this[HIDDEN], tag)) this[HIDDEN][tag] = false;
+             if (this === ObjectPrototype$2) setter.call(ObjectPrototypeSymbols, value);
+             if (has$1(this, HIDDEN) && has$1(this[HIDDEN], tag)) this[HIDDEN][tag] = false;
              setSymbolDescriptor(this, tag, createPropertyDescriptor(1, value));
            };
-           if (descriptors && USE_SETTER) setSymbolDescriptor(ObjectPrototype$1, tag, { configurable: true, set: setter });
-           return wrap(tag, description);
+           if (descriptors && USE_SETTER) setSymbolDescriptor(ObjectPrototype$2, tag, { configurable: true, set: setter });
+           return wrap$2(tag, description);
          };
 
          redefine($Symbol[PROTOTYPE$1], 'toString', function toString() {
-           return getInternalState$2(this).tag;
+           return getInternalState$3(this).tag;
          });
 
          redefine($Symbol, 'withoutSetter', function (description) {
-           return wrap(uid(description), description);
+           return wrap$2(uid(description), description);
          });
 
          objectPropertyIsEnumerable.f = $propertyIsEnumerable;
          objectGetOwnPropertySymbols.f = $getOwnPropertySymbols;
 
          wellKnownSymbolWrapped.f = function (name) {
-           return wrap(wellKnownSymbol(name), name);
+           return wrap$2(wellKnownSymbol(name), name);
          };
 
          if (descriptors) {
            // https://github.com/tc39/proposal-Symbol-description
-           nativeDefineProperty$1($Symbol[PROTOTYPE$1], 'description', {
+           nativeDefineProperty($Symbol[PROTOTYPE$1], 'description', {
              configurable: true,
              get: function description() {
-               return getInternalState$2(this).description;
+               return getInternalState$3(this).description;
              }
            });
            {
-             redefine(ObjectPrototype$1, 'propertyIsEnumerable', $propertyIsEnumerable, { unsafe: true });
+             redefine(ObjectPrototype$2, 'propertyIsEnumerable', $propertyIsEnumerable, { unsafe: true });
            }
          }
        }
          Symbol: $Symbol
        });
 
-       $forEach(objectKeys(WellKnownSymbolsStore$1), function (name) {
+       $forEach$2(objectKeys(WellKnownSymbolsStore), function (name) {
          defineWellKnownSymbol(name);
        });
 
          // https://tc39.es/ecma262/#sec-symbol.for
          'for': function (key) {
            var string = String(key);
-           if (has(StringToSymbolRegistry, string)) return StringToSymbolRegistry[string];
+           if (has$1(StringToSymbolRegistry, string)) return StringToSymbolRegistry[string];
            var symbol = $Symbol(string);
            StringToSymbolRegistry[string] = symbol;
            SymbolToStringRegistry[symbol] = string;
          // `Symbol.keyFor` method
          // https://tc39.es/ecma262/#sec-symbol.keyfor
          keyFor: function keyFor(sym) {
-           if (!isSymbol(sym)) throw TypeError(sym + ' is not a symbol');
-           if (has(SymbolToStringRegistry, sym)) return SymbolToStringRegistry[sym];
+           if (!isSymbol$1(sym)) throw TypeError(sym + ' is not a symbol');
+           if (has$1(SymbolToStringRegistry, sym)) return SymbolToStringRegistry[sym];
          },
          useSetter: function () { USE_SETTER = true; },
          useSimple: function () { USE_SETTER = false; }
              var $replacer;
              while (arguments.length > index) args.push(arguments[index++]);
              $replacer = replacer;
-             if (!isObject(replacer) && it === undefined || isSymbol(it)) return; // IE8 returns string on undefined
+             if (!isObject$4(replacer) && it === undefined || isSymbol$1(it)) return; // IE8 returns string on undefined
              if (!isArray(replacer)) replacer = function (key, value) {
                if (typeof $replacer == 'function') value = $replacer.call(this, key, value);
-               if (!isSymbol(value)) return value;
+               if (!isSymbol$1(value)) return value;
              };
              args[1] = replacer;
              return $stringify.apply(null, args);
        // https://tc39.es/ecma262/#sec-symbol.prototype-@@tostringtag
        setToStringTag($Symbol, SYMBOL);
 
-       hiddenKeys[HIDDEN] = true;
+       hiddenKeys$1[HIDDEN] = true;
 
-       var defineProperty$2 = objectDefineProperty.f;
+       var defineProperty$7 = objectDefineProperty.f;
 
 
-       var NativeSymbol = global_1.Symbol;
+       var NativeSymbol = global$2.Symbol;
 
        if (descriptors && typeof NativeSymbol == 'function' && (!('description' in NativeSymbol.prototype) ||
          // Safari 12 bug
          var symbolToString = symbolPrototype.toString;
          var native = String(NativeSymbol('test')) == 'Symbol(test)';
          var regexp = /^Symbol\((.*)\)[^)]+$/;
-         defineProperty$2(symbolPrototype, 'description', {
+         defineProperty$7(symbolPrototype, 'description', {
            configurable: true,
            get: function description() {
-             var symbol = isObject(this) ? this.valueOf() : this;
+             var symbol = isObject$4(this) ? this.valueOf() : this;
              var string = symbolToString.call(symbol);
-             if (has(EmptyStringDescriptionStore, symbol)) return '';
+             if (has$1(EmptyStringDescriptionStore, symbol)) return '';
              var desc = native ? string.slice(7, -1) : string.replace(regexp, '$1');
              return desc === '' ? undefined : desc;
            }
          });
        }
 
+       // eslint-disable-next-line es/no-typed-arrays -- safe
        var arrayBufferNative = typeof ArrayBuffer !== 'undefined' && typeof DataView !== 'undefined';
 
        var redefineAll = function (target, src, options) {
        };
 
        // IEEE754 conversions based on https://github.com/feross/ieee754
-       var abs = Math.abs;
-       var pow = Math.pow;
-       var floor$1 = Math.floor;
-       var log = Math.log;
+       var abs$4 = Math.abs;
+       var pow$2 = Math.pow;
+       var floor$6 = Math.floor;
+       var log$2 = Math.log;
        var LN2 = Math.LN2;
 
        var pack = function (number, mantissaLength, bytes) {
          var exponentLength = bytes * 8 - mantissaLength - 1;
          var eMax = (1 << exponentLength) - 1;
          var eBias = eMax >> 1;
-         var rt = mantissaLength === 23 ? pow(2, -24) - pow(2, -77) : 0;
+         var rt = mantissaLength === 23 ? pow$2(2, -24) - pow$2(2, -77) : 0;
          var sign = number < 0 || number === 0 && 1 / number < 0 ? 1 : 0;
          var index = 0;
          var exponent, mantissa, c;
-         number = abs(number);
+         number = abs$4(number);
          // eslint-disable-next-line no-self-compare -- NaN check
          if (number != number || number === Infinity) {
            // eslint-disable-next-line no-self-compare -- NaN check
            mantissa = number != number ? 1 : 0;
            exponent = eMax;
          } else {
-           exponent = floor$1(log(number) / LN2);
-           if (number * (c = pow(2, -exponent)) < 1) {
+           exponent = floor$6(log$2(number) / LN2);
+           if (number * (c = pow$2(2, -exponent)) < 1) {
              exponent--;
              c *= 2;
            }
            if (exponent + eBias >= 1) {
              number += rt / c;
            } else {
-             number += rt * pow(2, 1 - eBias);
+             number += rt * pow$2(2, 1 - eBias);
            }
            if (number * c >= 2) {
              exponent++;
              mantissa = 0;
              exponent = eMax;
            } else if (exponent + eBias >= 1) {
-             mantissa = (number * c - 1) * pow(2, mantissaLength);
+             mantissa = (number * c - 1) * pow$2(2, mantissaLength);
              exponent = exponent + eBias;
            } else {
-             mantissa = number * pow(2, eBias - 1) * pow(2, mantissaLength);
+             mantissa = number * pow$2(2, eBias - 1) * pow$2(2, mantissaLength);
              exponent = 0;
            }
          }
          } else if (exponent === eMax) {
            return mantissa ? NaN : sign ? -Infinity : Infinity;
          } else {
-           mantissa = mantissa + pow(2, mantissaLength);
+           mantissa = mantissa + pow$2(2, mantissaLength);
            exponent = exponent - eBias;
-         } return (sign ? -1 : 1) * mantissa * pow(2, exponent - mantissaLength);
+         } return (sign ? -1 : 1) * mantissa * pow$2(2, exponent - mantissaLength);
        };
 
-       var ieee754 = {
+       var ieee754$1 = {
          pack: pack,
          unpack: unpack
        };
          return O;
        };
 
-       var getOwnPropertyNames = objectGetOwnPropertyNames.f;
-       var defineProperty$3 = objectDefineProperty.f;
+       var getOwnPropertyNames$3 = objectGetOwnPropertyNames.f;
+       var defineProperty$6 = objectDefineProperty.f;
 
 
 
 
-       var getInternalState$3 = internalState.get;
-       var setInternalState$3 = internalState.set;
-       var ARRAY_BUFFER = 'ArrayBuffer';
+       var getInternalState$2 = internalState.get;
+       var setInternalState$4 = internalState.set;
+       var ARRAY_BUFFER$1 = 'ArrayBuffer';
        var DATA_VIEW = 'DataView';
-       var PROTOTYPE$2 = 'prototype';
+       var PROTOTYPE = 'prototype';
        var WRONG_LENGTH = 'Wrong length';
        var WRONG_INDEX = 'Wrong index';
-       var NativeArrayBuffer = global_1[ARRAY_BUFFER];
-       var $ArrayBuffer = NativeArrayBuffer;
-       var $DataView = global_1[DATA_VIEW];
-       var $DataViewPrototype = $DataView && $DataView[PROTOTYPE$2];
-       var ObjectPrototype$2 = Object.prototype;
-       var RangeError$1 = global_1.RangeError;
+       var NativeArrayBuffer$1 = global$2[ARRAY_BUFFER$1];
+       var $ArrayBuffer = NativeArrayBuffer$1;
+       var $DataView = global$2[DATA_VIEW];
+       var $DataViewPrototype = $DataView && $DataView[PROTOTYPE];
+       var ObjectPrototype$1 = Object.prototype;
+       var RangeError$1 = global$2.RangeError;
 
-       var packIEEE754 = ieee754.pack;
-       var unpackIEEE754 = ieee754.unpack;
+       var packIEEE754 = ieee754$1.pack;
+       var unpackIEEE754 = ieee754$1.unpack;
 
        var packInt8 = function (number) {
          return [number & 0xFF];
        };
 
        var addGetter = function (Constructor, key) {
-         defineProperty$3(Constructor[PROTOTYPE$2], key, { get: function () { return getInternalState$3(this)[key]; } });
+         defineProperty$6(Constructor[PROTOTYPE], key, { get: function () { return getInternalState$2(this)[key]; } });
        };
 
-       var get$1 = function (view, count, index, isLittleEndian) {
+       var get$4 = function (view, count, index, isLittleEndian) {
          var intIndex = toIndex(index);
-         var store = getInternalState$3(view);
+         var store = getInternalState$2(view);
          if (intIndex + count > store.byteLength) throw RangeError$1(WRONG_INDEX);
-         var bytes = getInternalState$3(store.buffer).bytes;
+         var bytes = getInternalState$2(store.buffer).bytes;
          var start = intIndex + store.byteOffset;
          var pack = bytes.slice(start, start + count);
          return isLittleEndian ? pack : pack.reverse();
        };
 
-       var set$1 = function (view, count, index, conversion, value, isLittleEndian) {
+       var set$3 = function (view, count, index, conversion, value, isLittleEndian) {
          var intIndex = toIndex(index);
-         var store = getInternalState$3(view);
+         var store = getInternalState$2(view);
          if (intIndex + count > store.byteLength) throw RangeError$1(WRONG_INDEX);
-         var bytes = getInternalState$3(store.buffer).bytes;
+         var bytes = getInternalState$2(store.buffer).bytes;
          var start = intIndex + store.byteOffset;
          var pack = conversion(+value);
          for (var i = 0; i < count; i++) bytes[start + i] = pack[isLittleEndian ? i : count - i - 1];
 
        if (!arrayBufferNative) {
          $ArrayBuffer = function ArrayBuffer(length) {
-           anInstance(this, $ArrayBuffer, ARRAY_BUFFER);
+           anInstance(this, $ArrayBuffer, ARRAY_BUFFER$1);
            var byteLength = toIndex(length);
-           setInternalState$3(this, {
+           setInternalState$4(this, {
              bytes: arrayFill.call(new Array(byteLength), 0),
              byteLength: byteLength
            });
          $DataView = function DataView(buffer, byteOffset, byteLength) {
            anInstance(this, $DataView, DATA_VIEW);
            anInstance(buffer, $ArrayBuffer, DATA_VIEW);
-           var bufferLength = getInternalState$3(buffer).byteLength;
+           var bufferLength = getInternalState$2(buffer).byteLength;
            var offset = toInteger(byteOffset);
            if (offset < 0 || offset > bufferLength) throw RangeError$1('Wrong offset');
            byteLength = byteLength === undefined ? bufferLength - offset : toLength(byteLength);
            if (offset + byteLength > bufferLength) throw RangeError$1(WRONG_LENGTH);
-           setInternalState$3(this, {
+           setInternalState$4(this, {
              buffer: buffer,
              byteLength: byteLength,
              byteOffset: offset
            addGetter($DataView, 'byteOffset');
          }
 
-         redefineAll($DataView[PROTOTYPE$2], {
+         redefineAll($DataView[PROTOTYPE], {
            getInt8: function getInt8(byteOffset) {
-             return get$1(this, 1, byteOffset)[0] << 24 >> 24;
+             return get$4(this, 1, byteOffset)[0] << 24 >> 24;
            },
            getUint8: function getUint8(byteOffset) {
-             return get$1(this, 1, byteOffset)[0];
+             return get$4(this, 1, byteOffset)[0];
            },
            getInt16: function getInt16(byteOffset /* , littleEndian */) {
-             var bytes = get$1(this, 2, byteOffset, arguments.length > 1 ? arguments[1] : undefined);
+             var bytes = get$4(this, 2, byteOffset, arguments.length > 1 ? arguments[1] : undefined);
              return (bytes[1] << 8 | bytes[0]) << 16 >> 16;
            },
            getUint16: function getUint16(byteOffset /* , littleEndian */) {
-             var bytes = get$1(this, 2, byteOffset, arguments.length > 1 ? arguments[1] : undefined);
+             var bytes = get$4(this, 2, byteOffset, arguments.length > 1 ? arguments[1] : undefined);
              return bytes[1] << 8 | bytes[0];
            },
            getInt32: function getInt32(byteOffset /* , littleEndian */) {
-             return unpackInt32(get$1(this, 4, byteOffset, arguments.length > 1 ? arguments[1] : undefined));
+             return unpackInt32(get$4(this, 4, byteOffset, arguments.length > 1 ? arguments[1] : undefined));
            },
            getUint32: function getUint32(byteOffset /* , littleEndian */) {
-             return unpackInt32(get$1(this, 4, byteOffset, arguments.length > 1 ? arguments[1] : undefined)) >>> 0;
+             return unpackInt32(get$4(this, 4, byteOffset, arguments.length > 1 ? arguments[1] : undefined)) >>> 0;
            },
            getFloat32: function getFloat32(byteOffset /* , littleEndian */) {
-             return unpackIEEE754(get$1(this, 4, byteOffset, arguments.length > 1 ? arguments[1] : undefined), 23);
+             return unpackIEEE754(get$4(this, 4, byteOffset, arguments.length > 1 ? arguments[1] : undefined), 23);
            },
            getFloat64: function getFloat64(byteOffset /* , littleEndian */) {
-             return unpackIEEE754(get$1(this, 8, byteOffset, arguments.length > 1 ? arguments[1] : undefined), 52);
+             return unpackIEEE754(get$4(this, 8, byteOffset, arguments.length > 1 ? arguments[1] : undefined), 52);
            },
            setInt8: function setInt8(byteOffset, value) {
-             set$1(this, 1, byteOffset, packInt8, value);
+             set$3(this, 1, byteOffset, packInt8, value);
            },
            setUint8: function setUint8(byteOffset, value) {
-             set$1(this, 1, byteOffset, packInt8, value);
+             set$3(this, 1, byteOffset, packInt8, value);
            },
            setInt16: function setInt16(byteOffset, value /* , littleEndian */) {
-             set$1(this, 2, byteOffset, packInt16, value, arguments.length > 2 ? arguments[2] : undefined);
+             set$3(this, 2, byteOffset, packInt16, value, arguments.length > 2 ? arguments[2] : undefined);
            },
            setUint16: function setUint16(byteOffset, value /* , littleEndian */) {
-             set$1(this, 2, byteOffset, packInt16, value, arguments.length > 2 ? arguments[2] : undefined);
+             set$3(this, 2, byteOffset, packInt16, value, arguments.length > 2 ? arguments[2] : undefined);
            },
            setInt32: function setInt32(byteOffset, value /* , littleEndian */) {
-             set$1(this, 4, byteOffset, packInt32, value, arguments.length > 2 ? arguments[2] : undefined);
+             set$3(this, 4, byteOffset, packInt32, value, arguments.length > 2 ? arguments[2] : undefined);
            },
            setUint32: function setUint32(byteOffset, value /* , littleEndian */) {
-             set$1(this, 4, byteOffset, packInt32, value, arguments.length > 2 ? arguments[2] : undefined);
+             set$3(this, 4, byteOffset, packInt32, value, arguments.length > 2 ? arguments[2] : undefined);
            },
            setFloat32: function setFloat32(byteOffset, value /* , littleEndian */) {
-             set$1(this, 4, byteOffset, packFloat32, value, arguments.length > 2 ? arguments[2] : undefined);
+             set$3(this, 4, byteOffset, packFloat32, value, arguments.length > 2 ? arguments[2] : undefined);
            },
            setFloat64: function setFloat64(byteOffset, value /* , littleEndian */) {
-             set$1(this, 8, byteOffset, packFloat64, value, arguments.length > 2 ? arguments[2] : undefined);
+             set$3(this, 8, byteOffset, packFloat64, value, arguments.length > 2 ? arguments[2] : undefined);
            }
          });
        } else {
          /* eslint-disable no-new -- required for testing */
          if (!fails(function () {
-           NativeArrayBuffer(1);
+           NativeArrayBuffer$1(1);
          }) || !fails(function () {
-           new NativeArrayBuffer(-1);
+           new NativeArrayBuffer$1(-1);
          }) || fails(function () {
-           new NativeArrayBuffer();
-           new NativeArrayBuffer(1.5);
-           new NativeArrayBuffer(NaN);
-           return NativeArrayBuffer.name != ARRAY_BUFFER;
+           new NativeArrayBuffer$1();
+           new NativeArrayBuffer$1(1.5);
+           new NativeArrayBuffer$1(NaN);
+           return NativeArrayBuffer$1.name != ARRAY_BUFFER$1;
          })) {
          /* eslint-enable no-new -- required for testing */
            $ArrayBuffer = function ArrayBuffer(length) {
              anInstance(this, $ArrayBuffer);
-             return new NativeArrayBuffer(toIndex(length));
+             return new NativeArrayBuffer$1(toIndex(length));
            };
-           var ArrayBufferPrototype = $ArrayBuffer[PROTOTYPE$2] = NativeArrayBuffer[PROTOTYPE$2];
-           for (var keys$1 = getOwnPropertyNames(NativeArrayBuffer), j = 0, key; keys$1.length > j;) {
-             if (!((key = keys$1[j++]) in $ArrayBuffer)) {
-               createNonEnumerableProperty($ArrayBuffer, key, NativeArrayBuffer[key]);
+           var ArrayBufferPrototype = $ArrayBuffer[PROTOTYPE] = NativeArrayBuffer$1[PROTOTYPE];
+           for (var keys$2 = getOwnPropertyNames$3(NativeArrayBuffer$1), j$2 = 0, key$1; keys$2.length > j$2;) {
+             if (!((key$1 = keys$2[j$2++]) in $ArrayBuffer)) {
+               createNonEnumerableProperty($ArrayBuffer, key$1, NativeArrayBuffer$1[key$1]);
              }
            }
            ArrayBufferPrototype.constructor = $ArrayBuffer;
          }
 
          // WebKit bug - the same parent prototype for typed arrays and data view
-         if (objectSetPrototypeOf && objectGetPrototypeOf($DataViewPrototype) !== ObjectPrototype$2) {
-           objectSetPrototypeOf($DataViewPrototype, ObjectPrototype$2);
+         if (objectSetPrototypeOf && objectGetPrototypeOf($DataViewPrototype) !== ObjectPrototype$1) {
+           objectSetPrototypeOf($DataViewPrototype, ObjectPrototype$1);
          }
 
          // iOS Safari 7.x bug
          var testView = new $DataView(new $ArrayBuffer(2));
-         var nativeSetInt8 = $DataViewPrototype.setInt8;
+         var $setInt8 = $DataViewPrototype.setInt8;
          testView.setInt8(0, 2147483648);
          testView.setInt8(1, 2147483649);
          if (testView.getInt8(0) || !testView.getInt8(1)) redefineAll($DataViewPrototype, {
            setInt8: function setInt8(byteOffset, value) {
-             nativeSetInt8.call(this, byteOffset, value << 24 >> 24);
+             $setInt8.call(this, byteOffset, value << 24 >> 24);
            },
            setUint8: function setUint8(byteOffset, value) {
-             nativeSetInt8.call(this, byteOffset, value << 24 >> 24);
+             $setInt8.call(this, byteOffset, value << 24 >> 24);
            }
          }, { unsafe: true });
        }
 
-       setToStringTag($ArrayBuffer, ARRAY_BUFFER);
+       setToStringTag($ArrayBuffer, ARRAY_BUFFER$1);
        setToStringTag($DataView, DATA_VIEW);
 
        var arrayBuffer = {
          DataView: $DataView
        };
 
-       // `DataView` constructor
-       // https://tc39.es/ecma262/#sec-dataview-constructor
-       _export({ global: true, forced: !arrayBufferNative }, {
-         DataView: arrayBuffer.DataView
-       });
-
-       var SPECIES$1 = wellKnownSymbol('species');
+       var SPECIES$5 = wellKnownSymbol('species');
 
        // `SpeciesConstructor` abstract operation
        // https://tc39.es/ecma262/#sec-speciesconstructor
        var speciesConstructor = function (O, defaultConstructor) {
          var C = anObject(O).constructor;
          var S;
-         return C === undefined || (S = anObject(C)[SPECIES$1]) == undefined ? defaultConstructor : aFunction$1(S);
+         return C === undefined || (S = anObject(C)[SPECIES$5]) == undefined ? defaultConstructor : aFunction(S);
        };
 
-       var ArrayBuffer$1 = arrayBuffer.ArrayBuffer;
+       var ArrayBuffer$3 = arrayBuffer.ArrayBuffer;
        var DataView$1 = arrayBuffer.DataView;
-       var nativeArrayBufferSlice = ArrayBuffer$1.prototype.slice;
+       var nativeArrayBufferSlice = ArrayBuffer$3.prototype.slice;
 
        var INCORRECT_SLICE = fails(function () {
-         return !new ArrayBuffer$1(2).slice(1, undefined).byteLength;
+         return !new ArrayBuffer$3(2).slice(1, undefined).byteLength;
        });
 
        // `ArrayBuffer.prototype.slice` method
            var length = anObject(this).byteLength;
            var first = toAbsoluteIndex(start, length);
            var fin = toAbsoluteIndex(end === undefined ? length : end, length);
-           var result = new (speciesConstructor(this, ArrayBuffer$1))(toLength(fin - first));
+           var result = new (speciesConstructor(this, ArrayBuffer$3))(toLength(fin - first));
            var viewSource = new DataView$1(this);
            var viewTarget = new DataView$1(result);
            var index = 0;
          }
        });
 
-       var defineProperty$4 = objectDefineProperty.f;
+       // `DataView` constructor
+       // https://tc39.es/ecma262/#sec-dataview-constructor
+       _export({ global: true, forced: !arrayBufferNative }, {
+         DataView: arrayBuffer.DataView
+       });
+
+       var defineProperty$5 = objectDefineProperty.f;
 
 
 
 
 
-       var Int8Array$1 = global_1.Int8Array;
-       var Int8ArrayPrototype = Int8Array$1 && Int8Array$1.prototype;
-       var Uint8ClampedArray = global_1.Uint8ClampedArray;
+       var Int8Array$3 = global$2.Int8Array;
+       var Int8ArrayPrototype = Int8Array$3 && Int8Array$3.prototype;
+       var Uint8ClampedArray = global$2.Uint8ClampedArray;
        var Uint8ClampedArrayPrototype = Uint8ClampedArray && Uint8ClampedArray.prototype;
-       var TypedArray = Int8Array$1 && objectGetPrototypeOf(Int8Array$1);
+       var TypedArray = Int8Array$3 && objectGetPrototypeOf(Int8Array$3);
        var TypedArrayPrototype = Int8ArrayPrototype && objectGetPrototypeOf(Int8ArrayPrototype);
-       var ObjectPrototype$3 = Object.prototype;
-       var isPrototypeOf = ObjectPrototype$3.isPrototypeOf;
+       var ObjectPrototype = Object.prototype;
+       var isPrototypeOf = ObjectPrototype.isPrototypeOf;
 
-       var TO_STRING_TAG$4 = wellKnownSymbol('toStringTag');
+       var TO_STRING_TAG = wellKnownSymbol('toStringTag');
        var TYPED_ARRAY_TAG = uid('TYPED_ARRAY_TAG');
        // Fixing native typed arrays in Opera Presto crashes the browser, see #595
-       var NATIVE_ARRAY_BUFFER_VIEWS = arrayBufferNative && !!objectSetPrototypeOf && classof(global_1.opera) !== 'Opera';
+       var NATIVE_ARRAY_BUFFER_VIEWS$2 = arrayBufferNative && !!objectSetPrototypeOf && classof(global$2.opera) !== 'Opera';
        var TYPED_ARRAY_TAG_REQIRED = false;
-       var NAME;
+       var NAME$1;
 
        var TypedArrayConstructorsList = {
          Int8Array: 1,
        };
 
        var isView = function isView(it) {
-         if (!isObject(it)) return false;
+         if (!isObject$4(it)) return false;
          var klass = classof(it);
          return klass === 'DataView'
-           || has(TypedArrayConstructorsList, klass)
-           || has(BigIntArrayConstructorsList, klass);
+           || has$1(TypedArrayConstructorsList, klass)
+           || has$1(BigIntArrayConstructorsList, klass);
        };
 
        var isTypedArray = function (it) {
-         if (!isObject(it)) return false;
+         if (!isObject$4(it)) return false;
          var klass = classof(it);
-         return has(TypedArrayConstructorsList, klass)
-           || has(BigIntArrayConstructorsList, klass);
+         return has$1(TypedArrayConstructorsList, klass)
+           || has$1(BigIntArrayConstructorsList, klass);
        };
 
-       var aTypedArray = function (it) {
+       var aTypedArray$m = function (it) {
          if (isTypedArray(it)) return it;
          throw TypeError('Target is not a typed array');
        };
 
-       var aTypedArrayConstructor = function (C) {
+       var aTypedArrayConstructor$4 = function (C) {
          if (objectSetPrototypeOf) {
            if (isPrototypeOf.call(TypedArray, C)) return C;
-         } else for (var ARRAY in TypedArrayConstructorsList) if (has(TypedArrayConstructorsList, NAME)) {
-           var TypedArrayConstructor = global_1[ARRAY];
+         } else for (var ARRAY in TypedArrayConstructorsList) if (has$1(TypedArrayConstructorsList, NAME$1)) {
+           var TypedArrayConstructor = global$2[ARRAY];
            if (TypedArrayConstructor && (C === TypedArrayConstructor || isPrototypeOf.call(TypedArrayConstructor, C))) {
              return C;
            }
          } throw TypeError('Target is not a typed array constructor');
        };
 
-       var exportTypedArrayMethod = function (KEY, property, forced) {
+       var exportTypedArrayMethod$n = function (KEY, property, forced) {
          if (!descriptors) return;
          if (forced) for (var ARRAY in TypedArrayConstructorsList) {
-           var TypedArrayConstructor = global_1[ARRAY];
-           if (TypedArrayConstructor && has(TypedArrayConstructor.prototype, KEY)) {
+           var TypedArrayConstructor = global$2[ARRAY];
+           if (TypedArrayConstructor && has$1(TypedArrayConstructor.prototype, KEY)) try {
              delete TypedArrayConstructor.prototype[KEY];
-           }
+           } catch (error) { /* empty */ }
          }
          if (!TypedArrayPrototype[KEY] || forced) {
            redefine(TypedArrayPrototype, KEY, forced ? property
-             : NATIVE_ARRAY_BUFFER_VIEWS && Int8ArrayPrototype[KEY] || property);
+             : NATIVE_ARRAY_BUFFER_VIEWS$2 && Int8ArrayPrototype[KEY] || property);
          }
        };
 
-       var exportTypedArrayStaticMethod = function (KEY, property, forced) {
+       var exportTypedArrayStaticMethod$1 = function (KEY, property, forced) {
          var ARRAY, TypedArrayConstructor;
          if (!descriptors) return;
          if (objectSetPrototypeOf) {
            if (forced) for (ARRAY in TypedArrayConstructorsList) {
-             TypedArrayConstructor = global_1[ARRAY];
-             if (TypedArrayConstructor && has(TypedArrayConstructor, KEY)) {
+             TypedArrayConstructor = global$2[ARRAY];
+             if (TypedArrayConstructor && has$1(TypedArrayConstructor, KEY)) try {
                delete TypedArrayConstructor[KEY];
-             }
+             } catch (error) { /* empty */ }
            }
            if (!TypedArray[KEY] || forced) {
              // V8 ~ Chrome 49-50 `%TypedArray%` methods are non-writable non-configurable
              try {
-               return redefine(TypedArray, KEY, forced ? property : NATIVE_ARRAY_BUFFER_VIEWS && Int8Array$1[KEY] || property);
+               return redefine(TypedArray, KEY, forced ? property : NATIVE_ARRAY_BUFFER_VIEWS$2 && TypedArray[KEY] || property);
              } catch (error) { /* empty */ }
            } else return;
          }
          for (ARRAY in TypedArrayConstructorsList) {
-           TypedArrayConstructor = global_1[ARRAY];
+           TypedArrayConstructor = global$2[ARRAY];
            if (TypedArrayConstructor && (!TypedArrayConstructor[KEY] || forced)) {
              redefine(TypedArrayConstructor, KEY, property);
            }
          }
        };
 
-       for (NAME in TypedArrayConstructorsList) {
-         if (!global_1[NAME]) NATIVE_ARRAY_BUFFER_VIEWS = false;
+       for (NAME$1 in TypedArrayConstructorsList) {
+         if (!global$2[NAME$1]) NATIVE_ARRAY_BUFFER_VIEWS$2 = false;
        }
 
        // WebKit bug - typed arrays constructors prototype is Object.prototype
-       if (!NATIVE_ARRAY_BUFFER_VIEWS || typeof TypedArray != 'function' || TypedArray === Function.prototype) {
+       if (!NATIVE_ARRAY_BUFFER_VIEWS$2 || typeof TypedArray != 'function' || TypedArray === Function.prototype) {
          // eslint-disable-next-line no-shadow -- safe
          TypedArray = function TypedArray() {
            throw TypeError('Incorrect invocation');
          };
-         if (NATIVE_ARRAY_BUFFER_VIEWS) for (NAME in TypedArrayConstructorsList) {
-           if (global_1[NAME]) objectSetPrototypeOf(global_1[NAME], TypedArray);
+         if (NATIVE_ARRAY_BUFFER_VIEWS$2) for (NAME$1 in TypedArrayConstructorsList) {
+           if (global$2[NAME$1]) objectSetPrototypeOf(global$2[NAME$1], TypedArray);
          }
        }
 
-       if (!NATIVE_ARRAY_BUFFER_VIEWS || !TypedArrayPrototype || TypedArrayPrototype === ObjectPrototype$3) {
+       if (!NATIVE_ARRAY_BUFFER_VIEWS$2 || !TypedArrayPrototype || TypedArrayPrototype === ObjectPrototype) {
          TypedArrayPrototype = TypedArray.prototype;
-         if (NATIVE_ARRAY_BUFFER_VIEWS) for (NAME in TypedArrayConstructorsList) {
-           if (global_1[NAME]) objectSetPrototypeOf(global_1[NAME].prototype, TypedArrayPrototype);
+         if (NATIVE_ARRAY_BUFFER_VIEWS$2) for (NAME$1 in TypedArrayConstructorsList) {
+           if (global$2[NAME$1]) objectSetPrototypeOf(global$2[NAME$1].prototype, TypedArrayPrototype);
          }
        }
 
        // WebKit bug - one more object in Uint8ClampedArray prototype chain
-       if (NATIVE_ARRAY_BUFFER_VIEWS && objectGetPrototypeOf(Uint8ClampedArrayPrototype) !== TypedArrayPrototype) {
+       if (NATIVE_ARRAY_BUFFER_VIEWS$2 && objectGetPrototypeOf(Uint8ClampedArrayPrototype) !== TypedArrayPrototype) {
          objectSetPrototypeOf(Uint8ClampedArrayPrototype, TypedArrayPrototype);
        }
 
-       if (descriptors && !has(TypedArrayPrototype, TO_STRING_TAG$4)) {
+       if (descriptors && !has$1(TypedArrayPrototype, TO_STRING_TAG)) {
          TYPED_ARRAY_TAG_REQIRED = true;
-         defineProperty$4(TypedArrayPrototype, TO_STRING_TAG$4, { get: function () {
-           return isObject(this) ? this[TYPED_ARRAY_TAG] : undefined;
+         defineProperty$5(TypedArrayPrototype, TO_STRING_TAG, { get: function () {
+           return isObject$4(this) ? this[TYPED_ARRAY_TAG] : undefined;
          } });
-         for (NAME in TypedArrayConstructorsList) if (global_1[NAME]) {
-           createNonEnumerableProperty(global_1[NAME], TYPED_ARRAY_TAG, NAME);
+         for (NAME$1 in TypedArrayConstructorsList) if (global$2[NAME$1]) {
+           createNonEnumerableProperty(global$2[NAME$1], TYPED_ARRAY_TAG, NAME$1);
          }
        }
 
        var arrayBufferViewCore = {
-         NATIVE_ARRAY_BUFFER_VIEWS: NATIVE_ARRAY_BUFFER_VIEWS,
+         NATIVE_ARRAY_BUFFER_VIEWS: NATIVE_ARRAY_BUFFER_VIEWS$2,
          TYPED_ARRAY_TAG: TYPED_ARRAY_TAG_REQIRED && TYPED_ARRAY_TAG,
-         aTypedArray: aTypedArray,
-         aTypedArrayConstructor: aTypedArrayConstructor,
-         exportTypedArrayMethod: exportTypedArrayMethod,
-         exportTypedArrayStaticMethod: exportTypedArrayStaticMethod,
+         aTypedArray: aTypedArray$m,
+         aTypedArrayConstructor: aTypedArrayConstructor$4,
+         exportTypedArrayMethod: exportTypedArrayMethod$n,
+         exportTypedArrayStaticMethod: exportTypedArrayStaticMethod$1,
          isView: isView,
          isTypedArray: isTypedArray,
          TypedArray: TypedArray,
          isView: arrayBufferViewCore.isView
        });
 
-       var SPECIES$2 = wellKnownSymbol('species');
+       var SPECIES$4 = wellKnownSymbol('species');
 
        var setSpecies = function (CONSTRUCTOR_NAME) {
          var Constructor = getBuiltIn(CONSTRUCTOR_NAME);
          var defineProperty = objectDefineProperty.f;
 
-         if (descriptors && Constructor && !Constructor[SPECIES$2]) {
-           defineProperty(Constructor, SPECIES$2, {
+         if (descriptors && Constructor && !Constructor[SPECIES$4]) {
+           defineProperty(Constructor, SPECIES$4, {
              configurable: true,
              get: function () { return this; }
            });
          }
        };
 
-       var ARRAY_BUFFER$1 = 'ArrayBuffer';
-       var ArrayBuffer$2 = arrayBuffer[ARRAY_BUFFER$1];
-       var NativeArrayBuffer$1 = global_1[ARRAY_BUFFER$1];
+       var ARRAY_BUFFER = 'ArrayBuffer';
+       var ArrayBuffer$2 = arrayBuffer[ARRAY_BUFFER];
+       var NativeArrayBuffer = global$2[ARRAY_BUFFER];
 
        // `ArrayBuffer` constructor
        // https://tc39.es/ecma262/#sec-arraybuffer-constructor
-       _export({ global: true, forced: NativeArrayBuffer$1 !== ArrayBuffer$2 }, {
+       _export({ global: true, forced: NativeArrayBuffer !== ArrayBuffer$2 }, {
          ArrayBuffer: ArrayBuffer$2
        });
 
-       setSpecies(ARRAY_BUFFER$1);
+       setSpecies(ARRAY_BUFFER);
 
        var arrayMethodIsStrict = function (METHOD_NAME, argument) {
          var method = [][METHOD_NAME];
          });
        };
 
-       var $indexOf = arrayIncludes.indexOf;
+       /* eslint-disable es/no-array-prototype-indexof -- required for testing */
+
+       var $indexOf$1 = arrayIncludes.indexOf;
 
 
        var nativeIndexOf = [].indexOf;
 
-       var NEGATIVE_ZERO = !!nativeIndexOf && 1 / [1].indexOf(1, -0) < 0;
-       var STRICT_METHOD = arrayMethodIsStrict('indexOf');
+       var NEGATIVE_ZERO$1 = !!nativeIndexOf && 1 / [1].indexOf(1, -0) < 0;
+       var STRICT_METHOD$7 = arrayMethodIsStrict('indexOf');
 
        // `Array.prototype.indexOf` method
        // https://tc39.es/ecma262/#sec-array.prototype.indexof
-       _export({ target: 'Array', proto: true, forced: NEGATIVE_ZERO || !STRICT_METHOD }, {
+       _export({ target: 'Array', proto: true, forced: NEGATIVE_ZERO$1 || !STRICT_METHOD$7 }, {
          indexOf: function indexOf(searchElement /* , fromIndex = 0 */) {
-           return NEGATIVE_ZERO
+           return NEGATIVE_ZERO$1
              // convert -0 to +0
              ? nativeIndexOf.apply(this, arguments) || 0
-             : $indexOf(this, searchElement, arguments.length > 1 ? arguments[1] : undefined);
+             : $indexOf$1(this, searchElement, arguments.length > 1 ? arguments[1] : undefined);
          }
        });
 
          });
        };
 
-       var $map = arrayIteration.map;
+       var $map$1 = arrayIteration.map;
 
 
-       var HAS_SPECIES_SUPPORT = arrayMethodHasSpeciesSupport('map');
+       var HAS_SPECIES_SUPPORT$3 = arrayMethodHasSpeciesSupport('map');
 
        // `Array.prototype.map` method
        // https://tc39.es/ecma262/#sec-array.prototype.map
        // with adding support of @@species
-       _export({ target: 'Array', proto: true, forced: !HAS_SPECIES_SUPPORT }, {
+       _export({ target: 'Array', proto: true, forced: !HAS_SPECIES_SUPPORT$3 }, {
          map: function map(callbackfn /* , thisArg */) {
-           return $map(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
+           return $map$1(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
          }
        });
 
        var $forEach$1 = arrayIteration.forEach;
 
 
-       var STRICT_METHOD$1 = arrayMethodIsStrict('forEach');
+       var STRICT_METHOD$6 = arrayMethodIsStrict('forEach');
 
        // `Array.prototype.forEach` method implementation
        // https://tc39.es/ecma262/#sec-array.prototype.foreach
-       var arrayForEach = !STRICT_METHOD$1 ? function forEach(callbackfn /* , thisArg */) {
+       var arrayForEach = !STRICT_METHOD$6 ? function forEach(callbackfn /* , thisArg */) {
          return $forEach$1(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
+       // eslint-disable-next-line es/no-array-prototype-foreach -- safe
        } : [].forEach;
 
        // `Array.prototype.forEach` method
        // https://tc39.es/ecma262/#sec-array.prototype.foreach
+       // eslint-disable-next-line es/no-array-prototype-foreach -- safe
        _export({ target: 'Array', proto: true, forced: [].forEach != arrayForEach }, {
          forEach: arrayForEach
        });
 
-       for (var COLLECTION_NAME$1 in domIterables) {
-         var Collection$1 = global_1[COLLECTION_NAME$1];
-         var CollectionPrototype$1 = Collection$1 && Collection$1.prototype;
+       for (var COLLECTION_NAME in domIterables) {
+         var Collection = global$2[COLLECTION_NAME];
+         var CollectionPrototype = Collection && Collection.prototype;
          // some Chrome versions have non-configurable methods on DOMTokenList
-         if (CollectionPrototype$1 && CollectionPrototype$1.forEach !== arrayForEach) try {
-           createNonEnumerableProperty(CollectionPrototype$1, 'forEach', arrayForEach);
+         if (CollectionPrototype && CollectionPrototype.forEach !== arrayForEach) try {
+           createNonEnumerableProperty(CollectionPrototype, 'forEach', arrayForEach);
          } catch (error) {
-           CollectionPrototype$1.forEach = arrayForEach;
+           CollectionPrototype.forEach = arrayForEach;
          }
        }
 
          isArray: isArray
        });
 
-       var nativeGetOwnPropertyNames$2 = objectGetOwnPropertyNamesExternal.f;
+       var getOwnPropertyNames$2 = objectGetOwnPropertyNamesExternal.f;
 
-       var FAILS_ON_PRIMITIVES = fails(function () { return !Object.getOwnPropertyNames(1); });
+       // eslint-disable-next-line es/no-object-getownpropertynames -- required for testing
+       var FAILS_ON_PRIMITIVES$4 = fails(function () { return !Object.getOwnPropertyNames(1); });
 
        // `Object.getOwnPropertyNames` method
        // https://tc39.es/ecma262/#sec-object.getownpropertynames
-       _export({ target: 'Object', stat: true, forced: FAILS_ON_PRIMITIVES }, {
-         getOwnPropertyNames: nativeGetOwnPropertyNames$2
+       _export({ target: 'Object', stat: true, forced: FAILS_ON_PRIMITIVES$4 }, {
+         getOwnPropertyNames: getOwnPropertyNames$2
        });
 
-       var nativePromiseConstructor = global_1.Promise;
+       var nativePromiseConstructor = global$2.Promise;
 
-       var ITERATOR$3 = wellKnownSymbol('iterator');
-       var ArrayPrototype$1 = Array.prototype;
+       var ITERATOR$5 = wellKnownSymbol('iterator');
+       var ArrayPrototype = Array.prototype;
 
        // check on default Array iterator
        var isArrayIteratorMethod = function (it) {
-         return it !== undefined && (iterators.Array === it || ArrayPrototype$1[ITERATOR$3] === it);
+         return it !== undefined && (iterators.Array === it || ArrayPrototype[ITERATOR$5] === it);
        };
 
        var ITERATOR$4 = wellKnownSymbol('iterator');
          } return new Result(false);
        };
 
-       var ITERATOR$5 = wellKnownSymbol('iterator');
+       var ITERATOR$3 = wellKnownSymbol('iterator');
        var SAFE_CLOSING = false;
 
        try {
              SAFE_CLOSING = true;
            }
          };
-         iteratorWithReturn[ITERATOR$5] = function () {
+         iteratorWithReturn[ITERATOR$3] = function () {
            return this;
          };
-         // eslint-disable-next-line no-throw-literal -- required for testing
+         // eslint-disable-next-line es/no-array-from, no-throw-literal -- required for testing
          Array.from(iteratorWithReturn, function () { throw 2; });
        } catch (error) { /* empty */ }
 
          var ITERATION_SUPPORT = false;
          try {
            var object = {};
-           object[ITERATOR$5] = function () {
+           object[ITERATOR$3] = function () {
              return {
                next: function () {
                  return { done: ITERATION_SUPPORT = true };
          return ITERATION_SUPPORT;
        };
 
-       var engineIsIos = /(iphone|ipod|ipad).*applewebkit/i.test(engineUserAgent);
+       var engineIsIos = /(?:iphone|ipod|ipad).*applewebkit/i.test(engineUserAgent);
+
+       var engineIsNode = classofRaw(global$2.process) == 'process';
 
-       var location$1 = global_1.location;
-       var set$2 = global_1.setImmediate;
-       var clear = global_1.clearImmediate;
-       var process$2 = global_1.process;
-       var MessageChannel = global_1.MessageChannel;
-       var Dispatch = global_1.Dispatch;
+       var location$1 = global$2.location;
+       var set$2 = global$2.setImmediate;
+       var clear = global$2.clearImmediate;
+       var process$3 = global$2.process;
+       var MessageChannel = global$2.MessageChannel;
+       var Dispatch$1 = global$2.Dispatch;
        var counter = 0;
        var queue = {};
        var ONREADYSTATECHANGE = 'onreadystatechange';
 
        var post = function (id) {
          // old engines have not location.origin
-         global_1.postMessage(id + '', location$1.protocol + '//' + location$1.host);
+         global$2.postMessage(id + '', location$1.protocol + '//' + location$1.host);
        };
 
        // Node.js 0.9+ & IE10+ has setImmediate, otherwise:
          // Node.js 0.8-
          if (engineIsNode) {
            defer = function (id) {
-             process$2.nextTick(runner(id));
+             process$3.nextTick(runner(id));
            };
          // Sphere (JS game engine) Dispatch API
-         } else if (Dispatch && Dispatch.now) {
+         } else if (Dispatch$1 && Dispatch$1.now) {
            defer = function (id) {
-             Dispatch.now(runner(id));
+             Dispatch$1.now(runner(id));
            };
          // Browsers with MessageChannel, includes WebWorkers
          // except iOS - https://github.com/zloirock/core-js/issues/624
          // Browsers with postMessage, skip WebWorkers
          // IE8 has postMessage, but it's sync & typeof its postMessage is 'object'
          } else if (
-           global_1.addEventListener &&
+           global$2.addEventListener &&
            typeof postMessage == 'function' &&
-           !global_1.importScripts &&
+           !global$2.importScripts &&
            location$1 && location$1.protocol !== 'file:' &&
            !fails(post)
          ) {
            defer = post;
-           global_1.addEventListener('message', listener, false);
+           global$2.addEventListener('message', listener, false);
          // IE8-
          } else if (ONREADYSTATECHANGE in documentCreateElement('script')) {
            defer = function (id) {
          }
        }
 
-       var task = {
+       var task$1 = {
          set: set$2,
          clear: clear
        };
 
        var engineIsWebosWebkit = /web0s(?!.*chrome)/i.test(engineUserAgent);
 
-       var getOwnPropertyDescriptor$2 = objectGetOwnPropertyDescriptor.f;
-       var macrotask = task.set;
+       var getOwnPropertyDescriptor$3 = objectGetOwnPropertyDescriptor.f;
+       var macrotask = task$1.set;
 
 
 
 
-       var MutationObserver = global_1.MutationObserver || global_1.WebKitMutationObserver;
-       var document$2 = global_1.document;
-       var process$3 = global_1.process;
-       var Promise$1 = global_1.Promise;
+       var MutationObserver = global$2.MutationObserver || global$2.WebKitMutationObserver;
+       var document$2 = global$2.document;
+       var process$2 = global$2.process;
+       var Promise$1 = global$2.Promise;
        // Node.js 11 shows ExperimentalWarning on getting `queueMicrotask`
-       var queueMicrotaskDescriptor = getOwnPropertyDescriptor$2(global_1, 'queueMicrotask');
+       var queueMicrotaskDescriptor = getOwnPropertyDescriptor$3(global$2, 'queueMicrotask');
        var queueMicrotask = queueMicrotaskDescriptor && queueMicrotaskDescriptor.value;
 
-       var flush, head, last, notify, toggle, node, promise, then;
+       var flush, head, last, notify$1, toggle, node, promise, then;
 
        // modern engines have queueMicrotask method
        if (!queueMicrotask) {
          flush = function () {
            var parent, fn;
-           if (engineIsNode && (parent = process$3.domain)) parent.exit();
+           if (engineIsNode && (parent = process$2.domain)) parent.exit();
            while (head) {
              fn = head.fn;
              head = head.next;
              try {
                fn();
              } catch (error) {
-               if (head) notify();
+               if (head) notify$1();
                else last = undefined;
                throw error;
              }
            toggle = true;
            node = document$2.createTextNode('');
            new MutationObserver(flush).observe(node, { characterData: true });
-           notify = function () {
+           notify$1 = function () {
              node.data = toggle = !toggle;
            };
          // environments with maybe non-completely correct, but existent Promise
          } else if (Promise$1 && Promise$1.resolve) {
            // Promise.resolve without an argument throws an error in LG WebOS 2
            promise = Promise$1.resolve(undefined);
+           // workaround of WebKit ~ iOS Safari 10.1 bug
+           promise.constructor = Promise$1;
            then = promise.then;
-           notify = function () {
+           notify$1 = function () {
              then.call(promise, flush);
            };
          // Node.js without promises
          } else if (engineIsNode) {
-           notify = function () {
-             process$3.nextTick(flush);
+           notify$1 = function () {
+             process$2.nextTick(flush);
            };
          // for other environments - macrotask based on:
          // - setImmediate
          // - onreadystatechange
          // - setTimeout
          } else {
-           notify = function () {
+           notify$1 = function () {
              // strange IE + webpack dev server bug - use .call(global)
-             macrotask.call(global_1, flush);
+             macrotask.call(global$2, flush);
            };
          }
        }
          if (last) last.next = task;
          if (!head) {
            head = task;
-           notify();
+           notify$1();
          } last = task;
        };
 
            resolve = $$resolve;
            reject = $$reject;
          });
-         this.resolve = aFunction$1(resolve);
-         this.reject = aFunction$1(reject);
+         this.resolve = aFunction(resolve);
+         this.reject = aFunction(reject);
        };
 
-       // 25.4.1.5 NewPromiseCapability(C)
-       var f$7 = function (C) {
+       // `NewPromiseCapability` abstract operation
+       // https://tc39.es/ecma262/#sec-newpromisecapability
+       var f = function (C) {
          return new PromiseCapability(C);
        };
 
-       var newPromiseCapability = {
-               f: f$7
+       var newPromiseCapability$1 = {
+               f: f
        };
 
        var promiseResolve = function (C, x) {
          anObject(C);
-         if (isObject(x) && x.constructor === C) return x;
-         var promiseCapability = newPromiseCapability.f(C);
+         if (isObject$4(x) && x.constructor === C) return x;
+         var promiseCapability = newPromiseCapability$1.f(C);
          var resolve = promiseCapability.resolve;
          resolve(x);
          return promiseCapability.promise;
        };
 
        var hostReportErrors = function (a, b) {
-         var console = global_1.console;
+         var console = global$2.console;
          if (console && console.error) {
            arguments.length === 1 ? console.error(a) : console.error(a, b);
          }
          }
        };
 
-       var task$1 = task.set;
+       var engineIsBrowser = typeof window == 'object';
 
+       var task = task$1.set;
 
 
 
 
 
 
-       var SPECIES$4 = wellKnownSymbol('species');
+
+
+       var SPECIES$2 = wellKnownSymbol('species');
        var PROMISE = 'Promise';
-       var getInternalState$4 = internalState.get;
-       var setInternalState$4 = internalState.set;
+       var getInternalState$1 = internalState.get;
+       var setInternalState$3 = internalState.set;
        var getInternalPromiseState = internalState.getterFor(PROMISE);
+       var NativePromisePrototype = nativePromiseConstructor && nativePromiseConstructor.prototype;
        var PromiseConstructor = nativePromiseConstructor;
-       var TypeError$1 = global_1.TypeError;
-       var document$3 = global_1.document;
-       var process$4 = global_1.process;
-       var $fetch = getBuiltIn('fetch');
-       var newPromiseCapability$1 = newPromiseCapability.f;
-       var newGenericPromiseCapability = newPromiseCapability$1;
-       var DISPATCH_EVENT = !!(document$3 && document$3.createEvent && global_1.dispatchEvent);
+       var PromiseConstructorPrototype = NativePromisePrototype;
+       var TypeError$1 = global$2.TypeError;
+       var document$1 = global$2.document;
+       var process$1 = global$2.process;
+       var newPromiseCapability = newPromiseCapability$1.f;
+       var newGenericPromiseCapability = newPromiseCapability;
+       var DISPATCH_EVENT = !!(document$1 && document$1.createEvent && global$2.dispatchEvent);
        var NATIVE_REJECTION_EVENT = typeof PromiseRejectionEvent == 'function';
        var UNHANDLED_REJECTION = 'unhandledrejection';
        var REJECTION_HANDLED = 'rejectionhandled';
        var REJECTED = 2;
        var HANDLED = 1;
        var UNHANDLED = 2;
+       var SUBCLASSING = false;
        var Internal, OwnPromiseCapability, PromiseWrapper, nativeThen;
 
-       var FORCED = isForced_1(PROMISE, function () {
-         var GLOBAL_CORE_JS_PROMISE = inspectSource(PromiseConstructor) !== String(PromiseConstructor);
-         if (!GLOBAL_CORE_JS_PROMISE) {
-           // V8 6.6 (Node 10 and Chrome 66) have a bug with resolving custom thenables
-           // https://bugs.chromium.org/p/chromium/issues/detail?id=830565
-           // We can't detect it synchronously, so just check versions
-           if (engineV8Version === 66) return true;
-           // Unhandled rejections tracking support, NodeJS Promise without it fails @@species test
-           if (!engineIsNode && !NATIVE_REJECTION_EVENT) return true;
-         }
+       var FORCED$f = isForced_1(PROMISE, function () {
+         var PROMISE_CONSTRUCTOR_SOURCE = inspectSource(PromiseConstructor);
+         var GLOBAL_CORE_JS_PROMISE = PROMISE_CONSTRUCTOR_SOURCE !== String(PromiseConstructor);
+         // V8 6.6 (Node 10 and Chrome 66) have a bug with resolving custom thenables
+         // https://bugs.chromium.org/p/chromium/issues/detail?id=830565
+         // We can't detect it synchronously, so just check versions
+         if (!GLOBAL_CORE_JS_PROMISE && engineV8Version === 66) return true;
          // We can't use @@species feature detection in V8 since it causes
          // deoptimization and performance degradation
          // https://github.com/zloirock/core-js/issues/679
-         if (engineV8Version >= 51 && /native code/.test(PromiseConstructor)) return false;
+         if (engineV8Version >= 51 && /native code/.test(PROMISE_CONSTRUCTOR_SOURCE)) return false;
          // Detect correctness of subclassing with @@species support
-         var promise = PromiseConstructor.resolve(1);
+         var promise = new PromiseConstructor(function (resolve) { resolve(1); });
          var FakePromise = function (exec) {
            exec(function () { /* empty */ }, function () { /* empty */ });
          };
          var constructor = promise.constructor = {};
-         constructor[SPECIES$4] = FakePromise;
-         return !(promise.then(function () { /* empty */ }) instanceof FakePromise);
+         constructor[SPECIES$2] = FakePromise;
+         SUBCLASSING = promise.then(function () { /* empty */ }) instanceof FakePromise;
+         if (!SUBCLASSING) return true;
+         // Unhandled rejections tracking support, NodeJS Promise without it fails @@species test
+         return !GLOBAL_CORE_JS_PROMISE && engineIsBrowser && !NATIVE_REJECTION_EVENT;
        });
 
-       var INCORRECT_ITERATION = FORCED || !checkCorrectnessOfIteration(function (iterable) {
+       var INCORRECT_ITERATION$1 = FORCED$f || !checkCorrectnessOfIteration(function (iterable) {
          PromiseConstructor.all(iterable)['catch'](function () { /* empty */ });
        });
 
        // helpers
        var isThenable = function (it) {
          var then;
-         return isObject(it) && typeof (then = it.then) == 'function' ? then : false;
+         return isObject$4(it) && typeof (then = it.then) == 'function' ? then : false;
        };
 
-       var notify$1 = function (state, isReject) {
+       var notify = function (state, isReject) {
          if (state.notified) return;
          state.notified = true;
          var chain = state.reactions;
          });
        };
 
-       var dispatchEvent = function (name, promise, reason) {
+       var dispatchEvent$1 = function (name, promise, reason) {
          var event, handler;
          if (DISPATCH_EVENT) {
-           event = document$3.createEvent('Event');
+           event = document$1.createEvent('Event');
            event.promise = promise;
            event.reason = reason;
            event.initEvent(name, false, true);
-           global_1.dispatchEvent(event);
+           global$2.dispatchEvent(event);
          } else event = { promise: promise, reason: reason };
-         if (!NATIVE_REJECTION_EVENT && (handler = global_1['on' + name])) handler(event);
+         if (!NATIVE_REJECTION_EVENT && (handler = global$2['on' + name])) handler(event);
          else if (name === UNHANDLED_REJECTION) hostReportErrors('Unhandled promise rejection', reason);
        };
 
        var onUnhandled = function (state) {
-         task$1.call(global_1, function () {
+         task.call(global$2, function () {
            var promise = state.facade;
            var value = state.value;
            var IS_UNHANDLED = isUnhandled(state);
            if (IS_UNHANDLED) {
              result = perform(function () {
                if (engineIsNode) {
-                 process$4.emit('unhandledRejection', value, promise);
-               } else dispatchEvent(UNHANDLED_REJECTION, promise, value);
+                 process$1.emit('unhandledRejection', value, promise);
+               } else dispatchEvent$1(UNHANDLED_REJECTION, promise, value);
              });
              // Browsers should not trigger `rejectionHandled` event if it was handled here, NodeJS - should
              state.rejection = engineIsNode || isUnhandled(state) ? UNHANDLED : HANDLED;
        };
 
        var onHandleUnhandled = function (state) {
-         task$1.call(global_1, function () {
+         task.call(global$2, function () {
            var promise = state.facade;
            if (engineIsNode) {
-             process$4.emit('rejectionHandled', promise);
-           } else dispatchEvent(REJECTION_HANDLED, promise, state.value);
+             process$1.emit('rejectionHandled', promise);
+           } else dispatchEvent$1(REJECTION_HANDLED, promise, state.value);
          });
        };
 
-       var bind = function (fn, state, unwrap) {
+       var bind$2 = function (fn, state, unwrap) {
          return function (value) {
            fn(state, value, unwrap);
          };
          if (unwrap) state = unwrap;
          state.value = value;
          state.state = REJECTED;
-         notify$1(state, true);
+         notify(state, true);
        };
 
        var internalResolve = function (state, value, unwrap) {
                var wrapper = { done: false };
                try {
                  then.call(value,
-                   bind(internalResolve, wrapper, state),
-                   bind(internalReject, wrapper, state)
+                   bind$2(internalResolve, wrapper, state),
+                   bind$2(internalReject, wrapper, state)
                  );
                } catch (error) {
                  internalReject(wrapper, error, state);
            } else {
              state.value = value;
              state.state = FULFILLED;
-             notify$1(state, false);
+             notify(state, false);
            }
          } catch (error) {
            internalReject({ done: false }, error, state);
        };
 
        // constructor polyfill
-       if (FORCED) {
+       if (FORCED$f) {
          // 25.4.3.1 Promise(executor)
          PromiseConstructor = function Promise(executor) {
            anInstance(this, PromiseConstructor, PROMISE);
-           aFunction$1(executor);
+           aFunction(executor);
            Internal.call(this);
-           var state = getInternalState$4(this);
+           var state = getInternalState$1(this);
            try {
-             executor(bind(internalResolve, state), bind(internalReject, state));
+             executor(bind$2(internalResolve, state), bind$2(internalReject, state));
            } catch (error) {
              internalReject(state, error);
            }
          };
+         PromiseConstructorPrototype = PromiseConstructor.prototype;
          // eslint-disable-next-line no-unused-vars -- required for `.length`
          Internal = function Promise(executor) {
-           setInternalState$4(this, {
+           setInternalState$3(this, {
              type: PROMISE,
              done: false,
              notified: false,
              value: undefined
            });
          };
-         Internal.prototype = redefineAll(PromiseConstructor.prototype, {
+         Internal.prototype = redefineAll(PromiseConstructorPrototype, {
            // `Promise.prototype.then` method
            // https://tc39.es/ecma262/#sec-promise.prototype.then
            then: function then(onFulfilled, onRejected) {
              var state = getInternalPromiseState(this);
-             var reaction = newPromiseCapability$1(speciesConstructor(this, PromiseConstructor));
+             var reaction = newPromiseCapability(speciesConstructor(this, PromiseConstructor));
              reaction.ok = typeof onFulfilled == 'function' ? onFulfilled : true;
              reaction.fail = typeof onRejected == 'function' && onRejected;
-             reaction.domain = engineIsNode ? process$4.domain : undefined;
+             reaction.domain = engineIsNode ? process$1.domain : undefined;
              state.parent = true;
              state.reactions.push(reaction);
-             if (state.state != PENDING) notify$1(state, false);
+             if (state.state != PENDING) notify(state, false);
              return reaction.promise;
            },
            // `Promise.prototype.catch` method
          });
          OwnPromiseCapability = function () {
            var promise = new Internal();
-           var state = getInternalState$4(promise);
+           var state = getInternalState$1(promise);
            this.promise = promise;
-           this.resolve = bind(internalResolve, state);
-           this.reject = bind(internalReject, state);
+           this.resolve = bind$2(internalResolve, state);
+           this.reject = bind$2(internalReject, state);
          };
-         newPromiseCapability.f = newPromiseCapability$1 = function (C) {
+         newPromiseCapability$1.f = newPromiseCapability = function (C) {
            return C === PromiseConstructor || C === PromiseWrapper
              ? new OwnPromiseCapability(C)
              : newGenericPromiseCapability(C);
          };
 
-         if ( typeof nativePromiseConstructor == 'function') {
-           nativeThen = nativePromiseConstructor.prototype.then;
+         if (typeof nativePromiseConstructor == 'function' && NativePromisePrototype !== Object.prototype) {
+           nativeThen = NativePromisePrototype.then;
 
-           // wrap native Promise#then for native async functions
-           redefine(nativePromiseConstructor.prototype, 'then', function then(onFulfilled, onRejected) {
-             var that = this;
-             return new PromiseConstructor(function (resolve, reject) {
-               nativeThen.call(that, resolve, reject);
-             }).then(onFulfilled, onRejected);
-           // https://github.com/zloirock/core-js/issues/640
-           }, { unsafe: true });
+           if (!SUBCLASSING) {
+             // make `Promise#then` return a polyfilled `Promise` for native promise-based APIs
+             redefine(NativePromisePrototype, 'then', function then(onFulfilled, onRejected) {
+               var that = this;
+               return new PromiseConstructor(function (resolve, reject) {
+                 nativeThen.call(that, resolve, reject);
+               }).then(onFulfilled, onRejected);
+             // https://github.com/zloirock/core-js/issues/640
+             }, { unsafe: true });
 
-           // wrap fetch result
-           if (typeof $fetch == 'function') _export({ global: true, enumerable: true, forced: true }, {
-             // eslint-disable-next-line no-unused-vars -- required for `.length`
-             fetch: function fetch(input /* , init */) {
-               return promiseResolve(PromiseConstructor, $fetch.apply(global_1, arguments));
-             }
-           });
+             // makes sure that native promise-based APIs `Promise#catch` properly works with patched `Promise#then`
+             redefine(NativePromisePrototype, 'catch', PromiseConstructorPrototype['catch'], { unsafe: true });
+           }
+
+           // make `.constructor === Promise` work for native promise-based APIs
+           try {
+             delete NativePromisePrototype.constructor;
+           } catch (error) { /* empty */ }
+
+           // make `instanceof Promise` work for native promise-based APIs
+           if (objectSetPrototypeOf) {
+             objectSetPrototypeOf(NativePromisePrototype, PromiseConstructorPrototype);
+           }
          }
        }
 
-       _export({ global: true, wrap: true, forced: FORCED }, {
+       _export({ global: true, wrap: true, forced: FORCED$f }, {
          Promise: PromiseConstructor
        });
 
        PromiseWrapper = getBuiltIn(PROMISE);
 
        // statics
-       _export({ target: PROMISE, stat: true, forced: FORCED }, {
+       _export({ target: PROMISE, stat: true, forced: FORCED$f }, {
          // `Promise.reject` method
          // https://tc39.es/ecma262/#sec-promise.reject
          reject: function reject(r) {
-           var capability = newPromiseCapability$1(this);
+           var capability = newPromiseCapability(this);
            capability.reject.call(undefined, r);
            return capability.promise;
          }
        });
 
-       _export({ target: PROMISE, stat: true, forced:  FORCED }, {
+       _export({ target: PROMISE, stat: true, forced: FORCED$f }, {
          // `Promise.resolve` method
          // https://tc39.es/ecma262/#sec-promise.resolve
          resolve: function resolve(x) {
-           return promiseResolve( this, x);
+           return promiseResolve(this, x);
          }
        });
 
-       _export({ target: PROMISE, stat: true, forced: INCORRECT_ITERATION }, {
+       _export({ target: PROMISE, stat: true, forced: INCORRECT_ITERATION$1 }, {
          // `Promise.all` method
          // https://tc39.es/ecma262/#sec-promise.all
          all: function all(iterable) {
            var C = this;
-           var capability = newPromiseCapability$1(C);
+           var capability = newPromiseCapability(C);
            var resolve = capability.resolve;
            var reject = capability.reject;
            var result = perform(function () {
-             var $promiseResolve = aFunction$1(C.resolve);
+             var $promiseResolve = aFunction(C.resolve);
              var values = [];
              var counter = 0;
              var remaining = 1;
          // https://tc39.es/ecma262/#sec-promise.race
          race: function race(iterable) {
            var C = this;
-           var capability = newPromiseCapability$1(C);
+           var capability = newPromiseCapability(C);
            var reject = capability.reject;
            var result = perform(function () {
-             var $promiseResolve = aFunction$1(C.resolve);
+             var $promiseResolve = aFunction(C.resolve);
              iterate(iterable, function (promise) {
                $promiseResolve.call(C, promise).then(capability.resolve, reject);
              });
 
        /* eslint-disable no-new -- required for testing */
 
+       var NATIVE_ARRAY_BUFFER_VIEWS = arrayBufferViewCore.NATIVE_ARRAY_BUFFER_VIEWS;
 
+       var ArrayBuffer$1 = global$2.ArrayBuffer;
+       var Int8Array$2 = global$2.Int8Array;
 
-       var NATIVE_ARRAY_BUFFER_VIEWS$2 = arrayBufferViewCore.NATIVE_ARRAY_BUFFER_VIEWS;
-
-       var ArrayBuffer$3 = global_1.ArrayBuffer;
-       var Int8Array$2 = global_1.Int8Array;
-
-       var typedArrayConstructorsRequireWrappers = !NATIVE_ARRAY_BUFFER_VIEWS$2 || !fails(function () {
+       var typedArrayConstructorsRequireWrappers = !NATIVE_ARRAY_BUFFER_VIEWS || !fails(function () {
          Int8Array$2(1);
        }) || !fails(function () {
          new Int8Array$2(-1);
          new Int8Array$2(iterable);
        }, true) || fails(function () {
          // Safari (11+) bug - a reason why even Safari 13 should load a typed array polyfill
-         return new Int8Array$2(new ArrayBuffer$3(2), 1, undefined).length !== 1;
+         return new Int8Array$2(new ArrayBuffer$1(2), 1, undefined).length !== 1;
        });
 
        var toPositiveInteger = function (it) {
          return offset;
        };
 
-       var aTypedArrayConstructor$1 = arrayBufferViewCore.aTypedArrayConstructor;
+       var aTypedArrayConstructor$3 = arrayBufferViewCore.aTypedArrayConstructor;
 
        var typedArrayFrom = function from(source /* , mapfn, thisArg */) {
          var O = toObject(source);
            mapfn = functionBindContext(mapfn, arguments[2], 2);
          }
          length = toLength(O.length);
-         result = new (aTypedArrayConstructor$1(this))(length);
+         result = new (aTypedArrayConstructor$3(this))(length);
          for (i = 0; length > i; i++) {
            result[i] = mapping ? mapfn(O[i], i) : O[i];
          }
            // we haven't completely correct pre-ES6 way for getting `new.target`, so use this
            typeof (NewTarget = dummy.constructor) == 'function' &&
            NewTarget !== Wrapper &&
-           isObject(NewTargetPrototype = NewTarget.prototype) &&
+           isObject$4(NewTargetPrototype = NewTarget.prototype) &&
            NewTargetPrototype !== Wrapper.prototype
          ) objectSetPrototypeOf($this, NewTargetPrototype);
          return $this;
        var nativeDefineProperty = objectDefineProperty.f;
        var nativeGetOwnPropertyDescriptor = objectGetOwnPropertyDescriptor.f;
        var round = Math.round;
-       var RangeError = global_1.RangeError;
+       var RangeError = global$2.RangeError;
        var ArrayBuffer = arrayBuffer.ArrayBuffer;
        var DataView = arrayBuffer.DataView;
        var NATIVE_ARRAY_BUFFER_VIEWS = arrayBufferViewCore.NATIVE_ARRAY_BUFFER_VIEWS;
 
        var wrappedDefineProperty = function defineProperty(target, key, descriptor) {
          if (isTypedArrayIndex(target, key = toPrimitive(key, true))
-           && isObject(descriptor)
-           && has(descriptor, 'value')
-           && !has(descriptor, 'get')
-           && !has(descriptor, 'set')
+           && isObject$4(descriptor)
+           && has$1(descriptor, 'value')
+           && !has$1(descriptor, 'get')
+           && !has$1(descriptor, 'set')
            // TODO: add validation descriptor w/o calling accessors
            && !descriptor.configurable
-           && (!has(descriptor, 'writable') || descriptor.writable)
-           && (!has(descriptor, 'enumerable') || descriptor.enumerable)
+           && (!has$1(descriptor, 'writable') || descriptor.writable)
+           && (!has$1(descriptor, 'enumerable') || descriptor.enumerable)
          ) {
            target[key] = descriptor.value;
            return target;
            var CONSTRUCTOR_NAME = TYPE + (CLAMPED ? 'Clamped' : '') + 'Array';
            var GETTER = 'get' + TYPE;
            var SETTER = 'set' + TYPE;
-           var NativeTypedArrayConstructor = global_1[CONSTRUCTOR_NAME];
+           var NativeTypedArrayConstructor = global$2[CONSTRUCTOR_NAME];
            var TypedArrayConstructor = NativeTypedArrayConstructor;
            var TypedArrayConstructorPrototype = TypedArrayConstructor && TypedArrayConstructor.prototype;
            var exported = {};
                var index = 0;
                var byteOffset = 0;
                var buffer, byteLength, length;
-               if (!isObject(data)) {
+               if (!isObject$4(data)) {
                  length = toIndex(data);
                  byteLength = length * BYTES;
                  buffer = new ArrayBuffer(byteLength);
              TypedArrayConstructor = wrapper(function (dummy, data, typedArrayOffset, $length) {
                anInstance(dummy, TypedArrayConstructor, CONSTRUCTOR_NAME);
                return inheritIfRequired(function () {
-                 if (!isObject(data)) return new NativeTypedArrayConstructor(toIndex(data));
+                 if (!isObject$4(data)) return new NativeTypedArrayConstructor(toIndex(data));
                  if (isArrayBuffer(data)) return $length !== undefined
                    ? new NativeTypedArrayConstructor(data, toOffset(typedArrayOffset, BYTES), $length)
                    : typedArrayOffset !== undefined
          };
        });
 
-       var min$2 = Math.min;
+       var min$7 = Math.min;
 
        // `Array.prototype.copyWithin` method implementation
        // https://tc39.es/ecma262/#sec-array.prototype.copywithin
+       // eslint-disable-next-line es/no-array-prototype-copywithin -- safe
        var arrayCopyWithin = [].copyWithin || function copyWithin(target /* = 0 */, start /* = 0, end = @length */) {
          var O = toObject(this);
          var len = toLength(O.length);
          var to = toAbsoluteIndex(target, len);
          var from = toAbsoluteIndex(start, len);
          var end = arguments.length > 2 ? arguments[2] : undefined;
-         var count = min$2((end === undefined ? len : toAbsoluteIndex(end, len)) - from, len - to);
+         var count = min$7((end === undefined ? len : toAbsoluteIndex(end, len)) - from, len - to);
          var inc = 1;
          if (from < to && to < from + count) {
            inc = -1;
          } return O;
        };
 
-       var aTypedArray$1 = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$1 = arrayBufferViewCore.exportTypedArrayMethod;
+       var aTypedArray$l = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$m = arrayBufferViewCore.exportTypedArrayMethod;
 
        // `%TypedArray%.prototype.copyWithin` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.copywithin
-       exportTypedArrayMethod$1('copyWithin', function copyWithin(target, start /* , end */) {
-         return arrayCopyWithin.call(aTypedArray$1(this), target, start, arguments.length > 2 ? arguments[2] : undefined);
+       exportTypedArrayMethod$m('copyWithin', function copyWithin(target, start /* , end */) {
+         return arrayCopyWithin.call(aTypedArray$l(this), target, start, arguments.length > 2 ? arguments[2] : undefined);
        });
 
-       var $every = arrayIteration.every;
+       var $every$1 = arrayIteration.every;
 
-       var aTypedArray$2 = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$2 = arrayBufferViewCore.exportTypedArrayMethod;
+       var aTypedArray$k = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$l = arrayBufferViewCore.exportTypedArrayMethod;
 
        // `%TypedArray%.prototype.every` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.every
-       exportTypedArrayMethod$2('every', function every(callbackfn /* , thisArg */) {
-         return $every(aTypedArray$2(this), callbackfn, arguments.length > 1 ? arguments[1] : undefined);
+       exportTypedArrayMethod$l('every', function every(callbackfn /* , thisArg */) {
+         return $every$1(aTypedArray$k(this), callbackfn, arguments.length > 1 ? arguments[1] : undefined);
        });
 
-       var aTypedArray$3 = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$3 = arrayBufferViewCore.exportTypedArrayMethod;
+       var aTypedArray$j = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$k = arrayBufferViewCore.exportTypedArrayMethod;
 
        // `%TypedArray%.prototype.fill` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.fill
        // eslint-disable-next-line no-unused-vars -- required for `.length`
-       exportTypedArrayMethod$3('fill', function fill(value /* , start, end */) {
-         return arrayFill.apply(aTypedArray$3(this), arguments);
+       exportTypedArrayMethod$k('fill', function fill(value /* , start, end */) {
+         return arrayFill.apply(aTypedArray$j(this), arguments);
        });
 
        var aTypedArrayConstructor$2 = arrayBufferViewCore.aTypedArrayConstructor;
          return result;
        };
 
-       var $filter = arrayIteration.filter;
+       var $filter$1 = arrayIteration.filter;
 
 
-       var aTypedArray$4 = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$4 = arrayBufferViewCore.exportTypedArrayMethod;
+       var aTypedArray$i = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$j = arrayBufferViewCore.exportTypedArrayMethod;
 
        // `%TypedArray%.prototype.filter` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.filter
-       exportTypedArrayMethod$4('filter', function filter(callbackfn /* , thisArg */) {
-         var list = $filter(aTypedArray$4(this), callbackfn, arguments.length > 1 ? arguments[1] : undefined);
+       exportTypedArrayMethod$j('filter', function filter(callbackfn /* , thisArg */) {
+         var list = $filter$1(aTypedArray$i(this), callbackfn, arguments.length > 1 ? arguments[1] : undefined);
          return typedArrayFromSpeciesAndList(this, list);
        });
 
-       var $find = arrayIteration.find;
+       var $find$1 = arrayIteration.find;
 
-       var aTypedArray$5 = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$5 = arrayBufferViewCore.exportTypedArrayMethod;
+       var aTypedArray$h = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$i = arrayBufferViewCore.exportTypedArrayMethod;
 
        // `%TypedArray%.prototype.find` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.find
-       exportTypedArrayMethod$5('find', function find(predicate /* , thisArg */) {
-         return $find(aTypedArray$5(this), predicate, arguments.length > 1 ? arguments[1] : undefined);
+       exportTypedArrayMethod$i('find', function find(predicate /* , thisArg */) {
+         return $find$1(aTypedArray$h(this), predicate, arguments.length > 1 ? arguments[1] : undefined);
        });
 
-       var $findIndex = arrayIteration.findIndex;
+       var $findIndex$1 = arrayIteration.findIndex;
 
-       var aTypedArray$6 = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$6 = arrayBufferViewCore.exportTypedArrayMethod;
+       var aTypedArray$g = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$h = arrayBufferViewCore.exportTypedArrayMethod;
 
        // `%TypedArray%.prototype.findIndex` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.findindex
-       exportTypedArrayMethod$6('findIndex', function findIndex(predicate /* , thisArg */) {
-         return $findIndex(aTypedArray$6(this), predicate, arguments.length > 1 ? arguments[1] : undefined);
+       exportTypedArrayMethod$h('findIndex', function findIndex(predicate /* , thisArg */) {
+         return $findIndex$1(aTypedArray$g(this), predicate, arguments.length > 1 ? arguments[1] : undefined);
        });
 
-       var $forEach$2 = arrayIteration.forEach;
+       var $forEach = arrayIteration.forEach;
 
-       var aTypedArray$7 = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$7 = arrayBufferViewCore.exportTypedArrayMethod;
+       var aTypedArray$f = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$g = arrayBufferViewCore.exportTypedArrayMethod;
 
        // `%TypedArray%.prototype.forEach` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.foreach
-       exportTypedArrayMethod$7('forEach', function forEach(callbackfn /* , thisArg */) {
-         $forEach$2(aTypedArray$7(this), callbackfn, arguments.length > 1 ? arguments[1] : undefined);
+       exportTypedArrayMethod$g('forEach', function forEach(callbackfn /* , thisArg */) {
+         $forEach(aTypedArray$f(this), callbackfn, arguments.length > 1 ? arguments[1] : undefined);
        });
 
-       var $includes = arrayIncludes.includes;
+       var $includes$1 = arrayIncludes.includes;
 
-       var aTypedArray$8 = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$8 = arrayBufferViewCore.exportTypedArrayMethod;
+       var aTypedArray$e = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$f = arrayBufferViewCore.exportTypedArrayMethod;
 
        // `%TypedArray%.prototype.includes` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.includes
-       exportTypedArrayMethod$8('includes', function includes(searchElement /* , fromIndex */) {
-         return $includes(aTypedArray$8(this), searchElement, arguments.length > 1 ? arguments[1] : undefined);
+       exportTypedArrayMethod$f('includes', function includes(searchElement /* , fromIndex */) {
+         return $includes$1(aTypedArray$e(this), searchElement, arguments.length > 1 ? arguments[1] : undefined);
        });
 
-       var $indexOf$1 = arrayIncludes.indexOf;
+       var $indexOf = arrayIncludes.indexOf;
 
-       var aTypedArray$9 = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$9 = arrayBufferViewCore.exportTypedArrayMethod;
+       var aTypedArray$d = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$e = arrayBufferViewCore.exportTypedArrayMethod;
 
        // `%TypedArray%.prototype.indexOf` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.indexof
-       exportTypedArrayMethod$9('indexOf', function indexOf(searchElement /* , fromIndex */) {
-         return $indexOf$1(aTypedArray$9(this), searchElement, arguments.length > 1 ? arguments[1] : undefined);
+       exportTypedArrayMethod$e('indexOf', function indexOf(searchElement /* , fromIndex */) {
+         return $indexOf(aTypedArray$d(this), searchElement, arguments.length > 1 ? arguments[1] : undefined);
        });
 
-       var ITERATOR$6 = wellKnownSymbol('iterator');
-       var Uint8Array$1 = global_1.Uint8Array;
+       var ITERATOR$2 = wellKnownSymbol('iterator');
+       var Uint8Array$2 = global$2.Uint8Array;
        var arrayValues = es_array_iterator.values;
        var arrayKeys = es_array_iterator.keys;
        var arrayEntries = es_array_iterator.entries;
-       var aTypedArray$a = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$a = arrayBufferViewCore.exportTypedArrayMethod;
-       var nativeTypedArrayIterator = Uint8Array$1 && Uint8Array$1.prototype[ITERATOR$6];
+       var aTypedArray$c = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$d = arrayBufferViewCore.exportTypedArrayMethod;
+       var nativeTypedArrayIterator = Uint8Array$2 && Uint8Array$2.prototype[ITERATOR$2];
 
        var CORRECT_ITER_NAME = !!nativeTypedArrayIterator
          && (nativeTypedArrayIterator.name == 'values' || nativeTypedArrayIterator.name == undefined);
 
        var typedArrayValues = function values() {
-         return arrayValues.call(aTypedArray$a(this));
+         return arrayValues.call(aTypedArray$c(this));
        };
 
        // `%TypedArray%.prototype.entries` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.entries
-       exportTypedArrayMethod$a('entries', function entries() {
-         return arrayEntries.call(aTypedArray$a(this));
+       exportTypedArrayMethod$d('entries', function entries() {
+         return arrayEntries.call(aTypedArray$c(this));
        });
        // `%TypedArray%.prototype.keys` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.keys
-       exportTypedArrayMethod$a('keys', function keys() {
-         return arrayKeys.call(aTypedArray$a(this));
+       exportTypedArrayMethod$d('keys', function keys() {
+         return arrayKeys.call(aTypedArray$c(this));
        });
        // `%TypedArray%.prototype.values` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.values
-       exportTypedArrayMethod$a('values', typedArrayValues, !CORRECT_ITER_NAME);
+       exportTypedArrayMethod$d('values', typedArrayValues, !CORRECT_ITER_NAME);
        // `%TypedArray%.prototype[@@iterator]` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype-@@iterator
-       exportTypedArrayMethod$a(ITERATOR$6, typedArrayValues, !CORRECT_ITER_NAME);
+       exportTypedArrayMethod$d(ITERATOR$2, typedArrayValues, !CORRECT_ITER_NAME);
 
        var aTypedArray$b = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$b = arrayBufferViewCore.exportTypedArrayMethod;
+       var exportTypedArrayMethod$c = arrayBufferViewCore.exportTypedArrayMethod;
        var $join = [].join;
 
        // `%TypedArray%.prototype.join` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.join
        // eslint-disable-next-line no-unused-vars -- required for `.length`
-       exportTypedArrayMethod$b('join', function join(separator) {
+       exportTypedArrayMethod$c('join', function join(separator) {
          return $join.apply(aTypedArray$b(this), arguments);
        });
 
-       var min$3 = Math.min;
-       var nativeLastIndexOf = [].lastIndexOf;
-       var NEGATIVE_ZERO$1 = !!nativeLastIndexOf && 1 / [1].lastIndexOf(1, -0) < 0;
-       var STRICT_METHOD$2 = arrayMethodIsStrict('lastIndexOf');
-       var FORCED$1 = NEGATIVE_ZERO$1 || !STRICT_METHOD$2;
+       /* eslint-disable es/no-array-prototype-lastindexof -- safe */
+
+
+
+
+
+       var min$6 = Math.min;
+       var $lastIndexOf = [].lastIndexOf;
+       var NEGATIVE_ZERO = !!$lastIndexOf && 1 / [1].lastIndexOf(1, -0) < 0;
+       var STRICT_METHOD$5 = arrayMethodIsStrict('lastIndexOf');
+       var FORCED$e = NEGATIVE_ZERO || !STRICT_METHOD$5;
 
        // `Array.prototype.lastIndexOf` method implementation
        // https://tc39.es/ecma262/#sec-array.prototype.lastindexof
-       var arrayLastIndexOf = FORCED$1 ? function lastIndexOf(searchElement /* , fromIndex = @[*-1] */) {
+       var arrayLastIndexOf = FORCED$e ? function lastIndexOf(searchElement /* , fromIndex = @[*-1] */) {
          // convert -0 to +0
-         if (NEGATIVE_ZERO$1) return nativeLastIndexOf.apply(this, arguments) || 0;
+         if (NEGATIVE_ZERO) return $lastIndexOf.apply(this, arguments) || 0;
          var O = toIndexedObject(this);
          var length = toLength(O.length);
          var index = length - 1;
-         if (arguments.length > 1) index = min$3(index, toInteger(arguments[1]));
+         if (arguments.length > 1) index = min$6(index, toInteger(arguments[1]));
          if (index < 0) index = length + index;
          for (;index >= 0; index--) if (index in O && O[index] === searchElement) return index || 0;
          return -1;
-       } : nativeLastIndexOf;
+       } : $lastIndexOf;
 
-       var aTypedArray$c = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$c = arrayBufferViewCore.exportTypedArrayMethod;
+       var aTypedArray$a = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$b = arrayBufferViewCore.exportTypedArrayMethod;
 
        // `%TypedArray%.prototype.lastIndexOf` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.lastindexof
        // eslint-disable-next-line no-unused-vars -- required for `.length`
-       exportTypedArrayMethod$c('lastIndexOf', function lastIndexOf(searchElement /* , fromIndex */) {
-         return arrayLastIndexOf.apply(aTypedArray$c(this), arguments);
+       exportTypedArrayMethod$b('lastIndexOf', function lastIndexOf(searchElement /* , fromIndex */) {
+         return arrayLastIndexOf.apply(aTypedArray$a(this), arguments);
        });
 
-       var $map$1 = arrayIteration.map;
+       var $map = arrayIteration.map;
 
 
-       var aTypedArray$d = arrayBufferViewCore.aTypedArray;
-       var aTypedArrayConstructor$3 = arrayBufferViewCore.aTypedArrayConstructor;
-       var exportTypedArrayMethod$d = arrayBufferViewCore.exportTypedArrayMethod;
+       var aTypedArray$9 = arrayBufferViewCore.aTypedArray;
+       var aTypedArrayConstructor$1 = arrayBufferViewCore.aTypedArrayConstructor;
+       var exportTypedArrayMethod$a = arrayBufferViewCore.exportTypedArrayMethod;
 
        // `%TypedArray%.prototype.map` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.map
-       exportTypedArrayMethod$d('map', function map(mapfn /* , thisArg */) {
-         return $map$1(aTypedArray$d(this), mapfn, arguments.length > 1 ? arguments[1] : undefined, function (O, length) {
-           return new (aTypedArrayConstructor$3(speciesConstructor(O, O.constructor)))(length);
+       exportTypedArrayMethod$a('map', function map(mapfn /* , thisArg */) {
+         return $map(aTypedArray$9(this), mapfn, arguments.length > 1 ? arguments[1] : undefined, function (O, length) {
+           return new (aTypedArrayConstructor$1(speciesConstructor(O, O.constructor)))(length);
          });
        });
 
        // `Array.prototype.{ reduce, reduceRight }` methods implementation
        var createMethod$3 = function (IS_RIGHT) {
          return function (that, callbackfn, argumentsLength, memo) {
-           aFunction$1(callbackfn);
+           aFunction(callbackfn);
            var O = toObject(that);
            var self = indexedObject(O);
            var length = toLength(O.length);
          right: createMethod$3(true)
        };
 
-       var $reduce = arrayReduce.left;
+       var $reduce$1 = arrayReduce.left;
 
-       var aTypedArray$e = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$e = arrayBufferViewCore.exportTypedArrayMethod;
+       var aTypedArray$8 = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$9 = arrayBufferViewCore.exportTypedArrayMethod;
 
        // `%TypedArray%.prototype.reduce` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.reduce
-       exportTypedArrayMethod$e('reduce', function reduce(callbackfn /* , initialValue */) {
-         return $reduce(aTypedArray$e(this), callbackfn, arguments.length, arguments.length > 1 ? arguments[1] : undefined);
+       exportTypedArrayMethod$9('reduce', function reduce(callbackfn /* , initialValue */) {
+         return $reduce$1(aTypedArray$8(this), callbackfn, arguments.length, arguments.length > 1 ? arguments[1] : undefined);
        });
 
        var $reduceRight = arrayReduce.right;
 
-       var aTypedArray$f = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$f = arrayBufferViewCore.exportTypedArrayMethod;
+       var aTypedArray$7 = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$8 = arrayBufferViewCore.exportTypedArrayMethod;
 
        // `%TypedArray%.prototype.reduceRicht` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.reduceright
-       exportTypedArrayMethod$f('reduceRight', function reduceRight(callbackfn /* , initialValue */) {
-         return $reduceRight(aTypedArray$f(this), callbackfn, arguments.length, arguments.length > 1 ? arguments[1] : undefined);
+       exportTypedArrayMethod$8('reduceRight', function reduceRight(callbackfn /* , initialValue */) {
+         return $reduceRight(aTypedArray$7(this), callbackfn, arguments.length, arguments.length > 1 ? arguments[1] : undefined);
        });
 
-       var aTypedArray$g = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$g = arrayBufferViewCore.exportTypedArrayMethod;
-       var floor$2 = Math.floor;
+       var aTypedArray$6 = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$7 = arrayBufferViewCore.exportTypedArrayMethod;
+       var floor$5 = Math.floor;
 
        // `%TypedArray%.prototype.reverse` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.reverse
-       exportTypedArrayMethod$g('reverse', function reverse() {
+       exportTypedArrayMethod$7('reverse', function reverse() {
          var that = this;
-         var length = aTypedArray$g(that).length;
-         var middle = floor$2(length / 2);
+         var length = aTypedArray$6(that).length;
+         var middle = floor$5(length / 2);
          var index = 0;
          var value;
          while (index < middle) {
          } return that;
        });
 
-       var aTypedArray$h = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$h = arrayBufferViewCore.exportTypedArrayMethod;
+       var aTypedArray$5 = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$6 = arrayBufferViewCore.exportTypedArrayMethod;
 
-       var FORCED$2 = fails(function () {
-         /* global Int8Array -- safe */
+       var FORCED$d = fails(function () {
+         // eslint-disable-next-line es/no-typed-arrays -- required for testing
          new Int8Array(1).set({});
        });
 
        // `%TypedArray%.prototype.set` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.set
-       exportTypedArrayMethod$h('set', function set(arrayLike /* , offset */) {
-         aTypedArray$h(this);
+       exportTypedArrayMethod$6('set', function set(arrayLike /* , offset */) {
+         aTypedArray$5(this);
          var offset = toOffset(arguments.length > 1 ? arguments[1] : undefined, 1);
          var length = this.length;
          var src = toObject(arrayLike);
          var index = 0;
          if (len + offset > length) throw RangeError('Wrong length');
          while (index < len) this[offset + index] = src[index++];
-       }, FORCED$2);
+       }, FORCED$d);
 
-       var aTypedArray$i = arrayBufferViewCore.aTypedArray;
-       var aTypedArrayConstructor$4 = arrayBufferViewCore.aTypedArrayConstructor;
-       var exportTypedArrayMethod$i = arrayBufferViewCore.exportTypedArrayMethod;
-       var $slice = [].slice;
+       var aTypedArray$4 = arrayBufferViewCore.aTypedArray;
+       var aTypedArrayConstructor = arrayBufferViewCore.aTypedArrayConstructor;
+       var exportTypedArrayMethod$5 = arrayBufferViewCore.exportTypedArrayMethod;
+       var $slice$1 = [].slice;
 
-       var FORCED$3 = fails(function () {
-         /* global Int8Array -- safe */
+       var FORCED$c = fails(function () {
+         // eslint-disable-next-line es/no-typed-arrays -- required for testing
          new Int8Array(1).slice();
        });
 
        // `%TypedArray%.prototype.slice` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.slice
-       exportTypedArrayMethod$i('slice', function slice(start, end) {
-         var list = $slice.call(aTypedArray$i(this), start, end);
+       exportTypedArrayMethod$5('slice', function slice(start, end) {
+         var list = $slice$1.call(aTypedArray$4(this), start, end);
          var C = speciesConstructor(this, this.constructor);
          var index = 0;
          var length = list.length;
-         var result = new (aTypedArrayConstructor$4(C))(length);
+         var result = new (aTypedArrayConstructor(C))(length);
          while (length > index) result[index] = list[index++];
          return result;
-       }, FORCED$3);
+       }, FORCED$c);
 
-       var $some = arrayIteration.some;
+       var $some$1 = arrayIteration.some;
 
-       var aTypedArray$j = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$j = arrayBufferViewCore.exportTypedArrayMethod;
+       var aTypedArray$3 = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$4 = arrayBufferViewCore.exportTypedArrayMethod;
 
        // `%TypedArray%.prototype.some` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.some
-       exportTypedArrayMethod$j('some', function some(callbackfn /* , thisArg */) {
-         return $some(aTypedArray$j(this), callbackfn, arguments.length > 1 ? arguments[1] : undefined);
+       exportTypedArrayMethod$4('some', function some(callbackfn /* , thisArg */) {
+         return $some$1(aTypedArray$3(this), callbackfn, arguments.length > 1 ? arguments[1] : undefined);
        });
 
-       var aTypedArray$k = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$k = arrayBufferViewCore.exportTypedArrayMethod;
-       var $sort = [].sort;
+       // TODO: use something more complex like timsort?
+       var floor$4 = Math.floor;
+
+       var mergeSort = function (array, comparefn) {
+         var length = array.length;
+         var middle = floor$4(length / 2);
+         return length < 8 ? insertionSort(array, comparefn) : merge$5(
+           mergeSort(array.slice(0, middle), comparefn),
+           mergeSort(array.slice(middle), comparefn),
+           comparefn
+         );
+       };
+
+       var insertionSort = function (array, comparefn) {
+         var length = array.length;
+         var i = 1;
+         var element, j;
+
+         while (i < length) {
+           j = i;
+           element = array[i];
+           while (j && comparefn(array[j - 1], element) > 0) {
+             array[j] = array[--j];
+           }
+           if (j !== i++) array[j] = element;
+         } return array;
+       };
+
+       var merge$5 = function (left, right, comparefn) {
+         var llength = left.length;
+         var rlength = right.length;
+         var lindex = 0;
+         var rindex = 0;
+         var result = [];
+
+         while (lindex < llength || rindex < rlength) {
+           if (lindex < llength && rindex < rlength) {
+             result.push(comparefn(left[lindex], right[rindex]) <= 0 ? left[lindex++] : right[rindex++]);
+           } else {
+             result.push(lindex < llength ? left[lindex++] : right[rindex++]);
+           }
+         } return result;
+       };
+
+       var arraySort = mergeSort;
+
+       var firefox = engineUserAgent.match(/firefox\/(\d+)/i);
+
+       var engineFfVersion = !!firefox && +firefox[1];
+
+       var engineIsIeOrEdge = /MSIE|Trident/.test(engineUserAgent);
+
+       var webkit = engineUserAgent.match(/AppleWebKit\/(\d+)\./);
+
+       var engineWebkitVersion = !!webkit && +webkit[1];
+
+       var aTypedArray$2 = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$3 = arrayBufferViewCore.exportTypedArrayMethod;
+       var Uint16Array = global$2.Uint16Array;
+       var nativeSort$1 = Uint16Array && Uint16Array.prototype.sort;
+
+       // WebKit
+       var ACCEPT_INCORRECT_ARGUMENTS = !!nativeSort$1 && !fails(function () {
+         var array = new Uint16Array(2);
+         array.sort(null);
+         array.sort({});
+       });
+
+       var STABLE_SORT$1 = !!nativeSort$1 && !fails(function () {
+         // feature detection can be too slow, so check engines versions
+         if (engineV8Version) return engineV8Version < 74;
+         if (engineFfVersion) return engineFfVersion < 67;
+         if (engineIsIeOrEdge) return true;
+         if (engineWebkitVersion) return engineWebkitVersion < 602;
+
+         var array = new Uint16Array(516);
+         var expected = Array(516);
+         var index, mod;
+
+         for (index = 0; index < 516; index++) {
+           mod = index % 4;
+           array[index] = 515 - index;
+           expected[index] = index - 2 * mod + 3;
+         }
+
+         array.sort(function (a, b) {
+           return (a / 4 | 0) - (b / 4 | 0);
+         });
+
+         for (index = 0; index < 516; index++) {
+           if (array[index] !== expected[index]) return true;
+         }
+       });
+
+       var getSortCompare$1 = function (comparefn) {
+         return function (x, y) {
+           if (comparefn !== undefined) return +comparefn(x, y) || 0;
+           // eslint-disable-next-line no-self-compare -- NaN check
+           if (y !== y) return -1;
+           // eslint-disable-next-line no-self-compare -- NaN check
+           if (x !== x) return 1;
+           if (x === 0 && y === 0) return 1 / x > 0 && 1 / y < 0 ? 1 : -1;
+           return x > y;
+         };
+       };
 
        // `%TypedArray%.prototype.sort` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.sort
-       exportTypedArrayMethod$k('sort', function sort(comparefn) {
-         return $sort.call(aTypedArray$k(this), comparefn);
-       });
+       exportTypedArrayMethod$3('sort', function sort(comparefn) {
+         var array = this;
+         if (comparefn !== undefined) aFunction(comparefn);
+         if (STABLE_SORT$1) return nativeSort$1.call(array, comparefn);
 
-       var aTypedArray$l = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$l = arrayBufferViewCore.exportTypedArrayMethod;
+         aTypedArray$2(array);
+         var arrayLength = toLength(array.length);
+         var items = Array(arrayLength);
+         var index;
+
+         for (index = 0; index < arrayLength; index++) {
+           items[index] = array[index];
+         }
+
+         items = arraySort(array, getSortCompare$1(comparefn));
+
+         for (index = 0; index < arrayLength; index++) {
+           array[index] = items[index];
+         }
+
+         return array;
+       }, !STABLE_SORT$1 || ACCEPT_INCORRECT_ARGUMENTS);
+
+       var aTypedArray$1 = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$2 = arrayBufferViewCore.exportTypedArrayMethod;
 
        // `%TypedArray%.prototype.subarray` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.subarray
-       exportTypedArrayMethod$l('subarray', function subarray(begin, end) {
-         var O = aTypedArray$l(this);
+       exportTypedArrayMethod$2('subarray', function subarray(begin, end) {
+         var O = aTypedArray$1(this);
          var length = O.length;
          var beginIndex = toAbsoluteIndex(begin, length);
          return new (speciesConstructor(O, O.constructor))(
          );
        });
 
-       var Int8Array$3 = global_1.Int8Array;
-       var aTypedArray$m = arrayBufferViewCore.aTypedArray;
-       var exportTypedArrayMethod$m = arrayBufferViewCore.exportTypedArrayMethod;
+       var Int8Array$1 = global$2.Int8Array;
+       var aTypedArray = arrayBufferViewCore.aTypedArray;
+       var exportTypedArrayMethod$1 = arrayBufferViewCore.exportTypedArrayMethod;
        var $toLocaleString = [].toLocaleString;
-       var $slice$1 = [].slice;
+       var $slice = [].slice;
 
        // iOS Safari 6.x fails here
-       var TO_LOCALE_STRING_BUG = !!Int8Array$3 && fails(function () {
-         $toLocaleString.call(new Int8Array$3(1));
+       var TO_LOCALE_STRING_BUG = !!Int8Array$1 && fails(function () {
+         $toLocaleString.call(new Int8Array$1(1));
        });
 
-       var FORCED$4 = fails(function () {
-         return [1, 2].toLocaleString() != new Int8Array$3([1, 2]).toLocaleString();
+       var FORCED$b = fails(function () {
+         return [1, 2].toLocaleString() != new Int8Array$1([1, 2]).toLocaleString();
        }) || !fails(function () {
-         Int8Array$3.prototype.toLocaleString.call([1, 2]);
+         Int8Array$1.prototype.toLocaleString.call([1, 2]);
        });
 
        // `%TypedArray%.prototype.toLocaleString` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.tolocalestring
-       exportTypedArrayMethod$m('toLocaleString', function toLocaleString() {
-         return $toLocaleString.apply(TO_LOCALE_STRING_BUG ? $slice$1.call(aTypedArray$m(this)) : aTypedArray$m(this), arguments);
-       }, FORCED$4);
+       exportTypedArrayMethod$1('toLocaleString', function toLocaleString() {
+         return $toLocaleString.apply(TO_LOCALE_STRING_BUG ? $slice.call(aTypedArray(this)) : aTypedArray(this), arguments);
+       }, FORCED$b);
 
-       var exportTypedArrayMethod$n = arrayBufferViewCore.exportTypedArrayMethod;
+       var exportTypedArrayMethod = arrayBufferViewCore.exportTypedArrayMethod;
 
 
 
-       var Uint8Array$2 = global_1.Uint8Array;
-       var Uint8ArrayPrototype = Uint8Array$2 && Uint8Array$2.prototype || {};
+       var Uint8Array$1 = global$2.Uint8Array;
+       var Uint8ArrayPrototype = Uint8Array$1 && Uint8Array$1.prototype || {};
        var arrayToString = [].toString;
        var arrayJoin = [].join;
 
 
        // `%TypedArray%.prototype.toString` method
        // https://tc39.es/ecma262/#sec-%typedarray%.prototype.tostring
-       exportTypedArrayMethod$n('toString', arrayToString, IS_NOT_ARRAY_METHOD);
+       exportTypedArrayMethod('toString', arrayToString, IS_NOT_ARRAY_METHOD);
 
        var nativeJoin = [].join;
 
        var ES3_STRINGS = indexedObject != Object;
-       var STRICT_METHOD$3 = arrayMethodIsStrict('join', ',');
+       var STRICT_METHOD$4 = arrayMethodIsStrict('join', ',');
 
        // `Array.prototype.join` method
        // https://tc39.es/ecma262/#sec-array.prototype.join
-       _export({ target: 'Array', proto: true, forced: ES3_STRINGS || !STRICT_METHOD$3 }, {
+       _export({ target: 'Array', proto: true, forced: ES3_STRINGS || !STRICT_METHOD$4 }, {
          join: function join(separator) {
            return nativeJoin.call(toIndexedObject(this), separator === undefined ? ',' : separator);
          }
          else object[propertyKey] = value;
        };
 
-       var HAS_SPECIES_SUPPORT$1 = arrayMethodHasSpeciesSupport('slice');
+       var HAS_SPECIES_SUPPORT$2 = arrayMethodHasSpeciesSupport('slice');
 
-       var SPECIES$5 = wellKnownSymbol('species');
+       var SPECIES$1 = wellKnownSymbol('species');
        var nativeSlice = [].slice;
-       var max$1 = Math.max;
+       var max$3 = Math.max;
 
        // `Array.prototype.slice` method
        // https://tc39.es/ecma262/#sec-array.prototype.slice
        // fallback for not array-like ES3 strings and DOM objects
-       _export({ target: 'Array', proto: true, forced: !HAS_SPECIES_SUPPORT$1 }, {
+       _export({ target: 'Array', proto: true, forced: !HAS_SPECIES_SUPPORT$2 }, {
          slice: function slice(start, end) {
            var O = toIndexedObject(this);
            var length = toLength(O.length);
              // cross-realm fallback
              if (typeof Constructor == 'function' && (Constructor === Array || isArray(Constructor.prototype))) {
                Constructor = undefined;
-             } else if (isObject(Constructor)) {
-               Constructor = Constructor[SPECIES$5];
+             } else if (isObject$4(Constructor)) {
+               Constructor = Constructor[SPECIES$1];
                if (Constructor === null) Constructor = undefined;
              }
              if (Constructor === Array || Constructor === undefined) {
                return nativeSlice.call(O, k, fin);
              }
            }
-           result = new (Constructor === undefined ? Array : Constructor)(max$1(fin - k, 0));
+           result = new (Constructor === undefined ? Array : Constructor)(max$3(fin - k, 0));
            for (n = 0; k < fin; k++, n++) if (k in O) createProperty(result, n, O[k]);
            result.length = n;
            return result;
          }
        });
 
-       var ITERATOR$7 = wellKnownSymbol('iterator');
+       var ITERATOR$1 = wellKnownSymbol('iterator');
 
        var nativeUrl = !fails(function () {
          var url = new URL('b?a=1&b=2&c=3', 'http://a');
            || url.href !== 'http://a/c%20d?a=1&c=3'
            || searchParams.get('c') !== '3'
            || String(new URLSearchParams('?a=1')) !== 'a=1'
-           || !searchParams[ITERATOR$7]
+           || !searchParams[ITERATOR$1]
            // throws in Edge
            || new URL('https://a@b').username !== 'a'
            || new URLSearchParams(new URLSearchParams('a=b')).get('a') !== 'b'
            || new URL('http://x', undefined).host !== 'x';
        });
 
-       var nativeAssign = Object.assign;
-       var defineProperty$5 = Object.defineProperty;
+       // eslint-disable-next-line es/no-object-assign -- safe
+       var $assign = Object.assign;
+       // eslint-disable-next-line es/no-object-defineproperty -- required for testing
+       var defineProperty$4 = Object.defineProperty;
 
        // `Object.assign` method
        // https://tc39.es/ecma262/#sec-object.assign
-       var objectAssign = !nativeAssign || fails(function () {
+       var objectAssign = !$assign || fails(function () {
          // should have correct order of operations (Edge bug)
-         if (descriptors && nativeAssign({ b: 1 }, nativeAssign(defineProperty$5({}, 'a', {
+         if (descriptors && $assign({ b: 1 }, $assign(defineProperty$4({}, 'a', {
            enumerable: true,
            get: function () {
-             defineProperty$5(this, 'b', {
+             defineProperty$4(this, 'b', {
                value: 3,
                enumerable: false
              });
          // should work with symbols and should have deterministic property order (V8 bug)
          var A = {};
          var B = {};
-         /* global Symbol -- required for testing */
+         // eslint-disable-next-line es/no-symbol -- safe
          var symbol = Symbol();
          var alphabet = 'abcdefghijklmnopqrst';
          A[symbol] = 7;
          alphabet.split('').forEach(function (chr) { B[chr] = chr; });
-         return nativeAssign({}, A)[symbol] != 7 || objectKeys(nativeAssign({}, B)).join('') != alphabet;
+         return $assign({}, A)[symbol] != 7 || objectKeys($assign({}, B)).join('') != alphabet;
        }) ? function assign(target, source) { // eslint-disable-line no-unused-vars -- required for `.length`
          var T = toObject(target);
          var argumentsLength = arguments.length;
              if (!descriptors || propertyIsEnumerable.call(S, key)) T[key] = S[key];
            }
          } return T;
-       } : nativeAssign;
+       } : $assign;
 
        // call something on iterator step with safe closing on error
        var callWithSafeIterationClosing = function (iterator, fn, value, ENTRIES) {
          try {
            return ENTRIES ? fn(anObject(value)[0], value[1]) : fn(value);
-         // 7.4.6 IteratorClose(iterator, completion)
          } catch (error) {
            iteratorClose(iterator);
            throw error;
 
 
 
-       var $fetch$1 = getBuiltIn('fetch');
-       var Headers = getBuiltIn('Headers');
-       var ITERATOR$8 = wellKnownSymbol('iterator');
+       var $fetch = getBuiltIn('fetch');
+       var Headers$1 = getBuiltIn('Headers');
+       var ITERATOR = wellKnownSymbol('iterator');
        var URL_SEARCH_PARAMS = 'URLSearchParams';
        var URL_SEARCH_PARAMS_ITERATOR = URL_SEARCH_PARAMS + 'Iterator';
-       var setInternalState$5 = internalState.set;
+       var setInternalState$2 = internalState.set;
        var getInternalParamsState = internalState.getterFor(URL_SEARCH_PARAMS);
        var getInternalIteratorState = internalState.getterFor(URL_SEARCH_PARAMS_ITERATOR);
 
          }
        };
 
-       var find = /[!'()~]|%20/g;
+       var find$1 = /[!'()~]|%20/g;
 
-       var replace = {
+       var replace$1 = {
          '!': '%21',
          "'": '%27',
          '(': '%28',
        };
 
        var replacer = function (match) {
-         return replace[match];
+         return replace$1[match];
        };
 
        var serialize = function (it) {
-         return encodeURIComponent(it).replace(find, replacer);
+         return encodeURIComponent(it).replace(find$1, replacer);
        };
 
        var parseSearchParams = function (result, query) {
        };
 
        var URLSearchParamsIterator = createIteratorConstructor(function Iterator(params, kind) {
-         setInternalState$5(this, {
+         setInternalState$2(this, {
            type: URL_SEARCH_PARAMS_ITERATOR,
            iterator: getIterator(getInternalParamsState(params).entries),
            kind: kind
          var entries = [];
          var iteratorMethod, iterator, next, step, entryIterator, entryNext, first, second, key;
 
-         setInternalState$5(that, {
+         setInternalState$2(that, {
            type: URL_SEARCH_PARAMS,
            entries: entries,
            updateURL: function () { /* empty */ },
          });
 
          if (init !== undefined) {
-           if (isObject(init)) {
+           if (isObject$4(init)) {
              iteratorMethod = getIteratorMethod(init);
              if (typeof iteratorMethod === 'function') {
                iterator = iteratorMethod.call(init);
                  ) throw TypeError('Expected sequence with length 2');
                  entries.push({ key: first.value + '', value: second.value + '' });
                }
-             } else for (key in init) if (has(init, key)) entries.push({ key: key, value: init[key] + '' });
+             } else for (key in init) if (has$1(init, key)) entries.push({ key: key, value: init[key] + '' });
            } else {
              parseSearchParams(entries, typeof init === 'string' ? init.charAt(0) === '?' ? init.slice(1) : init : init + '');
            }
        }, { enumerable: true });
 
        // `URLSearchParams.prototype[@@iterator]` method
-       redefine(URLSearchParamsPrototype, ITERATOR$8, URLSearchParamsPrototype.entries);
+       redefine(URLSearchParamsPrototype, ITERATOR, URLSearchParamsPrototype.entries);
 
        // `URLSearchParams.prototype.toString` method
        // https://url.spec.whatwg.org/#urlsearchparams-stringification-behavior
 
        // Wrap `fetch` for correct work with polyfilled `URLSearchParams`
        // https://github.com/zloirock/core-js/issues/674
-       if (!nativeUrl && typeof $fetch$1 == 'function' && typeof Headers == 'function') {
+       if (!nativeUrl && typeof $fetch == 'function' && typeof Headers$1 == 'function') {
          _export({ global: true, enumerable: true, forced: true }, {
            fetch: function fetch(input /* , init */) {
              var args = [input];
              var init, body, headers;
              if (arguments.length > 1) {
                init = arguments[1];
-               if (isObject(init)) {
+               if (isObject$4(init)) {
                  body = init.body;
                  if (classof(body) === URL_SEARCH_PARAMS) {
-                   headers = init.headers ? new Headers(init.headers) : new Headers();
+                   headers = init.headers ? new Headers$1(init.headers) : new Headers$1();
                    if (!headers.has('content-type')) {
                      headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8');
                    }
                  }
                }
                args.push(init);
-             } return $fetch$1.apply(this, args);
+             } return $fetch.apply(this, args);
            }
          });
        }
 
 
 
-       var NativeURL = global_1.URL;
+       var NativeURL = global$2.URL;
        var URLSearchParams$1 = web_urlSearchParams.URLSearchParams;
        var getInternalSearchParamsState = web_urlSearchParams.getState;
-       var setInternalState$6 = internalState.set;
+       var setInternalState$1 = internalState.set;
        var getInternalURLState = internalState.getterFor('URL');
-       var floor$4 = Math.floor;
+       var floor$2 = Math.floor;
        var pow$1 = Math.pow;
 
        var INVALID_AUTHORITY = 'Invalid authority';
        var INVALID_PORT = 'Invalid port';
 
        var ALPHA = /[A-Za-z]/;
+       // eslint-disable-next-line regexp/no-obscure-range -- safe
        var ALPHANUMERIC = /[\d+-.A-Za-z]/;
        var DIGIT = /\d/;
-       var HEX_START = /^(0x|0X)/;
+       var HEX_START = /^0x/i;
        var OCT = /^[0-7]+$/;
        var DEC = /^\d+$/;
        var HEX = /^[\dA-Fa-f]+$/;
        /* eslint-disable no-control-regex -- safe */
-       var FORBIDDEN_HOST_CODE_POINT = /[\u0000\t\u000A\u000D #%/:?@[\\]]/;
-       var FORBIDDEN_HOST_CODE_POINT_EXCLUDING_PERCENT = /[\u0000\t\u000A\u000D #/:?@[\\]]/;
+       var FORBIDDEN_HOST_CODE_POINT = /[\0\t\n\r #%/:<>?@[\\\]^|]/;
+       var FORBIDDEN_HOST_CODE_POINT_EXCLUDING_PERCENT = /[\0\t\n\r #/:<>?@[\\\]^|]/;
        var LEADING_AND_TRAILING_C0_CONTROL_OR_SPACE = /^[\u0000-\u001F ]+|[\u0000-\u001F ]+$/g;
-       var TAB_AND_NEW_LINE = /[\t\u000A\u000D]/g;
+       var TAB_AND_NEW_LINE = /[\t\n\r]/g;
        /* eslint-enable no-control-regex -- safe */
        var EOF;
 
            result = [];
            for (index = 0; index < 4; index++) {
              result.unshift(host % 256);
-             host = floor$4(host / 256);
+             host = floor$2(host / 256);
            } return result.join('.');
          // ipv6
          } else if (typeof host == 'object') {
 
        var percentEncode = function (char, set) {
          var code = codeAt(char, 0);
-         return code > 0x20 && code < 0x7F && !has(set, char) ? char : encodeURIComponent(char);
+         return code > 0x20 && code < 0x7F && !has$1(set, char) ? char : encodeURIComponent(char);
        };
 
        var specialSchemes = {
        };
 
        var isSpecial = function (url) {
-         return has(specialSchemes, url.scheme);
+         return has$1(specialSchemes, url.scheme);
        };
 
        var includesCredentials = function (url) {
                  buffer += char.toLowerCase();
                } else if (char == ':') {
                  if (stateOverride && (
-                   (isSpecial(url) != has(specialSchemes, buffer)) ||
+                   (isSpecial(url) != has$1(specialSchemes, buffer)) ||
                    (buffer == 'file' && (includesCredentials(url) || url.port !== null)) ||
                    (url.scheme == 'file' && !url.host)
                  )) return;
          var that = anInstance(this, URLConstructor, 'URL');
          var base = arguments.length > 1 ? arguments[1] : undefined;
          var urlString = String(url);
-         var state = setInternalState$6(that, { type: 'URL' });
+         var state = setInternalState$1(that, { type: 'URL' });
          var baseState, failure;
          if (base !== undefined) {
            if (base instanceof URLConstructor) baseState = getInternalURLState(base);
          var scheme = url.scheme;
          var port = url.port;
          if (scheme == 'blob') try {
-           return new URL(scheme.path[0]).origin;
+           return new URLConstructor(scheme.path[0]).origin;
          } catch (error) {
            return 'null';
          }
          return result;
        };
 
-       var TO_STRING$1 = 'toString';
-       var RegExpPrototype = RegExp.prototype;
-       var nativeToString = RegExpPrototype[TO_STRING$1];
+       var TO_STRING = 'toString';
+       var RegExpPrototype$2 = RegExp.prototype;
+       var nativeToString = RegExpPrototype$2[TO_STRING];
 
        var NOT_GENERIC = fails(function () { return nativeToString.call({ source: 'a', flags: 'b' }) != '/a/b'; });
        // FF44- RegExp#toString has a wrong name
-       var INCORRECT_NAME = nativeToString.name != TO_STRING$1;
+       var INCORRECT_NAME = nativeToString.name != TO_STRING;
 
        // `RegExp.prototype.toString` method
        // https://tc39.es/ecma262/#sec-regexp.prototype.tostring
        if (NOT_GENERIC || INCORRECT_NAME) {
-         redefine(RegExp.prototype, TO_STRING$1, function toString() {
+         redefine(RegExp.prototype, TO_STRING, function toString() {
            var R = anObject(this);
            var p = String(R.source);
            var rf = R.flags;
-           var f = String(rf === undefined && R instanceof RegExp && !('flags' in RegExpPrototype) ? regexpFlags.call(R) : rf);
+           var f = String(rf === undefined && R instanceof RegExp && !('flags' in RegExpPrototype$2) ? regexpFlags.call(R) : rf);
            return '/' + p + '/' + f;
          }, { unsafe: true });
        }
 
        // babel-minify transpiles RegExp('a', 'y') -> /a/y and it causes SyntaxError,
-       // so we use an intermediate function.
-       function RE(s, f) {
+       var RE = function (s, f) {
          return RegExp(s, f);
-       }
+       };
 
-       var UNSUPPORTED_Y = fails(function () {
-         // babel-minify transpiles RegExp('a', 'y') -> /a/y and it causes SyntaxError
+       var UNSUPPORTED_Y$3 = fails(function () {
          var re = RE('a', 'y');
          re.lastIndex = 2;
          return re.exec('abcd') != null;
        });
 
        var regexpStickyHelpers = {
-               UNSUPPORTED_Y: UNSUPPORTED_Y,
+               UNSUPPORTED_Y: UNSUPPORTED_Y$3,
                BROKEN_CARET: BROKEN_CARET
        };
 
+       var regexpUnsupportedDotAll = fails(function () {
+         // babel-minify transpiles RegExp('.', 's') -> /./s and it causes SyntaxError
+         var re = RegExp('.', (typeof '').charAt(0));
+         return !(re.dotAll && re.exec('\n') && re.flags === 's');
+       });
+
+       var regexpUnsupportedNcg = fails(function () {
+         // babel-minify transpiles RegExp('.', 'g') -> /./g and it causes SyntaxError
+         var re = RegExp('(?<a>b)', (typeof '').charAt(5));
+         return re.exec('b').groups.a !== 'b' ||
+           'b'.replace(re, '$<a>c') !== 'bc';
+       });
+
+       /* eslint-disable regexp/no-assertion-capturing-group, regexp/no-empty-group, regexp/no-lazy-ends -- testing */
+       /* eslint-disable regexp/no-useless-quantifier -- testing */
+
+
+
+
+       var getInternalState = internalState.get;
+
+
+
        var nativeExec = RegExp.prototype.exec;
-       // This always refers to the native implementation, because the
-       // String#replace polyfill uses ./fix-regexp-well-known-symbol-logic.js,
-       // which loads this file before patching the method.
-       var nativeReplace = String.prototype.replace;
+       var nativeReplace = shared('native-string-replace', String.prototype.replace);
 
        var patchedExec = nativeExec;
 
          return re1.lastIndex !== 0 || re2.lastIndex !== 0;
        })();
 
-       var UNSUPPORTED_Y$1 = regexpStickyHelpers.UNSUPPORTED_Y || regexpStickyHelpers.BROKEN_CARET;
+       var UNSUPPORTED_Y$2 = regexpStickyHelpers.UNSUPPORTED_Y || regexpStickyHelpers.BROKEN_CARET;
 
        // nonparticipating capturing group, copied from es5-shim's String#split patch.
-       // eslint-disable-next-line regexp/no-assertion-capturing-group, regexp/no-empty-group -- required for testing
        var NPCG_INCLUDED = /()??/.exec('')[1] !== undefined;
 
-       var PATCH = UPDATES_LAST_INDEX_WRONG || NPCG_INCLUDED || UNSUPPORTED_Y$1;
+       var PATCH = UPDATES_LAST_INDEX_WRONG || NPCG_INCLUDED || UNSUPPORTED_Y$2 || regexpUnsupportedDotAll || regexpUnsupportedNcg;
 
        if (PATCH) {
+         // eslint-disable-next-line max-statements -- TODO
          patchedExec = function exec(str) {
            var re = this;
-           var lastIndex, reCopy, match, i;
-           var sticky = UNSUPPORTED_Y$1 && re.sticky;
+           var state = getInternalState(re);
+           var raw = state.raw;
+           var result, reCopy, lastIndex, match, i, object, group;
+
+           if (raw) {
+             raw.lastIndex = re.lastIndex;
+             result = patchedExec.call(raw, str);
+             re.lastIndex = raw.lastIndex;
+             return result;
+           }
+
+           var groups = state.groups;
+           var sticky = UNSUPPORTED_Y$2 && re.sticky;
            var flags = regexpFlags.call(re);
            var source = re.source;
            var charsAdded = 0;
              });
            }
 
+           if (match && groups) {
+             match.groups = object = objectCreate(null);
+             for (i = 0; i < groups.length; i++) {
+               group = groups[i];
+               object[group[0]] = match[group[1]];
+             }
+           }
+
            return match;
          };
        }
 
 
 
-       var SPECIES$6 = wellKnownSymbol('species');
-
-       var REPLACE_SUPPORTS_NAMED_GROUPS = !fails(function () {
-         // #replace needs built-in support for named groups.
-         // #match works fine because it just return the exec results, even if it has
-         // a "grops" property.
-         var re = /./;
-         re.exec = function () {
-           var result = [];
-           result.groups = { a: '7' };
-           return result;
-         };
-         return ''.replace(re, '$<a>') !== '7';
-       });
-
-       // IE <= 11 replaces $0 with the whole match, as if it was $&
-       // https://stackoverflow.com/questions/6024666/getting-ie-to-replace-a-regex-with-the-literal-string-0
-       var REPLACE_KEEPS_$0 = (function () {
-         return 'a'.replace(/./, '$0') === '$0';
-       })();
-
-       var REPLACE = wellKnownSymbol('replace');
-       // Safari <= 13.0.3(?) substitutes nth capture where n>m with an empty string
-       var REGEXP_REPLACE_SUBSTITUTES_UNDEFINED_CAPTURE = (function () {
-         if (/./[REPLACE]) {
-           return /./[REPLACE]('a', '$0') === '';
-         }
-         return false;
-       })();
-
-       // Chrome 51 has a buggy "split" implementation when RegExp#exec !== nativeExec
-       // Weex JS has frozen built-in prototypes, so use try / catch wrapper
-       var SPLIT_WORKS_WITH_OVERWRITTEN_EXEC = !fails(function () {
-         // eslint-disable-next-line regexp/no-empty-group -- required for testing
-         var re = /(?:)/;
-         var originalExec = re.exec;
-         re.exec = function () { return originalExec.apply(this, arguments); };
-         var result = 'ab'.split(re);
-         return result.length !== 2 || result[0] !== 'a' || result[1] !== 'b';
-       });
+       var SPECIES = wellKnownSymbol('species');
+       var RegExpPrototype$1 = RegExp.prototype;
 
-       var fixRegexpWellKnownSymbolLogic = function (KEY, length, exec, sham) {
+       var fixRegexpWellKnownSymbolLogic = function (KEY, exec, FORCED, SHAM) {
          var SYMBOL = wellKnownSymbol(KEY);
 
          var DELEGATES_TO_SYMBOL = !fails(function () {
              // RegExp[@@split] doesn't call the regex's exec method, but first creates
              // a new one. We need to return the patched regex when creating the new one.
              re.constructor = {};
-             re.constructor[SPECIES$6] = function () { return re; };
+             re.constructor[SPECIES] = function () { return re; };
              re.flags = '';
              re[SYMBOL] = /./[SYMBOL];
            }
          if (
            !DELEGATES_TO_SYMBOL ||
            !DELEGATES_TO_EXEC ||
-           (KEY === 'replace' && !(
-             REPLACE_SUPPORTS_NAMED_GROUPS &&
-             REPLACE_KEEPS_$0 &&
-             !REGEXP_REPLACE_SUBSTITUTES_UNDEFINED_CAPTURE
-           )) ||
-           (KEY === 'split' && !SPLIT_WORKS_WITH_OVERWRITTEN_EXEC)
+           FORCED
          ) {
            var nativeRegExpMethod = /./[SYMBOL];
            var methods = exec(SYMBOL, ''[KEY], function (nativeMethod, regexp, str, arg2, forceStringMethod) {
-             if (regexp.exec === regexpExec) {
+             var $exec = regexp.exec;
+             if ($exec === regexpExec || $exec === RegExpPrototype$1.exec) {
                if (DELEGATES_TO_SYMBOL && !forceStringMethod) {
                  // The native String method already delegates to @@method (this
                  // polyfilled function), leasing to infinite recursion.
                return { done: true, value: nativeMethod.call(str, regexp, arg2) };
              }
              return { done: false };
-           }, {
-             REPLACE_KEEPS_$0: REPLACE_KEEPS_$0,
-             REGEXP_REPLACE_SUBSTITUTES_UNDEFINED_CAPTURE: REGEXP_REPLACE_SUBSTITUTES_UNDEFINED_CAPTURE
-           });
-           var stringMethod = methods[0];
-           var regexMethod = methods[1];
-
-           redefine(String.prototype, KEY, stringMethod);
-           redefine(RegExp.prototype, SYMBOL, length == 2
-             // 21.2.5.8 RegExp.prototype[@@replace](string, replaceValue)
-             // 21.2.5.11 RegExp.prototype[@@split](string, limit)
-             ? function (string, arg) { return regexMethod.call(string, this, arg); }
-             // 21.2.5.6 RegExp.prototype[@@match](string)
-             // 21.2.5.9 RegExp.prototype[@@search](string)
-             : function (string) { return regexMethod.call(string, this); }
-           );
+           });
+
+           redefine(String.prototype, KEY, methods[0]);
+           redefine(RegExpPrototype$1, SYMBOL, methods[1]);
          }
 
-         if (sham) createNonEnumerableProperty(RegExp.prototype[SYMBOL], 'sham', true);
+         if (SHAM) createNonEnumerableProperty(RegExpPrototype$1[SYMBOL], 'sham', true);
        };
 
-       var charAt$1 = stringMultibyte.charAt;
+       var charAt = stringMultibyte.charAt;
 
        // `AdvanceStringIndex` abstract operation
        // https://tc39.es/ecma262/#sec-advancestringindex
        var advanceStringIndex = function (S, index, unicode) {
-         return index + (unicode ? charAt$1(S, index).length : 1);
+         return index + (unicode ? charAt(S, index).length : 1);
        };
 
-       var floor$5 = Math.floor;
-       var replace$1 = ''.replace;
+       var floor$1 = Math.floor;
+       var replace = ''.replace;
        var SUBSTITUTION_SYMBOLS = /\$([$&'`]|\d{1,2}|<[^>]*>)/g;
        var SUBSTITUTION_SYMBOLS_NO_NAMED = /\$([$&'`]|\d{1,2})/g;
 
+       // `GetSubstitution` abstract operation
        // https://tc39.es/ecma262/#sec-getsubstitution
        var getSubstitution = function (matched, str, position, captures, namedCaptures, replacement) {
          var tailPos = position + matched.length;
            namedCaptures = toObject(namedCaptures);
            symbols = SUBSTITUTION_SYMBOLS;
          }
-         return replace$1.call(replacement, symbols, function (match, ch) {
+         return replace.call(replacement, symbols, function (match, ch) {
            var capture;
            switch (ch.charAt(0)) {
              case '$': return '$';
                var n = +ch;
                if (n === 0) return match;
                if (n > m) {
-                 var f = floor$5(n / 10);
+                 var f = floor$1(n / 10);
                  if (f === 0) return match;
                  if (f <= m) return captures[f - 1] === undefined ? ch.charAt(1) : captures[f - 1] + ch.charAt(1);
                  return match;
          return regexpExec.call(R, S);
        };
 
+       var REPLACE = wellKnownSymbol('replace');
        var max$2 = Math.max;
-       var min$4 = Math.min;
+       var min$5 = Math.min;
 
        var maybeToString = function (it) {
          return it === undefined ? it : String(it);
        };
 
+       // IE <= 11 replaces $0 with the whole match, as if it was $&
+       // https://stackoverflow.com/questions/6024666/getting-ie-to-replace-a-regex-with-the-literal-string-0
+       var REPLACE_KEEPS_$0 = (function () {
+         // eslint-disable-next-line regexp/prefer-escape-replacement-dollar-char -- required for testing
+         return 'a'.replace(/./, '$0') === '$0';
+       })();
+
+       // Safari <= 13.0.3(?) substitutes nth capture where n>m with an empty string
+       var REGEXP_REPLACE_SUBSTITUTES_UNDEFINED_CAPTURE = (function () {
+         if (/./[REPLACE]) {
+           return /./[REPLACE]('a', '$0') === '';
+         }
+         return false;
+       })();
+
+       var REPLACE_SUPPORTS_NAMED_GROUPS = !fails(function () {
+         var re = /./;
+         re.exec = function () {
+           var result = [];
+           result.groups = { a: '7' };
+           return result;
+         };
+         return ''.replace(re, '$<a>') !== '7';
+       });
+
        // @@replace logic
-       fixRegexpWellKnownSymbolLogic('replace', 2, function (REPLACE, nativeReplace, maybeCallNative, reason) {
-         var REGEXP_REPLACE_SUBSTITUTES_UNDEFINED_CAPTURE = reason.REGEXP_REPLACE_SUBSTITUTES_UNDEFINED_CAPTURE;
-         var REPLACE_KEEPS_$0 = reason.REPLACE_KEEPS_$0;
+       fixRegexpWellKnownSymbolLogic('replace', function (_, nativeReplace, maybeCallNative) {
          var UNSAFE_SUBSTITUTE = REGEXP_REPLACE_SUBSTITUTES_UNDEFINED_CAPTURE ? '$' : '$0';
 
          return [
            },
            // `RegExp.prototype[@@replace]` method
            // https://tc39.es/ecma262/#sec-regexp.prototype-@@replace
-           function (regexp, replaceValue) {
+           function (string, replaceValue) {
              if (
-               (!REGEXP_REPLACE_SUBSTITUTES_UNDEFINED_CAPTURE && REPLACE_KEEPS_$0) ||
-               (typeof replaceValue === 'string' && replaceValue.indexOf(UNSAFE_SUBSTITUTE) === -1)
+               typeof replaceValue === 'string' &&
+               replaceValue.indexOf(UNSAFE_SUBSTITUTE) === -1 &&
+               replaceValue.indexOf('$<') === -1
              ) {
-               var res = maybeCallNative(nativeReplace, regexp, this, replaceValue);
+               var res = maybeCallNative(nativeReplace, this, string, replaceValue);
                if (res.done) return res.value;
              }
 
-             var rx = anObject(regexp);
-             var S = String(this);
+             var rx = anObject(this);
+             var S = String(string);
 
              var functionalReplace = typeof replaceValue === 'function';
              if (!functionalReplace) replaceValue = String(replaceValue);
                result = results[i];
 
                var matched = String(result[0]);
-               var position = max$2(min$4(toInteger(result.index), S.length), 0);
+               var position = max$2(min$5(toInteger(result.index), S.length), 0);
                var captures = [];
                // NOTE: This is equivalent to
                //   captures = result.slice(1).map(maybeToString)
              return accumulatedResult + S.slice(nextSourcePosition);
            }
          ];
-       });
+       }, !REPLACE_SUPPORTS_NAMED_GROUPS || !REPLACE_KEEPS_$0 || REGEXP_REPLACE_SUBSTITUTES_UNDEFINED_CAPTURE);
 
-       var MATCH = wellKnownSymbol('match');
+       var MATCH$2 = wellKnownSymbol('match');
 
        // `IsRegExp` abstract operation
        // https://tc39.es/ecma262/#sec-isregexp
        var isRegexp = function (it) {
          var isRegExp;
-         return isObject(it) && ((isRegExp = it[MATCH]) !== undefined ? !!isRegExp : classofRaw(it) == 'RegExp');
+         return isObject$4(it) && ((isRegExp = it[MATCH$2]) !== undefined ? !!isRegExp : classofRaw(it) == 'RegExp');
        };
 
+       var UNSUPPORTED_Y$1 = regexpStickyHelpers.UNSUPPORTED_Y;
        var arrayPush = [].push;
-       var min$5 = Math.min;
+       var min$4 = Math.min;
        var MAX_UINT32 = 0xFFFFFFFF;
 
-       // babel-minify transpiles RegExp('x', 'y') -> /x/y and it causes SyntaxError
-       var SUPPORTS_Y = !fails(function () { return !RegExp(MAX_UINT32, 'y'); });
+       // Chrome 51 has a buggy "split" implementation when RegExp#exec !== nativeExec
+       // Weex JS has frozen built-in prototypes, so use try / catch wrapper
+       var SPLIT_WORKS_WITH_OVERWRITTEN_EXEC = !fails(function () {
+         // eslint-disable-next-line regexp/no-empty-group -- required for testing
+         var re = /(?:)/;
+         var originalExec = re.exec;
+         re.exec = function () { return originalExec.apply(this, arguments); };
+         var result = 'ab'.split(re);
+         return result.length !== 2 || result[0] !== 'a' || result[1] !== 'b';
+       });
 
        // @@split logic
-       fixRegexpWellKnownSymbolLogic('split', 2, function (SPLIT, nativeSplit, maybeCallNative) {
+       fixRegexpWellKnownSymbolLogic('split', function (SPLIT, nativeSplit, maybeCallNative) {
          var internalSplit;
          if (
            'abbc'.split(/(b)*/)[1] == 'c' ||
            //
            // NOTE: This cannot be properly polyfilled in engines that don't support
            // the 'y' flag.
-           function (regexp, limit) {
-             var res = maybeCallNative(internalSplit, regexp, this, limit, internalSplit !== nativeSplit);
+           function (string, limit) {
+             var res = maybeCallNative(internalSplit, this, string, limit, internalSplit !== nativeSplit);
              if (res.done) return res.value;
 
-             var rx = anObject(regexp);
-             var S = String(this);
+             var rx = anObject(this);
+             var S = String(string);
              var C = speciesConstructor(rx, RegExp);
 
              var unicodeMatching = rx.unicode;
              var flags = (rx.ignoreCase ? 'i' : '') +
                          (rx.multiline ? 'm' : '') +
                          (rx.unicode ? 'u' : '') +
-                         (SUPPORTS_Y ? 'y' : 'g');
+                         (UNSUPPORTED_Y$1 ? 'g' : 'y');
 
              // ^(? + rx + ) is needed, in combination with some S slicing, to
              // simulate the 'y' flag.
-             var splitter = new C(SUPPORTS_Y ? rx : '^(?:' + rx.source + ')', flags);
+             var splitter = new C(UNSUPPORTED_Y$1 ? '^(?:' + rx.source + ')' : rx, flags);
              var lim = limit === undefined ? MAX_UINT32 : limit >>> 0;
              if (lim === 0) return [];
              if (S.length === 0) return regexpExecAbstract(splitter, S) === null ? [S] : [];
              var q = 0;
              var A = [];
              while (q < S.length) {
-               splitter.lastIndex = SUPPORTS_Y ? q : 0;
-               var z = regexpExecAbstract(splitter, SUPPORTS_Y ? S : S.slice(q));
+               splitter.lastIndex = UNSUPPORTED_Y$1 ? 0 : q;
+               var z = regexpExecAbstract(splitter, UNSUPPORTED_Y$1 ? S.slice(q) : S);
                var e;
                if (
                  z === null ||
-                 (e = min$5(toLength(splitter.lastIndex + (SUPPORTS_Y ? 0 : q)), S.length)) === p
+                 (e = min$4(toLength(splitter.lastIndex + (UNSUPPORTED_Y$1 ? q : 0)), S.length)) === p
                ) {
                  q = advanceStringIndex(S, q, unicodeMatching);
                } else {
              return A;
            }
          ];
-       }, !SUPPORTS_Y);
+       }, !SPLIT_WORKS_WITH_OVERWRITTEN_EXEC, UNSUPPORTED_Y$1);
 
        // a string of all valid unicode whitespaces
        var whitespaces = '\u0009\u000A\u000B\u000C\u000D\u0020\u00A0\u1680\u2000\u2001\u2002' +
 
        var whitespace = '[' + whitespaces + ']';
        var ltrim = RegExp('^' + whitespace + whitespace + '*');
-       var rtrim = RegExp(whitespace + whitespace + '*$');
+       var rtrim$2 = RegExp(whitespace + whitespace + '*$');
 
        // `String.prototype.{ trim, trimStart, trimEnd, trimLeft, trimRight }` methods implementation
-       var createMethod$4 = function (TYPE) {
+       var createMethod$2 = function (TYPE) {
          return function ($this) {
            var string = String(requireObjectCoercible($this));
            if (TYPE & 1) string = string.replace(ltrim, '');
-           if (TYPE & 2) string = string.replace(rtrim, '');
+           if (TYPE & 2) string = string.replace(rtrim$2, '');
            return string;
          };
        };
        var stringTrim = {
          // `String.prototype.{ trimLeft, trimStart }` methods
          // https://tc39.es/ecma262/#sec-string.prototype.trimstart
-         start: createMethod$4(1),
+         start: createMethod$2(1),
          // `String.prototype.{ trimRight, trimEnd }` methods
          // https://tc39.es/ecma262/#sec-string.prototype.trimend
-         end: createMethod$4(2),
+         end: createMethod$2(2),
          // `String.prototype.trim` method
          // https://tc39.es/ecma262/#sec-string.prototype.trim
-         trim: createMethod$4(3)
+         trim: createMethod$2(3)
        };
 
        var non = '\u200B\u0085\u180E';
          }
        });
 
-       var defineProperty$6 = objectDefineProperty.f;
+       var defineProperty$3 = objectDefineProperty.f;
 
        var FunctionPrototype = Function.prototype;
        var FunctionPrototypeToString = FunctionPrototype.toString;
        var nameRE = /^\s*function ([^ (]*)/;
-       var NAME$1 = 'name';
+       var NAME = 'name';
 
        // Function instances `.name` property
        // https://tc39.es/ecma262/#sec-function-instances-name
-       if (descriptors && !(NAME$1 in FunctionPrototype)) {
-         defineProperty$6(FunctionPrototype, NAME$1, {
+       if (descriptors && !(NAME in FunctionPrototype)) {
+         defineProperty$3(FunctionPrototype, NAME, {
            configurable: true,
            get: function () {
              try {
          create: objectCreate
        });
 
-       var slice = [].slice;
+       var slice$3 = [].slice;
        var MSIE = /MSIE .\./.test(engineUserAgent); // <- dirty ie9- check
 
        var wrap$1 = function (scheduler) {
          return function (handler, timeout /* , ...arguments */) {
            var boundArgs = arguments.length > 2;
-           var args = boundArgs ? slice.call(arguments, 2) : undefined;
+           var args = boundArgs ? slice$3.call(arguments, 2) : undefined;
            return scheduler(boundArgs ? function () {
              // eslint-disable-next-line no-new-func -- spec requirement
              (typeof handler == 'function' ? handler : Function(handler)).apply(this, args);
        _export({ global: true, bind: true, forced: MSIE }, {
          // `setTimeout` method
          // https://html.spec.whatwg.org/multipage/timers-and-user-prompts.html#dom-settimeout
-         setTimeout: wrap$1(global_1.setTimeout),
+         setTimeout: wrap$1(global$2.setTimeout),
          // `setInterval` method
          // https://html.spec.whatwg.org/multipage/timers-and-user-prompts.html#dom-setinterval
-         setInterval: wrap$1(global_1.setInterval)
+         setInterval: wrap$1(global$2.setInterval)
        });
 
        var global$1 = typeof globalThis !== 'undefined' && globalThis || typeof self !== 'undefined' && self || typeof global$1 !== 'undefined' && global$1;
          return iterator;
        }
 
-       function Headers$1(headers) {
+       function Headers(headers) {
          this.map = {};
 
-         if (headers instanceof Headers$1) {
+         if (headers instanceof Headers) {
            headers.forEach(function (value, name) {
              this.append(name, value);
            }, this);
          }
        }
 
-       Headers$1.prototype.append = function (name, value) {
+       Headers.prototype.append = function (name, value) {
          name = normalizeName(name);
          value = normalizeValue(value);
          var oldValue = this.map[name];
          this.map[name] = oldValue ? oldValue + ', ' + value : value;
        };
 
-       Headers$1.prototype['delete'] = function (name) {
+       Headers.prototype['delete'] = function (name) {
          delete this.map[normalizeName(name)];
        };
 
-       Headers$1.prototype.get = function (name) {
+       Headers.prototype.get = function (name) {
          name = normalizeName(name);
          return this.has(name) ? this.map[name] : null;
        };
 
-       Headers$1.prototype.has = function (name) {
+       Headers.prototype.has = function (name) {
          return this.map.hasOwnProperty(normalizeName(name));
        };
 
-       Headers$1.prototype.set = function (name, value) {
+       Headers.prototype.set = function (name, value) {
          this.map[normalizeName(name)] = normalizeValue(value);
        };
 
-       Headers$1.prototype.forEach = function (callback, thisArg) {
+       Headers.prototype.forEach = function (callback, thisArg) {
          for (var name in this.map) {
            if (this.map.hasOwnProperty(name)) {
              callback.call(thisArg, this.map[name], name, this);
          }
        };
 
-       Headers$1.prototype.keys = function () {
+       Headers.prototype.keys = function () {
          var items = [];
          this.forEach(function (value, name) {
            items.push(name);
          return iteratorFor(items);
        };
 
-       Headers$1.prototype.values = function () {
+       Headers.prototype.values = function () {
          var items = [];
          this.forEach(function (value) {
            items.push(value);
          return iteratorFor(items);
        };
 
-       Headers$1.prototype.entries = function () {
+       Headers.prototype.entries = function () {
          var items = [];
          this.forEach(function (value, name) {
            items.push([name, value]);
        };
 
        if (support.iterable) {
-         Headers$1.prototype[Symbol.iterator] = Headers$1.prototype.entries;
+         Headers.prototype[Symbol.iterator] = Headers.prototype.entries;
        }
 
        function consumed(body) {
            this.credentials = input.credentials;
 
            if (!options.headers) {
-             this.headers = new Headers$1(input.headers);
+             this.headers = new Headers(input.headers);
            }
 
            this.method = input.method;
          this.credentials = options.credentials || this.credentials || 'same-origin';
 
          if (options.headers || !this.headers) {
-           this.headers = new Headers$1(options.headers);
+           this.headers = new Headers(options.headers);
          }
 
          this.method = normalizeMethod(options.method || this.method || 'GET');
        }
 
        function parseHeaders(rawHeaders) {
-         var headers = new Headers$1(); // Replace instances of \r\n and \n followed by at least one space or horizontal tab with a space
+         var headers = new Headers(); // Replace instances of \r\n and \n followed by at least one space or horizontal tab with a space
          // https://tools.ietf.org/html/rfc7230#section-3.2
 
          var preProcessedHeaders = rawHeaders.replace(/\r?\n[\t ]+/g, ' '); // Avoiding split via regex to work around a common IE11 bug with the core-js 3.6.0 regex polyfill
          this.status = options.status === undefined ? 200 : options.status;
          this.ok = this.status >= 200 && this.status < 300;
          this.statusText = options.statusText === undefined ? '' : '' + options.statusText;
-         this.headers = new Headers$1(options.headers);
+         this.headers = new Headers(options.headers);
          this.url = options.url || '';
 
          this._initBody(bodyInit);
          return new Response(this._bodyInit, {
            status: this.status,
            statusText: this.statusText,
-           headers: new Headers$1(this.headers),
+           headers: new Headers(this.headers),
            url: this.url
          });
        };
          });
        };
 
-       var DOMException$1 = global$1.DOMException;
+       var DOMException$2 = global$1.DOMException;
 
        try {
-         new DOMException$1();
+         new DOMException$2();
        } catch (err) {
-         DOMException$1 = function DOMException(message, name) {
+         DOMException$2 = function DOMException(message, name) {
            this.message = message;
            this.name = name;
            var error = Error(message);
            this.stack = error.stack;
          };
 
-         DOMException$1.prototype = Object.create(Error.prototype);
-         DOMException$1.prototype.constructor = DOMException$1;
+         DOMException$2.prototype = Object.create(Error.prototype);
+         DOMException$2.prototype.constructor = DOMException$2;
        }
 
        function fetch$1(input, init) {
            var request = new Request(input, init);
 
            if (request.signal && request.signal.aborted) {
-             return reject(new DOMException$1('Aborted', 'AbortError'));
+             return reject(new DOMException$2('Aborted', 'AbortError'));
            }
 
            var xhr = new XMLHttpRequest();
 
            xhr.onabort = function () {
              setTimeout(function () {
-               reject(new DOMException$1('Aborted', 'AbortError'));
+               reject(new DOMException$2('Aborted', 'AbortError'));
              }, 0);
            };
 
              }
            }
 
-           if (init && _typeof(init.headers) === 'object' && !(init.headers instanceof Headers$1)) {
+           if (init && _typeof(init.headers) === 'object' && !(init.headers instanceof Headers)) {
              Object.getOwnPropertyNames(init.headers).forEach(function (name) {
                xhr.setRequestHeader(name, normalizeValue(init.headers[name]));
              });
 
        if (!global$1.fetch) {
          global$1.fetch = fetch$1;
-         global$1.Headers = Headers$1;
+         global$1.Headers = Headers;
          global$1.Request = Request;
          global$1.Response = Response;
        }
          setPrototypeOf: objectSetPrototypeOf
        });
 
-       var FAILS_ON_PRIMITIVES$1 = fails(function () { objectGetPrototypeOf(1); });
+       var FAILS_ON_PRIMITIVES$3 = fails(function () { objectGetPrototypeOf(1); });
 
        // `Object.getPrototypeOf` method
        // https://tc39.es/ecma262/#sec-object.getprototypeof
-       _export({ target: 'Object', stat: true, forced: FAILS_ON_PRIMITIVES$1, sham: !correctPrototypeGetter }, {
+       _export({ target: 'Object', stat: true, forced: FAILS_ON_PRIMITIVES$3, sham: !correctPrototypeGetter }, {
          getPrototypeOf: function getPrototypeOf(it) {
            return objectGetPrototypeOf(toObject(it));
          }
        });
 
-       var slice$1 = [].slice;
+       var slice$2 = [].slice;
        var factories = {};
 
        var construct = function (C, argsLength, args) {
        // `Function.prototype.bind` method implementation
        // https://tc39.es/ecma262/#sec-function.prototype.bind
        var functionBind = Function.bind || function bind(that /* , ...args */) {
-         var fn = aFunction$1(this);
-         var partArgs = slice$1.call(arguments, 1);
+         var fn = aFunction(this);
+         var partArgs = slice$2.call(arguments, 1);
          var boundFunction = function bound(/* args... */) {
-           var args = partArgs.concat(slice$1.call(arguments));
+           var args = partArgs.concat(slice$2.call(arguments));
            return this instanceof boundFunction ? construct(fn, args.length, args) : fn.apply(that, args);
          };
-         if (isObject(fn.prototype)) boundFunction.prototype = fn.prototype;
+         if (isObject$4(fn.prototype)) boundFunction.prototype = fn.prototype;
          return boundFunction;
        };
 
        var ARGS_BUG = !fails(function () {
          nativeConstruct(function () { /* empty */ });
        });
-       var FORCED$5 = NEW_TARGET_BUG || ARGS_BUG;
+       var FORCED$a = NEW_TARGET_BUG || ARGS_BUG;
 
-       _export({ target: 'Reflect', stat: true, forced: FORCED$5, sham: FORCED$5 }, {
+       _export({ target: 'Reflect', stat: true, forced: FORCED$a, sham: FORCED$a }, {
          construct: function construct(Target, args /* , newTarget */) {
-           aFunction$1(Target);
+           aFunction(Target);
            anObject(args);
-           var newTarget = arguments.length < 3 ? Target : aFunction$1(arguments[2]);
+           var newTarget = arguments.length < 3 ? Target : aFunction(arguments[2]);
            if (ARGS_BUG && !NEW_TARGET_BUG) return nativeConstruct(Target, args, newTarget);
            if (Target == newTarget) {
              // w/o altered newTarget, optimization for 0-4 arguments
            }
            // with altered newTarget, not support built-in constructors
            var proto = newTarget.prototype;
-           var instance = objectCreate(isObject(proto) ? proto : Object.prototype);
+           var instance = objectCreate(isObject$4(proto) ? proto : Object.prototype);
            var result = Function.apply.call(Target, instance, args);
-           return isObject(result) ? result : instance;
+           return isObject$4(result) ? result : instance;
          }
        });
 
        // `Reflect.get` method
        // https://tc39.es/ecma262/#sec-reflect.get
-       function get$2(target, propertyKey /* , receiver */) {
+       function get$3(target, propertyKey /* , receiver */) {
          var receiver = arguments.length < 3 ? target : arguments[2];
          var descriptor, prototype;
          if (anObject(target) === receiver) return target[propertyKey];
-         if (descriptor = objectGetOwnPropertyDescriptor.f(target, propertyKey)) return has(descriptor, 'value')
+         if (descriptor = objectGetOwnPropertyDescriptor.f(target, propertyKey)) return has$1(descriptor, 'value')
            ? descriptor.value
            : descriptor.get === undefined
              ? undefined
              : descriptor.get.call(receiver);
-         if (isObject(prototype = objectGetPrototypeOf(target))) return get$2(prototype, propertyKey, receiver);
+         if (isObject$4(prototype = objectGetPrototypeOf(target))) return get$3(prototype, propertyKey, receiver);
        }
 
        _export({ target: 'Reflect', stat: true }, {
-         get: get$2
+         get: get$3
        });
 
-       var nativeGetOwnPropertyDescriptor$2 = objectGetOwnPropertyDescriptor.f;
+       var nativeGetOwnPropertyDescriptor = objectGetOwnPropertyDescriptor.f;
 
 
-       var FAILS_ON_PRIMITIVES$2 = fails(function () { nativeGetOwnPropertyDescriptor$2(1); });
-       var FORCED$6 = !descriptors || FAILS_ON_PRIMITIVES$2;
+       var FAILS_ON_PRIMITIVES$2 = fails(function () { nativeGetOwnPropertyDescriptor(1); });
+       var FORCED$9 = !descriptors || FAILS_ON_PRIMITIVES$2;
 
        // `Object.getOwnPropertyDescriptor` method
        // https://tc39.es/ecma262/#sec-object.getownpropertydescriptor
-       _export({ target: 'Object', stat: true, forced: FORCED$6, sham: !descriptors }, {
+       _export({ target: 'Object', stat: true, forced: FORCED$9, sham: !descriptors }, {
          getOwnPropertyDescriptor: function getOwnPropertyDescriptor(it, key) {
-           return nativeGetOwnPropertyDescriptor$2(toIndexedObject(it), key);
+           return nativeGetOwnPropertyDescriptor(toIndexedObject(it), key);
          }
        });
 
-       var HAS_SPECIES_SUPPORT$2 = arrayMethodHasSpeciesSupport('splice');
+       var HAS_SPECIES_SUPPORT$1 = arrayMethodHasSpeciesSupport('splice');
 
-       var max$3 = Math.max;
-       var min$6 = Math.min;
-       var MAX_SAFE_INTEGER = 0x1FFFFFFFFFFFFF;
+       var max$1 = Math.max;
+       var min$3 = Math.min;
+       var MAX_SAFE_INTEGER$1 = 0x1FFFFFFFFFFFFF;
        var MAXIMUM_ALLOWED_LENGTH_EXCEEDED = 'Maximum allowed length exceeded';
 
        // `Array.prototype.splice` method
        // https://tc39.es/ecma262/#sec-array.prototype.splice
        // with adding support of @@species
-       _export({ target: 'Array', proto: true, forced: !HAS_SPECIES_SUPPORT$2 }, {
+       _export({ target: 'Array', proto: true, forced: !HAS_SPECIES_SUPPORT$1 }, {
          splice: function splice(start, deleteCount /* , ...items */) {
            var O = toObject(this);
            var len = toLength(O.length);
              actualDeleteCount = len - actualStart;
            } else {
              insertCount = argumentsLength - 2;
-             actualDeleteCount = min$6(max$3(toInteger(deleteCount), 0), len - actualStart);
+             actualDeleteCount = min$3(max$1(toInteger(deleteCount), 0), len - actualStart);
            }
-           if (len + insertCount - actualDeleteCount > MAX_SAFE_INTEGER) {
+           if (len + insertCount - actualDeleteCount > MAX_SAFE_INTEGER$1) {
              throw TypeError(MAXIMUM_ALLOWED_LENGTH_EXCEEDED);
            }
            A = arraySpeciesCreate(O, actualDeleteCount);
        // https://tc39.es/ecma262/#sec-symbol.tostringtag
        defineWellKnownSymbol('toStringTag');
 
+       // JSON[@@toStringTag] property
+       // https://tc39.es/ecma262/#sec-json-@@tostringtag
+       setToStringTag(global$2.JSON, 'JSON', true);
+
        // Math[@@toStringTag] property
        // https://tc39.es/ecma262/#sec-math-@@tostringtag
        setToStringTag(Math, 'Math', true);
 
-       // JSON[@@toStringTag] property
-       // https://tc39.es/ecma262/#sec-json-@@tostringtag
-       setToStringTag(global_1.JSON, 'JSON', true);
-
        (function (factory) {
-          factory();
+         factory();
        })(function () {
 
          function _classCallCheck(instance, Constructor) {
            if (typeof Proxy === "function") return true;
 
            try {
-             Date.prototype.toString.call(Reflect.construct(Date, [], function () {}));
+             Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {}));
              return true;
            } catch (e) {
              return false;
        }
 
        var IS_CONCAT_SPREADABLE = wellKnownSymbol('isConcatSpreadable');
-       var MAX_SAFE_INTEGER$1 = 0x1FFFFFFFFFFFFF;
+       var MAX_SAFE_INTEGER = 0x1FFFFFFFFFFFFF;
        var MAXIMUM_ALLOWED_INDEX_EXCEEDED = 'Maximum allowed index exceeded';
 
        // We can't use this feature detection in V8 since it causes
        var SPECIES_SUPPORT = arrayMethodHasSpeciesSupport('concat');
 
        var isConcatSpreadable = function (O) {
-         if (!isObject(O)) return false;
+         if (!isObject$4(O)) return false;
          var spreadable = O[IS_CONCAT_SPREADABLE];
          return spreadable !== undefined ? !!spreadable : isArray(O);
        };
 
-       var FORCED$7 = !IS_CONCAT_SPREADABLE_SUPPORT || !SPECIES_SUPPORT;
+       var FORCED$8 = !IS_CONCAT_SPREADABLE_SUPPORT || !SPECIES_SUPPORT;
 
        // `Array.prototype.concat` method
        // https://tc39.es/ecma262/#sec-array.prototype.concat
        // with adding support of @@isConcatSpreadable and @@species
-       _export({ target: 'Array', proto: true, forced: FORCED$7 }, {
+       _export({ target: 'Array', proto: true, forced: FORCED$8 }, {
          // eslint-disable-next-line no-unused-vars -- required for `.length`
          concat: function concat(arg) {
            var O = toObject(this);
              E = i === -1 ? O : arguments[i];
              if (isConcatSpreadable(E)) {
                len = toLength(E.length);
-               if (n + len > MAX_SAFE_INTEGER$1) throw TypeError(MAXIMUM_ALLOWED_INDEX_EXCEEDED);
+               if (n + len > MAX_SAFE_INTEGER) throw TypeError(MAXIMUM_ALLOWED_INDEX_EXCEEDED);
                for (k = 0; k < len; k++, n++) if (k in E) createProperty(A, n, E[k]);
              } else {
-               if (n >= MAX_SAFE_INTEGER$1) throw TypeError(MAXIMUM_ALLOWED_INDEX_EXCEEDED);
+               if (n >= MAX_SAFE_INTEGER) throw TypeError(MAXIMUM_ALLOWED_INDEX_EXCEEDED);
                createProperty(A, n++, E);
              }
            }
 
        // `Object.assign` method
        // https://tc39.es/ecma262/#sec-object.assign
+       // eslint-disable-next-line es/no-object-assign -- required for testing
        _export({ target: 'Object', stat: true, forced: Object.assign !== objectAssign }, {
          assign: objectAssign
        });
 
-       var $filter$1 = arrayIteration.filter;
+       var $filter = arrayIteration.filter;
 
 
-       var HAS_SPECIES_SUPPORT$3 = arrayMethodHasSpeciesSupport('filter');
+       var HAS_SPECIES_SUPPORT = arrayMethodHasSpeciesSupport('filter');
 
        // `Array.prototype.filter` method
        // https://tc39.es/ecma262/#sec-array.prototype.filter
        // with adding support of @@species
-       _export({ target: 'Array', proto: true, forced: !HAS_SPECIES_SUPPORT$3 }, {
+       _export({ target: 'Array', proto: true, forced: !HAS_SPECIES_SUPPORT }, {
          filter: function filter(callbackfn /* , thisArg */) {
-           return $filter$1(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
+           return $filter(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
          }
        });
 
-       var FAILS_ON_PRIMITIVES$3 = fails(function () { objectKeys(1); });
+       var FAILS_ON_PRIMITIVES$1 = fails(function () { objectKeys(1); });
 
        // `Object.keys` method
        // https://tc39.es/ecma262/#sec-object.keys
-       _export({ target: 'Object', stat: true, forced: FAILS_ON_PRIMITIVES$3 }, {
+       _export({ target: 'Object', stat: true, forced: FAILS_ON_PRIMITIVES$1 }, {
          keys: function keys(it) {
            return objectKeys(toObject(it));
          }
          }
        });
 
-       var trim = stringTrim.trim;
+       var trim$4 = stringTrim.trim;
 
 
-       var $parseFloat = global_1.parseFloat;
-       var FORCED$8 = 1 / $parseFloat(whitespaces + '-0') !== -Infinity;
+       var $parseFloat = global$2.parseFloat;
+       var FORCED$7 = 1 / $parseFloat(whitespaces + '-0') !== -Infinity;
 
        // `parseFloat` method
        // https://tc39.es/ecma262/#sec-parsefloat-string
-       var numberParseFloat = FORCED$8 ? function parseFloat(string) {
-         var trimmedString = trim(String(string));
+       var numberParseFloat = FORCED$7 ? function parseFloat(string) {
+         var trimmedString = trim$4(String(string));
          var result = $parseFloat(trimmedString);
          return result === 0 && trimmedString.charAt(0) == '-' ? -0 : result;
        } : $parseFloat;
          tidal_channel: true
        };
 
-       var trim$1 = stringTrim.trim;
+       var trim$3 = stringTrim.trim;
 
 
-       var $parseInt = global_1.parseInt;
-       var hex$1 = /^[+-]?0[Xx]/;
-       var FORCED$9 = $parseInt(whitespaces + '08') !== 8 || $parseInt(whitespaces + '0x16') !== 22;
+       var $parseInt = global$2.parseInt;
+       var hex$2 = /^[+-]?0[Xx]/;
+       var FORCED$6 = $parseInt(whitespaces + '08') !== 8 || $parseInt(whitespaces + '0x16') !== 22;
 
        // `parseInt` method
        // https://tc39.es/ecma262/#sec-parseint-string-radix
-       var numberParseInt = FORCED$9 ? function parseInt(string, radix) {
-         var S = trim$1(String(string));
-         return $parseInt(S, (radix >>> 0) || (hex$1.test(S) ? 16 : 10));
+       var numberParseInt = FORCED$6 ? function parseInt(string, radix) {
+         var S = trim$3(String(string));
+         return $parseInt(S, (radix >>> 0) || (hex$2.test(S) ? 16 : 10));
        } : $parseInt;
 
        // `parseInt` method
        });
 
        var freezing = !fails(function () {
+         // eslint-disable-next-line es/no-object-isextensible, es/no-object-preventextensions -- required for testing
          return Object.isExtensible(Object.preventExtensions({}));
        });
 
        var METADATA = uid('meta');
        var id = 0;
 
+       // eslint-disable-next-line es/no-object-isextensible -- safe
        var isExtensible = Object.isExtensible || function () {
          return true;
        };
 
        var setMetadata = function (it) {
          defineProperty(it, METADATA, { value: {
-           objectID: 'O' + ++id, // object ID
+           objectID: 'O' + id++, // object ID
            weakData: {}          // weak collections IDs
          } });
        };
 
        var fastKey = function (it, create) {
          // return a primitive with prefix
-         if (!isObject(it)) return typeof it == 'symbol' ? it : (typeof it == 'string' ? 'S' : 'P') + it;
-         if (!has(it, METADATA)) {
+         if (!isObject$4(it)) return typeof it == 'symbol' ? it : (typeof it == 'string' ? 'S' : 'P') + it;
+         if (!has$1(it, METADATA)) {
            // can't set metadata to uncaught frozen object
            if (!isExtensible(it)) return 'F';
            // not necessary to add metadata
        };
 
        var getWeakData = function (it, create) {
-         if (!has(it, METADATA)) {
+         if (!has$1(it, METADATA)) {
            // can't set metadata to uncaught frozen object
            if (!isExtensible(it)) return true;
            // not necessary to add metadata
 
        // add metadata on freeze-family methods calling
        var onFreeze = function (it) {
-         if (freezing && meta.REQUIRED && isExtensible(it) && !has(it, METADATA)) setMetadata(it);
+         if (freezing && meta.REQUIRED && isExtensible(it) && !has$1(it, METADATA)) setMetadata(it);
          return it;
        };
 
          onFreeze: onFreeze
        };
 
-       hiddenKeys[METADATA] = true;
+       hiddenKeys$1[METADATA] = true;
        });
 
        var collection = function (CONSTRUCTOR_NAME, wrapper, common) {
          var IS_MAP = CONSTRUCTOR_NAME.indexOf('Map') !== -1;
          var IS_WEAK = CONSTRUCTOR_NAME.indexOf('Weak') !== -1;
          var ADDER = IS_MAP ? 'set' : 'add';
-         var NativeConstructor = global_1[CONSTRUCTOR_NAME];
+         var NativeConstructor = global$2[CONSTRUCTOR_NAME];
          var NativePrototype = NativeConstructor && NativeConstructor.prototype;
          var Constructor = NativeConstructor;
          var exported = {};
                nativeMethod.call(this, value === 0 ? 0 : value);
                return this;
              } : KEY == 'delete' ? function (key) {
-               return IS_WEAK && !isObject(key) ? false : nativeMethod.call(this, key === 0 ? 0 : key);
+               return IS_WEAK && !isObject$4(key) ? false : nativeMethod.call(this, key === 0 ? 0 : key);
              } : KEY == 'get' ? function get(key) {
-               return IS_WEAK && !isObject(key) ? undefined : nativeMethod.call(this, key === 0 ? 0 : key);
+               return IS_WEAK && !isObject$4(key) ? undefined : nativeMethod.call(this, key === 0 ? 0 : key);
              } : KEY == 'has' ? function has(key) {
-               return IS_WEAK && !isObject(key) ? false : nativeMethod.call(this, key === 0 ? 0 : key);
+               return IS_WEAK && !isObject$4(key) ? false : nativeMethod.call(this, key === 0 ? 0 : key);
              } : function set(key, value) {
                nativeMethod.call(this, key === 0 ? 0 : key, value);
                return this;
          return Constructor;
        };
 
-       var defineProperty$7 = objectDefineProperty.f;
+       var defineProperty$2 = objectDefineProperty.f;
 
 
 
        var fastKey = internalMetadata.fastKey;
 
 
-       var setInternalState$7 = internalState.set;
+       var setInternalState = internalState.set;
        var internalStateGetterFor = internalState.getterFor;
 
        var collectionStrong = {
          getConstructor: function (wrapper, CONSTRUCTOR_NAME, IS_MAP, ADDER) {
            var C = wrapper(function (that, iterable) {
              anInstance(that, C, CONSTRUCTOR_NAME);
-             setInternalState$7(that, {
+             setInternalState(that, {
                type: CONSTRUCTOR_NAME,
                index: objectCreate(null),
                first: undefined,
            };
 
            redefineAll(C.prototype, {
-             // 23.1.3.1 Map.prototype.clear()
-             // 23.2.3.2 Set.prototype.clear()
+             // `{ Map, Set }.prototype.clear()` methods
+             // https://tc39.es/ecma262/#sec-map.prototype.clear
+             // https://tc39.es/ecma262/#sec-set.prototype.clear
              clear: function clear() {
                var that = this;
                var state = getInternalState(that);
                if (descriptors) state.size = 0;
                else that.size = 0;
              },
-             // 23.1.3.3 Map.prototype.delete(key)
-             // 23.2.3.4 Set.prototype.delete(value)
+             // `{ Map, Set }.prototype.delete(key)` methods
+             // https://tc39.es/ecma262/#sec-map.prototype.delete
+             // https://tc39.es/ecma262/#sec-set.prototype.delete
              'delete': function (key) {
                var that = this;
                var state = getInternalState(that);
                  else that.size--;
                } return !!entry;
              },
-             // 23.2.3.6 Set.prototype.forEach(callbackfn, thisArg = undefined)
-             // 23.1.3.5 Map.prototype.forEach(callbackfn, thisArg = undefined)
+             // `{ Map, Set }.prototype.forEach(callbackfn, thisArg = undefined)` methods
+             // https://tc39.es/ecma262/#sec-map.prototype.foreach
+             // https://tc39.es/ecma262/#sec-set.prototype.foreach
              forEach: function forEach(callbackfn /* , that = undefined */) {
                var state = getInternalState(this);
                var boundFunction = functionBindContext(callbackfn, arguments.length > 1 ? arguments[1] : undefined, 3);
                  while (entry && entry.removed) entry = entry.previous;
                }
              },
-             // 23.1.3.7 Map.prototype.has(key)
-             // 23.2.3.7 Set.prototype.has(value)
+             // `{ Map, Set}.prototype.has(key)` methods
+             // https://tc39.es/ecma262/#sec-map.prototype.has
+             // https://tc39.es/ecma262/#sec-set.prototype.has
              has: function has(key) {
                return !!getEntry(this, key);
              }
            });
 
            redefineAll(C.prototype, IS_MAP ? {
-             // 23.1.3.6 Map.prototype.get(key)
+             // `Map.prototype.get(key)` method
+             // https://tc39.es/ecma262/#sec-map.prototype.get
              get: function get(key) {
                var entry = getEntry(this, key);
                return entry && entry.value;
              },
-             // 23.1.3.9 Map.prototype.set(key, value)
+             // `Map.prototype.set(key, value)` method
+             // https://tc39.es/ecma262/#sec-map.prototype.set
              set: function set(key, value) {
                return define(this, key === 0 ? 0 : key, value);
              }
            } : {
-             // 23.2.3.1 Set.prototype.add(value)
+             // `Set.prototype.add(value)` method
+             // https://tc39.es/ecma262/#sec-set.prototype.add
              add: function add(value) {
                return define(this, value = value === 0 ? 0 : value, value);
              }
            });
-           if (descriptors) defineProperty$7(C.prototype, 'size', {
+           if (descriptors) defineProperty$2(C.prototype, 'size', {
              get: function () {
                return getInternalState(this).size;
              }
            var ITERATOR_NAME = CONSTRUCTOR_NAME + ' Iterator';
            var getInternalCollectionState = internalStateGetterFor(CONSTRUCTOR_NAME);
            var getInternalIteratorState = internalStateGetterFor(ITERATOR_NAME);
-           // add .keys, .values, .entries, [@@iterator]
-           // 23.1.3.4, 23.1.3.8, 23.1.3.11, 23.1.3.12, 23.2.3.5, 23.2.3.8, 23.2.3.10, 23.2.3.11
+           // `{ Map, Set }.prototype.{ keys, values, entries, @@iterator }()` methods
+           // https://tc39.es/ecma262/#sec-map.prototype.entries
+           // https://tc39.es/ecma262/#sec-map.prototype.keys
+           // https://tc39.es/ecma262/#sec-map.prototype.values
+           // https://tc39.es/ecma262/#sec-map.prototype-@@iterator
+           // https://tc39.es/ecma262/#sec-set.prototype.entries
+           // https://tc39.es/ecma262/#sec-set.prototype.keys
+           // https://tc39.es/ecma262/#sec-set.prototype.values
+           // https://tc39.es/ecma262/#sec-set.prototype-@@iterator
            defineIterator(C, CONSTRUCTOR_NAME, function (iterated, kind) {
-             setInternalState$7(this, {
+             setInternalState(this, {
                type: ITERATOR_NAME,
                target: iterated,
                state: getInternalCollectionState(iterated),
              return { value: [entry.key, entry.value], done: false };
            }, IS_MAP ? 'entries' : 'values', !IS_MAP, true);
 
-           // add [@@species], 23.1.2.2, 23.2.2.2
+           // `{ Map, Set }.prototype[@@species]` accessors
+           // https://tc39.es/ecma262/#sec-get-map-@@species
+           // https://tc39.es/ecma262/#sec-get-set-@@species
            setSpecies(CONSTRUCTOR_NAME);
          }
        };
 
        // `Set` constructor
        // https://tc39.es/ecma262/#sec-set-objects
-       var es_set = collection('Set', function (init) {
+       collection('Set', function (init) {
          return function Set() { return init(this, arguments.length ? arguments[0] : undefined); };
        }, collectionStrong);
 
        // https://tc39.es/ecma262/#sec-symbol.asynciterator
        defineWellKnownSymbol('asyncIterator');
 
-       var runtime_1 = createCommonjsModule(function (module) {
-         /**
-          * Copyright (c) 2014-present, Facebook, Inc.
-          *
-          * This source code is licensed under the MIT license found in the
-          * LICENSE file in the root directory of this source tree.
-          */
+       createCommonjsModule(function (module) {
          var runtime = function (exports) {
 
            var Op = Object.prototype;
          // as the regeneratorRuntime namespace. Otherwise create a new empty
          // object. Either way, the resulting object will be used to initialize
          // the regeneratorRuntime variable at the top of this file.
-          module.exports );
+         module.exports );
 
          try {
            regeneratorRuntime = runtime;
          }
        });
 
-       var _marked = /*#__PURE__*/regeneratorRuntime.mark(numbers);
+       var _marked$2 = /*#__PURE__*/regeneratorRuntime.mark(numbers);
 
-       function number (x) {
+       function number$1 (x) {
          return x === null ? NaN : +x;
        }
        function numbers(values, valueof) {
                  return _context.stop();
              }
            }
-         }, _marked, null, [[2, 13, 16, 19], [23, 34, 37, 40]]);
+         }, _marked$2, null, [[2, 13, 16, 19], [23, 34, 37, 40]]);
        }
 
        var ascendingBisect = d3_bisector(d3_ascending);
        var bisectRight = ascendingBisect.right;
-       var bisectCenter = d3_bisector(number).center;
+       d3_bisector(number$1).center;
 
-       var INCORRECT_ITERATION$1 = !checkCorrectnessOfIteration(function (iterable) {
+       var INCORRECT_ITERATION = !checkCorrectnessOfIteration(function (iterable) {
+         // eslint-disable-next-line es/no-array-from -- required for testing
          Array.from(iterable);
        });
 
        // `Array.from` method
        // https://tc39.es/ecma262/#sec-array.from
-       _export({ target: 'Array', stat: true, forced: INCORRECT_ITERATION$1 }, {
+       _export({ target: 'Array', stat: true, forced: INCORRECT_ITERATION }, {
          from: arrayFrom
        });
 
        // https://tc39.es/ecma262/#sec-array.prototype-@@unscopables
        addToUnscopables('fill');
 
-       var $some$1 = arrayIteration.some;
+       var $some = arrayIteration.some;
 
 
-       var STRICT_METHOD$4 = arrayMethodIsStrict('some');
+       var STRICT_METHOD$3 = arrayMethodIsStrict('some');
 
        // `Array.prototype.some` method
        // https://tc39.es/ecma262/#sec-array.prototype.some
-       _export({ target: 'Array', proto: true, forced: !STRICT_METHOD$4 }, {
+       _export({ target: 'Array', proto: true, forced: !STRICT_METHOD$3 }, {
          some: function some(callbackfn /* , thisArg */) {
-           return $some$1(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
+           return $some(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
          }
        });
 
-       var exportTypedArrayStaticMethod$1 = arrayBufferViewCore.exportTypedArrayStaticMethod;
+       var exportTypedArrayStaticMethod = arrayBufferViewCore.exportTypedArrayStaticMethod;
 
 
        // `%TypedArray%.from` method
        // https://tc39.es/ecma262/#sec-%typedarray%.from
-       exportTypedArrayStaticMethod$1('from', typedArrayFrom, typedArrayConstructorsRequireWrappers);
+       exportTypedArrayStaticMethod('from', typedArrayFrom, typedArrayConstructorsRequireWrappers);
 
        // `Float64Array` constructor
        // https://tc39.es/ecma262/#sec-typedarray-objects
        // https://github.com/python/cpython/blob/a74eea238f5baba15797e2e8b570d153bc8690a7/Modules/mathmodule.c#L1423
        var Adder = /*#__PURE__*/function () {
          function Adder() {
-           _classCallCheck(this, Adder);
+           _classCallCheck$1(this, Adder);
 
            this._partials = new Float64Array(32);
            this._n = 0;
          }
 
-         _createClass(Adder, [{
+         _createClass$1(Adder, [{
            key: "add",
            value: function add(x) {
              var p = this._partials;
 
        // `Map` constructor
        // https://tc39.es/ecma262/#sec-map-objects
-       var es_map = collection('Map', function (init) {
+       collection('Map', function (init) {
          return function Map() { return init(this, arguments.length ? arguments[0] : undefined); };
        }, collectionStrong);
 
-       var test$2 = [];
-       var nativeSort = test$2.sort;
+       var test = [];
+       var nativeSort = test.sort;
 
        // IE8-
        var FAILS_ON_UNDEFINED = fails(function () {
-         test$2.sort(undefined);
+         test.sort(undefined);
        });
        // V8 bug
        var FAILS_ON_NULL = fails(function () {
-         test$2.sort(null);
+         test.sort(null);
        });
        // Old WebKit
-       var STRICT_METHOD$5 = arrayMethodIsStrict('sort');
+       var STRICT_METHOD$2 = arrayMethodIsStrict('sort');
+
+       var STABLE_SORT = !fails(function () {
+         // feature detection can be too slow, so check engines versions
+         if (engineV8Version) return engineV8Version < 70;
+         if (engineFfVersion && engineFfVersion > 3) return;
+         if (engineIsIeOrEdge) return true;
+         if (engineWebkitVersion) return engineWebkitVersion < 603;
+
+         var result = '';
+         var code, chr, value, index;
+
+         // generate an array with more 512 elements (Chakra and old V8 fails only in this case)
+         for (code = 65; code < 76; code++) {
+           chr = String.fromCharCode(code);
+
+           switch (code) {
+             case 66: case 69: case 70: case 72: value = 3; break;
+             case 68: case 71: value = 4; break;
+             default: value = 2;
+           }
+
+           for (index = 0; index < 47; index++) {
+             test.push({ k: chr + index, v: value });
+           }
+         }
+
+         test.sort(function (a, b) { return b.v - a.v; });
 
-       var FORCED$a = FAILS_ON_UNDEFINED || !FAILS_ON_NULL || !STRICT_METHOD$5;
+         for (index = 0; index < test.length; index++) {
+           chr = test[index].k.charAt(0);
+           if (result.charAt(result.length - 1) !== chr) result += chr;
+         }
+
+         return result !== 'DGBEFHACIJK';
+       });
+
+       var FORCED$5 = FAILS_ON_UNDEFINED || !FAILS_ON_NULL || !STRICT_METHOD$2 || !STABLE_SORT;
+
+       var getSortCompare = function (comparefn) {
+         return function (x, y) {
+           if (y === undefined) return -1;
+           if (x === undefined) return 1;
+           if (comparefn !== undefined) return +comparefn(x, y) || 0;
+           return String(x) > String(y) ? 1 : -1;
+         };
+       };
 
        // `Array.prototype.sort` method
        // https://tc39.es/ecma262/#sec-array.prototype.sort
-       _export({ target: 'Array', proto: true, forced: FORCED$a }, {
+       _export({ target: 'Array', proto: true, forced: FORCED$5 }, {
          sort: function sort(comparefn) {
-           return comparefn === undefined
-             ? nativeSort.call(toObject(this))
-             : nativeSort.call(toObject(this), aFunction$1(comparefn));
+           if (comparefn !== undefined) aFunction(comparefn);
+
+           var array = toObject(this);
+
+           if (STABLE_SORT) return comparefn === undefined ? nativeSort.call(array) : nativeSort.call(array, comparefn);
+
+           var items = [];
+           var arrayLength = toLength(array.length);
+           var itemsLength, index;
+
+           for (index = 0; index < arrayLength; index++) {
+             if (index in array) items.push(array[index]);
+           }
+
+           items = arraySort(items, getSortCompare(comparefn));
+           itemsLength = items.length;
+           index = 0;
+
+           while (index < itemsLength) array[index] = items[index++];
+           while (index < arrayLength) delete array[index++];
+
+           return array;
          }
        });
 
          if ((step = tickIncrement(start, stop, count)) === 0 || !isFinite(step)) return [];
 
          if (step > 0) {
-           start = Math.ceil(start / step);
-           stop = Math.floor(stop / step);
-           ticks = new Array(n = Math.ceil(stop - start + 1));
+           var r0 = Math.round(start / step),
+               r1 = Math.round(stop / step);
+           if (r0 * step < start) ++r0;
+           if (r1 * step > stop) --r1;
+           ticks = new Array(n = r1 - r0 + 1);
 
            while (++i < n) {
-             ticks[i] = (start + i) * step;
+             ticks[i] = (r0 + i) * step;
            }
          } else {
            step = -step;
-           start = Math.ceil(start * step);
-           stop = Math.floor(stop * step);
-           ticks = new Array(n = Math.ceil(stop - start + 1));
+
+           var _r = Math.round(start * step),
+               _r2 = Math.round(stop * step);
+
+           if (_r / step < start) ++_r;
+           if (_r2 / step > stop) --_r2;
+           ticks = new Array(n = _r2 - _r + 1);
 
            while (++i < n) {
-             ticks[i] = (start + i) / step;
+             ticks[i] = (_r + i) / step;
            }
          }
 
          return stop < start ? -step1 : step1;
        }
 
-       function max$4(values, valueof) {
+       function max(values, valueof) {
          var max;
 
          if (valueof === undefined) {
          return max;
        }
 
-       function min$7(values, valueof) {
+       function min$2(values, valueof) {
          var min;
 
          if (valueof === undefined) {
 
        // ISC license, Copyright 2018 Vladimir Agafonkin.
 
-       function quickselect(array, k) {
+       function quickselect$2(array, k) {
          var left = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 0;
          var right = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : array.length - 1;
          var compare = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : d3_ascending;
              var sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1);
              var newLeft = Math.max(left, Math.floor(k - m * s / n + sd));
              var newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd));
-             quickselect(array, k, newLeft, newRight, compare);
+             quickselect$2(array, k, newLeft, newRight, compare);
            }
 
            var t = array[k];
            var i = left;
            var j = right;
-           swap(array, left, k);
-           if (compare(array[right], t) > 0) swap(array, left, right);
+           swap$1(array, left, k);
+           if (compare(array[right], t) > 0) swap$1(array, left, right);
 
            while (i < j) {
-             swap(array, i, j), ++i, --j;
+             swap$1(array, i, j), ++i, --j;
 
              while (compare(array[i], t) < 0) {
                ++i;
              }
            }
 
-           if (compare(array[left], t) === 0) swap(array, left, j);else ++j, swap(array, j, right);
+           if (compare(array[left], t) === 0) swap$1(array, left, j);else ++j, swap$1(array, j, right);
            if (j <= k) left = j + 1;
            if (k <= j) right = j - 1;
          }
          return array;
        }
 
-       function swap(array, i, j) {
+       function swap$1(array, i, j) {
          var t = array[i];
          array[i] = array[j];
          array[j] = t;
        function quantile(values, p, valueof) {
          values = Float64Array.from(numbers(values, valueof));
          if (!(n = values.length)) return;
-         if ((p = +p) <= 0 || n < 2) return min$7(values);
-         if (p >= 1) return max$4(values);
+         if ((p = +p) <= 0 || n < 2) return min$2(values);
+         if (p >= 1) return max(values);
          var n,
              i = (n - 1) * p,
              i0 = Math.floor(i),
-             value0 = max$4(quickselect(values, i0).subarray(0, i0 + 1)),
-             value1 = min$7(values.subarray(i0 + 1));
+             value0 = max(quickselect$2(values, i0).subarray(0, i0 + 1)),
+             value1 = min$2(values.subarray(i0 + 1));
          return value0 + (value1 - value0) * (i - i0);
        }
 
          }, _marked$1, null, [[1, 10, 13, 16]]);
        }
 
-       function merge(arrays) {
+       function merge$4(arrays) {
          return Array.from(flatten(arrays));
        }
 
-       function range (start, stop, step) {
+       function range$1 (start, stop, step) {
          start = +start, stop = +stop, step = (n = arguments.length) < 2 ? (stop = start, start = 0, 1) : n < 3 ? 1 : +step;
          var i = -1,
              n = Math.max(0, Math.ceil((stop - start) / step)) | 0,
 
        // `SameValue` abstract operation
        // https://tc39.es/ecma262/#sec-samevalue
+       // eslint-disable-next-line es/no-object-is -- safe
        var sameValue = Object.is || function is(x, y) {
          // eslint-disable-next-line no-self-compare -- NaN check
          return x === y ? x !== 0 || 1 / x === 1 / y : x != x && y != y;
        };
 
+       // eslint-disable-next-line es/no-math-hypot -- required for testing
        var $hypot = Math.hypot;
-       var abs$1 = Math.abs;
-       var sqrt = Math.sqrt;
+       var abs$3 = Math.abs;
+       var sqrt$1 = Math.sqrt;
 
        // Chrome 77 bug
        // https://bugs.chromium.org/p/v8/issues/detail?id=9546
            var larg = 0;
            var arg, div;
            while (i < aLen) {
-             arg = abs$1(arguments[i++]);
+             arg = abs$3(arguments[i++]);
              if (larg < arg) {
                div = larg / arg;
                sum = sum * div * div + 1;
                sum += div * div;
              } else sum += arg;
            }
-           return larg === Infinity ? Infinity : larg * sqrt(sum);
+           return larg === Infinity ? Infinity : larg * sqrt$1(sum);
          }
        });
 
        // `Math.sign` method implementation
        // https://tc39.es/ecma262/#sec-math.sign
+       // eslint-disable-next-line es/no-math-sign -- safe
        var mathSign = Math.sign || function sign(x) {
          // eslint-disable-next-line no-self-compare -- NaN check
          return (x = +x) == 0 || x != x ? x : x < 0 ? -1 : 1;
          sign: mathSign
        });
 
-       var epsilon = 1e-6;
-       var epsilon2 = 1e-12;
+       var epsilon$1 = 1e-6;
+       var epsilon2$1 = 1e-12;
        var pi = Math.PI;
        var halfPi = pi / 2;
        var quarterPi = pi / 4;
        var tau = pi * 2;
-       var degrees = 180 / pi;
+       var degrees$1 = 180 / pi;
        var radians = pi / 180;
        var abs$2 = Math.abs;
        var atan = Math.atan;
        var atan2 = Math.atan2;
        var cos = Math.cos;
-       var exp = Math.exp;
-       var hypot = Math.hypot;
+       var exp$2 = Math.exp;
        var log$1 = Math.log;
        var sin = Math.sin;
        var sign = Math.sign || function (x) {
          return x > 0 ? 1 : x < 0 ? -1 : 0;
        };
-       var sqrt$1 = Math.sqrt;
+       var sqrt = Math.sqrt;
        var tan = Math.tan;
        function acos(x) {
          return x > 1 ? 0 : x < -1 ? pi : Math.acos(x);
          return x > 1 ? halfPi : x < -1 ? -halfPi : Math.asin(x);
        }
 
-       function noop() {}
+       function noop$1() {}
 
        function streamGeometry(geometry, stream) {
          if (geometry && streamGeometryType.hasOwnProperty(geometry.type)) {
          }
        }
 
-       var areaRingSum = new Adder(); // hello?
+       var areaRingSum$1 = new Adder(); // hello?
 
-       var areaSum = new Adder(),
-           lambda00,
-           phi00,
-           lambda0,
-           cosPhi0,
-           sinPhi0;
-       var areaStream = {
-         point: noop,
-         lineStart: noop,
-         lineEnd: noop,
+       var areaSum$1 = new Adder(),
+           lambda00$1,
+           phi00$1,
+           lambda0$2,
+           cosPhi0$1,
+           sinPhi0$1;
+       var areaStream$1 = {
+         point: noop$1,
+         lineStart: noop$1,
+         lineEnd: noop$1,
          polygonStart: function polygonStart() {
-           areaRingSum = new Adder();
-           areaStream.lineStart = areaRingStart;
-           areaStream.lineEnd = areaRingEnd;
+           areaRingSum$1 = new Adder();
+           areaStream$1.lineStart = areaRingStart$1;
+           areaStream$1.lineEnd = areaRingEnd$1;
          },
          polygonEnd: function polygonEnd() {
-           var areaRing = +areaRingSum;
-           areaSum.add(areaRing < 0 ? tau + areaRing : areaRing);
-           this.lineStart = this.lineEnd = this.point = noop;
+           var areaRing = +areaRingSum$1;
+           areaSum$1.add(areaRing < 0 ? tau + areaRing : areaRing);
+           this.lineStart = this.lineEnd = this.point = noop$1;
          },
          sphere: function sphere() {
-           areaSum.add(tau);
+           areaSum$1.add(tau);
          }
        };
 
-       function areaRingStart() {
-         areaStream.point = areaPointFirst;
+       function areaRingStart$1() {
+         areaStream$1.point = areaPointFirst$1;
        }
 
-       function areaRingEnd() {
-         areaPoint(lambda00, phi00);
+       function areaRingEnd$1() {
+         areaPoint$1(lambda00$1, phi00$1);
        }
 
-       function areaPointFirst(lambda, phi) {
-         areaStream.point = areaPoint;
-         lambda00 = lambda, phi00 = phi;
+       function areaPointFirst$1(lambda, phi) {
+         areaStream$1.point = areaPoint$1;
+         lambda00$1 = lambda, phi00$1 = phi;
          lambda *= radians, phi *= radians;
-         lambda0 = lambda, cosPhi0 = cos(phi = phi / 2 + quarterPi), sinPhi0 = sin(phi);
+         lambda0$2 = lambda, cosPhi0$1 = cos(phi = phi / 2 + quarterPi), sinPhi0$1 = sin(phi);
        }
 
-       function areaPoint(lambda, phi) {
+       function areaPoint$1(lambda, phi) {
          lambda *= radians, phi *= radians;
          phi = phi / 2 + quarterPi; // half the angular distance from south pole
          // Spherical excess E for a spherical triangle with vertices: south pole,
          // previous point, current point.  Uses a formula derived from Cagnoli’s
          // theorem.  See Todhunter, Spherical Trig. (1871), Sec. 103, Eq. (2).
 
-         var dLambda = lambda - lambda0,
+         var dLambda = lambda - lambda0$2,
              sdLambda = dLambda >= 0 ? 1 : -1,
              adLambda = sdLambda * dLambda,
              cosPhi = cos(phi),
              sinPhi = sin(phi),
-             k = sinPhi0 * sinPhi,
-             u = cosPhi0 * cosPhi + k * cos(adLambda),
+             k = sinPhi0$1 * sinPhi,
+             u = cosPhi0$1 * cosPhi + k * cos(adLambda),
              v = k * sdLambda * sin(adLambda);
-         areaRingSum.add(atan2(v, u)); // Advance the previous points.
+         areaRingSum$1.add(atan2(v, u)); // Advance the previous points.
 
-         lambda0 = lambda, cosPhi0 = cosPhi, sinPhi0 = sinPhi;
+         lambda0$2 = lambda, cosPhi0$1 = cosPhi, sinPhi0$1 = sinPhi;
        }
 
        function d3_geoArea (object) {
-         areaSum = new Adder();
-         d3_geoStream(object, areaStream);
-         return areaSum * 2;
+         areaSum$1 = new Adder();
+         d3_geoStream(object, areaStream$1);
+         return areaSum$1 * 2;
        }
 
        function spherical(cartesian) {
        } // TODO return d
 
        function cartesianNormalizeInPlace(d) {
-         var l = sqrt$1(d[0] * d[0] + d[1] * d[1] + d[2] * d[2]);
+         var l = sqrt(d[0] * d[0] + d[1] * d[1] + d[2] * d[2]);
          d[0] /= l, d[1] /= l, d[2] /= l;
        }
 
        var lambda0$1, phi0, lambda1, phi1, // bounds
        lambda2, // previous lambda-coordinate
-       lambda00$1, phi00$1, // first point
+       lambda00, phi00, // first point
        p0, // previous 3D point
-       deltaSum, ranges, range$1;
-       var boundsStream = {
-         point: boundsPoint,
+       deltaSum, ranges, range;
+       var boundsStream$1 = {
+         point: boundsPoint$1,
          lineStart: boundsLineStart,
          lineEnd: boundsLineEnd,
          polygonStart: function polygonStart() {
-           boundsStream.point = boundsRingPoint;
-           boundsStream.lineStart = boundsRingStart;
-           boundsStream.lineEnd = boundsRingEnd;
+           boundsStream$1.point = boundsRingPoint;
+           boundsStream$1.lineStart = boundsRingStart;
+           boundsStream$1.lineEnd = boundsRingEnd;
            deltaSum = new Adder();
-           areaStream.polygonStart();
+           areaStream$1.polygonStart();
          },
          polygonEnd: function polygonEnd() {
-           areaStream.polygonEnd();
-           boundsStream.point = boundsPoint;
-           boundsStream.lineStart = boundsLineStart;
-           boundsStream.lineEnd = boundsLineEnd;
-           if (areaRingSum < 0) lambda0$1 = -(lambda1 = 180), phi0 = -(phi1 = 90);else if (deltaSum > epsilon) phi1 = 90;else if (deltaSum < -epsilon) phi0 = -90;
-           range$1[0] = lambda0$1, range$1[1] = lambda1;
+           areaStream$1.polygonEnd();
+           boundsStream$1.point = boundsPoint$1;
+           boundsStream$1.lineStart = boundsLineStart;
+           boundsStream$1.lineEnd = boundsLineEnd;
+           if (areaRingSum$1 < 0) lambda0$1 = -(lambda1 = 180), phi0 = -(phi1 = 90);else if (deltaSum > epsilon$1) phi1 = 90;else if (deltaSum < -epsilon$1) phi0 = -90;
+           range[0] = lambda0$1, range[1] = lambda1;
          },
          sphere: function sphere() {
            lambda0$1 = -(lambda1 = 180), phi0 = -(phi1 = 90);
          }
        };
 
-       function boundsPoint(lambda, phi) {
-         ranges.push(range$1 = [lambda0$1 = lambda, lambda1 = lambda]);
+       function boundsPoint$1(lambda, phi) {
+         ranges.push(range = [lambda0$1 = lambda, lambda1 = lambda]);
          if (phi < phi0) phi0 = phi;
          if (phi > phi1) phi1 = phi;
        }
            inflection = spherical(inflection);
            var delta = lambda - lambda2,
                sign = delta > 0 ? 1 : -1,
-               lambdai = inflection[0] * degrees * sign,
+               lambdai = inflection[0] * degrees$1 * sign,
                phii,
                antimeridian = abs$2(delta) > 180;
 
            if (antimeridian ^ (sign * lambda2 < lambdai && lambdai < sign * lambda)) {
-             phii = inflection[1] * degrees;
+             phii = inflection[1] * degrees$1;
              if (phii > phi1) phi1 = phii;
            } else if (lambdai = (lambdai + 360) % 360 - 180, antimeridian ^ (sign * lambda2 < lambdai && lambdai < sign * lambda)) {
-             phii = -inflection[1] * degrees;
+             phii = -inflection[1] * degrees$1;
              if (phii < phi0) phi0 = phii;
            } else {
              if (phi < phi0) phi0 = phi;
              }
            }
          } else {
-           ranges.push(range$1 = [lambda0$1 = lambda, lambda1 = lambda]);
+           ranges.push(range = [lambda0$1 = lambda, lambda1 = lambda]);
          }
 
          if (phi < phi0) phi0 = phi;
        }
 
        function boundsLineStart() {
-         boundsStream.point = linePoint;
+         boundsStream$1.point = linePoint;
        }
 
        function boundsLineEnd() {
-         range$1[0] = lambda0$1, range$1[1] = lambda1;
-         boundsStream.point = boundsPoint;
+         range[0] = lambda0$1, range[1] = lambda1;
+         boundsStream$1.point = boundsPoint$1;
          p0 = null;
        }
 
            var delta = lambda - lambda2;
            deltaSum.add(abs$2(delta) > 180 ? delta + (delta > 0 ? 360 : -360) : delta);
          } else {
-           lambda00$1 = lambda, phi00$1 = phi;
+           lambda00 = lambda, phi00 = phi;
          }
 
-         areaStream.point(lambda, phi);
+         areaStream$1.point(lambda, phi);
          linePoint(lambda, phi);
        }
 
        function boundsRingStart() {
-         areaStream.lineStart();
+         areaStream$1.lineStart();
        }
 
        function boundsRingEnd() {
-         boundsRingPoint(lambda00$1, phi00$1);
-         areaStream.lineEnd();
-         if (abs$2(deltaSum) > epsilon) lambda0$1 = -(lambda1 = 180);
-         range$1[0] = lambda0$1, range$1[1] = lambda1;
+         boundsRingPoint(lambda00, phi00);
+         areaStream$1.lineEnd();
+         if (abs$2(deltaSum) > epsilon$1) lambda0$1 = -(lambda1 = 180);
+         range[0] = lambda0$1, range[1] = lambda1;
          p0 = null;
        } // Finds the left-right distance between two longitudes.
        // This is almost the same as (lambda1 - lambda0 + 360°) % 360°, except that we want
          var i, n, a, b, merged, deltaMax, delta;
          phi1 = lambda1 = -(lambda0$1 = phi0 = Infinity);
          ranges = [];
-         d3_geoStream(feature, boundsStream); // First, sort ranges by their minimum longitudes.
+         d3_geoStream(feature, boundsStream$1); // First, sort ranges by their minimum longitudes.
 
          if (n = ranges.length) {
            ranges.sort(rangeCompare); // Then, merge any ranges that overlap.
            }
          }
 
-         ranges = range$1 = null;
+         ranges = range = null;
          return lambda0$1 === Infinity || phi0 === Infinity ? [[NaN, NaN], [NaN, NaN]] : [[lambda0$1, phi0], [lambda1, phi1]];
        }
 
-       var W0, W1, X0, Y0, Z0, X1, Y1, Z1, X2, Y2, Z2, lambda00$2, phi00$2, // first point
-       x0, y0, z0; // previous point
-
-       var centroidStream = {
-         sphere: noop,
-         point: centroidPoint,
-         lineStart: centroidLineStart,
-         lineEnd: centroidLineEnd,
-         polygonStart: function polygonStart() {
-           centroidStream.lineStart = centroidRingStart;
-           centroidStream.lineEnd = centroidRingEnd;
-         },
-         polygonEnd: function polygonEnd() {
-           centroidStream.lineStart = centroidLineStart;
-           centroidStream.lineEnd = centroidLineEnd;
-         }
-       }; // Arithmetic mean of Cartesian vectors.
-
-       function centroidPoint(lambda, phi) {
-         lambda *= radians, phi *= radians;
-         var cosPhi = cos(phi);
-         centroidPointCartesian(cosPhi * cos(lambda), cosPhi * sin(lambda), sin(phi));
-       }
-
-       function centroidPointCartesian(x, y, z) {
-         ++W0;
-         X0 += (x - X0) / W0;
-         Y0 += (y - Y0) / W0;
-         Z0 += (z - Z0) / W0;
-       }
-
-       function centroidLineStart() {
-         centroidStream.point = centroidLinePointFirst;
-       }
-
-       function centroidLinePointFirst(lambda, phi) {
-         lambda *= radians, phi *= radians;
-         var cosPhi = cos(phi);
-         x0 = cosPhi * cos(lambda);
-         y0 = cosPhi * sin(lambda);
-         z0 = sin(phi);
-         centroidStream.point = centroidLinePoint;
-         centroidPointCartesian(x0, y0, z0);
-       }
-
-       function centroidLinePoint(lambda, phi) {
-         lambda *= radians, phi *= radians;
-         var cosPhi = cos(phi),
-             x = cosPhi * cos(lambda),
-             y = cosPhi * sin(lambda),
-             z = sin(phi),
-             w = atan2(sqrt$1((w = y0 * z - z0 * y) * w + (w = z0 * x - x0 * z) * w + (w = x0 * y - y0 * x) * w), x0 * x + y0 * y + z0 * z);
-         W1 += w;
-         X1 += w * (x0 + (x0 = x));
-         Y1 += w * (y0 + (y0 = y));
-         Z1 += w * (z0 + (z0 = z));
-         centroidPointCartesian(x0, y0, z0);
-       }
-
-       function centroidLineEnd() {
-         centroidStream.point = centroidPoint;
-       } // See J. E. Brock, The Inertia Tensor for a Spherical Triangle,
-       // J. Applied Mechanics 42, 239 (1975).
-
-
-       function centroidRingStart() {
-         centroidStream.point = centroidRingPointFirst;
-       }
-
-       function centroidRingEnd() {
-         centroidRingPoint(lambda00$2, phi00$2);
-         centroidStream.point = centroidPoint;
-       }
-
-       function centroidRingPointFirst(lambda, phi) {
-         lambda00$2 = lambda, phi00$2 = phi;
-         lambda *= radians, phi *= radians;
-         centroidStream.point = centroidRingPoint;
-         var cosPhi = cos(phi);
-         x0 = cosPhi * cos(lambda);
-         y0 = cosPhi * sin(lambda);
-         z0 = sin(phi);
-         centroidPointCartesian(x0, y0, z0);
-       }
-
-       function centroidRingPoint(lambda, phi) {
-         lambda *= radians, phi *= radians;
-         var cosPhi = cos(phi),
-             x = cosPhi * cos(lambda),
-             y = cosPhi * sin(lambda),
-             z = sin(phi),
-             cx = y0 * z - z0 * y,
-             cy = z0 * x - x0 * z,
-             cz = x0 * y - y0 * x,
-             m = hypot(cx, cy, cz),
-             w = asin(m),
-             // line weight = angle
-         v = m && -w / m; // area weight multiplier
-
-         X2.add(v * cx);
-         Y2.add(v * cy);
-         Z2.add(v * cz);
-         W1 += w;
-         X1 += w * (x0 + (x0 = x));
-         Y1 += w * (y0 + (y0 = y));
-         Z1 += w * (z0 + (z0 = z));
-         centroidPointCartesian(x0, y0, z0);
-       }
-
-       function d3_geoCentroid (object) {
-         W0 = W1 = X0 = Y0 = Z0 = X1 = Y1 = Z1 = 0;
-         X2 = new Adder();
-         Y2 = new Adder();
-         Z2 = new Adder();
-         d3_geoStream(object, centroidStream);
-         var x = +X2,
-             y = +Y2,
-             z = +Z2,
-             m = hypot(x, y, z); // If the area-weighted ccentroid is undefined, fall back to length-weighted ccentroid.
-
-         if (m < epsilon2) {
-           x = X1, y = Y1, z = Z1; // If the feature has zero length, fall back to arithmetic mean of point vectors.
-
-           if (W1 < epsilon) x = X0, y = Y0, z = Z0;
-           m = hypot(x, y, z); // If the feature still has an undefined ccentroid, then return.
-
-           if (m < epsilon2) return [NaN, NaN];
-         }
-
-         return [atan2(y, x) * degrees, asin(z / m) * degrees];
-       }
-
        function compose (a, b) {
          function compose(x, y) {
            return x = a(x, y), b(x[0], x[1]);
 
          function forward(coordinates) {
            coordinates = rotate(coordinates[0] * radians, coordinates[1] * radians);
-           return coordinates[0] *= degrees, coordinates[1] *= degrees, coordinates;
+           return coordinates[0] *= degrees$1, coordinates[1] *= degrees$1, coordinates;
          }
 
          forward.invert = function (coordinates) {
            coordinates = rotate.invert(coordinates[0] * radians, coordinates[1] * radians);
-           return coordinates[0] *= degrees, coordinates[1] *= degrees, coordinates;
+           return coordinates[0] *= degrees$1, coordinates[1] *= degrees$1, coordinates;
          };
 
          return forward;
          point = cartesian(point), point[0] -= cosRadius;
          cartesianNormalizeInPlace(point);
          var radius = acos(-point[1]);
-         return ((-point[2] < 0 ? -radius : radius) + tau - epsilon) % tau;
+         return ((-point[2] < 0 ? -radius : radius) + tau - epsilon$1) % tau;
        }
 
        function clipBuffer () {
            lineStart: function lineStart() {
              lines.push(line = []);
            },
-           lineEnd: noop,
+           lineEnd: noop$1,
            rejoin: function rejoin() {
              if (lines.length > 1) lines.push(lines.pop().concat(lines.shift()));
            },
        }
 
        function pointEqual (a, b) {
-         return abs$2(a[0] - b[0]) < epsilon && abs$2(a[1] - b[1]) < epsilon;
+         return abs$2(a[0] - b[0]) < epsilon$1 && abs$2(a[1] - b[1]) < epsilon$1;
        }
 
        function Intersection(point, points, other, entry) {
              } // handle degenerate cases by moving the point
 
 
-             p1[0] += 2 * epsilon;
+             p1[0] += 2 * epsilon$1;
            }
 
            subject.push(x = new Intersection(p0, segment, null, true));
              angle = 0,
              winding = 0;
          var sum = new Adder();
-         if (sinPhi === 1) phi = halfPi + epsilon;else if (sinPhi === -1) phi = -halfPi - epsilon;
+         if (sinPhi === 1) phi = halfPi + epsilon$1;else if (sinPhi === -1) phi = -halfPi - epsilon$1;
 
          for (var i = 0, n = polygon.length; i < n; ++i) {
            if (!(m = (ring = polygon[i]).length)) continue;
          // same side as the South pole.
 
 
-         return (angle < -epsilon || angle < epsilon && sum < -epsilon2) ^ winding & 1;
+         return (angle < -epsilon$1 || angle < epsilon$1 && sum < -epsilon2$1) ^ winding & 1;
        }
 
        function clip (pointVisible, clipLine, interpolate, start) {
                clip.point = point;
                clip.lineStart = lineStart;
                clip.lineEnd = lineEnd;
-               segments = merge(segments);
+               segments = merge$4(segments);
                var startInside = polygonContains(polygon, start);
 
                if (segments.length) {
 
 
        function compareIntersection(a, b) {
-         return ((a = a.x)[0] < 0 ? a[1] - halfPi - epsilon : halfPi - a[1]) - ((b = b.x)[0] < 0 ? b[1] - halfPi - epsilon : halfPi - b[1]);
+         return ((a = a.x)[0] < 0 ? a[1] - halfPi - epsilon$1 : halfPi - a[1]) - ((b = b.x)[0] < 0 ? b[1] - halfPi - epsilon$1 : halfPi - b[1]);
        }
 
        var clipAntimeridian = clip(function () {
              var sign1 = lambda1 > 0 ? pi : -pi,
                  delta = abs$2(lambda1 - lambda0);
 
-             if (abs$2(delta - pi) < epsilon) {
+             if (abs$2(delta - pi) < epsilon$1) {
                // line crosses a pole
                stream.point(lambda0, phi0 = (phi0 + phi1) / 2 > 0 ? halfPi : -halfPi);
                stream.point(sign0, phi0);
                _clean = 0;
              } else if (sign0 !== sign1 && delta >= pi) {
                // line crosses antimeridian
-               if (abs$2(lambda0 - sign0) < epsilon) lambda0 -= sign0 * epsilon; // handle degeneracies
+               if (abs$2(lambda0 - sign0) < epsilon$1) lambda0 -= sign0 * epsilon$1; // handle degeneracies
 
-               if (abs$2(lambda1 - sign1) < epsilon) lambda1 -= sign1 * epsilon;
+               if (abs$2(lambda1 - sign1) < epsilon$1) lambda1 -= sign1 * epsilon$1;
                phi0 = clipAntimeridianIntersect(lambda0, phi0, lambda1, phi1);
                stream.point(sign0, phi0);
                stream.lineEnd();
          var cosPhi0,
              cosPhi1,
              sinLambda0Lambda1 = sin(lambda0 - lambda1);
-         return abs$2(sinLambda0Lambda1) > epsilon ? atan((sin(phi0) * (cosPhi1 = cos(phi1)) * sin(lambda1) - sin(phi1) * (cosPhi0 = cos(phi0)) * sin(lambda0)) / (cosPhi0 * cosPhi1 * sinLambda0Lambda1)) : (phi0 + phi1) / 2;
+         return abs$2(sinLambda0Lambda1) > epsilon$1 ? atan((sin(phi0) * (cosPhi1 = cos(phi1)) * sin(lambda1) - sin(phi1) * (cosPhi0 = cos(phi0)) * sin(lambda0)) / (cosPhi0 * cosPhi1 * sinLambda0Lambda1)) : (phi0 + phi1) / 2;
        }
 
        function clipAntimeridianInterpolate(from, to, direction, stream) {
            stream.point(-pi, -phi);
            stream.point(-pi, 0);
            stream.point(-pi, phi);
-         } else if (abs$2(from[0] - to[0]) > epsilon) {
+         } else if (abs$2(from[0] - to[0]) > epsilon$1) {
            var lambda = from[0] < to[0] ? pi : -pi;
            phi = direction * lambda / 2;
            stream.point(-lambda, phi);
          var cr = cos(radius),
              delta = 6 * radians,
              smallRadius = cr > 0,
-             notHemisphere = abs$2(cr) > epsilon; // TODO optimise for this common case
+             notHemisphere = abs$2(cr) > epsilon$1; // TODO optimise for this common case
 
          function interpolate(from, to, direction, stream) {
            circleStream(stream, radius, delta, direction, from, to);
                uu = cartesianDot(u, u),
                t2 = w * w - uu * (cartesianDot(A, A) - 1);
            if (t2 < 0) return;
-           var t = sqrt$1(t2),
+           var t = sqrt(t2),
                q = cartesianScale(u, (-w - t) / uu);
            cartesianAddInPlace(q, A);
            q = spherical(q);
                z;
            if (lambda1 < lambda0) z = lambda0, lambda0 = lambda1, lambda1 = z;
            var delta = lambda1 - lambda0,
-               polar = abs$2(delta - pi) < epsilon,
-               meridian = polar || delta < epsilon;
+               polar = abs$2(delta - pi) < epsilon$1,
+               meridian = polar || delta < epsilon$1;
            if (!polar && phi1 < phi0) z = phi0, phi0 = phi1, phi1 = z; // Check that the first point is between a and b.
 
-           if (meridian ? polar ? phi0 + phi1 > 0 ^ q[1] < (abs$2(q[0] - lambda0) < epsilon ? phi0 : phi1) : phi0 <= q[1] && q[1] <= phi1 : delta > pi ^ (lambda0 <= q[0] && q[0] <= lambda1)) {
+           if (meridian ? polar ? phi0 + phi1 > 0 ^ q[1] < (abs$2(q[0] - lambda0) < epsilon$1 ? phi0 : phi1) : phi0 <= q[1] && q[1] <= phi1 : delta > pi ^ (lambda0 <= q[0] && q[0] <= lambda1)) {
              var q1 = cartesianScale(u, (-w + t) / uu);
              cartesianAddInPlace(q1, A);
              return [q, spherical(q1)];
          }
 
          function corner(p, direction) {
-           return abs$2(p[0] - x0) < epsilon ? direction > 0 ? 0 : 3 : abs$2(p[0] - x1) < epsilon ? direction > 0 ? 2 : 1 : abs$2(p[1] - y0) < epsilon ? direction > 0 ? 1 : 0 : direction > 0 ? 3 : 2; // abs(p[1] - y1) < epsilon
+           return abs$2(p[0] - x0) < epsilon$1 ? direction > 0 ? 0 : 3 : abs$2(p[0] - x1) < epsilon$1 ? direction > 0 ? 2 : 1 : abs$2(p[1] - y0) < epsilon$1 ? direction > 0 ? 1 : 0 : direction > 0 ? 3 : 2; // abs(p[1] - y1) < epsilon
          }
 
          function compareIntersection(a, b) {
            function polygonEnd() {
              var startInside = polygonInside(),
                  cleanInside = clean && startInside,
-                 visible = (segments = merge(segments)).length;
+                 visible = (segments = merge$4(segments)).length;
 
              if (cleanInside || visible) {
                stream.polygonStart();
          };
        }
 
-       var lengthSum, lambda0$2, sinPhi0$1, cosPhi0$1;
-       var lengthStream = {
-         sphere: noop,
-         point: noop,
+       var lengthSum$1, lambda0, sinPhi0, cosPhi0;
+       var lengthStream$1 = {
+         sphere: noop$1,
+         point: noop$1,
          lineStart: lengthLineStart,
-         lineEnd: noop,
-         polygonStart: noop,
-         polygonEnd: noop
+         lineEnd: noop$1,
+         polygonStart: noop$1,
+         polygonEnd: noop$1
        };
 
        function lengthLineStart() {
-         lengthStream.point = lengthPointFirst;
-         lengthStream.lineEnd = lengthLineEnd;
+         lengthStream$1.point = lengthPointFirst$1;
+         lengthStream$1.lineEnd = lengthLineEnd;
        }
 
        function lengthLineEnd() {
-         lengthStream.point = lengthStream.lineEnd = noop;
+         lengthStream$1.point = lengthStream$1.lineEnd = noop$1;
        }
 
-       function lengthPointFirst(lambda, phi) {
+       function lengthPointFirst$1(lambda, phi) {
          lambda *= radians, phi *= radians;
-         lambda0$2 = lambda, sinPhi0$1 = sin(phi), cosPhi0$1 = cos(phi);
-         lengthStream.point = lengthPoint;
+         lambda0 = lambda, sinPhi0 = sin(phi), cosPhi0 = cos(phi);
+         lengthStream$1.point = lengthPoint$1;
        }
 
-       function lengthPoint(lambda, phi) {
+       function lengthPoint$1(lambda, phi) {
          lambda *= radians, phi *= radians;
          var sinPhi = sin(phi),
              cosPhi = cos(phi),
-             delta = abs$2(lambda - lambda0$2),
+             delta = abs$2(lambda - lambda0),
              cosDelta = cos(delta),
              sinDelta = sin(delta),
              x = cosPhi * sinDelta,
-             y = cosPhi0$1 * sinPhi - sinPhi0$1 * cosPhi * cosDelta,
-             z = sinPhi0$1 * sinPhi + cosPhi0$1 * cosPhi * cosDelta;
-         lengthSum.add(atan2(sqrt$1(x * x + y * y), z));
-         lambda0$2 = lambda, sinPhi0$1 = sinPhi, cosPhi0$1 = cosPhi;
+             y = cosPhi0 * sinPhi - sinPhi0 * cosPhi * cosDelta,
+             z = sinPhi0 * sinPhi + cosPhi0 * cosPhi * cosDelta;
+         lengthSum$1.add(atan2(sqrt(x * x + y * y), z));
+         lambda0 = lambda, sinPhi0 = sinPhi, cosPhi0 = cosPhi;
        }
 
        function d3_geoLength (object) {
-         lengthSum = new Adder();
-         d3_geoStream(object, lengthStream);
-         return +lengthSum;
+         lengthSum$1 = new Adder();
+         d3_geoStream(object, lengthStream$1);
+         return +lengthSum$1;
        }
 
-       var identity = (function (x) {
+       var identity$4 = (function (x) {
          return x;
        });
 
-       var areaSum$1 = new Adder(),
-           areaRingSum$1 = new Adder(),
-           x00,
-           y00,
-           x0$1,
-           y0$1;
-       var areaStream$1 = {
-         point: noop,
-         lineStart: noop,
-         lineEnd: noop,
+       var areaSum = new Adder(),
+           areaRingSum = new Adder(),
+           x00$2,
+           y00$2,
+           x0$3,
+           y0$3;
+       var areaStream = {
+         point: noop$1,
+         lineStart: noop$1,
+         lineEnd: noop$1,
          polygonStart: function polygonStart() {
-           areaStream$1.lineStart = areaRingStart$1;
-           areaStream$1.lineEnd = areaRingEnd$1;
+           areaStream.lineStart = areaRingStart;
+           areaStream.lineEnd = areaRingEnd;
          },
          polygonEnd: function polygonEnd() {
-           areaStream$1.lineStart = areaStream$1.lineEnd = areaStream$1.point = noop;
-           areaSum$1.add(abs$2(areaRingSum$1));
-           areaRingSum$1 = new Adder();
+           areaStream.lineStart = areaStream.lineEnd = areaStream.point = noop$1;
+           areaSum.add(abs$2(areaRingSum));
+           areaRingSum = new Adder();
          },
          result: function result() {
-           var area = areaSum$1 / 2;
-           areaSum$1 = new Adder();
+           var area = areaSum / 2;
+           areaSum = new Adder();
            return area;
          }
        };
 
-       function areaRingStart$1() {
-         areaStream$1.point = areaPointFirst$1;
+       function areaRingStart() {
+         areaStream.point = areaPointFirst;
        }
 
-       function areaPointFirst$1(x, y) {
-         areaStream$1.point = areaPoint$1;
-         x00 = x0$1 = x, y00 = y0$1 = y;
+       function areaPointFirst(x, y) {
+         areaStream.point = areaPoint;
+         x00$2 = x0$3 = x, y00$2 = y0$3 = y;
        }
 
-       function areaPoint$1(x, y) {
-         areaRingSum$1.add(y0$1 * x - x0$1 * y);
-         x0$1 = x, y0$1 = y;
+       function areaPoint(x, y) {
+         areaRingSum.add(y0$3 * x - x0$3 * y);
+         x0$3 = x, y0$3 = y;
        }
 
-       function areaRingEnd$1() {
-         areaPoint$1(x00, y00);
+       function areaRingEnd() {
+         areaPoint(x00$2, y00$2);
        }
 
        var x0$2 = Infinity,
            y0$2 = x0$2,
            x1 = -x0$2,
            y1 = x1;
-       var boundsStream$1 = {
-         point: boundsPoint$1,
-         lineStart: noop,
-         lineEnd: noop,
-         polygonStart: noop,
-         polygonEnd: noop,
+       var boundsStream = {
+         point: boundsPoint,
+         lineStart: noop$1,
+         lineEnd: noop$1,
+         polygonStart: noop$1,
+         polygonEnd: noop$1,
          result: function result() {
            var bounds = [[x0$2, y0$2], [x1, y1]];
            x1 = y1 = -(y0$2 = x0$2 = Infinity);
          }
        };
 
-       function boundsPoint$1(x, y) {
+       function boundsPoint(x, y) {
          if (x < x0$2) x0$2 = x;
          if (x > x1) x1 = x;
          if (y < y0$2) y0$2 = y;
          if (y > y1) y1 = y;
        }
 
-       var X0$1 = 0,
-           Y0$1 = 0,
-           Z0$1 = 0,
-           X1$1 = 0,
-           Y1$1 = 0,
-           Z1$1 = 0,
-           X2$1 = 0,
-           Y2$1 = 0,
-           Z2$1 = 0,
+       var X0 = 0,
+           Y0 = 0,
+           Z0 = 0,
+           X1 = 0,
+           Y1 = 0,
+           Z1 = 0,
+           X2 = 0,
+           Y2 = 0,
+           Z2 = 0,
            x00$1,
            y00$1,
-           x0$3,
-           y0$3;
-       var centroidStream$1 = {
-         point: centroidPoint$1,
-         lineStart: centroidLineStart$1,
-         lineEnd: centroidLineEnd$1,
+           x0$1,
+           y0$1;
+       var centroidStream = {
+         point: centroidPoint,
+         lineStart: centroidLineStart,
+         lineEnd: centroidLineEnd,
          polygonStart: function polygonStart() {
-           centroidStream$1.lineStart = centroidRingStart$1;
-           centroidStream$1.lineEnd = centroidRingEnd$1;
+           centroidStream.lineStart = centroidRingStart;
+           centroidStream.lineEnd = centroidRingEnd;
          },
          polygonEnd: function polygonEnd() {
-           centroidStream$1.point = centroidPoint$1;
-           centroidStream$1.lineStart = centroidLineStart$1;
-           centroidStream$1.lineEnd = centroidLineEnd$1;
+           centroidStream.point = centroidPoint;
+           centroidStream.lineStart = centroidLineStart;
+           centroidStream.lineEnd = centroidLineEnd;
          },
          result: function result() {
-           var centroid = Z2$1 ? [X2$1 / Z2$1, Y2$1 / Z2$1] : Z1$1 ? [X1$1 / Z1$1, Y1$1 / Z1$1] : Z0$1 ? [X0$1 / Z0$1, Y0$1 / Z0$1] : [NaN, NaN];
-           X0$1 = Y0$1 = Z0$1 = X1$1 = Y1$1 = Z1$1 = X2$1 = Y2$1 = Z2$1 = 0;
+           var centroid = Z2 ? [X2 / Z2, Y2 / Z2] : Z1 ? [X1 / Z1, Y1 / Z1] : Z0 ? [X0 / Z0, Y0 / Z0] : [NaN, NaN];
+           X0 = Y0 = Z0 = X1 = Y1 = Z1 = X2 = Y2 = Z2 = 0;
            return centroid;
          }
        };
 
-       function centroidPoint$1(x, y) {
-         X0$1 += x;
-         Y0$1 += y;
-         ++Z0$1;
+       function centroidPoint(x, y) {
+         X0 += x;
+         Y0 += y;
+         ++Z0;
        }
 
-       function centroidLineStart$1() {
-         centroidStream$1.point = centroidPointFirstLine;
+       function centroidLineStart() {
+         centroidStream.point = centroidPointFirstLine;
        }
 
        function centroidPointFirstLine(x, y) {
-         centroidStream$1.point = centroidPointLine;
-         centroidPoint$1(x0$3 = x, y0$3 = y);
+         centroidStream.point = centroidPointLine;
+         centroidPoint(x0$1 = x, y0$1 = y);
        }
 
        function centroidPointLine(x, y) {
-         var dx = x - x0$3,
-             dy = y - y0$3,
-             z = sqrt$1(dx * dx + dy * dy);
-         X1$1 += z * (x0$3 + x) / 2;
-         Y1$1 += z * (y0$3 + y) / 2;
-         Z1$1 += z;
-         centroidPoint$1(x0$3 = x, y0$3 = y);
+         var dx = x - x0$1,
+             dy = y - y0$1,
+             z = sqrt(dx * dx + dy * dy);
+         X1 += z * (x0$1 + x) / 2;
+         Y1 += z * (y0$1 + y) / 2;
+         Z1 += z;
+         centroidPoint(x0$1 = x, y0$1 = y);
        }
 
-       function centroidLineEnd$1() {
-         centroidStream$1.point = centroidPoint$1;
+       function centroidLineEnd() {
+         centroidStream.point = centroidPoint;
        }
 
-       function centroidRingStart$1() {
-         centroidStream$1.point = centroidPointFirstRing;
+       function centroidRingStart() {
+         centroidStream.point = centroidPointFirstRing;
        }
 
-       function centroidRingEnd$1() {
+       function centroidRingEnd() {
          centroidPointRing(x00$1, y00$1);
        }
 
        function centroidPointFirstRing(x, y) {
-         centroidStream$1.point = centroidPointRing;
-         centroidPoint$1(x00$1 = x0$3 = x, y00$1 = y0$3 = y);
+         centroidStream.point = centroidPointRing;
+         centroidPoint(x00$1 = x0$1 = x, y00$1 = y0$1 = y);
        }
 
        function centroidPointRing(x, y) {
-         var dx = x - x0$3,
-             dy = y - y0$3,
-             z = sqrt$1(dx * dx + dy * dy);
-         X1$1 += z * (x0$3 + x) / 2;
-         Y1$1 += z * (y0$3 + y) / 2;
-         Z1$1 += z;
-         z = y0$3 * x - x0$3 * y;
-         X2$1 += z * (x0$3 + x);
-         Y2$1 += z * (y0$3 + y);
-         Z2$1 += z * 3;
-         centroidPoint$1(x0$3 = x, y0$3 = y);
+         var dx = x - x0$1,
+             dy = y - y0$1,
+             z = sqrt(dx * dx + dy * dy);
+         X1 += z * (x0$1 + x) / 2;
+         Y1 += z * (y0$1 + y) / 2;
+         Z1 += z;
+         z = y0$1 * x - x0$1 * y;
+         X2 += z * (x0$1 + x);
+         Y2 += z * (y0$1 + y);
+         Z2 += z * 3;
+         centroidPoint(x0$1 = x, y0$1 = y);
        }
 
        function PathContext(context) {
                }
            }
          },
-         result: noop
+         result: noop$1
        };
 
-       var lengthSum$1 = new Adder(),
+       var lengthSum = new Adder(),
            lengthRing,
-           x00$2,
-           y00$2,
-           x0$4,
-           y0$4;
-       var lengthStream$1 = {
-         point: noop,
+           x00,
+           y00,
+           x0,
+           y0;
+       var lengthStream = {
+         point: noop$1,
          lineStart: function lineStart() {
-           lengthStream$1.point = lengthPointFirst$1;
+           lengthStream.point = lengthPointFirst;
          },
          lineEnd: function lineEnd() {
-           if (lengthRing) lengthPoint$1(x00$2, y00$2);
-           lengthStream$1.point = noop;
+           if (lengthRing) lengthPoint(x00, y00);
+           lengthStream.point = noop$1;
          },
          polygonStart: function polygonStart() {
            lengthRing = true;
            lengthRing = null;
          },
          result: function result() {
-           var length = +lengthSum$1;
-           lengthSum$1 = new Adder();
+           var length = +lengthSum;
+           lengthSum = new Adder();
            return length;
          }
        };
 
-       function lengthPointFirst$1(x, y) {
-         lengthStream$1.point = lengthPoint$1;
-         x00$2 = x0$4 = x, y00$2 = y0$4 = y;
+       function lengthPointFirst(x, y) {
+         lengthStream.point = lengthPoint;
+         x00 = x0 = x, y00 = y0 = y;
        }
 
-       function lengthPoint$1(x, y) {
-         x0$4 -= x, y0$4 -= y;
-         lengthSum$1.add(sqrt$1(x0$4 * x0$4 + y0$4 * y0$4));
-         x0$4 = x, y0$4 = y;
+       function lengthPoint(x, y) {
+         x0 -= x, y0 -= y;
+         lengthSum.add(sqrt(x0 * x0 + y0 * y0));
+         x0 = x, y0 = y;
        }
 
        function PathString() {
          }
 
          path.area = function (object) {
-           d3_geoStream(object, projectionStream(areaStream$1));
-           return areaStream$1.result();
+           d3_geoStream(object, projectionStream(areaStream));
+           return areaStream.result();
          };
 
          path.measure = function (object) {
-           d3_geoStream(object, projectionStream(lengthStream$1));
-           return lengthStream$1.result();
+           d3_geoStream(object, projectionStream(lengthStream));
+           return lengthStream.result();
          };
 
          path.bounds = function (object) {
-           d3_geoStream(object, projectionStream(boundsStream$1));
-           return boundsStream$1.result();
+           d3_geoStream(object, projectionStream(boundsStream));
+           return boundsStream.result();
          };
 
          path.centroid = function (object) {
-           d3_geoStream(object, projectionStream(centroidStream$1));
-           return centroidStream$1.result();
+           d3_geoStream(object, projectionStream(centroidStream));
+           return centroidStream.result();
          };
 
          path.projection = function (_) {
-           return arguments.length ? (projectionStream = _ == null ? (projection = null, identity) : (projection = _).stream, path) : projection;
+           return arguments.length ? (projectionStream = _ == null ? (projection = null, identity$4) : (projection = _).stream, path) : projection;
          };
 
          path.context = function (_) {
 
        function d3_geoTransform (methods) {
          return {
-           stream: transformer(methods)
+           stream: transformer$1(methods)
          };
        }
-       function transformer(methods) {
+       function transformer$1(methods) {
          return function (stream) {
            var s = new TransformStream();
 
          var clip = projection.clipExtent && projection.clipExtent();
          projection.scale(150).translate([0, 0]);
          if (clip != null) projection.clipExtent(null);
-         d3_geoStream(object, projection.stream(boundsStream$1));
-         fitBounds(boundsStream$1.result());
+         d3_geoStream(object, projection.stream(boundsStream));
+         fitBounds(boundsStream.result());
          if (clip != null) projection.clipExtent(clip);
          return projection;
        }
        }
 
        function resampleNone(project) {
-         return transformer({
+         return transformer$1({
            point: function point(x, y) {
              x = project(x, y);
              this.stream.point(x[0], x[1]);
              var a = a0 + a1,
                  b = b0 + b1,
                  c = c0 + c1,
-                 m = sqrt$1(a * a + b * b + c * c),
+                 m = sqrt(a * a + b * b + c * c),
                  phi2 = asin(c /= m),
-                 lambda2 = abs$2(abs$2(c) - 1) < epsilon || abs$2(lambda0 - lambda1) < epsilon ? (lambda0 + lambda1) / 2 : atan2(b, a),
+                 lambda2 = abs$2(abs$2(c) - 1) < epsilon$1 || abs$2(lambda0 - lambda1) < epsilon$1 ? (lambda0 + lambda1) / 2 : atan2(b, a),
                  p = project(lambda2, phi2),
                  x2 = p[0],
                  y2 = p[1],
          };
        }
 
-       var transformRadians = transformer({
+       var transformRadians = transformer$1({
          point: function point(x, y) {
            this.stream.point(x * radians, y * radians);
          }
        });
 
        function transformRotate(rotate) {
-         return transformer({
+         return transformer$1({
            point: function point(x, y) {
              var r = rotate(x, y);
              return this.stream.point(r[0], r[1]);
              y0,
              x1,
              y1,
-             postclip = identity,
+             postclip = identity$4,
              // post-clip extent
          delta2 = 0.5,
              // precision
 
          function invert(point) {
            point = projectRotateTransform.invert(point[0], point[1]);
-           return point && [point[0] * degrees, point[1] * degrees];
+           return point && [point[0] * degrees$1, point[1] * degrees$1];
          }
 
          projection.stream = function (stream) {
          };
 
          projection.clipAngle = function (_) {
-           return arguments.length ? (preclip = +_ ? clipCircle(theta = _ * radians) : (theta = null, clipAntimeridian), reset()) : theta * degrees;
+           return arguments.length ? (preclip = +_ ? clipCircle(theta = _ * radians) : (theta = null, clipAntimeridian), reset()) : theta * degrees$1;
          };
 
          projection.clipExtent = function (_) {
-           return arguments.length ? (postclip = _ == null ? (x0 = y0 = x1 = y1 = null, identity) : clipRectangle(x0 = +_[0][0], y0 = +_[0][1], x1 = +_[1][0], y1 = +_[1][1]), reset()) : x0 == null ? null : [[x0, y0], [x1, y1]];
+           return arguments.length ? (postclip = _ == null ? (x0 = y0 = x1 = y1 = null, identity$4) : clipRectangle(x0 = +_[0][0], y0 = +_[0][1], x1 = +_[1][0], y1 = +_[1][1]), reset()) : x0 == null ? null : [[x0, y0], [x1, y1]];
          };
 
          projection.scale = function (_) {
          };
 
          projection.center = function (_) {
-           return arguments.length ? (lambda = _[0] % 360 * radians, phi = _[1] % 360 * radians, recenter()) : [lambda * degrees, phi * degrees];
+           return arguments.length ? (lambda = _[0] % 360 * radians, phi = _[1] % 360 * radians, recenter()) : [lambda * degrees$1, phi * degrees$1];
          };
 
          projection.rotate = function (_) {
-           return arguments.length ? (deltaLambda = _[0] % 360 * radians, deltaPhi = _[1] % 360 * radians, deltaGamma = _.length > 2 ? _[2] % 360 * radians : 0, recenter()) : [deltaLambda * degrees, deltaPhi * degrees, deltaGamma * degrees];
+           return arguments.length ? (deltaLambda = _[0] % 360 * radians, deltaPhi = _[1] % 360 * radians, deltaGamma = _.length > 2 ? _[2] % 360 * radians : 0, recenter()) : [deltaLambda * degrees$1, deltaPhi * degrees$1, deltaGamma * degrees$1];
          };
 
          projection.angle = function (_) {
-           return arguments.length ? (alpha = _ % 360 * radians, recenter()) : alpha * degrees;
+           return arguments.length ? (alpha = _ % 360 * radians, recenter()) : alpha * degrees$1;
          };
 
          projection.reflectX = function (_) {
          };
 
          projection.precision = function (_) {
-           return arguments.length ? (projectResample = resample(projectTransform, delta2 = _ * _), reset()) : sqrt$1(delta2);
+           return arguments.length ? (projectResample = resample(projectTransform, delta2 = _ * _), reset()) : sqrt(delta2);
          };
 
          projection.fitExtent = function (extent, object) {
        }
 
        mercatorRaw.invert = function (x, y) {
-         return [x, 2 * atan(exp(y)) - halfPi];
+         return [x, 2 * atan(exp$2(y)) - halfPi];
        };
 
        function mercator () {
              // clip extent
          kx = 1,
              ky = 1,
-             transform = transformer({
+             transform = transformer$1({
            point: function point(x, y) {
              var p = projection([x, y]);
              this.stream.point(p[0], p[1]);
            }
          }),
-             postclip = identity,
+             postclip = identity$4,
              cache,
              cacheStream;
 
          };
 
          projection.clipExtent = function (_) {
-           return arguments.length ? (postclip = _ == null ? (x0 = y0 = x1 = y1 = null, identity) : clipRectangle(x0 = +_[0][0], y0 = +_[0][1], x1 = +_[1][0], y1 = +_[1][1]), reset()) : x0 == null ? null : [[x0, y0], [x1, y1]];
+           return arguments.length ? (postclip = _ == null ? (x0 = y0 = x1 = y1 = null, identity$4) : clipRectangle(x0 = +_[0][0], y0 = +_[0][1], x1 = +_[1][0], y1 = +_[1][1]), reset()) : x0 == null ? null : [[x0, y0], [x1, y1]];
          };
 
          projection.scale = function (_) {
          };
 
          projection.angle = function (_) {
-           return arguments.length ? (alpha = _ % 360 * radians, sa = sin(alpha), ca = cos(alpha), reset()) : alpha * degrees;
+           return arguments.length ? (alpha = _ % 360 * radians, sa = sin(alpha), ca = cos(alpha), reset()) : alpha * degrees$1;
          };
 
          projection.reflectX = function (_) {
        // constants
        var TAU = 2 * Math.PI;
        var EQUATORIAL_RADIUS = 6356752.314245179;
-       var POLAR_RADIUS = 6378137.0;
+       var POLAR_RADIUS$1 = 6378137.0;
        function geoLatToMeters(dLat) {
-         return dLat * (TAU * POLAR_RADIUS / 360);
+         return dLat * (TAU * POLAR_RADIUS$1 / 360);
        }
        function geoLonToMeters(dLon, atLat) {
          return Math.abs(atLat) >= 90 ? 0 : dLon * (TAU * EQUATORIAL_RADIUS / 360) * Math.abs(Math.cos(atLat * (Math.PI / 180)));
        }
        function geoMetersToLat(m) {
-         return m / (TAU * POLAR_RADIUS / 360);
+         return m / (TAU * POLAR_RADIUS$1 / 360);
        }
        function geoMetersToLon(m, atLat) {
          return Math.abs(atLat) >= 90 ? 0 : m / (TAU * EQUATORIAL_RADIUS / 360) / Math.abs(Math.cos(atLat * (Math.PI / 180)));
        }
        function geoMetersToOffset(meters, tileSize) {
          tileSize = tileSize || 256;
-         return [meters[0] * tileSize / (TAU * EQUATORIAL_RADIUS), -meters[1] * tileSize / (TAU * POLAR_RADIUS)];
+         return [meters[0] * tileSize / (TAU * EQUATORIAL_RADIUS), -meters[1] * tileSize / (TAU * POLAR_RADIUS$1)];
        }
        function geoOffsetToMeters(offset, tileSize) {
          tileSize = tileSize || 256;
-         return [offset[0] * TAU * EQUATORIAL_RADIUS / tileSize, -offset[1] * TAU * POLAR_RADIUS / tileSize];
+         return [offset[0] * TAU * EQUATORIAL_RADIUS / tileSize, -offset[1] * TAU * POLAR_RADIUS$1 / tileSize];
        } // Equirectangular approximation of spherical distances on Earth
 
        function geoSphericalDistance(a, b) {
          }
        });
 
-       var $every$1 = arrayIteration.every;
+       var $every = arrayIteration.every;
 
 
-       var STRICT_METHOD$6 = arrayMethodIsStrict('every');
+       var STRICT_METHOD$1 = arrayMethodIsStrict('every');
 
        // `Array.prototype.every` method
        // https://tc39.es/ecma262/#sec-array.prototype.every
-       _export({ target: 'Array', proto: true, forced: !STRICT_METHOD$6 }, {
+       _export({ target: 'Array', proto: true, forced: !STRICT_METHOD$1 }, {
          every: function every(callbackfn /* , thisArg */) {
-           return $every$1(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
+           return $every(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
          }
        });
 
-       var $reduce$1 = arrayReduce.left;
+       var $reduce = arrayReduce.left;
 
 
 
 
-       var STRICT_METHOD$7 = arrayMethodIsStrict('reduce');
+       var STRICT_METHOD = arrayMethodIsStrict('reduce');
        // Chrome 80-82 has a critical bug
        // https://bugs.chromium.org/p/chromium/issues/detail?id=1049982
        var CHROME_BUG = !engineIsNode && engineV8Version > 79 && engineV8Version < 83;
 
        // `Array.prototype.reduce` method
        // https://tc39.es/ecma262/#sec-array.prototype.reduce
-       _export({ target: 'Array', proto: true, forced: !STRICT_METHOD$7 || CHROME_BUG }, {
+       _export({ target: 'Array', proto: true, forced: !STRICT_METHOD || CHROME_BUG }, {
          reduce: function reduce(callbackfn /* , initialValue */) {
-           return $reduce$1(this, callbackfn, arguments.length, arguments.length > 1 ? arguments[1] : undefined);
+           return $reduce(this, callbackfn, arguments.length, arguments.length > 1 ? arguments[1] : undefined);
          }
        });
 
 
          var x = 0;
          var y = 0;
-         if (point[0] > dimensions[0] - pad[1]) x = -10;
-         if (point[0] < pad[3]) x = 10;
-         if (point[1] > dimensions[1] - pad[2]) y = -10;
-         if (point[1] < pad[0]) y = 10;
+
+         if (point[0] > dimensions[0] - pad[1]) {
+           x = -10;
+         }
+
+         if (point[0] < pad[3]) {
+           x = 10;
+         }
+
+         if (point[1] > dimensions[1] - pad[2]) {
+           y = -10;
+         }
+
+         if (point[1] < pad[0]) {
+           y = 10;
+         }
 
          if (x || y) {
            return [x, y];
          }
        }
 
-       var noop$1 = {
+       var noop = {
          value: function value() {}
        };
 
-       function dispatch() {
+       function dispatch$8() {
          for (var i = 0, n = arguments.length, _ = {}, t; i < n; ++i) {
            if (!(t = arguments[i] + "") || t in _ || /[\s.]/.test(t)) throw new Error("illegal type: " + t);
            _[t] = [];
          }
 
-         return new Dispatch$1(_);
+         return new Dispatch(_);
        }
 
-       function Dispatch$1(_) {
+       function Dispatch(_) {
          this._ = _;
        }
 
-       function parseTypenames(typenames, types) {
+       function parseTypenames$1(typenames, types) {
          return typenames.trim().split(/^|\s+/).map(function (t) {
            var name = "",
                i = t.indexOf(".");
          });
        }
 
-       Dispatch$1.prototype = dispatch.prototype = {
-         constructor: Dispatch$1,
+       Dispatch.prototype = dispatch$8.prototype = {
+         constructor: Dispatch,
          on: function on(typename, callback) {
            var _ = this._,
-               T = parseTypenames(typename + "", _),
+               T = parseTypenames$1(typename + "", _),
                t,
                i = -1,
                n = T.length; // If no callback was specified, return the callback of the given type and name.
 
            if (arguments.length < 2) {
              while (++i < n) {
-               if ((t = (typename = T[i]).type) && (t = get$3(_[t], typename.name))) return t;
+               if ((t = (typename = T[i]).type) && (t = get$2(_[t], typename.name))) return t;
              }
 
              return;
            if (callback != null && typeof callback !== "function") throw new Error("invalid callback: " + callback);
 
            while (++i < n) {
-             if (t = (typename = T[i]).type) _[t] = set$3(_[t], typename.name, callback);else if (callback == null) for (t in _) {
-               _[t] = set$3(_[t], typename.name, null);
+             if (t = (typename = T[i]).type) _[t] = set$1(_[t], typename.name, callback);else if (callback == null) for (t in _) {
+               _[t] = set$1(_[t], typename.name, null);
              }
            }
 
              copy[t] = _[t].slice();
            }
 
-           return new Dispatch$1(copy);
+           return new Dispatch(copy);
          },
          call: function call(type, that) {
            if ((n = arguments.length - 2) > 0) for (var args = new Array(n), i = 0, n, t; i < n; ++i) {
          }
        };
 
-       function get$3(type, name) {
+       function get$2(type, name) {
          for (var i = 0, n = type.length, c; i < n; ++i) {
            if ((c = type[i]).name === name) {
              return c.value;
          }
        }
 
-       function set$3(type, name, callback) {
+       function set$1(type, name, callback) {
          for (var i = 0, n = type.length; i < n; ++i) {
            if (type[i].name === name) {
-             type[i] = noop$1, type = type.slice(0, i).concat(type.slice(i + 1));
+             type[i] = noop, type = type.slice(0, i).concat(type.slice(i + 1));
              break;
            }
          }
            }
          }
 
-         return new Selection(subgroups, this._parents);
+         return new Selection$1(subgroups, this._parents);
        }
 
        function array (x) {
            }
          }
 
-         return new Selection(subgroups, parents);
+         return new Selection$1(subgroups, parents);
        }
 
-       var $find$1 = arrayIteration.find;
+       var $find = arrayIteration.find;
 
 
        var FIND = 'find';
-       var SKIPS_HOLES = true;
+       var SKIPS_HOLES$1 = true;
 
        // Shouldn't skip holes
-       if (FIND in []) Array(1)[FIND](function () { SKIPS_HOLES = false; });
+       if (FIND in []) Array(1)[FIND](function () { SKIPS_HOLES$1 = false; });
 
        // `Array.prototype.find` method
        // https://tc39.es/ecma262/#sec-array.prototype.find
-       _export({ target: 'Array', proto: true, forced: SKIPS_HOLES }, {
+       _export({ target: 'Array', proto: true, forced: SKIPS_HOLES$1 }, {
          find: function find(callbackfn /* , that = undefined */) {
-           return $find$1(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
+           return $find(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
          }
        });
 
          };
        }
 
-       var find$1 = Array.prototype.find;
+       var find = Array.prototype.find;
 
        function childFind(match) {
          return function () {
-           return find$1.call(this.children, match);
+           return find.call(this.children, match);
          };
        }
 
            }
          }
 
-         return new Selection(subgroups, this._parents);
+         return new Selection$1(subgroups, this._parents);
        }
 
        function sparse (update) {
        }
 
        function selection_enter () {
-         return new Selection(this._enter || this._groups.map(sparse), this._parents);
+         return new Selection$1(this._enter || this._groups.map(sparse), this._parents);
        }
        function EnterNode(parent, datum) {
          this.ownerDocument = parent.ownerDocument;
          }
        };
 
-       function constant (x) {
+       function constant$3 (x) {
          return function () {
            return x;
          };
          var bind = key ? bindKey : bindIndex,
              parents = this._parents,
              groups = this._groups;
-         if (typeof value !== "function") value = constant(value);
+         if (typeof value !== "function") value = constant$3(value);
 
          for (var m = groups.length, update = new Array(m), enter = new Array(m), exit = new Array(m), j = 0; j < m; ++j) {
            var parent = parents[j],
            }
          }
 
-         update = new Selection(update, parents);
+         update = new Selection$1(update, parents);
          update._enter = enter;
          update._exit = exit;
          return update;
        }
 
        function selection_exit () {
-         return new Selection(this._exit || this._groups.map(sparse), this._parents);
+         return new Selection$1(this._exit || this._groups.map(sparse), this._parents);
        }
 
        function selection_join (onenter, onupdate, onexit) {
        }
 
        function selection_merge (selection) {
-         if (!(selection instanceof Selection)) throw new Error("invalid merge");
+         if (!(selection instanceof Selection$1)) throw new Error("invalid merge");
 
          for (var groups0 = this._groups, groups1 = selection._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
            for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
            merges[j] = groups0[j];
          }
 
-         return new Selection(merges, this._parents);
+         return new Selection$1(merges, this._parents);
        }
 
        function selection_order () {
            sortgroup.sort(compareNode);
          }
 
-         return new Selection(sortgroups, this._parents).order();
+         return new Selection$1(sortgroups, this._parents).order();
        }
 
        function ascending(a, b) {
          return this;
        }
 
-       function attrRemove(name) {
+       function attrRemove$1(name) {
          return function () {
            this.removeAttribute(name);
          };
        }
 
-       function attrRemoveNS(fullname) {
+       function attrRemoveNS$1(fullname) {
          return function () {
            this.removeAttributeNS(fullname.space, fullname.local);
          };
        }
 
-       function attrConstant(name, value) {
+       function attrConstant$1(name, value) {
          return function () {
            this.setAttribute(name, value);
          };
        }
 
-       function attrConstantNS(fullname, value) {
+       function attrConstantNS$1(fullname, value) {
          return function () {
            this.setAttributeNS(fullname.space, fullname.local, value);
          };
        }
 
-       function attrFunction(name, value) {
+       function attrFunction$1(name, value) {
          return function () {
            var v = value.apply(this, arguments);
            if (v == null) this.removeAttribute(name);else this.setAttribute(name, v);
          };
        }
 
-       function attrFunctionNS(fullname, value) {
+       function attrFunctionNS$1(fullname, value) {
          return function () {
            var v = value.apply(this, arguments);
            if (v == null) this.removeAttributeNS(fullname.space, fullname.local);else this.setAttributeNS(fullname.space, fullname.local, v);
            return fullname.local ? node.getAttributeNS(fullname.space, fullname.local) : node.getAttribute(fullname);
          }
 
-         return this.each((value == null ? fullname.local ? attrRemoveNS : attrRemove : typeof value === "function" ? fullname.local ? attrFunctionNS : attrFunction : fullname.local ? attrConstantNS : attrConstant)(fullname, value));
+         return this.each((value == null ? fullname.local ? attrRemoveNS$1 : attrRemove$1 : typeof value === "function" ? fullname.local ? attrFunctionNS$1 : attrFunction$1 : fullname.local ? attrConstantNS$1 : attrConstant$1)(fullname, value));
        }
 
        function defaultView (node) {
          || node.defaultView; // node is a Document
        }
 
-       function styleRemove(name) {
+       function styleRemove$1(name) {
          return function () {
            this.style.removeProperty(name);
          };
        }
 
-       function styleConstant(name, value, priority) {
+       function styleConstant$1(name, value, priority) {
          return function () {
            this.style.setProperty(name, value, priority);
          };
        }
 
-       function styleFunction(name, value, priority) {
+       function styleFunction$1(name, value, priority) {
          return function () {
            var v = value.apply(this, arguments);
            if (v == null) this.style.removeProperty(name);else this.style.setProperty(name, v, priority);
        }
 
        function selection_style (name, value, priority) {
-         return arguments.length > 1 ? this.each((value == null ? styleRemove : typeof value === "function" ? styleFunction : styleConstant)(name, value, priority == null ? "" : priority)) : styleValue(this.node(), name);
+         return arguments.length > 1 ? this.each((value == null ? styleRemove$1 : typeof value === "function" ? styleFunction$1 : styleConstant$1)(name, value, priority == null ? "" : priority)) : styleValue(this.node(), name);
        }
        function styleValue(node, name) {
          return node.style.getPropertyValue(name) || defaultView(node).getComputedStyle(node, null).getPropertyValue(name);
          this.textContent = "";
        }
 
-       function textConstant(value) {
+       function textConstant$1(value) {
          return function () {
            this.textContent = value;
          };
        }
 
-       function textFunction(value) {
+       function textFunction$1(value) {
          return function () {
            var v = value.apply(this, arguments);
            this.textContent = v == null ? "" : v;
        }
 
        function selection_text (value) {
-         return arguments.length ? this.each(value == null ? textRemove : (typeof value === "function" ? textFunction : textConstant)(value)) : this.node().textContent;
+         return arguments.length ? this.each(value == null ? textRemove : (typeof value === "function" ? textFunction$1 : textConstant$1)(value)) : this.node().textContent;
        }
 
        function htmlRemove() {
          });
        }
 
-       function remove() {
+       function remove$7() {
          var parent = this.parentNode;
          if (parent) parent.removeChild(this);
        }
 
        function selection_remove () {
-         return this.each(remove);
+         return this.each(remove$7);
        }
 
        function selection_cloneShallow() {
          };
        }
 
-       function parseTypenames$1(typenames) {
+       function parseTypenames(typenames) {
          return typenames.trim().split(/^|\s+/).map(function (t) {
            var name = "",
                i = t.indexOf(".");
        }
 
        function selection_on (typename, value, options) {
-         var typenames = parseTypenames$1(typename + ""),
+         var typenames = parseTypenames(typename + ""),
              i,
              n = typenames.length,
              t;
          return this;
        }
 
-       function dispatchEvent$1(node, type, params) {
+       function dispatchEvent(node, type, params) {
          var window = defaultView(node),
              event = window.CustomEvent;
 
 
        function dispatchConstant(type, params) {
          return function () {
-           return dispatchEvent$1(this, type, params);
+           return dispatchEvent(this, type, params);
          };
        }
 
        function dispatchFunction(type, params) {
          return function () {
-           return dispatchEvent$1(this, type, params.apply(this, arguments));
+           return dispatchEvent(this, type, params.apply(this, arguments));
          };
        }
 
          return this.each((typeof params === "function" ? dispatchFunction : dispatchConstant)(type, params));
        }
 
-       var _marked$2 = /*#__PURE__*/regeneratorRuntime.mark(_callee);
+       var _marked = /*#__PURE__*/regeneratorRuntime.mark(_callee);
 
        function _callee() {
          var groups, j, m, group, i, n, node;
                  return _context.stop();
              }
            }
-         }, _marked$2, this);
+         }, _marked, this);
        }
 
-       var root = [null];
-       function Selection(groups, parents) {
+       var root$1 = [null];
+       function Selection$1(groups, parents) {
          this._groups = groups;
          this._parents = parents;
        }
 
        function selection() {
-         return new Selection([[document.documentElement]], root);
+         return new Selection$1([[document.documentElement]], root$1);
        }
 
        function selection_selection() {
          return this;
        }
 
-       Selection.prototype = selection.prototype = _defineProperty({
-         constructor: Selection,
+       Selection$1.prototype = selection.prototype = _defineProperty({
+         constructor: Selection$1,
          select: selection_select,
          selectAll: selection_selectAll,
          selectChild: selection_selectChild,
        }, Symbol.iterator, _callee);
 
        function select (selector) {
-         return typeof selector === "string" ? new Selection([[document.querySelector(selector)]], [document.documentElement]) : new Selection([[selector]], root);
+         return typeof selector === "string" ? new Selection$1([[document.querySelector(selector)]], [document.documentElement]) : new Selection$1([[selector]], root$1);
        }
 
        function sourceEvent (event) {
        }
 
        function selectAll (selector) {
-         return typeof selector === "string" ? new Selection([document.querySelectorAll(selector)], [document.documentElement]) : new Selection([selector == null ? [] : array(selector)], root);
+         return typeof selector === "string" ? new Selection$1([document.querySelectorAll(selector)], [document.documentElement]) : new Selection$1([selector == null ? [] : array(selector)], root$1);
        }
 
-       function nopropagation(event) {
+       function nopropagation$1(event) {
          event.stopImmediatePropagation();
        }
-       function noevent (event) {
+       function noevent$1 (event) {
          event.preventDefault();
          event.stopImmediatePropagation();
        }
 
        function dragDisable (view) {
          var root = view.document.documentElement,
-             selection = select(view).on("dragstart.drag", noevent, true);
+             selection = select(view).on("dragstart.drag", noevent$1, true);
 
          if ("onselectstart" in root) {
-           selection.on("selectstart.drag", noevent, true);
+           selection.on("selectstart.drag", noevent$1, true);
          } else {
            root.__noselect = root.style.MozUserSelect;
            root.style.MozUserSelect = "none";
              selection = select(view).on("dragstart.drag", null);
 
          if (noclick) {
-           selection.on("click.drag", noevent, true);
+           selection.on("click.drag", noevent$1, true);
            setTimeout(function () {
              selection.on("click.drag", null);
            }, 0);
          }
        }
 
-       var constant$1 = (function (x) {
+       var constant$2 = (function (x) {
          return function () {
            return x;
          };
          return value === this._ ? this : value;
        };
 
-       function defaultFilter(event) {
+       function defaultFilter$2(event) {
          return !event.ctrlKey && !event.button;
        }
 
          } : d;
        }
 
-       function defaultTouchable() {
+       function defaultTouchable$1() {
          return navigator.maxTouchPoints || "ontouchstart" in this;
        }
 
        function d3_drag () {
-         var filter = defaultFilter,
+         var filter = defaultFilter$2,
              container = defaultContainer,
              subject = defaultSubject,
-             touchable = defaultTouchable,
+             touchable = defaultTouchable$1,
              gestures = {},
-             listeners = dispatch("start", "drag", "end"),
+             listeners = dispatch$8("start", "drag", "end"),
              active = 0,
              mousedownx,
              mousedowny,
            if (!gesture) return;
            select(event.view).on("mousemove.drag", mousemoved, true).on("mouseup.drag", mouseupped, true);
            dragDisable(event.view);
-           nopropagation(event);
+           nopropagation$1(event);
            mousemoving = false;
            mousedownx = event.clientX;
            mousedowny = event.clientY;
          }
 
          function mousemoved(event) {
-           noevent(event);
+           noevent$1(event);
 
            if (!mousemoving) {
              var dx = event.clientX - mousedownx,
          function mouseupped(event) {
            select(event.view).on("mousemove.drag mouseup.drag", null);
            yesdrag(event.view, mousemoving);
-           noevent(event);
+           noevent$1(event);
            gestures.mouse("end", event);
          }
 
 
            for (i = 0; i < n; ++i) {
              if (gesture = beforestart(this, c, event, d, touches[i].identifier, touches[i])) {
-               nopropagation(event);
+               nopropagation$1(event);
                gesture("start", event, touches[i]);
              }
            }
 
            for (i = 0; i < n; ++i) {
              if (gesture = gestures[touches[i].identifier]) {
-               noevent(event);
+               noevent$1(event);
                gesture("drag", event, touches[i]);
              }
            }
 
            for (i = 0; i < n; ++i) {
              if (gesture = gestures[touches[i].identifier]) {
-               nopropagation(event);
+               nopropagation$1(event);
                gesture("end", event, touches[i]);
              }
            }
          }
 
          drag.filter = function (_) {
-           return arguments.length ? (filter = typeof _ === "function" ? _ : constant$1(!!_), drag) : filter;
+           return arguments.length ? (filter = typeof _ === "function" ? _ : constant$2(!!_), drag) : filter;
          };
 
          drag.container = function (_) {
-           return arguments.length ? (container = typeof _ === "function" ? _ : constant$1(_), drag) : container;
+           return arguments.length ? (container = typeof _ === "function" ? _ : constant$2(_), drag) : container;
          };
 
          drag.subject = function (_) {
-           return arguments.length ? (subject = typeof _ === "function" ? _ : constant$1(_), drag) : subject;
+           return arguments.length ? (subject = typeof _ === "function" ? _ : constant$2(_), drag) : subject;
          };
 
          drag.touchable = function (_) {
-           return arguments.length ? (touchable = typeof _ === "function" ? _ : constant$1(!!_), drag) : touchable;
+           return arguments.length ? (touchable = typeof _ === "function" ? _ : constant$2(!!_), drag) : touchable;
          };
 
          drag.on = function () {
          return drag;
        }
 
-       var defineProperty$8 = objectDefineProperty.f;
+       var defineProperty$1 = objectDefineProperty.f;
        var getOwnPropertyNames$1 = objectGetOwnPropertyNames.f;
 
 
 
 
 
-       var setInternalState$8 = internalState.set;
+
+       var enforceInternalState = internalState.enforce;
+
+
 
 
 
        var MATCH$1 = wellKnownSymbol('match');
-       var NativeRegExp = global_1.RegExp;
-       var RegExpPrototype$1 = NativeRegExp.prototype;
+       var NativeRegExp = global$2.RegExp;
+       var RegExpPrototype = NativeRegExp.prototype;
+       // TODO: Use only propper RegExpIdentifierName
+       var IS_NCG = /^\?<[^\s\d!#%&*+<=>@^][^\s!#%&*+<=>@^]*>/;
        var re1 = /a/g;
        var re2 = /a/g;
 
        // "new" should create a new object, old webkit bug
        var CORRECT_NEW = new NativeRegExp(re1) !== re1;
 
-       var UNSUPPORTED_Y$2 = regexpStickyHelpers.UNSUPPORTED_Y;
+       var UNSUPPORTED_Y = regexpStickyHelpers.UNSUPPORTED_Y;
+
+       var BASE_FORCED = descriptors &&
+         (!CORRECT_NEW || UNSUPPORTED_Y || regexpUnsupportedDotAll || regexpUnsupportedNcg || fails(function () {
+           re2[MATCH$1] = false;
+           // RegExp constructor can alter flags and IsRegExp works correct with @@match
+           return NativeRegExp(re1) != re1 || NativeRegExp(re2) == re2 || NativeRegExp(re1, 'i') != '/a/i';
+         }));
+
+       var handleDotAll = function (string) {
+         var length = string.length;
+         var index = 0;
+         var result = '';
+         var brackets = false;
+         var chr;
+         for (; index <= length; index++) {
+           chr = string.charAt(index);
+           if (chr === '\\') {
+             result += chr + string.charAt(++index);
+             continue;
+           }
+           if (!brackets && chr === '.') {
+             result += '[\\s\\S]';
+           } else {
+             if (chr === '[') {
+               brackets = true;
+             } else if (chr === ']') {
+               brackets = false;
+             } result += chr;
+           }
+         } return result;
+       };
 
-       var FORCED$b = descriptors && isForced_1('RegExp', (!CORRECT_NEW || UNSUPPORTED_Y$2 || fails(function () {
-         re2[MATCH$1] = false;
-         // RegExp constructor can alter flags and IsRegExp works correct with @@match
-         return NativeRegExp(re1) != re1 || NativeRegExp(re2) == re2 || NativeRegExp(re1, 'i') != '/a/i';
-       })));
+       var handleNCG = function (string) {
+         var length = string.length;
+         var index = 0;
+         var result = '';
+         var named = [];
+         var names = {};
+         var brackets = false;
+         var ncg = false;
+         var groupid = 0;
+         var groupname = '';
+         var chr;
+         for (; index <= length; index++) {
+           chr = string.charAt(index);
+           if (chr === '\\') {
+             chr = chr + string.charAt(++index);
+           } else if (chr === ']') {
+             brackets = false;
+           } else if (!brackets) switch (true) {
+             case chr === '[':
+               brackets = true;
+               break;
+             case chr === '(':
+               if (IS_NCG.test(string.slice(index + 1))) {
+                 index += 2;
+                 ncg = true;
+               }
+               result += chr;
+               groupid++;
+               continue;
+             case chr === '>' && ncg:
+               if (groupname === '' || has$1(names, groupname)) {
+                 throw new SyntaxError('Invalid capture group name');
+               }
+               names[groupname] = true;
+               named.push([groupname, groupid]);
+               ncg = false;
+               groupname = '';
+               continue;
+           }
+           if (ncg) groupname += chr;
+           else result += chr;
+         } return [result, named];
+       };
 
        // `RegExp` constructor
        // https://tc39.es/ecma262/#sec-regexp-constructor
-       if (FORCED$b) {
+       if (isForced_1('RegExp', BASE_FORCED)) {
          var RegExpWrapper = function RegExp(pattern, flags) {
            var thisIsRegExp = this instanceof RegExpWrapper;
            var patternIsRegExp = isRegexp(pattern);
            var flagsAreUndefined = flags === undefined;
-           var sticky;
+           var groups = [];
+           var rawPattern = pattern;
+           var rawFlags, dotAll, sticky, handled, result, state;
 
-           if (!thisIsRegExp && patternIsRegExp && pattern.constructor === RegExpWrapper && flagsAreUndefined) {
+           if (!thisIsRegExp && patternIsRegExp && flagsAreUndefined && pattern.constructor === RegExpWrapper) {
              return pattern;
            }
 
-           if (CORRECT_NEW) {
-             if (patternIsRegExp && !flagsAreUndefined) pattern = pattern.source;
-           } else if (pattern instanceof RegExpWrapper) {
-             if (flagsAreUndefined) flags = regexpFlags.call(pattern);
+           if (patternIsRegExp || pattern instanceof RegExpWrapper) {
              pattern = pattern.source;
+             if (flagsAreUndefined) flags = 'flags' in rawPattern ? rawPattern.flags : regexpFlags.call(rawPattern);
+           }
+
+           pattern = pattern === undefined ? '' : String(pattern);
+           flags = flags === undefined ? '' : String(flags);
+           rawPattern = pattern;
+
+           if (regexpUnsupportedDotAll && 'dotAll' in re1) {
+             dotAll = !!flags && flags.indexOf('s') > -1;
+             if (dotAll) flags = flags.replace(/s/g, '');
            }
 
-           if (UNSUPPORTED_Y$2) {
+           rawFlags = flags;
+
+           if (UNSUPPORTED_Y && 'sticky' in re1) {
              sticky = !!flags && flags.indexOf('y') > -1;
              if (sticky) flags = flags.replace(/y/g, '');
            }
 
-           var result = inheritIfRequired(
-             CORRECT_NEW ? new NativeRegExp(pattern, flags) : NativeRegExp(pattern, flags),
-             thisIsRegExp ? this : RegExpPrototype$1,
-             RegExpWrapper
-           );
+           if (regexpUnsupportedNcg) {
+             handled = handleNCG(pattern);
+             pattern = handled[0];
+             groups = handled[1];
+           }
+
+           result = inheritIfRequired(NativeRegExp(pattern, flags), thisIsRegExp ? this : RegExpPrototype, RegExpWrapper);
+
+           if (dotAll || sticky || groups.length) {
+             state = enforceInternalState(result);
+             if (dotAll) {
+               state.dotAll = true;
+               state.raw = RegExpWrapper(handleDotAll(pattern), rawFlags);
+             }
+             if (sticky) state.sticky = true;
+             if (groups.length) state.groups = groups;
+           }
 
-           if (UNSUPPORTED_Y$2 && sticky) setInternalState$8(result, { sticky: sticky });
+           if (pattern !== rawPattern) try {
+             // fails in old engines, but we have no alternatives for unsupported regex syntax
+             createNonEnumerableProperty(result, 'source', rawPattern === '' ? '(?:)' : rawPattern);
+           } catch (error) { /* empty */ }
 
            return result;
          };
+
          var proxy = function (key) {
-           key in RegExpWrapper || defineProperty$8(RegExpWrapper, key, {
+           key in RegExpWrapper || defineProperty$1(RegExpWrapper, key, {
              configurable: true,
              get: function () { return NativeRegExp[key]; },
              set: function (it) { NativeRegExp[key] = it; }
            });
          };
-         var keys$2 = getOwnPropertyNames$1(NativeRegExp);
-         var index = 0;
-         while (keys$2.length > index) proxy(keys$2[index++]);
-         RegExpPrototype$1.constructor = RegExpWrapper;
-         RegExpWrapper.prototype = RegExpPrototype$1;
-         redefine(global_1, 'RegExp', RegExpWrapper);
+
+         for (var keys$1 = getOwnPropertyNames$1(NativeRegExp), index$1 = 0; keys$1.length > index$1;) {
+           proxy(keys$1[index$1++]);
+         }
+
+         RegExpPrototype.constructor = RegExpWrapper;
+         RegExpWrapper.prototype = RegExpPrototype;
+         redefine(global$2, 'RegExp', RegExpWrapper);
        }
 
        // https://tc39.es/ecma262/#sec-get-regexp-@@species
          constructor.prototype = factory.prototype = prototype;
          prototype.constructor = constructor;
        }
-       function extend(parent, definition) {
+       function extend$3(parent, definition) {
          var prototype = Object.create(parent.prototype);
 
          for (var key in definition) {
          this.b = +b;
          this.opacity = +opacity;
        }
-       define(Rgb, rgb, extend(Color, {
+       define(Rgb, rgb, extend$3(Color, {
          brighter: function brighter(k) {
            k = k == null ? _brighter : Math.pow(_brighter, k);
            return new Rgb(this.r * k, this.g * k, this.b * k, this.opacity);
        }));
 
        function rgb_formatHex() {
-         return "#" + hex$2(this.r) + hex$2(this.g) + hex$2(this.b);
+         return "#" + hex$1(this.r) + hex$1(this.g) + hex$1(this.b);
        }
 
        function rgb_formatRgb() {
          return (a === 1 ? "rgb(" : "rgba(") + Math.max(0, Math.min(255, Math.round(this.r) || 0)) + ", " + Math.max(0, Math.min(255, Math.round(this.g) || 0)) + ", " + Math.max(0, Math.min(255, Math.round(this.b) || 0)) + (a === 1 ? ")" : ", " + a + ")");
        }
 
-       function hex$2(value) {
+       function hex$1(value) {
          value = Math.max(0, Math.min(255, Math.round(value) || 0));
          return (value < 16 ? "0" : "") + value.toString(16);
        }
          this.opacity = +opacity;
        }
 
-       define(Hsl, hsl, extend(Color, {
+       define(Hsl, hsl, extend$3(Color, {
          brighter: function brighter(k) {
            k = k == null ? _brighter : Math.pow(_brighter, k);
            return new Hsl(this.h, this.s, this.l * k, this.opacity);
          return (h < 60 ? m1 + (m2 - m1) * h / 60 : h < 180 ? m2 : h < 240 ? m1 + (m2 - m1) * (240 - h) / 60 : m1) * 255;
        }
 
-       var constant$2 = (function (x) {
+       var constant$1 = (function (x) {
          return function () {
            return x;
          };
        });
 
-       function linear(a, d) {
+       function linear$2(a, d) {
          return function (t) {
            return a + t * d;
          };
        }
        function gamma(y) {
          return (y = +y) === 1 ? nogamma : function (a, b) {
-           return b - a ? exponential(a, b, y) : constant$2(isNaN(a) ? b : a);
+           return b - a ? exponential(a, b, y) : constant$1(isNaN(a) ? b : a);
          };
        }
        function nogamma(a, b) {
          var d = b - a;
-         return d ? linear(a, d) : constant$2(isNaN(a) ? b : a);
+         return d ? linear$2(a, d) : constant$1(isNaN(a) ? b : a);
        }
 
        var d3_interpolateRgb = (function rgbGamma(y) {
              i;
 
          for (i = 0; i < na; ++i) {
-           x[i] = interpolate(a[i], b[i]);
+           x[i] = interpolate$1(a[i], b[i]);
          }
 
          for (; i < nb; ++i) {
 
          for (k in b) {
            if (k in a) {
-             i[k] = interpolate(a[k], b[k]);
+             i[k] = interpolate$1(a[k], b[k]);
            } else {
              c[k] = b[k];
            }
          });
        }
 
-       function interpolate (a, b) {
+       function interpolate$1 (a, b) {
          var t = _typeof(b),
              c;
 
-         return b == null || t === "boolean" ? constant$2(b) : (t === "number" ? d3_interpolateNumber : t === "string" ? (c = color(b)) ? (b = c, d3_interpolateRgb) : interpolateString : b instanceof color ? d3_interpolateRgb : b instanceof Date ? date : isNumberArray(b) ? numberArray : Array.isArray(b) ? genericArray : typeof b.valueOf !== "function" && typeof b.toString !== "function" || isNaN(b) ? object : d3_interpolateNumber)(a, b);
+         return b == null || t === "boolean" ? constant$1(b) : (t === "number" ? d3_interpolateNumber : t === "string" ? (c = color(b)) ? (b = c, d3_interpolateRgb) : interpolateString : b instanceof color ? d3_interpolateRgb : b instanceof Date ? date : isNumberArray(b) ? numberArray : Array.isArray(b) ? genericArray : typeof b.valueOf !== "function" && typeof b.toString !== "function" || isNaN(b) ? object : d3_interpolateNumber)(a, b);
        }
 
        function interpolateRound (a, b) {
          };
        }
 
-       var degrees$1 = 180 / Math.PI;
-       var identity$1 = {
+       var degrees = 180 / Math.PI;
+       var identity$3 = {
          translateX: 0,
          translateY: 0,
          rotate: 0,
          return {
            translateX: e,
            translateY: f,
-           rotate: Math.atan2(b, a) * degrees$1,
-           skewX: Math.atan(skewX) * degrees$1,
+           rotate: Math.atan2(b, a) * degrees,
+           skewX: Math.atan(skewX) * degrees,
            scaleX: scaleX,
            scaleY: scaleY
          };
 
        function parseCss(value) {
          var m = new (typeof DOMMatrix === "function" ? DOMMatrix : WebKitCSSMatrix)(value + "");
-         return m.isIdentity ? identity$1 : decompose(m.a, m.b, m.c, m.d, m.e, m.f);
+         return m.isIdentity ? identity$3 : decompose(m.a, m.b, m.c, m.d, m.e, m.f);
        }
        function parseSvg(value) {
-         if (value == null) return identity$1;
+         if (value == null) return identity$3;
          if (!svgNode) svgNode = document.createElementNS("http://www.w3.org/2000/svg", "g");
          svgNode.setAttribute("transform", value);
-         if (!(value = svgNode.transform.baseVal.consolidate())) return identity$1;
+         if (!(value = svgNode.transform.baseVal.consolidate())) return identity$3;
          value = value.matrix;
          return decompose(value.a, value.b, value.c, value.d, value.e, value.f);
        }
        var interpolateTransformCss = interpolateTransform(parseCss, "px, ", "px)", "deg)");
        var interpolateTransformSvg = interpolateTransform(parseSvg, ", ", ")", ")");
 
-       var epsilon2$1 = 1e-12;
+       var epsilon2 = 1e-12;
 
        function cosh(x) {
          return ((x = Math.exp(x)) + 1 / x) / 2;
                i,
                S; // Special case for u0 ≅ u1.
 
-           if (d2 < epsilon2$1) {
+           if (d2 < epsilon2) {
              S = Math.log(w1 / w0) / rho;
 
              i = function i(t) {
            setFrame = (typeof window === "undefined" ? "undefined" : _typeof(window)) === "object" && window.requestAnimationFrame ? window.requestAnimationFrame.bind(window) : function (f) {
          setTimeout(f, 17);
        };
-       function now() {
+       function now$1() {
          return clockNow || (setFrame(clearNow), clockNow = clock.now() + clockSkew);
        }
 
          constructor: Timer,
          restart: function restart(callback, delay, time) {
            if (typeof callback !== "function") throw new TypeError("callback is not a function");
-           time = (time == null ? now() : +time) + (delay == null ? 0 : +delay);
+           time = (time == null ? now$1() : +time) + (delay == null ? 0 : +delay);
 
            if (!this._next && taskTail !== this) {
              if (taskTail) taskTail._next = this;else taskHead = this;
          return t;
        }
        function timerFlush() {
-         now(); // Get the current time, if not already set.
+         now$1(); // Get the current time, if not already set.
 
          ++frame; // Pretend we’ve set an alarm, if we haven’t already.
 
          return t;
        }
 
-       var emptyOn = dispatch("start", "end", "cancel", "interrupt");
+       var emptyOn = dispatch$8("start", "end", "cancel", "interrupt");
        var emptyTween = [];
        var CREATED = 0;
        var SCHEDULED = 1;
        function schedule (node, name, id, index, group, timing) {
          var schedules = node.__transition;
          if (!schedules) node.__transition = {};else if (id in schedules) return;
-         create(node, id, {
+         create$2(node, id, {
            name: name,
            index: index,
            // For context during callback.
          });
        }
        function init(node, id) {
-         var schedule = get$4(node, id);
+         var schedule = get$1(node, id);
          if (schedule.state > CREATED) throw new Error("too late; already scheduled");
          return schedule;
        }
-       function set$4(node, id) {
-         var schedule = get$4(node, id);
+       function set(node, id) {
+         var schedule = get$1(node, id);
          if (schedule.state > STARTED) throw new Error("too late; already running");
          return schedule;
        }
-       function get$4(node, id) {
+       function get$1(node, id) {
          var schedule = node.__transition;
          if (!schedule || !(schedule = schedule[id])) throw new Error("transition not found");
          return schedule;
        }
 
-       function create(node, id, self) {
+       function create$2(node, id, self) {
          var schedules = node.__transition,
              tween; // Initialize the self timer when the transition is created.
          // Note the actual delay is not known until the first callback!
        function tweenRemove(id, name) {
          var tween0, tween1;
          return function () {
-           var schedule = set$4(this, id),
+           var schedule = set(this, id),
                tween = schedule.tween; // If this node shared tween with the previous node,
            // just assign the updated shared tween and we’re done!
            // Otherwise, copy-on-write.
          var tween0, tween1;
          if (typeof value !== "function") throw new Error();
          return function () {
-           var schedule = set$4(this, id),
+           var schedule = set(this, id),
                tween = schedule.tween; // If this node shared tween with the previous node,
            // just assign the updated shared tween and we’re done!
            // Otherwise, copy-on-write.
          name += "";
 
          if (arguments.length < 2) {
-           var tween = get$4(this.node(), id).tween;
+           var tween = get$1(this.node(), id).tween;
 
            for (var i = 0, n = tween.length, t; i < n; ++i) {
              if ((t = tween[i]).name === name) {
        function tweenValue(transition, name, value) {
          var id = transition._id;
          transition.each(function () {
-           var schedule = set$4(this, id);
+           var schedule = set(this, id);
            (schedule.value || (schedule.value = {}))[name] = value.apply(this, arguments);
          });
          return function (node) {
-           return get$4(node, id).value[name];
+           return get$1(node, id).value[name];
          };
        }
 
-       function interpolate$1 (a, b) {
+       function interpolate (a, b) {
          var c;
          return (typeof b === "number" ? d3_interpolateNumber : b instanceof color ? d3_interpolateRgb : (c = color(b)) ? (b = c, d3_interpolateRgb) : interpolateString)(a, b);
        }
 
-       function attrRemove$1(name) {
+       function attrRemove(name) {
          return function () {
            this.removeAttribute(name);
          };
        }
 
-       function attrRemoveNS$1(fullname) {
+       function attrRemoveNS(fullname) {
          return function () {
            this.removeAttributeNS(fullname.space, fullname.local);
          };
        }
 
-       function attrConstant$1(name, interpolate, value1) {
+       function attrConstant(name, interpolate, value1) {
          var string00,
              string1 = value1 + "",
              interpolate0;
          };
        }
 
-       function attrConstantNS$1(fullname, interpolate, value1) {
+       function attrConstantNS(fullname, interpolate, value1) {
          var string00,
              string1 = value1 + "",
              interpolate0;
          };
        }
 
-       function attrFunction$1(name, interpolate, value) {
+       function attrFunction(name, interpolate, value) {
          var string00, string10, interpolate0;
          return function () {
            var string0,
          };
        }
 
-       function attrFunctionNS$1(fullname, interpolate, value) {
+       function attrFunctionNS(fullname, interpolate, value) {
          var string00, string10, interpolate0;
          return function () {
            var string0,
 
        function transition_attr (name, value) {
          var fullname = namespace(name),
-             i = fullname === "transform" ? interpolateTransformSvg : interpolate$1;
-         return this.attrTween(name, typeof value === "function" ? (fullname.local ? attrFunctionNS$1 : attrFunction$1)(fullname, i, tweenValue(this, "attr." + name, value)) : value == null ? (fullname.local ? attrRemoveNS$1 : attrRemove$1)(fullname) : (fullname.local ? attrConstantNS$1 : attrConstant$1)(fullname, i, value));
+             i = fullname === "transform" ? interpolateTransformSvg : interpolate;
+         return this.attrTween(name, typeof value === "function" ? (fullname.local ? attrFunctionNS : attrFunction)(fullname, i, tweenValue(this, "attr." + name, value)) : value == null ? (fullname.local ? attrRemoveNS : attrRemove)(fullname) : (fullname.local ? attrConstantNS : attrConstant)(fullname, i, value));
        }
 
        function attrInterpolate(name, i) {
 
        function transition_delay (value) {
          var id = this._id;
-         return arguments.length ? this.each((typeof value === "function" ? delayFunction : delayConstant)(id, value)) : get$4(this.node(), id).delay;
+         return arguments.length ? this.each((typeof value === "function" ? delayFunction : delayConstant)(id, value)) : get$1(this.node(), id).delay;
        }
 
        function durationFunction(id, value) {
          return function () {
-           set$4(this, id).duration = +value.apply(this, arguments);
+           set(this, id).duration = +value.apply(this, arguments);
          };
        }
 
        function durationConstant(id, value) {
          return value = +value, function () {
-           set$4(this, id).duration = value;
+           set(this, id).duration = value;
          };
        }
 
        function transition_duration (value) {
          var id = this._id;
-         return arguments.length ? this.each((typeof value === "function" ? durationFunction : durationConstant)(id, value)) : get$4(this.node(), id).duration;
+         return arguments.length ? this.each((typeof value === "function" ? durationFunction : durationConstant)(id, value)) : get$1(this.node(), id).duration;
        }
 
        function easeConstant(id, value) {
          if (typeof value !== "function") throw new Error();
          return function () {
-           set$4(this, id).ease = value;
+           set(this, id).ease = value;
          };
        }
 
        function transition_ease (value) {
          var id = this._id;
-         return arguments.length ? this.each(easeConstant(id, value)) : get$4(this.node(), id).ease;
+         return arguments.length ? this.each(easeConstant(id, value)) : get$1(this.node(), id).ease;
        }
 
        function easeVarying(id, value) {
          return function () {
            var v = value.apply(this, arguments);
            if (typeof v !== "function") throw new Error();
-           set$4(this, id).ease = v;
+           set(this, id).ease = v;
          };
        }
 
        function onFunction(id, name, listener) {
          var on0,
              on1,
-             sit = start(name) ? init : set$4;
+             sit = start(name) ? init : set;
          return function () {
            var schedule = sit(this, id),
                on = schedule.on; // If this node shared a dispatch with the previous node,
 
        function transition_on (name, listener) {
          var id = this._id;
-         return arguments.length < 2 ? get$4(this.node(), id).on.on(name) : this.each(onFunction(id, name, listener));
+         return arguments.length < 2 ? get$1(this.node(), id).on.on(name) : this.each(onFunction(id, name, listener));
        }
 
        function removeFunction(id) {
              if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
                if ("__data__" in node) subnode.__data__ = node.__data__;
                subgroup[i] = subnode;
-               schedule(subgroup[i], name, id, i, subgroup, get$4(node, id));
+               schedule(subgroup[i], name, id, i, subgroup, get$1(node, id));
              }
            }
          }
          for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
            for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
              if (node = group[i]) {
-               for (var children = select.call(node, node.__data__, i, group), child, inherit = get$4(node, id), k = 0, l = children.length; k < l; ++k) {
+               for (var children = select.call(node, node.__data__, i, group), child, inherit = get$1(node, id), k = 0, l = children.length; k < l; ++k) {
                  if (child = children[k]) {
                    schedule(child, name, id, k, children, inherit);
                  }
          return new Transition(subgroups, parents, name, id);
        }
 
-       var Selection$1 = selection.prototype.constructor;
+       var Selection = selection.prototype.constructor;
        function transition_selection () {
-         return new Selection$1(this._groups, this._parents);
+         return new Selection(this._groups, this._parents);
        }
 
        function styleNull(name, interpolate) {
          };
        }
 
-       function styleRemove$1(name) {
+       function styleRemove(name) {
          return function () {
            this.style.removeProperty(name);
          };
        }
 
-       function styleConstant$1(name, interpolate, value1) {
+       function styleConstant(name, interpolate, value1) {
          var string00,
              string1 = value1 + "",
              interpolate0;
          };
        }
 
-       function styleFunction$1(name, interpolate, value) {
+       function styleFunction(name, interpolate, value) {
          var string00, string10, interpolate0;
          return function () {
            var string0 = styleValue(this, name),
              event = "end." + key,
              remove;
          return function () {
-           var schedule = set$4(this, id),
+           var schedule = set(this, id),
                on = schedule.on,
-               listener = schedule.value[key] == null ? remove || (remove = styleRemove$1(name)) : undefined; // If this node shared a dispatch with the previous node,
+               listener = schedule.value[key] == null ? remove || (remove = styleRemove(name)) : undefined; // If this node shared a dispatch with the previous node,
            // just assign the updated shared dispatch and we’re done!
            // Otherwise, copy-on-write.
 
        }
 
        function transition_style (name, value, priority) {
-         var i = (name += "") === "transform" ? interpolateTransformCss : interpolate$1;
-         return value == null ? this.styleTween(name, styleNull(name, i)).on("end.style." + name, styleRemove$1(name)) : typeof value === "function" ? this.styleTween(name, styleFunction$1(name, i, tweenValue(this, "style." + name, value))).each(styleMaybeRemove(this._id, name)) : this.styleTween(name, styleConstant$1(name, i, value), priority).on("end.style." + name, null);
+         var i = (name += "") === "transform" ? interpolateTransformCss : interpolate;
+         return value == null ? this.styleTween(name, styleNull(name, i)).on("end.style." + name, styleRemove(name)) : typeof value === "function" ? this.styleTween(name, styleFunction(name, i, tweenValue(this, "style." + name, value))).each(styleMaybeRemove(this._id, name)) : this.styleTween(name, styleConstant(name, i, value), priority).on("end.style." + name, null);
        }
 
        function styleInterpolate(name, i, priority) {
          return this.tween(key, styleTween(name, value, priority == null ? "" : priority));
        }
 
-       function textConstant$1(value) {
+       function textConstant(value) {
          return function () {
            this.textContent = value;
          };
        }
 
-       function textFunction$1(value) {
+       function textFunction(value) {
          return function () {
            var value1 = value(this);
            this.textContent = value1 == null ? "" : value1;
        }
 
        function transition_text (value) {
-         return this.tween("text", typeof value === "function" ? textFunction$1(tweenValue(this, "text", value)) : textConstant$1(value == null ? "" : value + ""));
+         return this.tween("text", typeof value === "function" ? textFunction(tweenValue(this, "text", value)) : textConstant(value == null ? "" : value + ""));
        }
 
        function textInterpolate(i) {
          for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {
            for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
              if (node = group[i]) {
-               var inherit = get$4(node, id0);
+               var inherit = get$1(node, id0);
                schedule(node, name, id1, i, group, {
                  time: inherit.time + inherit.delay + inherit.duration,
                  delay: 0,
              }
            };
            that.each(function () {
-             var schedule = set$4(this, id),
+             var schedule = set(this, id),
                  on = schedule.on; // If this node shared a dispatch with the previous node,
              // just assign the updated shared dispatch and we’re done!
              // Otherwise, copy-on-write.
          });
        }
 
-       var id$1 = 0;
+       var id = 0;
        function Transition(groups, parents, name, id) {
          this._groups = groups;
          this._parents = parents;
          this._name = name;
          this._id = id;
        }
-       function transition(name) {
-         return selection().transition(name);
-       }
        function newId() {
-         return ++id$1;
+         return ++id;
        }
        var selection_prototype = selection.prototype;
-       Transition.prototype = transition.prototype = _defineProperty({
+       Transition.prototype = _defineProperty({
          constructor: Transition,
          select: transition_select,
          selectAll: transition_selectAll,
          if (name instanceof Transition) {
            id = name._id, name = name._name;
          } else {
-           id = newId(), (timing = defaultTiming).time = now(), name = name == null ? null : name + "";
+           id = newId(), (timing = defaultTiming).time = now$1(), name = name == null ? null : name + "";
          }
 
          for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {
        selection.prototype.interrupt = selection_interrupt;
        selection.prototype.transition = selection_transition;
 
-       var constant$3 = (function (x) {
+       var constant = (function (x) {
          return function () {
            return x;
          };
        };
        var identity$2 = new Transform(1, 0, 0);
 
-       function nopropagation$1(event) {
+       function nopropagation(event) {
          event.stopImmediatePropagation();
        }
-       function noevent$1 (event) {
+       function noevent (event) {
          event.preventDefault();
          event.stopImmediatePropagation();
        }
          return (!event.ctrlKey || event.type === 'wheel') && !event.button;
        }
 
-       function defaultExtent() {
+       function defaultExtent$1() {
          var e = this;
 
          if (e instanceof SVGElement) {
          return this.__zoom || identity$2;
        }
 
-       function defaultWheelDelta(event) {
+       function defaultWheelDelta$1(event) {
          return -event.deltaY * (event.deltaMode === 1 ? 0.05 : event.deltaMode ? 1 : 0.002) * (event.ctrlKey ? 10 : 1);
        }
 
-       function defaultTouchable$1() {
+       function defaultTouchable() {
          return navigator.maxTouchPoints || "ontouchstart" in this;
        }
 
-       function defaultConstrain(transform, extent, translateExtent) {
+       function defaultConstrain$1(transform, extent, translateExtent) {
          var dx0 = transform.invertX(extent[0][0]) - translateExtent[0][0],
              dx1 = transform.invertX(extent[1][0]) - translateExtent[1][0],
              dy0 = transform.invertY(extent[0][1]) - translateExtent[0][1],
 
        function d3_zoom () {
          var filter = defaultFilter$1,
-             extent = defaultExtent,
-             constrain = defaultConstrain,
-             wheelDelta = defaultWheelDelta,
-             touchable = defaultTouchable$1,
+             extent = defaultExtent$1,
+             constrain = defaultConstrain$1,
+             wheelDelta = defaultWheelDelta$1,
+             touchable = defaultTouchable,
              scaleExtent = [0, Infinity],
              translateExtent = [[-Infinity, -Infinity], [Infinity, Infinity]],
              duration = 250,
              interpolate = interpolateZoom,
-             listeners = dispatch("start", "zoom", "end"),
+             listeners = dispatch$8("start", "zoom", "end"),
              touchstarting,
              touchfirst,
              touchending,
                  g.start();
                }
 
-           noevent$1(event);
+           noevent(event);
            g.wheel = setTimeout(wheelidled, wheelDelay);
            g.zoom("mouse", constrain(translate(scale(t, k), g.mouse[0], g.mouse[1]), g.extent, translateExtent));
 
                x0 = event.clientX,
                y0 = event.clientY;
            dragDisable(event.view);
-           nopropagation$1(event);
+           nopropagation(event);
            g.mouse = [p, this.__zoom.invert(p)];
            interrupt(this);
            g.start();
 
            function mousemoved(event) {
-             noevent$1(event);
+             noevent(event);
 
              if (!g.moved) {
                var dx = event.clientX - x0,
            function mouseupped(event) {
              v.on("mousemove.zoom mouseup.zoom", null);
              yesdrag(event.view, g.moved);
-             noevent$1(event);
+             noevent(event);
              g.event(event).end();
            }
          }
                p1 = t0.invert(p0),
                k1 = t0.k * (event.shiftKey ? 0.5 : 2),
                t1 = constrain(translate(scale(t0, k1), p0, p1), extent.apply(this, args), translateExtent);
-           noevent$1(event);
+           noevent(event);
            if (duration > 0) select(this).transition().duration(duration).call(schedule, t1, p0, event);else select(this).call(zoom.transform, t1, p0, event);
          }
 
                i,
                t,
                p;
-           nopropagation$1(event);
+           nopropagation(event);
 
            for (i = 0; i < n; ++i) {
              t = touches[i], p = pointer(t, this);
                t,
                p,
                l;
-           noevent$1(event);
+           noevent(event);
 
            for (i = 0; i < n; ++i) {
              t = touches[i], p = pointer(t, this);
                n = touches.length,
                i,
                t;
-           nopropagation$1(event);
+           nopropagation(event);
            if (touchending) clearTimeout(touchending);
            touchending = setTimeout(function () {
              touchending = null;
          }
 
          zoom.wheelDelta = function (_) {
-           return arguments.length ? (wheelDelta = typeof _ === "function" ? _ : constant$3(+_), zoom) : wheelDelta;
+           return arguments.length ? (wheelDelta = typeof _ === "function" ? _ : constant(+_), zoom) : wheelDelta;
          };
 
          zoom.filter = function (_) {
-           return arguments.length ? (filter = typeof _ === "function" ? _ : constant$3(!!_), zoom) : filter;
+           return arguments.length ? (filter = typeof _ === "function" ? _ : constant(!!_), zoom) : filter;
          };
 
          zoom.touchable = function (_) {
-           return arguments.length ? (touchable = typeof _ === "function" ? _ : constant$3(!!_), zoom) : touchable;
+           return arguments.length ? (touchable = typeof _ === "function" ? _ : constant(!!_), zoom) : touchable;
          };
 
          zoom.extent = function (_) {
-           return arguments.length ? (extent = typeof _ === "function" ? _ : constant$3([[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]]), zoom) : extent;
+           return arguments.length ? (extent = typeof _ === "function" ? _ : constant([[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]]), zoom) : extent;
          };
 
          zoom.scaleExtent = function (_) {
 
        var onFreeze = internalMetadata.onFreeze;
 
-       var nativeFreeze = Object.freeze;
-       var FAILS_ON_PRIMITIVES$4 = fails(function () { nativeFreeze(1); });
+       // eslint-disable-next-line es/no-object-freeze -- safe
+       var $freeze = Object.freeze;
+       var FAILS_ON_PRIMITIVES = fails(function () { $freeze(1); });
 
        // `Object.freeze` method
        // https://tc39.es/ecma262/#sec-object.freeze
-       _export({ target: 'Object', stat: true, forced: FAILS_ON_PRIMITIVES$4, sham: !freezing }, {
+       _export({ target: 'Object', stat: true, forced: FAILS_ON_PRIMITIVES, sham: !freezing }, {
          freeze: function freeze(it) {
-           return nativeFreeze && isObject(it) ? nativeFreeze(onFreeze(it)) : it;
+           return $freeze && isObject$4(it) ? $freeze(onFreeze(it)) : it;
          }
        });
 
        }
 
        // @@match logic
-       fixRegexpWellKnownSymbolLogic('match', 1, function (MATCH, nativeMatch, maybeCallNative) {
+       fixRegexpWellKnownSymbolLogic('match', function (MATCH, nativeMatch, maybeCallNative) {
          return [
            // `String.prototype.match` method
            // https://tc39.es/ecma262/#sec-string.prototype.match
            },
            // `RegExp.prototype[@@match]` method
            // https://tc39.es/ecma262/#sec-regexp.prototype-@@match
-           function (regexp) {
-             var res = maybeCallNative(nativeMatch, regexp, this);
+           function (string) {
+             var res = maybeCallNative(nativeMatch, this, string);
              if (res.done) return res.value;
 
-             var rx = anObject(regexp);
-             var S = String(this);
+             var rx = anObject(this);
+             var S = String(string);
 
              if (!rx.global) return regexpExecAbstract(rx, S);
 
          ];
        });
 
-       var remove$1 = removeDiacritics;
+       var remove$6 = removeDiacritics;
        var replacementList = [{
          base: ' ',
          chars: "\xA0"
        }];
        var diacriticsMap = {};
 
-       for (var i = 0; i < replacementList.length; i += 1) {
-         var chars = replacementList[i].chars;
+       for (var i$1 = 0; i$1 < replacementList.length; i$1 += 1) {
+         var chars = replacementList[i$1].chars;
 
          for (var j$1 = 0; j$1 < chars.length; j$1 += 1) {
-           diacriticsMap[chars[j$1]] = replacementList[i].base;
+           diacriticsMap[chars[j$1]] = replacementList[i$1].base;
          }
        }
 
        var replacementList_1 = replacementList;
        var diacriticsMap_1 = diacriticsMap;
        var diacritics = {
-         remove: remove$1,
+         remove: remove$6,
          replacementList: replacementList_1,
          diacriticsMap: diacriticsMap_1
        };
 
-       var isArabic_1 = createCommonjsModule(function (module, exports) {
-
-         Object.defineProperty(exports, "__esModule", {
-           value: true
-         });
-         var arabicBlocks = [[0x0600, 0x06FF], [0x0750, 0x077F], [0x08A0, 0x08FF], [0xFB50, 0xFDFF], [0xFE70, 0xFEFF], [0x10E60, 0x10E7F], [0x1EC70, 0x1ECBF], [0x1EE00, 0x1EEFF] // Mathematical Alphabetic symbols https://www.unicode.org/charts/PDF/U1EE00.pdf
-         ];
+       var arabicBlocks = [[0x0600, 0x06FF], [0x0750, 0x077F], [0x08A0, 0x08FF], [0xFB50, 0xFDFF], [0xFE70, 0xFEFF], [0x10E60, 0x10E7F], [0x1EC70, 0x1ECBF], [0x1EE00, 0x1EEFF] // Mathematical Alphabetic symbols https://www.unicode.org/charts/PDF/U1EE00.pdf
+       ];
 
-         function isArabic(_char) {
-           if (_char.length > 1) {
-             // allow the newer chars?
-             throw new Error('isArabic works on only one-character strings');
-           }
+       function isArabic(_char) {
+         if (_char.length > 1) {
+           // allow the newer chars?
+           throw new Error('isArabic works on only one-character strings');
+         }
 
-           var code = _char.charCodeAt(0);
+         var code = _char.charCodeAt(0);
 
-           for (var i = 0; i < arabicBlocks.length; i++) {
-             var block = arabicBlocks[i];
+         for (var i = 0; i < arabicBlocks.length; i++) {
+           var block = arabicBlocks[i];
 
-             if (code >= block[0] && code <= block[1]) {
-               return true;
-             }
+           if (code >= block[0] && code <= block[1]) {
+             return true;
            }
-
-           return false;
          }
 
-         exports.isArabic = isArabic;
-
-         function isMath(_char2) {
-           if (_char2.length > 2) {
-             // allow the newer chars?
-             throw new Error('isMath works on only one-character strings');
-           }
+         return false;
+       }
 
-           var code = _char2.charCodeAt(0);
+       var isArabic_2 = isArabic;
 
-           return code >= 0x660 && code <= 0x66C || code >= 0x6F0 && code <= 0x6F9;
+       function isMath(_char2) {
+         if (_char2.length > 2) {
+           // allow the newer chars?
+           throw new Error('isMath works on only one-character strings');
          }
 
-         exports.isMath = isMath;
-       });
+         var code = _char2.charCodeAt(0);
 
-       var unicodeArabic = createCommonjsModule(function (module, exports) {
+         return code >= 0x660 && code <= 0x66C || code >= 0x6F0 && code <= 0x6F9;
+       }
 
-         Object.defineProperty(exports, "__esModule", {
-           value: true
-         });
-         var arabicReference = {
-           "alef": {
-             "normal": ["\u0627"],
-             "madda_above": {
-               "normal": ["\u0627\u0653", "\u0622"],
-               "isolated": "\uFE81",
-               "final": "\uFE82"
-             },
-             "hamza_above": {
-               "normal": ["\u0627\u0654", "\u0623"],
-               "isolated": "\uFE83",
-               "final": "\uFE84"
-             },
-             "hamza_below": {
-               "normal": ["\u0627\u0655", "\u0625"],
-               "isolated": "\uFE87",
-               "final": "\uFE88"
-             },
-             "wasla": {
-               "normal": "\u0671",
-               "isolated": "\uFB50",
-               "final": "\uFB51"
-             },
-             "wavy_hamza_above": ["\u0672"],
-             "wavy_hamza_below": ["\u0627\u065F", "\u0673"],
-             "high_hamza": ["\u0675", "\u0627\u0674"],
-             "indic_two_above": ["\u0773"],
-             "indic_three_above": ["\u0774"],
-             "fathatan": {
-               "normal": ["\u0627\u064B"],
-               "final": "\uFD3C",
-               "isolated": "\uFD3D"
-             },
-             "isolated": "\uFE8D",
-             "final": "\uFE8E"
-           },
-           "beh": {
-             "normal": ["\u0628"],
-             "dotless": ["\u066E"],
-             "three_dots_horizontally_below": ["\u0750"],
-             "dot_below_three_dots_above": ["\u0751"],
-             "three_dots_pointing_upwards_below": ["\u0752"],
-             "three_dots_pointing_upwards_below_two_dots_above": ["\u0753"],
-             "two_dots_below_dot_above": ["\u0754"],
-             "inverted_small_v_below": ["\u0755"],
-             "small_v": ["\u0756"],
-             "small_v_below": ["\u08A0"],
-             "hamza_above": ["\u08A1"],
-             "small_meem_above": ["\u08B6"],
-             "isolated": "\uFE8F",
-             "final": "\uFE90",
-             "initial": "\uFE91",
-             "medial": "\uFE92"
-           },
-           "teh marbuta": {
-             "normal": ["\u0629"],
-             "isolated": "\uFE93",
-             "final": "\uFE94"
-           },
-           "teh": {
-             "normal": ["\u062A"],
-             "ring": ["\u067C"],
-             "three_dots_above_downwards": ["\u067D"],
-             "small_teh_above": ["\u08B8"],
-             "isolated": "\uFE95",
-             "final": "\uFE96",
-             "initial": "\uFE97",
-             "medial": "\uFE98"
-           },
-           "theh": {
-             "normal": ["\u062B"],
-             "isolated": "\uFE99",
-             "final": "\uFE9A",
-             "initial": "\uFE9B",
-             "medial": "\uFE9C"
-           },
-           "jeem": {
-             "normal": ["\u062C"],
-             "two_dots_above": ["\u08A2"],
-             "isolated": "\uFE9D",
-             "final": "\uFE9E",
-             "initial": "\uFE9F",
-             "medial": "\uFEA0"
-           },
-           "hah": {
-             "normal": ["\u062D"],
-             "hamza_above": ["\u0681"],
-             "two_dots_vertical_above": ["\u0682"],
-             "three_dots_above": ["\u0685"],
-             "two_dots_above": ["\u0757"],
-             "three_dots_pointing_upwards_below": ["\u0758"],
-             "small_tah_below": ["\u076E"],
-             "small_tah_two_dots": ["\u076F"],
-             "small_tah_above": ["\u0772"],
-             "indic_four_below": ["\u077C"],
-             "isolated": "\uFEA1",
-             "final": "\uFEA2",
-             "initial": "\uFEA3",
-             "medial": "\uFEA4"
-           },
-           "khah": {
-             "normal": ["\u062E"],
-             "isolated": "\uFEA5",
-             "final": "\uFEA6",
-             "initial": "\uFEA7",
-             "medial": "\uFEA8"
-           },
-           "dal": {
-             "normal": ["\u062F"],
-             "ring": ["\u0689"],
-             "dot_below": ["\u068A"],
-             "dot_below_small_tah": ["\u068B"],
-             "three_dots_above_downwards": ["\u068F"],
-             "four_dots_above": ["\u0690"],
-             "inverted_v": ["\u06EE"],
-             "two_dots_vertically_below_small_tah": ["\u0759"],
-             "inverted_small_v_below": ["\u075A"],
-             "three_dots_below": ["\u08AE"],
-             "isolated": "\uFEA9",
-             "final": "\uFEAA"
-           },
-           "thal": {
-             "normal": ["\u0630"],
-             "isolated": "\uFEAB",
-             "final": "\uFEAC"
-           },
-           "reh": {
-             "normal": ["\u0631"],
-             "small_v": ["\u0692"],
-             "ring": ["\u0693"],
-             "dot_below": ["\u0694"],
-             "small_v_below": ["\u0695"],
-             "dot_below_dot_above": ["\u0696"],
-             "two_dots_above": ["\u0697"],
-             "four_dots_above": ["\u0699"],
-             "inverted_v": ["\u06EF"],
-             "stroke": ["\u075B"],
-             "two_dots_vertically_above": ["\u076B"],
-             "hamza_above": ["\u076C"],
-             "small_tah_two_dots": ["\u0771"],
-             "loop": ["\u08AA"],
-             "small_noon_above": ["\u08B9"],
-             "isolated": "\uFEAD",
-             "final": "\uFEAE"
-           },
-           "zain": {
-             "normal": ["\u0632"],
-             "inverted_v_above": ["\u08B2"],
-             "isolated": "\uFEAF",
-             "final": "\uFEB0"
-           },
-           "seen": {
-             "normal": ["\u0633"],
-             "dot_below_dot_above": ["\u069A"],
-             "three_dots_below": ["\u069B"],
-             "three_dots_below_three_dots_above": ["\u069C"],
-             "four_dots_above": ["\u075C"],
-             "two_dots_vertically_above": ["\u076D"],
-             "small_tah_two_dots": ["\u0770"],
-             "indic_four_above": ["\u077D"],
-             "inverted_v": ["\u077E"],
-             "isolated": "\uFEB1",
-             "final": "\uFEB2",
-             "initial": "\uFEB3",
-             "medial": "\uFEB4"
-           },
-           "sheen": {
-             "normal": ["\u0634"],
-             "dot_below": ["\u06FA"],
-             "isolated": "\uFEB5",
-             "final": "\uFEB6",
-             "initial": "\uFEB7",
-             "medial": "\uFEB8"
-           },
-           "sad": {
-             "normal": ["\u0635"],
-             "two_dots_below": ["\u069D"],
-             "three_dots_above": ["\u069E"],
-             "three_dots_below": ["\u08AF"],
-             "isolated": "\uFEB9",
-             "final": "\uFEBA",
-             "initial": "\uFEBB",
-             "medial": "\uFEBC"
-           },
-           "dad": {
-             "normal": ["\u0636"],
-             "dot_below": ["\u06FB"],
-             "isolated": "\uFEBD",
-             "final": "\uFEBE",
-             "initial": "\uFEBF",
-             "medial": "\uFEC0"
-           },
-           "tah": {
-             "normal": ["\u0637"],
-             "three_dots_above": ["\u069F"],
-             "two_dots_above": ["\u08A3"],
-             "isolated": "\uFEC1",
-             "final": "\uFEC2",
-             "initial": "\uFEC3",
-             "medial": "\uFEC4"
-           },
-           "zah": {
-             "normal": ["\u0638"],
-             "isolated": "\uFEC5",
-             "final": "\uFEC6",
-             "initial": "\uFEC7",
-             "medial": "\uFEC8"
-           },
-           "ain": {
-             "normal": ["\u0639"],
-             "three_dots_above": ["\u06A0"],
-             "two_dots_above": ["\u075D"],
-             "three_dots_pointing_downwards_above": ["\u075E"],
-             "two_dots_vertically_above": ["\u075F"],
-             "three_dots_below": ["\u08B3"],
-             "isolated": "\uFEC9",
-             "final": "\uFECA",
-             "initial": "\uFECB",
-             "medial": "\uFECC"
-           },
-           "ghain": {
-             "normal": ["\u063A"],
-             "dot_below": ["\u06FC"],
-             "isolated": "\uFECD",
-             "final": "\uFECE",
-             "initial": "\uFECF",
-             "medial": "\uFED0"
-           },
-           "feh": {
-             "normal": ["\u0641"],
-             "dotless": ["\u06A1"],
-             "dot_moved_below": ["\u06A2"],
-             "dot_below": ["\u06A3"],
-             "three_dots_below": ["\u06A5"],
-             "two_dots_below": ["\u0760"],
-             "three_dots_pointing_upwards_below": ["\u0761"],
-             "dot_below_three_dots_above": ["\u08A4"],
-             "isolated": "\uFED1",
-             "final": "\uFED2",
-             "initial": "\uFED3",
-             "medial": "\uFED4"
-           },
-           "qaf": {
-             "normal": ["\u0642"],
-             "dotless": ["\u066F"],
-             "dot_above": ["\u06A7"],
-             "three_dots_above": ["\u06A8"],
-             "dot_below": ["\u08A5"],
-             "isolated": "\uFED5",
-             "final": "\uFED6",
-             "initial": "\uFED7",
-             "medial": "\uFED8"
-           },
-           "kaf": {
-             "normal": ["\u0643"],
-             "swash": ["\u06AA"],
-             "ring": ["\u06AB"],
-             "dot_above": ["\u06AC"],
-             "three_dots_below": ["\u06AE"],
-             "two_dots_above": ["\u077F"],
-             "dot_below": ["\u08B4"],
-             "isolated": "\uFED9",
-             "final": "\uFEDA",
-             "initial": "\uFEDB",
-             "medial": "\uFEDC"
-           },
-           "lam": {
-             "normal": ["\u0644"],
-             "small_v": ["\u06B5"],
-             "dot_above": ["\u06B6"],
-             "three_dots_above": ["\u06B7"],
-             "three_dots_below": ["\u06B8"],
-             "bar": ["\u076A"],
-             "double_bar": ["\u08A6"],
-             "isolated": "\uFEDD",
-             "final": "\uFEDE",
-             "initial": "\uFEDF",
-             "medial": "\uFEE0"
-           },
-           "meem": {
-             "normal": ["\u0645"],
-             "dot_above": ["\u0765"],
-             "dot_below": ["\u0766"],
-             "three_dots_above": ["\u08A7"],
-             "isolated": "\uFEE1",
-             "final": "\uFEE2",
-             "initial": "\uFEE3",
-             "medial": "\uFEE4"
-           },
-           "noon": {
-             "normal": ["\u0646"],
-             "dot_below": ["\u06B9"],
-             "ring": ["\u06BC"],
-             "three_dots_above": ["\u06BD"],
-             "two_dots_below": ["\u0767"],
-             "small_tah": ["\u0768"],
-             "small_v": ["\u0769"],
-             "isolated": "\uFEE5",
-             "final": "\uFEE6",
-             "initial": "\uFEE7",
-             "medial": "\uFEE8"
-           },
-           "heh": {
-             "normal": ["\u0647"],
-             "isolated": "\uFEE9",
-             "final": "\uFEEA",
-             "initial": "\uFEEB",
-             "medial": "\uFEEC"
-           },
-           "waw": {
-             "normal": ["\u0648"],
-             "hamza_above": {
-               "normal": ["\u0624", "\u0648\u0654"],
-               "isolated": "\uFE85",
-               "final": "\uFE86"
-             },
-             "high_hamza": ["\u0676", "\u0648\u0674"],
-             "ring": ["\u06C4"],
-             "two_dots_above": ["\u06CA"],
-             "dot_above": ["\u06CF"],
-             "indic_two_above": ["\u0778"],
-             "indic_three_above": ["\u0779"],
-             "dot_within": ["\u08AB"],
-             "isolated": "\uFEED",
-             "final": "\uFEEE"
-           },
-           "alef_maksura": {
-             "normal": ["\u0649"],
-             "hamza_above": ["\u0626", "\u064A\u0654"],
-             "initial": "\uFBE8",
-             "medial": "\uFBE9",
-             "isolated": "\uFEEF",
-             "final": "\uFEF0"
-           },
-           "yeh": {
-             "normal": ["\u064A"],
-             "hamza_above": {
-               "normal": ["\u0626", "\u0649\u0654"],
-               "isolated": "\uFE89",
-               "final": "\uFE8A",
-               "initial": "\uFE8B",
-               "medial": "\uFE8C"
-             },
-             "two_dots_below_hamza_above": ["\u08A8"],
-             "high_hamza": ["\u0678", "\u064A\u0674"],
-             "tail": ["\u06CD"],
-             "small_v": ["\u06CE"],
-             "three_dots_below": ["\u06D1"],
-             "two_dots_below_dot_above": ["\u08A9"],
-             "two_dots_below_small_noon_above": ["\u08BA"],
-             "isolated": "\uFEF1",
-             "final": "\uFEF2",
-             "initial": "\uFEF3",
-             "medial": "\uFEF4"
-           },
-           "tteh": {
-             "normal": ["\u0679"],
-             "isolated": "\uFB66",
-             "final": "\uFB67",
-             "initial": "\uFB68",
-             "medial": "\uFB69"
-           },
-           "tteheh": {
-             "normal": ["\u067A"],
-             "isolated": "\uFB5E",
-             "final": "\uFB5F",
-             "initial": "\uFB60",
-             "medial": "\uFB61"
-           },
-           "beeh": {
-             "normal": ["\u067B"],
-             "isolated": "\uFB52",
-             "final": "\uFB53",
-             "initial": "\uFB54",
-             "medial": "\uFB55"
-           },
-           "peh": {
-             "normal": ["\u067E"],
-             "small_meem_above": ["\u08B7"],
-             "isolated": "\uFB56",
-             "final": "\uFB57",
-             "initial": "\uFB58",
-             "medial": "\uFB59"
-           },
-           "teheh": {
-             "normal": ["\u067F"],
-             "isolated": "\uFB62",
-             "final": "\uFB63",
-             "initial": "\uFB64",
-             "medial": "\uFB65"
-           },
-           "beheh": {
-             "normal": ["\u0680"],
-             "isolated": "\uFB5A",
-             "final": "\uFB5B",
-             "initial": "\uFB5C",
-             "medial": "\uFB5D"
-           },
-           "nyeh": {
-             "normal": ["\u0683"],
-             "isolated": "\uFB76",
-             "final": "\uFB77",
-             "initial": "\uFB78",
-             "medial": "\uFB79"
-           },
-           "dyeh": {
-             "normal": ["\u0684"],
-             "isolated": "\uFB72",
-             "final": "\uFB73",
-             "initial": "\uFB74",
-             "medial": "\uFB75"
-           },
-           "tcheh": {
-             "normal": ["\u0686"],
-             "dot_above": ["\u06BF"],
-             "isolated": "\uFB7A",
-             "final": "\uFB7B",
-             "initial": "\uFB7C",
-             "medial": "\uFB7D"
-           },
-           "tcheheh": {
-             "normal": ["\u0687"],
-             "isolated": "\uFB7E",
-             "final": "\uFB7F",
-             "initial": "\uFB80",
-             "medial": "\uFB81"
-           },
-           "ddal": {
-             "normal": ["\u0688"],
-             "isolated": "\uFB88",
-             "final": "\uFB89"
-           },
-           "dahal": {
-             "normal": ["\u068C"],
-             "isolated": "\uFB84",
-             "final": "\uFB85"
-           },
-           "ddahal": {
-             "normal": ["\u068D"],
-             "isolated": "\uFB82",
-             "final": "\uFB83"
-           },
-           "dul": {
-             "normal": ["\u068F", "\u068E"],
-             "isolated": "\uFB86",
-             "final": "\uFB87"
-           },
-           "rreh": {
-             "normal": ["\u0691"],
-             "isolated": "\uFB8C",
-             "final": "\uFB8D"
-           },
-           "jeh": {
-             "normal": ["\u0698"],
-             "isolated": "\uFB8A",
-             "final": "\uFB8B"
-           },
-           "veh": {
-             "normal": ["\u06A4"],
-             "isolated": "\uFB6A",
-             "final": "\uFB6B",
-             "initial": "\uFB6C",
-             "medial": "\uFB6D"
-           },
-           "peheh": {
-             "normal": ["\u06A6"],
-             "isolated": "\uFB6E",
-             "final": "\uFB6F",
-             "initial": "\uFB70",
-             "medial": "\uFB71"
-           },
-           "keheh": {
-             "normal": ["\u06A9"],
-             "dot_above": ["\u0762"],
-             "three_dots_above": ["\u0763"],
-             "three_dots_pointing_upwards_below": ["\u0764"],
-             "isolated": "\uFB8E",
-             "final": "\uFB8F",
-             "initial": "\uFB90",
-             "medial": "\uFB91"
-           },
-           "ng": {
-             "normal": ["\u06AD"],
-             "isolated": "\uFBD3",
-             "final": "\uFBD4",
-             "initial": "\uFBD5",
-             "medial": "\uFBD6"
-           },
-           "gaf": {
-             "normal": ["\u06AF"],
-             "ring": ["\u06B0"],
-             "two_dots_below": ["\u06B2"],
-             "three_dots_above": ["\u06B4"],
-             "inverted_stroke": ["\u08B0"],
-             "isolated": "\uFB92",
-             "final": "\uFB93",
-             "initial": "\uFB94",
-             "medial": "\uFB95"
-           },
-           "ngoeh": {
-             "normal": ["\u06B1"],
-             "isolated": "\uFB9A",
-             "final": "\uFB9B",
-             "initial": "\uFB9C",
-             "medial": "\uFB9D"
-           },
-           "gueh": {
-             "normal": ["\u06B3"],
-             "isolated": "\uFB96",
-             "final": "\uFB97",
-             "initial": "\uFB98",
-             "medial": "\uFB99"
-           },
-           "noon ghunna": {
-             "normal": ["\u06BA"],
-             "isolated": "\uFB9E",
-             "final": "\uFB9F"
-           },
-           "rnoon": {
-             "normal": ["\u06BB"],
-             "isolated": "\uFBA0",
-             "final": "\uFBA1",
-             "initial": "\uFBA2",
-             "medial": "\uFBA3"
-           },
-           "heh doachashmee": {
-             "normal": ["\u06BE"],
-             "isolated": "\uFBAA",
-             "final": "\uFBAB",
-             "initial": "\uFBAC",
-             "medial": "\uFBAD"
-           },
-           "heh goal": {
-             "normal": ["\u06C1"],
-             "hamza_above": ["\u06C1\u0654", "\u06C2"],
-             "isolated": "\uFBA6",
-             "final": "\uFBA7",
-             "initial": "\uFBA8",
-             "medial": "\uFBA9"
-           },
-           "teh marbuta goal": {
-             "normal": ["\u06C3"]
-           },
-           "kirghiz oe": {
-             "normal": ["\u06C5"],
-             "isolated": "\uFBE0",
-             "final": "\uFBE1"
-           },
-           "oe": {
-             "normal": ["\u06C6"],
-             "isolated": "\uFBD9",
-             "final": "\uFBDA"
-           },
-           "u": {
-             "normal": ["\u06C7"],
-             "hamza_above": {
-               "normal": ["\u0677", "\u06C7\u0674"],
-               "isolated": "\uFBDD"
-             },
-             "isolated": "\uFBD7",
-             "final": "\uFBD8"
-           },
-           "yu": {
-             "normal": ["\u06C8"],
-             "isolated": "\uFBDB",
-             "final": "\uFBDC"
-           },
-           "kirghiz yu": {
-             "normal": ["\u06C9"],
-             "isolated": "\uFBE2",
-             "final": "\uFBE3"
-           },
-           "ve": {
-             "normal": ["\u06CB"],
-             "isolated": "\uFBDE",
-             "final": "\uFBDF"
-           },
-           "farsi yeh": {
-             "normal": ["\u06CC"],
-             "indic_two_above": ["\u0775"],
-             "indic_three_above": ["\u0776"],
-             "indic_four_above": ["\u0777"],
-             "isolated": "\uFBFC",
-             "final": "\uFBFD",
-             "initial": "\uFBFE",
-             "medial": "\uFBFF"
+       var isMath_1 = isMath;
+       var isArabic_1 = /*#__PURE__*/Object.defineProperty({
+         isArabic: isArabic_2,
+         isMath: isMath_1
+       }, '__esModule', {
+         value: true
+       });
+
+       var arabicReference = {
+         "alef": {
+           "normal": ["\u0627"],
+           "madda_above": {
+             "normal": ["\u0627\u0653", "\u0622"],
+             "isolated": "\uFE81",
+             "final": "\uFE82"
            },
-           "e": {
-             "normal": ["\u06D0"],
-             "isolated": "\uFBE4",
-             "final": "\uFBE5",
-             "initial": "\uFBE6",
-             "medial": "\uFBE7"
+           "hamza_above": {
+             "normal": ["\u0627\u0654", "\u0623"],
+             "isolated": "\uFE83",
+             "final": "\uFE84"
            },
-           "yeh barree": {
-             "normal": ["\u06D2"],
-             "hamza_above": {
-               "normal": ["\u06D2\u0654", "\u06D3"],
-               "isolated": "\uFBB0",
-               "final": "\uFBB1"
-             },
-             "indic_two_above": ["\u077A"],
-             "indic_three_above": ["\u077B"],
-             "isolated": "\uFBAE",
-             "final": "\uFBAF"
+           "hamza_below": {
+             "normal": ["\u0627\u0655", "\u0625"],
+             "isolated": "\uFE87",
+             "final": "\uFE88"
            },
-           "ae": {
-             "normal": ["\u06D5"],
-             "isolated": "\u06D5",
-             "final": "\uFEEA",
-             "yeh_above": {
-               "normal": ["\u06C0", "\u06D5\u0654"],
-               "isolated": "\uFBA4",
-               "final": "\uFBA5"
-             }
+           "wasla": {
+             "normal": "\u0671",
+             "isolated": "\uFB50",
+             "final": "\uFB51"
            },
-           "rohingya yeh": {
-             "normal": ["\u08AC"]
+           "wavy_hamza_above": ["\u0672"],
+           "wavy_hamza_below": ["\u0627\u065F", "\u0673"],
+           "high_hamza": ["\u0675", "\u0627\u0674"],
+           "indic_two_above": ["\u0773"],
+           "indic_three_above": ["\u0774"],
+           "fathatan": {
+             "normal": ["\u0627\u064B"],
+             "final": "\uFD3C",
+             "isolated": "\uFD3D"
            },
-           "low alef": {
-             "normal": ["\u08AD"]
+           "isolated": "\uFE8D",
+           "final": "\uFE8E"
+         },
+         "beh": {
+           "normal": ["\u0628"],
+           "dotless": ["\u066E"],
+           "three_dots_horizontally_below": ["\u0750"],
+           "dot_below_three_dots_above": ["\u0751"],
+           "three_dots_pointing_upwards_below": ["\u0752"],
+           "three_dots_pointing_upwards_below_two_dots_above": ["\u0753"],
+           "two_dots_below_dot_above": ["\u0754"],
+           "inverted_small_v_below": ["\u0755"],
+           "small_v": ["\u0756"],
+           "small_v_below": ["\u08A0"],
+           "hamza_above": ["\u08A1"],
+           "small_meem_above": ["\u08B6"],
+           "isolated": "\uFE8F",
+           "final": "\uFE90",
+           "initial": "\uFE91",
+           "medial": "\uFE92"
+         },
+         "teh marbuta": {
+           "normal": ["\u0629"],
+           "isolated": "\uFE93",
+           "final": "\uFE94"
+         },
+         "teh": {
+           "normal": ["\u062A"],
+           "ring": ["\u067C"],
+           "three_dots_above_downwards": ["\u067D"],
+           "small_teh_above": ["\u08B8"],
+           "isolated": "\uFE95",
+           "final": "\uFE96",
+           "initial": "\uFE97",
+           "medial": "\uFE98"
+         },
+         "theh": {
+           "normal": ["\u062B"],
+           "isolated": "\uFE99",
+           "final": "\uFE9A",
+           "initial": "\uFE9B",
+           "medial": "\uFE9C"
+         },
+         "jeem": {
+           "normal": ["\u062C"],
+           "two_dots_above": ["\u08A2"],
+           "isolated": "\uFE9D",
+           "final": "\uFE9E",
+           "initial": "\uFE9F",
+           "medial": "\uFEA0"
+         },
+         "hah": {
+           "normal": ["\u062D"],
+           "hamza_above": ["\u0681"],
+           "two_dots_vertical_above": ["\u0682"],
+           "three_dots_above": ["\u0685"],
+           "two_dots_above": ["\u0757"],
+           "three_dots_pointing_upwards_below": ["\u0758"],
+           "small_tah_below": ["\u076E"],
+           "small_tah_two_dots": ["\u076F"],
+           "small_tah_above": ["\u0772"],
+           "indic_four_below": ["\u077C"],
+           "isolated": "\uFEA1",
+           "final": "\uFEA2",
+           "initial": "\uFEA3",
+           "medial": "\uFEA4"
+         },
+         "khah": {
+           "normal": ["\u062E"],
+           "isolated": "\uFEA5",
+           "final": "\uFEA6",
+           "initial": "\uFEA7",
+           "medial": "\uFEA8"
+         },
+         "dal": {
+           "normal": ["\u062F"],
+           "ring": ["\u0689"],
+           "dot_below": ["\u068A"],
+           "dot_below_small_tah": ["\u068B"],
+           "three_dots_above_downwards": ["\u068F"],
+           "four_dots_above": ["\u0690"],
+           "inverted_v": ["\u06EE"],
+           "two_dots_vertically_below_small_tah": ["\u0759"],
+           "inverted_small_v_below": ["\u075A"],
+           "three_dots_below": ["\u08AE"],
+           "isolated": "\uFEA9",
+           "final": "\uFEAA"
+         },
+         "thal": {
+           "normal": ["\u0630"],
+           "isolated": "\uFEAB",
+           "final": "\uFEAC"
+         },
+         "reh": {
+           "normal": ["\u0631"],
+           "small_v": ["\u0692"],
+           "ring": ["\u0693"],
+           "dot_below": ["\u0694"],
+           "small_v_below": ["\u0695"],
+           "dot_below_dot_above": ["\u0696"],
+           "two_dots_above": ["\u0697"],
+           "four_dots_above": ["\u0699"],
+           "inverted_v": ["\u06EF"],
+           "stroke": ["\u075B"],
+           "two_dots_vertically_above": ["\u076B"],
+           "hamza_above": ["\u076C"],
+           "small_tah_two_dots": ["\u0771"],
+           "loop": ["\u08AA"],
+           "small_noon_above": ["\u08B9"],
+           "isolated": "\uFEAD",
+           "final": "\uFEAE"
+         },
+         "zain": {
+           "normal": ["\u0632"],
+           "inverted_v_above": ["\u08B2"],
+           "isolated": "\uFEAF",
+           "final": "\uFEB0"
+         },
+         "seen": {
+           "normal": ["\u0633"],
+           "dot_below_dot_above": ["\u069A"],
+           "three_dots_below": ["\u069B"],
+           "three_dots_below_three_dots_above": ["\u069C"],
+           "four_dots_above": ["\u075C"],
+           "two_dots_vertically_above": ["\u076D"],
+           "small_tah_two_dots": ["\u0770"],
+           "indic_four_above": ["\u077D"],
+           "inverted_v": ["\u077E"],
+           "isolated": "\uFEB1",
+           "final": "\uFEB2",
+           "initial": "\uFEB3",
+           "medial": "\uFEB4"
+         },
+         "sheen": {
+           "normal": ["\u0634"],
+           "dot_below": ["\u06FA"],
+           "isolated": "\uFEB5",
+           "final": "\uFEB6",
+           "initial": "\uFEB7",
+           "medial": "\uFEB8"
+         },
+         "sad": {
+           "normal": ["\u0635"],
+           "two_dots_below": ["\u069D"],
+           "three_dots_above": ["\u069E"],
+           "three_dots_below": ["\u08AF"],
+           "isolated": "\uFEB9",
+           "final": "\uFEBA",
+           "initial": "\uFEBB",
+           "medial": "\uFEBC"
+         },
+         "dad": {
+           "normal": ["\u0636"],
+           "dot_below": ["\u06FB"],
+           "isolated": "\uFEBD",
+           "final": "\uFEBE",
+           "initial": "\uFEBF",
+           "medial": "\uFEC0"
+         },
+         "tah": {
+           "normal": ["\u0637"],
+           "three_dots_above": ["\u069F"],
+           "two_dots_above": ["\u08A3"],
+           "isolated": "\uFEC1",
+           "final": "\uFEC2",
+           "initial": "\uFEC3",
+           "medial": "\uFEC4"
+         },
+         "zah": {
+           "normal": ["\u0638"],
+           "isolated": "\uFEC5",
+           "final": "\uFEC6",
+           "initial": "\uFEC7",
+           "medial": "\uFEC8"
+         },
+         "ain": {
+           "normal": ["\u0639"],
+           "three_dots_above": ["\u06A0"],
+           "two_dots_above": ["\u075D"],
+           "three_dots_pointing_downwards_above": ["\u075E"],
+           "two_dots_vertically_above": ["\u075F"],
+           "three_dots_below": ["\u08B3"],
+           "isolated": "\uFEC9",
+           "final": "\uFECA",
+           "initial": "\uFECB",
+           "medial": "\uFECC"
+         },
+         "ghain": {
+           "normal": ["\u063A"],
+           "dot_below": ["\u06FC"],
+           "isolated": "\uFECD",
+           "final": "\uFECE",
+           "initial": "\uFECF",
+           "medial": "\uFED0"
+         },
+         "feh": {
+           "normal": ["\u0641"],
+           "dotless": ["\u06A1"],
+           "dot_moved_below": ["\u06A2"],
+           "dot_below": ["\u06A3"],
+           "three_dots_below": ["\u06A5"],
+           "two_dots_below": ["\u0760"],
+           "three_dots_pointing_upwards_below": ["\u0761"],
+           "dot_below_three_dots_above": ["\u08A4"],
+           "isolated": "\uFED1",
+           "final": "\uFED2",
+           "initial": "\uFED3",
+           "medial": "\uFED4"
+         },
+         "qaf": {
+           "normal": ["\u0642"],
+           "dotless": ["\u066F"],
+           "dot_above": ["\u06A7"],
+           "three_dots_above": ["\u06A8"],
+           "dot_below": ["\u08A5"],
+           "isolated": "\uFED5",
+           "final": "\uFED6",
+           "initial": "\uFED7",
+           "medial": "\uFED8"
+         },
+         "kaf": {
+           "normal": ["\u0643"],
+           "swash": ["\u06AA"],
+           "ring": ["\u06AB"],
+           "dot_above": ["\u06AC"],
+           "three_dots_below": ["\u06AE"],
+           "two_dots_above": ["\u077F"],
+           "dot_below": ["\u08B4"],
+           "isolated": "\uFED9",
+           "final": "\uFEDA",
+           "initial": "\uFEDB",
+           "medial": "\uFEDC"
+         },
+         "lam": {
+           "normal": ["\u0644"],
+           "small_v": ["\u06B5"],
+           "dot_above": ["\u06B6"],
+           "three_dots_above": ["\u06B7"],
+           "three_dots_below": ["\u06B8"],
+           "bar": ["\u076A"],
+           "double_bar": ["\u08A6"],
+           "isolated": "\uFEDD",
+           "final": "\uFEDE",
+           "initial": "\uFEDF",
+           "medial": "\uFEE0"
+         },
+         "meem": {
+           "normal": ["\u0645"],
+           "dot_above": ["\u0765"],
+           "dot_below": ["\u0766"],
+           "three_dots_above": ["\u08A7"],
+           "isolated": "\uFEE1",
+           "final": "\uFEE2",
+           "initial": "\uFEE3",
+           "medial": "\uFEE4"
+         },
+         "noon": {
+           "normal": ["\u0646"],
+           "dot_below": ["\u06B9"],
+           "ring": ["\u06BC"],
+           "three_dots_above": ["\u06BD"],
+           "two_dots_below": ["\u0767"],
+           "small_tah": ["\u0768"],
+           "small_v": ["\u0769"],
+           "isolated": "\uFEE5",
+           "final": "\uFEE6",
+           "initial": "\uFEE7",
+           "medial": "\uFEE8"
+         },
+         "heh": {
+           "normal": ["\u0647"],
+           "isolated": "\uFEE9",
+           "final": "\uFEEA",
+           "initial": "\uFEEB",
+           "medial": "\uFEEC"
+         },
+         "waw": {
+           "normal": ["\u0648"],
+           "hamza_above": {
+             "normal": ["\u0624", "\u0648\u0654"],
+             "isolated": "\uFE85",
+             "final": "\uFE86"
            },
-           "straight waw": {
-             "normal": ["\u08B1"]
+           "high_hamza": ["\u0676", "\u0648\u0674"],
+           "ring": ["\u06C4"],
+           "two_dots_above": ["\u06CA"],
+           "dot_above": ["\u06CF"],
+           "indic_two_above": ["\u0778"],
+           "indic_three_above": ["\u0779"],
+           "dot_within": ["\u08AB"],
+           "isolated": "\uFEED",
+           "final": "\uFEEE"
+         },
+         "alef_maksura": {
+           "normal": ["\u0649"],
+           "hamza_above": ["\u0626", "\u064A\u0654"],
+           "initial": "\uFBE8",
+           "medial": "\uFBE9",
+           "isolated": "\uFEEF",
+           "final": "\uFEF0"
+         },
+         "yeh": {
+           "normal": ["\u064A"],
+           "hamza_above": {
+             "normal": ["\u0626", "\u0649\u0654"],
+             "isolated": "\uFE89",
+             "final": "\uFE8A",
+             "initial": "\uFE8B",
+             "medial": "\uFE8C"
            },
-           "african feh": {
-             "normal": ["\u08BB"]
+           "two_dots_below_hamza_above": ["\u08A8"],
+           "high_hamza": ["\u0678", "\u064A\u0674"],
+           "tail": ["\u06CD"],
+           "small_v": ["\u06CE"],
+           "three_dots_below": ["\u06D1"],
+           "two_dots_below_dot_above": ["\u08A9"],
+           "two_dots_below_small_noon_above": ["\u08BA"],
+           "isolated": "\uFEF1",
+           "final": "\uFEF2",
+           "initial": "\uFEF3",
+           "medial": "\uFEF4"
+         },
+         "tteh": {
+           "normal": ["\u0679"],
+           "isolated": "\uFB66",
+           "final": "\uFB67",
+           "initial": "\uFB68",
+           "medial": "\uFB69"
+         },
+         "tteheh": {
+           "normal": ["\u067A"],
+           "isolated": "\uFB5E",
+           "final": "\uFB5F",
+           "initial": "\uFB60",
+           "medial": "\uFB61"
+         },
+         "beeh": {
+           "normal": ["\u067B"],
+           "isolated": "\uFB52",
+           "final": "\uFB53",
+           "initial": "\uFB54",
+           "medial": "\uFB55"
+         },
+         "peh": {
+           "normal": ["\u067E"],
+           "small_meem_above": ["\u08B7"],
+           "isolated": "\uFB56",
+           "final": "\uFB57",
+           "initial": "\uFB58",
+           "medial": "\uFB59"
+         },
+         "teheh": {
+           "normal": ["\u067F"],
+           "isolated": "\uFB62",
+           "final": "\uFB63",
+           "initial": "\uFB64",
+           "medial": "\uFB65"
+         },
+         "beheh": {
+           "normal": ["\u0680"],
+           "isolated": "\uFB5A",
+           "final": "\uFB5B",
+           "initial": "\uFB5C",
+           "medial": "\uFB5D"
+         },
+         "nyeh": {
+           "normal": ["\u0683"],
+           "isolated": "\uFB76",
+           "final": "\uFB77",
+           "initial": "\uFB78",
+           "medial": "\uFB79"
+         },
+         "dyeh": {
+           "normal": ["\u0684"],
+           "isolated": "\uFB72",
+           "final": "\uFB73",
+           "initial": "\uFB74",
+           "medial": "\uFB75"
+         },
+         "tcheh": {
+           "normal": ["\u0686"],
+           "dot_above": ["\u06BF"],
+           "isolated": "\uFB7A",
+           "final": "\uFB7B",
+           "initial": "\uFB7C",
+           "medial": "\uFB7D"
+         },
+         "tcheheh": {
+           "normal": ["\u0687"],
+           "isolated": "\uFB7E",
+           "final": "\uFB7F",
+           "initial": "\uFB80",
+           "medial": "\uFB81"
+         },
+         "ddal": {
+           "normal": ["\u0688"],
+           "isolated": "\uFB88",
+           "final": "\uFB89"
+         },
+         "dahal": {
+           "normal": ["\u068C"],
+           "isolated": "\uFB84",
+           "final": "\uFB85"
+         },
+         "ddahal": {
+           "normal": ["\u068D"],
+           "isolated": "\uFB82",
+           "final": "\uFB83"
+         },
+         "dul": {
+           "normal": ["\u068F", "\u068E"],
+           "isolated": "\uFB86",
+           "final": "\uFB87"
+         },
+         "rreh": {
+           "normal": ["\u0691"],
+           "isolated": "\uFB8C",
+           "final": "\uFB8D"
+         },
+         "jeh": {
+           "normal": ["\u0698"],
+           "isolated": "\uFB8A",
+           "final": "\uFB8B"
+         },
+         "veh": {
+           "normal": ["\u06A4"],
+           "isolated": "\uFB6A",
+           "final": "\uFB6B",
+           "initial": "\uFB6C",
+           "medial": "\uFB6D"
+         },
+         "peheh": {
+           "normal": ["\u06A6"],
+           "isolated": "\uFB6E",
+           "final": "\uFB6F",
+           "initial": "\uFB70",
+           "medial": "\uFB71"
+         },
+         "keheh": {
+           "normal": ["\u06A9"],
+           "dot_above": ["\u0762"],
+           "three_dots_above": ["\u0763"],
+           "three_dots_pointing_upwards_below": ["\u0764"],
+           "isolated": "\uFB8E",
+           "final": "\uFB8F",
+           "initial": "\uFB90",
+           "medial": "\uFB91"
+         },
+         "ng": {
+           "normal": ["\u06AD"],
+           "isolated": "\uFBD3",
+           "final": "\uFBD4",
+           "initial": "\uFBD5",
+           "medial": "\uFBD6"
+         },
+         "gaf": {
+           "normal": ["\u06AF"],
+           "ring": ["\u06B0"],
+           "two_dots_below": ["\u06B2"],
+           "three_dots_above": ["\u06B4"],
+           "inverted_stroke": ["\u08B0"],
+           "isolated": "\uFB92",
+           "final": "\uFB93",
+           "initial": "\uFB94",
+           "medial": "\uFB95"
+         },
+         "ngoeh": {
+           "normal": ["\u06B1"],
+           "isolated": "\uFB9A",
+           "final": "\uFB9B",
+           "initial": "\uFB9C",
+           "medial": "\uFB9D"
+         },
+         "gueh": {
+           "normal": ["\u06B3"],
+           "isolated": "\uFB96",
+           "final": "\uFB97",
+           "initial": "\uFB98",
+           "medial": "\uFB99"
+         },
+         "noon ghunna": {
+           "normal": ["\u06BA"],
+           "isolated": "\uFB9E",
+           "final": "\uFB9F"
+         },
+         "rnoon": {
+           "normal": ["\u06BB"],
+           "isolated": "\uFBA0",
+           "final": "\uFBA1",
+           "initial": "\uFBA2",
+           "medial": "\uFBA3"
+         },
+         "heh doachashmee": {
+           "normal": ["\u06BE"],
+           "isolated": "\uFBAA",
+           "final": "\uFBAB",
+           "initial": "\uFBAC",
+           "medial": "\uFBAD"
+         },
+         "heh goal": {
+           "normal": ["\u06C1"],
+           "hamza_above": ["\u06C1\u0654", "\u06C2"],
+           "isolated": "\uFBA6",
+           "final": "\uFBA7",
+           "initial": "\uFBA8",
+           "medial": "\uFBA9"
+         },
+         "teh marbuta goal": {
+           "normal": ["\u06C3"]
+         },
+         "kirghiz oe": {
+           "normal": ["\u06C5"],
+           "isolated": "\uFBE0",
+           "final": "\uFBE1"
+         },
+         "oe": {
+           "normal": ["\u06C6"],
+           "isolated": "\uFBD9",
+           "final": "\uFBDA"
+         },
+         "u": {
+           "normal": ["\u06C7"],
+           "hamza_above": {
+             "normal": ["\u0677", "\u06C7\u0674"],
+             "isolated": "\uFBDD"
            },
-           "african qaf": {
-             "normal": ["\u08BC"]
+           "isolated": "\uFBD7",
+           "final": "\uFBD8"
+         },
+         "yu": {
+           "normal": ["\u06C8"],
+           "isolated": "\uFBDB",
+           "final": "\uFBDC"
+         },
+         "kirghiz yu": {
+           "normal": ["\u06C9"],
+           "isolated": "\uFBE2",
+           "final": "\uFBE3"
+         },
+         "ve": {
+           "normal": ["\u06CB"],
+           "isolated": "\uFBDE",
+           "final": "\uFBDF"
+         },
+         "farsi yeh": {
+           "normal": ["\u06CC"],
+           "indic_two_above": ["\u0775"],
+           "indic_three_above": ["\u0776"],
+           "indic_four_above": ["\u0777"],
+           "isolated": "\uFBFC",
+           "final": "\uFBFD",
+           "initial": "\uFBFE",
+           "medial": "\uFBFF"
+         },
+         "e": {
+           "normal": ["\u06D0"],
+           "isolated": "\uFBE4",
+           "final": "\uFBE5",
+           "initial": "\uFBE6",
+           "medial": "\uFBE7"
+         },
+         "yeh barree": {
+           "normal": ["\u06D2"],
+           "hamza_above": {
+             "normal": ["\u06D2\u0654", "\u06D3"],
+             "isolated": "\uFBB0",
+             "final": "\uFBB1"
            },
-           "african noon": {
-             "normal": ["\u08BD"]
+           "indic_two_above": ["\u077A"],
+           "indic_three_above": ["\u077B"],
+           "isolated": "\uFBAE",
+           "final": "\uFBAF"
+         },
+         "ae": {
+           "normal": ["\u06D5"],
+           "isolated": "\u06D5",
+           "final": "\uFEEA",
+           "yeh_above": {
+             "normal": ["\u06C0", "\u06D5\u0654"],
+             "isolated": "\uFBA4",
+             "final": "\uFBA5"
            }
-         };
-         exports["default"] = arabicReference;
+         },
+         "rohingya yeh": {
+           "normal": ["\u08AC"]
+         },
+         "low alef": {
+           "normal": ["\u08AD"]
+         },
+         "straight waw": {
+           "normal": ["\u08B1"]
+         },
+         "african feh": {
+           "normal": ["\u08BB"]
+         },
+         "african qaf": {
+           "normal": ["\u08BC"]
+         },
+         "african noon": {
+           "normal": ["\u08BD"]
+         }
+       };
+       var _default$3 = arabicReference;
+       var unicodeArabic = /*#__PURE__*/Object.defineProperty({
+         "default": _default$3
+       }, '__esModule', {
+         value: true
+       });
+
+       var ligatureReference = {
+         "\u0626\u0627": {
+           "isolated": "\uFBEA",
+           "final": "\uFBEB"
+         },
+         "\u0626\u06D5": {
+           "isolated": "\uFBEC",
+           "final": "\uFBED"
+         },
+         "\u0626\u0648": {
+           "isolated": "\uFBEE",
+           "final": "\uFBEF"
+         },
+         "\u0626\u06C7": {
+           "isolated": "\uFBF0",
+           "final": "\uFBF1"
+         },
+         "\u0626\u06C6": {
+           "isolated": "\uFBF2",
+           "final": "\uFBF3"
+         },
+         "\u0626\u06C8": {
+           "isolated": "\uFBF4",
+           "final": "\uFBF5"
+         },
+         "\u0626\u06D0": {
+           "isolated": "\uFBF6",
+           "final": "\uFBF7",
+           "initial": "\uFBF8"
+         },
+         "\u0626\u0649": {
+           "uighur_kirghiz": {
+             "isolated": "\uFBF9",
+             "final": "\uFBFA",
+             "initial": "\uFBFB"
+           },
+           "isolated": "\uFC03",
+           "final": "\uFC68"
+         },
+         "\u0626\u062C": {
+           "isolated": "\uFC00",
+           "initial": "\uFC97"
+         },
+         "\u0626\u062D": {
+           "isolated": "\uFC01",
+           "initial": "\uFC98"
+         },
+         "\u0626\u0645": {
+           "isolated": "\uFC02",
+           "final": "\uFC66",
+           "initial": "\uFC9A",
+           "medial": "\uFCDF"
+         },
+         "\u0626\u064A": {
+           "isolated": "\uFC04",
+           "final": "\uFC69"
+         },
+         "\u0628\u062C": {
+           "isolated": "\uFC05",
+           "initial": "\uFC9C"
+         },
+         "\u0628\u062D": {
+           "isolated": "\uFC06",
+           "initial": "\uFC9D"
+         },
+         "\u0628\u062E": {
+           "isolated": "\uFC07",
+           "initial": "\uFC9E"
+         },
+         "\u0628\u0645": {
+           "isolated": "\uFC08",
+           "final": "\uFC6C",
+           "initial": "\uFC9F",
+           "medial": "\uFCE1"
+         },
+         "\u0628\u0649": {
+           "isolated": "\uFC09",
+           "final": "\uFC6E"
+         },
+         "\u0628\u064A": {
+           "isolated": "\uFC0A",
+           "final": "\uFC6F"
+         },
+         "\u062A\u062C": {
+           "isolated": "\uFC0B",
+           "initial": "\uFCA1"
+         },
+         "\u062A\u062D": {
+           "isolated": "\uFC0C",
+           "initial": "\uFCA2"
+         },
+         "\u062A\u062E": {
+           "isolated": "\uFC0D",
+           "initial": "\uFCA3"
+         },
+         "\u062A\u0645": {
+           "isolated": "\uFC0E",
+           "final": "\uFC72",
+           "initial": "\uFCA4",
+           "medial": "\uFCE3"
+         },
+         "\u062A\u0649": {
+           "isolated": "\uFC0F",
+           "final": "\uFC74"
+         },
+         "\u062A\u064A": {
+           "isolated": "\uFC10",
+           "final": "\uFC75"
+         },
+         "\u062B\u062C": {
+           "isolated": "\uFC11"
+         },
+         "\u062B\u0645": {
+           "isolated": "\uFC12",
+           "final": "\uFC78",
+           "initial": "\uFCA6",
+           "medial": "\uFCE5"
+         },
+         "\u062B\u0649": {
+           "isolated": "\uFC13",
+           "final": "\uFC7A"
+         },
+         "\u062B\u0648": {
+           "isolated": "\uFC14"
+         },
+         "\u062C\u062D": {
+           "isolated": "\uFC15",
+           "initial": "\uFCA7"
+         },
+         "\u062C\u0645": {
+           "isolated": "\uFC16",
+           "initial": "\uFCA8"
+         },
+         "\u062D\u062C": {
+           "isolated": "\uFC17",
+           "initial": "\uFCA9"
+         },
+         "\u062D\u0645": {
+           "isolated": "\uFC18",
+           "initial": "\uFCAA"
+         },
+         "\u062E\u062C": {
+           "isolated": "\uFC19",
+           "initial": "\uFCAB"
+         },
+         "\u062E\u062D": {
+           "isolated": "\uFC1A"
+         },
+         "\u062E\u0645": {
+           "isolated": "\uFC1B",
+           "initial": "\uFCAC"
+         },
+         "\u0633\u062C": {
+           "isolated": "\uFC1C",
+           "initial": "\uFCAD",
+           "medial": "\uFD34"
+         },
+         "\u0633\u062D": {
+           "isolated": "\uFC1D",
+           "initial": "\uFCAE",
+           "medial": "\uFD35"
+         },
+         "\u0633\u062E": {
+           "isolated": "\uFC1E",
+           "initial": "\uFCAF",
+           "medial": "\uFD36"
+         },
+         "\u0633\u0645": {
+           "isolated": "\uFC1F",
+           "initial": "\uFCB0",
+           "medial": "\uFCE7"
+         },
+         "\u0635\u062D": {
+           "isolated": "\uFC20",
+           "initial": "\uFCB1"
+         },
+         "\u0635\u0645": {
+           "isolated": "\uFC21",
+           "initial": "\uFCB3"
+         },
+         "\u0636\u062C": {
+           "isolated": "\uFC22",
+           "initial": "\uFCB4"
+         },
+         "\u0636\u062D": {
+           "isolated": "\uFC23",
+           "initial": "\uFCB5"
+         },
+         "\u0636\u062E": {
+           "isolated": "\uFC24",
+           "initial": "\uFCB6"
+         },
+         "\u0636\u0645": {
+           "isolated": "\uFC25",
+           "initial": "\uFCB7"
+         },
+         "\u0637\u062D": {
+           "isolated": "\uFC26",
+           "initial": "\uFCB8"
+         },
+         "\u0637\u0645": {
+           "isolated": "\uFC27",
+           "initial": "\uFD33",
+           "medial": "\uFD3A"
+         },
+         "\u0638\u0645": {
+           "isolated": "\uFC28",
+           "initial": "\uFCB9",
+           "medial": "\uFD3B"
+         },
+         "\u0639\u062C": {
+           "isolated": "\uFC29",
+           "initial": "\uFCBA"
+         },
+         "\u0639\u0645": {
+           "isolated": "\uFC2A",
+           "initial": "\uFCBB"
+         },
+         "\u063A\u062C": {
+           "isolated": "\uFC2B",
+           "initial": "\uFCBC"
+         },
+         "\u063A\u0645": {
+           "isolated": "\uFC2C",
+           "initial": "\uFCBD"
+         },
+         "\u0641\u062C": {
+           "isolated": "\uFC2D",
+           "initial": "\uFCBE"
+         },
+         "\u0641\u062D": {
+           "isolated": "\uFC2E",
+           "initial": "\uFCBF"
+         },
+         "\u0641\u062E": {
+           "isolated": "\uFC2F",
+           "initial": "\uFCC0"
+         },
+         "\u0641\u0645": {
+           "isolated": "\uFC30",
+           "initial": "\uFCC1"
+         },
+         "\u0641\u0649": {
+           "isolated": "\uFC31",
+           "final": "\uFC7C"
+         },
+         "\u0641\u064A": {
+           "isolated": "\uFC32",
+           "final": "\uFC7D"
+         },
+         "\u0642\u062D": {
+           "isolated": "\uFC33",
+           "initial": "\uFCC2"
+         },
+         "\u0642\u0645": {
+           "isolated": "\uFC34",
+           "initial": "\uFCC3"
+         },
+         "\u0642\u0649": {
+           "isolated": "\uFC35",
+           "final": "\uFC7E"
+         },
+         "\u0642\u064A": {
+           "isolated": "\uFC36",
+           "final": "\uFC7F"
+         },
+         "\u0643\u0627": {
+           "isolated": "\uFC37",
+           "final": "\uFC80"
+         },
+         "\u0643\u062C": {
+           "isolated": "\uFC38",
+           "initial": "\uFCC4"
+         },
+         "\u0643\u062D": {
+           "isolated": "\uFC39",
+           "initial": "\uFCC5"
+         },
+         "\u0643\u062E": {
+           "isolated": "\uFC3A",
+           "initial": "\uFCC6"
+         },
+         "\u0643\u0644": {
+           "isolated": "\uFC3B",
+           "final": "\uFC81",
+           "initial": "\uFCC7",
+           "medial": "\uFCEB"
+         },
+         "\u0643\u0645": {
+           "isolated": "\uFC3C",
+           "final": "\uFC82",
+           "initial": "\uFCC8",
+           "medial": "\uFCEC"
+         },
+         "\u0643\u0649": {
+           "isolated": "\uFC3D",
+           "final": "\uFC83"
+         },
+         "\u0643\u064A": {
+           "isolated": "\uFC3E",
+           "final": "\uFC84"
+         },
+         "\u0644\u062C": {
+           "isolated": "\uFC3F",
+           "initial": "\uFCC9"
+         },
+         "\u0644\u062D": {
+           "isolated": "\uFC40",
+           "initial": "\uFCCA"
+         },
+         "\u0644\u062E": {
+           "isolated": "\uFC41",
+           "initial": "\uFCCB"
+         },
+         "\u0644\u0645": {
+           "isolated": "\uFC42",
+           "final": "\uFC85",
+           "initial": "\uFCCC",
+           "medial": "\uFCED"
+         },
+         "\u0644\u0649": {
+           "isolated": "\uFC43",
+           "final": "\uFC86"
+         },
+         "\u0644\u064A": {
+           "isolated": "\uFC44",
+           "final": "\uFC87"
+         },
+         "\u0645\u062C": {
+           "isolated": "\uFC45",
+           "initial": "\uFCCE"
+         },
+         "\u0645\u062D": {
+           "isolated": "\uFC46",
+           "initial": "\uFCCF"
+         },
+         "\u0645\u062E": {
+           "isolated": "\uFC47",
+           "initial": "\uFCD0"
+         },
+         "\u0645\u0645": {
+           "isolated": "\uFC48",
+           "final": "\uFC89",
+           "initial": "\uFCD1"
+         },
+         "\u0645\u0649": {
+           "isolated": "\uFC49"
+         },
+         "\u0645\u064A": {
+           "isolated": "\uFC4A"
+         },
+         "\u0646\u062C": {
+           "isolated": "\uFC4B",
+           "initial": "\uFCD2"
+         },
+         "\u0646\u062D": {
+           "isolated": "\uFC4C",
+           "initial": "\uFCD3"
+         },
+         "\u0646\u062E": {
+           "isolated": "\uFC4D",
+           "initial": "\uFCD4"
+         },
+         "\u0646\u0645": {
+           "isolated": "\uFC4E",
+           "final": "\uFC8C",
+           "initial": "\uFCD5",
+           "medial": "\uFCEE"
+         },
+         "\u0646\u0649": {
+           "isolated": "\uFC4F",
+           "final": "\uFC8E"
+         },
+         "\u0646\u064A": {
+           "isolated": "\uFC50",
+           "final": "\uFC8F"
+         },
+         "\u0647\u062C": {
+           "isolated": "\uFC51",
+           "initial": "\uFCD7"
+         },
+         "\u0647\u0645": {
+           "isolated": "\uFC52",
+           "initial": "\uFCD8"
+         },
+         "\u0647\u0649": {
+           "isolated": "\uFC53"
+         },
+         "\u0647\u064A": {
+           "isolated": "\uFC54"
+         },
+         "\u064A\u062C": {
+           "isolated": "\uFC55",
+           "initial": "\uFCDA"
+         },
+         "\u064A\u062D": {
+           "isolated": "\uFC56",
+           "initial": "\uFCDB"
+         },
+         "\u064A\u062E": {
+           "isolated": "\uFC57",
+           "initial": "\uFCDC"
+         },
+         "\u064A\u0645": {
+           "isolated": "\uFC58",
+           "final": "\uFC93",
+           "initial": "\uFCDD",
+           "medial": "\uFCF0"
+         },
+         "\u064A\u0649": {
+           "isolated": "\uFC59",
+           "final": "\uFC95"
+         },
+         "\u064A\u064A": {
+           "isolated": "\uFC5A",
+           "final": "\uFC96"
+         },
+         "\u0630\u0670": {
+           "isolated": "\uFC5B"
+         },
+         "\u0631\u0670": {
+           "isolated": "\uFC5C"
+         },
+         "\u0649\u0670": {
+           "isolated": "\uFC5D",
+           "final": "\uFC90"
+         },
+         "\u064C\u0651": {
+           "isolated": "\uFC5E"
+         },
+         "\u064D\u0651": {
+           "isolated": "\uFC5F"
+         },
+         "\u064E\u0651": {
+           "isolated": "\uFC60"
+         },
+         "\u064F\u0651": {
+           "isolated": "\uFC61"
+         },
+         "\u0650\u0651": {
+           "isolated": "\uFC62"
+         },
+         "\u0651\u0670": {
+           "isolated": "\uFC63"
+         },
+         "\u0626\u0631": {
+           "final": "\uFC64"
+         },
+         "\u0626\u0632": {
+           "final": "\uFC65"
+         },
+         "\u0626\u0646": {
+           "final": "\uFC67"
+         },
+         "\u0628\u0631": {
+           "final": "\uFC6A"
+         },
+         "\u0628\u0632": {
+           "final": "\uFC6B"
+         },
+         "\u0628\u0646": {
+           "final": "\uFC6D"
+         },
+         "\u062A\u0631": {
+           "final": "\uFC70"
+         },
+         "\u062A\u0632": {
+           "final": "\uFC71"
+         },
+         "\u062A\u0646": {
+           "final": "\uFC73"
+         },
+         "\u062B\u0631": {
+           "final": "\uFC76"
+         },
+         "\u062B\u0632": {
+           "final": "\uFC77"
+         },
+         "\u062B\u0646": {
+           "final": "\uFC79"
+         },
+         "\u062B\u064A": {
+           "final": "\uFC7B"
+         },
+         "\u0645\u0627": {
+           "final": "\uFC88"
+         },
+         "\u0646\u0631": {
+           "final": "\uFC8A"
+         },
+         "\u0646\u0632": {
+           "final": "\uFC8B"
+         },
+         "\u0646\u0646": {
+           "final": "\uFC8D"
+         },
+         "\u064A\u0631": {
+           "final": "\uFC91"
+         },
+         "\u064A\u0632": {
+           "final": "\uFC92"
+         },
+         "\u064A\u0646": {
+           "final": "\uFC94"
+         },
+         "\u0626\u062E": {
+           "initial": "\uFC99"
+         },
+         "\u0626\u0647": {
+           "initial": "\uFC9B",
+           "medial": "\uFCE0"
+         },
+         "\u0628\u0647": {
+           "initial": "\uFCA0",
+           "medial": "\uFCE2"
+         },
+         "\u062A\u0647": {
+           "initial": "\uFCA5",
+           "medial": "\uFCE4"
+         },
+         "\u0635\u062E": {
+           "initial": "\uFCB2"
+         },
+         "\u0644\u0647": {
+           "initial": "\uFCCD"
+         },
+         "\u0646\u0647": {
+           "initial": "\uFCD6",
+           "medial": "\uFCEF"
+         },
+         "\u0647\u0670": {
+           "initial": "\uFCD9"
+         },
+         "\u064A\u0647": {
+           "initial": "\uFCDE",
+           "medial": "\uFCF1"
+         },
+         "\u062B\u0647": {
+           "medial": "\uFCE6"
+         },
+         "\u0633\u0647": {
+           "medial": "\uFCE8",
+           "initial": "\uFD31"
+         },
+         "\u0634\u0645": {
+           "medial": "\uFCE9",
+           "isolated": "\uFD0C",
+           "final": "\uFD28",
+           "initial": "\uFD30"
+         },
+         "\u0634\u0647": {
+           "medial": "\uFCEA",
+           "initial": "\uFD32"
+         },
+         "\u0640\u064E\u0651": {
+           "medial": "\uFCF2"
+         },
+         "\u0640\u064F\u0651": {
+           "medial": "\uFCF3"
+         },
+         "\u0640\u0650\u0651": {
+           "medial": "\uFCF4"
+         },
+         "\u0637\u0649": {
+           "isolated": "\uFCF5",
+           "final": "\uFD11"
+         },
+         "\u0637\u064A": {
+           "isolated": "\uFCF6",
+           "final": "\uFD12"
+         },
+         "\u0639\u0649": {
+           "isolated": "\uFCF7",
+           "final": "\uFD13"
+         },
+         "\u0639\u064A": {
+           "isolated": "\uFCF8",
+           "final": "\uFD14"
+         },
+         "\u063A\u0649": {
+           "isolated": "\uFCF9",
+           "final": "\uFD15"
+         },
+         "\u063A\u064A": {
+           "isolated": "\uFCFA",
+           "final": "\uFD16"
+         },
+         "\u0633\u0649": {
+           "isolated": "\uFCFB"
+         },
+         "\u0633\u064A": {
+           "isolated": "\uFCFC",
+           "final": "\uFD18"
+         },
+         "\u0634\u0649": {
+           "isolated": "\uFCFD",
+           "final": "\uFD19"
+         },
+         "\u0634\u064A": {
+           "isolated": "\uFCFE",
+           "final": "\uFD1A"
+         },
+         "\u062D\u0649": {
+           "isolated": "\uFCFF",
+           "final": "\uFD1B"
+         },
+         "\u062D\u064A": {
+           "isolated": "\uFD00",
+           "final": "\uFD1C"
+         },
+         "\u062C\u0649": {
+           "isolated": "\uFD01",
+           "final": "\uFD1D"
+         },
+         "\u062C\u064A": {
+           "isolated": "\uFD02",
+           "final": "\uFD1E"
+         },
+         "\u062E\u0649": {
+           "isolated": "\uFD03",
+           "final": "\uFD1F"
+         },
+         "\u062E\u064A": {
+           "isolated": "\uFD04",
+           "final": "\uFD20"
+         },
+         "\u0635\u0649": {
+           "isolated": "\uFD05",
+           "final": "\uFD21"
+         },
+         "\u0635\u064A": {
+           "isolated": "\uFD06",
+           "final": "\uFD22"
+         },
+         "\u0636\u0649": {
+           "isolated": "\uFD07",
+           "final": "\uFD23"
+         },
+         "\u0636\u064A": {
+           "isolated": "\uFD08",
+           "final": "\uFD24"
+         },
+         "\u0634\u062C": {
+           "isolated": "\uFD09",
+           "final": "\uFD25",
+           "initial": "\uFD2D",
+           "medial": "\uFD37"
+         },
+         "\u0634\u062D": {
+           "isolated": "\uFD0A",
+           "final": "\uFD26",
+           "initial": "\uFD2E",
+           "medial": "\uFD38"
+         },
+         "\u0634\u062E": {
+           "isolated": "\uFD0B",
+           "final": "\uFD27",
+           "initial": "\uFD2F",
+           "medial": "\uFD39"
+         },
+         "\u0634\u0631": {
+           "isolated": "\uFD0D",
+           "final": "\uFD29"
+         },
+         "\u0633\u0631": {
+           "isolated": "\uFD0E",
+           "final": "\uFD2A"
+         },
+         "\u0635\u0631": {
+           "isolated": "\uFD0F",
+           "final": "\uFD2B"
+         },
+         "\u0636\u0631": {
+           "isolated": "\uFD10",
+           "final": "\uFD2C"
+         },
+         "\u0633\u0639": {
+           "final": "\uFD17"
+         },
+         "\u062A\u062C\u0645": {
+           "initial": "\uFD50"
+         },
+         "\u062A\u062D\u062C": {
+           "final": "\uFD51",
+           "initial": "\uFD52"
+         },
+         "\u062A\u062D\u0645": {
+           "initial": "\uFD53"
+         },
+         "\u062A\u062E\u0645": {
+           "initial": "\uFD54"
+         },
+         "\u062A\u0645\u062C": {
+           "initial": "\uFD55"
+         },
+         "\u062A\u0645\u062D": {
+           "initial": "\uFD56"
+         },
+         "\u062A\u0645\u062E": {
+           "initial": "\uFD57"
+         },
+         "\u062C\u0645\u062D": {
+           "final": "\uFD58",
+           "initial": "\uFD59"
+         },
+         "\u062D\u0645\u064A": {
+           "final": "\uFD5A"
+         },
+         "\u062D\u0645\u0649": {
+           "final": "\uFD5B"
+         },
+         "\u0633\u062D\u062C": {
+           "initial": "\uFD5C"
+         },
+         "\u0633\u062C\u062D": {
+           "initial": "\uFD5D"
+         },
+         "\u0633\u062C\u0649": {
+           "final": "\uFD5E"
+         },
+         "\u0633\u0645\u062D": {
+           "final": "\uFD5F",
+           "initial": "\uFD60"
+         },
+         "\u0633\u0645\u062C": {
+           "initial": "\uFD61"
+         },
+         "\u0633\u0645\u0645": {
+           "final": "\uFD62",
+           "initial": "\uFD63"
+         },
+         "\u0635\u062D\u062D": {
+           "final": "\uFD64",
+           "initial": "\uFD65"
+         },
+         "\u0635\u0645\u0645": {
+           "final": "\uFD66",
+           "initial": "\uFDC5"
+         },
+         "\u0634\u062D\u0645": {
+           "final": "\uFD67",
+           "initial": "\uFD68"
+         },
+         "\u0634\u062C\u064A": {
+           "final": "\uFD69"
+         },
+         "\u0634\u0645\u062E": {
+           "final": "\uFD6A",
+           "initial": "\uFD6B"
+         },
+         "\u0634\u0645\u0645": {
+           "final": "\uFD6C",
+           "initial": "\uFD6D"
+         },
+         "\u0636\u062D\u0649": {
+           "final": "\uFD6E"
+         },
+         "\u0636\u062E\u0645": {
+           "final": "\uFD6F",
+           "initial": "\uFD70"
+         },
+         "\u0636\u0645\u062D": {
+           "final": "\uFD71"
+         },
+         "\u0637\u0645\u062D": {
+           "initial": "\uFD72"
+         },
+         "\u0637\u0645\u0645": {
+           "initial": "\uFD73"
+         },
+         "\u0637\u0645\u064A": {
+           "final": "\uFD74"
+         },
+         "\u0639\u062C\u0645": {
+           "final": "\uFD75",
+           "initial": "\uFDC4"
+         },
+         "\u0639\u0645\u0645": {
+           "final": "\uFD76",
+           "initial": "\uFD77"
+         },
+         "\u0639\u0645\u0649": {
+           "final": "\uFD78"
+         },
+         "\u063A\u0645\u0645": {
+           "final": "\uFD79"
+         },
+         "\u063A\u0645\u064A": {
+           "final": "\uFD7A"
+         },
+         "\u063A\u0645\u0649": {
+           "final": "\uFD7B"
+         },
+         "\u0641\u062E\u0645": {
+           "final": "\uFD7C",
+           "initial": "\uFD7D"
+         },
+         "\u0642\u0645\u062D": {
+           "final": "\uFD7E",
+           "initial": "\uFDB4"
+         },
+         "\u0642\u0645\u0645": {
+           "final": "\uFD7F"
+         },
+         "\u0644\u062D\u0645": {
+           "final": "\uFD80",
+           "initial": "\uFDB5"
+         },
+         "\u0644\u062D\u064A": {
+           "final": "\uFD81"
+         },
+         "\u0644\u062D\u0649": {
+           "final": "\uFD82"
+         },
+         "\u0644\u062C\u062C": {
+           "initial": "\uFD83",
+           "final": "\uFD84"
+         },
+         "\u0644\u062E\u0645": {
+           "final": "\uFD85",
+           "initial": "\uFD86"
+         },
+         "\u0644\u0645\u062D": {
+           "final": "\uFD87",
+           "initial": "\uFD88"
+         },
+         "\u0645\u062D\u062C": {
+           "initial": "\uFD89"
+         },
+         "\u0645\u062D\u0645": {
+           "initial": "\uFD8A"
+         },
+         "\u0645\u062D\u064A": {
+           "final": "\uFD8B"
+         },
+         "\u0645\u062C\u062D": {
+           "initial": "\uFD8C"
+         },
+         "\u0645\u062C\u0645": {
+           "initial": "\uFD8D"
+         },
+         "\u0645\u062E\u062C": {
+           "initial": "\uFD8E"
+         },
+         "\u0645\u062E\u0645": {
+           "initial": "\uFD8F"
+         },
+         "\u0645\u062C\u062E": {
+           "initial": "\uFD92"
+         },
+         "\u0647\u0645\u062C": {
+           "initial": "\uFD93"
+         },
+         "\u0647\u0645\u0645": {
+           "initial": "\uFD94"
+         },
+         "\u0646\u062D\u0645": {
+           "initial": "\uFD95"
+         },
+         "\u0646\u062D\u0649": {
+           "final": "\uFD96"
+         },
+         "\u0646\u062C\u0645": {
+           "final": "\uFD97",
+           "initial": "\uFD98"
+         },
+         "\u0646\u062C\u0649": {
+           "final": "\uFD99"
+         },
+         "\u0646\u0645\u064A": {
+           "final": "\uFD9A"
+         },
+         "\u0646\u0645\u0649": {
+           "final": "\uFD9B"
+         },
+         "\u064A\u0645\u0645": {
+           "final": "\uFD9C",
+           "initial": "\uFD9D"
+         },
+         "\u0628\u062E\u064A": {
+           "final": "\uFD9E"
+         },
+         "\u062A\u062C\u064A": {
+           "final": "\uFD9F"
+         },
+         "\u062A\u062C\u0649": {
+           "final": "\uFDA0"
+         },
+         "\u062A\u062E\u064A": {
+           "final": "\uFDA1"
+         },
+         "\u062A\u062E\u0649": {
+           "final": "\uFDA2"
+         },
+         "\u062A\u0645\u064A": {
+           "final": "\uFDA3"
+         },
+         "\u062A\u0645\u0649": {
+           "final": "\uFDA4"
+         },
+         "\u062C\u0645\u064A": {
+           "final": "\uFDA5"
+         },
+         "\u062C\u062D\u0649": {
+           "final": "\uFDA6"
+         },
+         "\u062C\u0645\u0649": {
+           "final": "\uFDA7"
+         },
+         "\u0633\u062E\u0649": {
+           "final": "\uFDA8"
+         },
+         "\u0635\u062D\u064A": {
+           "final": "\uFDA9"
+         },
+         "\u0634\u062D\u064A": {
+           "final": "\uFDAA"
+         },
+         "\u0636\u062D\u064A": {
+           "final": "\uFDAB"
+         },
+         "\u0644\u062C\u064A": {
+           "final": "\uFDAC"
+         },
+         "\u0644\u0645\u064A": {
+           "final": "\uFDAD"
+         },
+         "\u064A\u062D\u064A": {
+           "final": "\uFDAE"
+         },
+         "\u064A\u062C\u064A": {
+           "final": "\uFDAF"
+         },
+         "\u064A\u0645\u064A": {
+           "final": "\uFDB0"
+         },
+         "\u0645\u0645\u064A": {
+           "final": "\uFDB1"
+         },
+         "\u0642\u0645\u064A": {
+           "final": "\uFDB2"
+         },
+         "\u0646\u062D\u064A": {
+           "final": "\uFDB3"
+         },
+         "\u0639\u0645\u064A": {
+           "final": "\uFDB6"
+         },
+         "\u0643\u0645\u064A": {
+           "final": "\uFDB7"
+         },
+         "\u0646\u062C\u062D": {
+           "initial": "\uFDB8",
+           "final": "\uFDBD"
+         },
+         "\u0645\u062E\u064A": {
+           "final": "\uFDB9"
+         },
+         "\u0644\u062C\u0645": {
+           "initial": "\uFDBA",
+           "final": "\uFDBC"
+         },
+         "\u0643\u0645\u0645": {
+           "final": "\uFDBB",
+           "initial": "\uFDC3"
+         },
+         "\u062C\u062D\u064A": {
+           "final": "\uFDBE"
+         },
+         "\u062D\u062C\u064A": {
+           "final": "\uFDBF"
+         },
+         "\u0645\u062C\u064A": {
+           "final": "\uFDC0"
+         },
+         "\u0641\u0645\u064A": {
+           "final": "\uFDC1"
+         },
+         "\u0628\u062D\u064A": {
+           "final": "\uFDC2"
+         },
+         "\u0633\u062E\u064A": {
+           "final": "\uFDC6"
+         },
+         "\u0646\u062C\u064A": {
+           "final": "\uFDC7"
+         },
+         "\u0644\u0622": {
+           "isolated": "\uFEF5",
+           "final": "\uFEF6"
+         },
+         "\u0644\u0623": {
+           "isolated": "\uFEF7",
+           "final": "\uFEF8"
+         },
+         "\u0644\u0625": {
+           "isolated": "\uFEF9",
+           "final": "\uFEFA"
+         },
+         "\u0644\u0627": {
+           "isolated": "\uFEFB",
+           "final": "\uFEFC"
+         },
+         "words": {
+           "\u0635\u0644\u06D2": "\uFDF0",
+           "\u0642\u0644\u06D2": "\uFDF1",
+           "\u0627\u0644\u0644\u0647": "\uFDF2",
+           "\u0627\u0643\u0628\u0631": "\uFDF3",
+           "\u0645\u062D\u0645\u062F": "\uFDF4",
+           "\u0635\u0644\u0639\u0645": "\uFDF5",
+           "\u0631\u0633\u0648\u0644": "\uFDF6",
+           "\u0639\u0644\u064A\u0647": "\uFDF7",
+           "\u0648\u0633\u0644\u0645": "\uFDF8",
+           "\u0635\u0644\u0649": "\uFDF9",
+           "\u0635\u0644\u0649\u0627\u0644\u0644\u0647\u0639\u0644\u064A\u0647\u0648\u0633\u0644\u0645": "\uFDFA",
+           "\u062C\u0644\u062C\u0644\u0627\u0644\u0647": "\uFDFB",
+           "\u0631\u06CC\u0627\u0644": "\uFDFC"
+         }
+       };
+       var _default$2 = ligatureReference;
+       var unicodeLigatures = /*#__PURE__*/Object.defineProperty({
+         "default": _default$2
+       }, '__esModule', {
+         value: true
        });
 
-       var unicodeLigatures = createCommonjsModule(function (module, exports) {
-
-         Object.defineProperty(exports, "__esModule", {
-           value: true
-         });
-         var ligatureReference = {
-           "\u0626\u0627": {
-             "isolated": "\uFBEA",
-             "final": "\uFBEB"
-           },
-           "\u0626\u06D5": {
-             "isolated": "\uFBEC",
-             "final": "\uFBED"
-           },
-           "\u0626\u0648": {
-             "isolated": "\uFBEE",
-             "final": "\uFBEF"
-           },
-           "\u0626\u06C7": {
-             "isolated": "\uFBF0",
-             "final": "\uFBF1"
-           },
-           "\u0626\u06C6": {
-             "isolated": "\uFBF2",
-             "final": "\uFBF3"
-           },
-           "\u0626\u06C8": {
-             "isolated": "\uFBF4",
-             "final": "\uFBF5"
-           },
-           "\u0626\u06D0": {
-             "isolated": "\uFBF6",
-             "final": "\uFBF7",
-             "initial": "\uFBF8"
-           },
-           "\u0626\u0649": {
-             "uighur_kirghiz": {
-               "isolated": "\uFBF9",
-               "final": "\uFBFA",
-               "initial": "\uFBFB"
-             },
-             "isolated": "\uFC03",
-             "final": "\uFC68"
-           },
-           "\u0626\u062C": {
-             "isolated": "\uFC00",
-             "initial": "\uFC97"
-           },
-           "\u0626\u062D": {
-             "isolated": "\uFC01",
-             "initial": "\uFC98"
-           },
-           "\u0626\u0645": {
-             "isolated": "\uFC02",
-             "final": "\uFC66",
-             "initial": "\uFC9A",
-             "medial": "\uFCDF"
-           },
-           "\u0626\u064A": {
-             "isolated": "\uFC04",
-             "final": "\uFC69"
-           },
-           "\u0628\u062C": {
-             "isolated": "\uFC05",
-             "initial": "\uFC9C"
-           },
-           "\u0628\u062D": {
-             "isolated": "\uFC06",
-             "initial": "\uFC9D"
-           },
-           "\u0628\u062E": {
-             "isolated": "\uFC07",
-             "initial": "\uFC9E"
-           },
-           "\u0628\u0645": {
-             "isolated": "\uFC08",
-             "final": "\uFC6C",
-             "initial": "\uFC9F",
-             "medial": "\uFCE1"
-           },
-           "\u0628\u0649": {
-             "isolated": "\uFC09",
-             "final": "\uFC6E"
-           },
-           "\u0628\u064A": {
-             "isolated": "\uFC0A",
-             "final": "\uFC6F"
-           },
-           "\u062A\u062C": {
-             "isolated": "\uFC0B",
-             "initial": "\uFCA1"
-           },
-           "\u062A\u062D": {
-             "isolated": "\uFC0C",
-             "initial": "\uFCA2"
-           },
-           "\u062A\u062E": {
-             "isolated": "\uFC0D",
-             "initial": "\uFCA3"
-           },
-           "\u062A\u0645": {
-             "isolated": "\uFC0E",
-             "final": "\uFC72",
-             "initial": "\uFCA4",
-             "medial": "\uFCE3"
-           },
-           "\u062A\u0649": {
-             "isolated": "\uFC0F",
-             "final": "\uFC74"
-           },
-           "\u062A\u064A": {
-             "isolated": "\uFC10",
-             "final": "\uFC75"
-           },
-           "\u062B\u062C": {
-             "isolated": "\uFC11"
-           },
-           "\u062B\u0645": {
-             "isolated": "\uFC12",
-             "final": "\uFC78",
-             "initial": "\uFCA6",
-             "medial": "\uFCE5"
-           },
-           "\u062B\u0649": {
-             "isolated": "\uFC13",
-             "final": "\uFC7A"
-           },
-           "\u062B\u0648": {
-             "isolated": "\uFC14"
-           },
-           "\u062C\u062D": {
-             "isolated": "\uFC15",
-             "initial": "\uFCA7"
-           },
-           "\u062C\u0645": {
-             "isolated": "\uFC16",
-             "initial": "\uFCA8"
-           },
-           "\u062D\u062C": {
-             "isolated": "\uFC17",
-             "initial": "\uFCA9"
-           },
-           "\u062D\u0645": {
-             "isolated": "\uFC18",
-             "initial": "\uFCAA"
-           },
-           "\u062E\u062C": {
-             "isolated": "\uFC19",
-             "initial": "\uFCAB"
-           },
-           "\u062E\u062D": {
-             "isolated": "\uFC1A"
-           },
-           "\u062E\u0645": {
-             "isolated": "\uFC1B",
-             "initial": "\uFCAC"
-           },
-           "\u0633\u062C": {
-             "isolated": "\uFC1C",
-             "initial": "\uFCAD",
-             "medial": "\uFD34"
-           },
-           "\u0633\u062D": {
-             "isolated": "\uFC1D",
-             "initial": "\uFCAE",
-             "medial": "\uFD35"
-           },
-           "\u0633\u062E": {
-             "isolated": "\uFC1E",
-             "initial": "\uFCAF",
-             "medial": "\uFD36"
-           },
-           "\u0633\u0645": {
-             "isolated": "\uFC1F",
-             "initial": "\uFCB0",
-             "medial": "\uFCE7"
-           },
-           "\u0635\u062D": {
-             "isolated": "\uFC20",
-             "initial": "\uFCB1"
-           },
-           "\u0635\u0645": {
-             "isolated": "\uFC21",
-             "initial": "\uFCB3"
-           },
-           "\u0636\u062C": {
-             "isolated": "\uFC22",
-             "initial": "\uFCB4"
-           },
-           "\u0636\u062D": {
-             "isolated": "\uFC23",
-             "initial": "\uFCB5"
-           },
-           "\u0636\u062E": {
-             "isolated": "\uFC24",
-             "initial": "\uFCB6"
-           },
-           "\u0636\u0645": {
-             "isolated": "\uFC25",
-             "initial": "\uFCB7"
-           },
-           "\u0637\u062D": {
-             "isolated": "\uFC26",
-             "initial": "\uFCB8"
-           },
-           "\u0637\u0645": {
-             "isolated": "\uFC27",
-             "initial": "\uFD33",
-             "medial": "\uFD3A"
-           },
-           "\u0638\u0645": {
-             "isolated": "\uFC28",
-             "initial": "\uFCB9",
-             "medial": "\uFD3B"
-           },
-           "\u0639\u062C": {
-             "isolated": "\uFC29",
-             "initial": "\uFCBA"
-           },
-           "\u0639\u0645": {
-             "isolated": "\uFC2A",
-             "initial": "\uFCBB"
-           },
-           "\u063A\u062C": {
-             "isolated": "\uFC2B",
-             "initial": "\uFCBC"
-           },
-           "\u063A\u0645": {
-             "isolated": "\uFC2C",
-             "initial": "\uFCBD"
-           },
-           "\u0641\u062C": {
-             "isolated": "\uFC2D",
-             "initial": "\uFCBE"
-           },
-           "\u0641\u062D": {
-             "isolated": "\uFC2E",
-             "initial": "\uFCBF"
-           },
-           "\u0641\u062E": {
-             "isolated": "\uFC2F",
-             "initial": "\uFCC0"
-           },
-           "\u0641\u0645": {
-             "isolated": "\uFC30",
-             "initial": "\uFCC1"
-           },
-           "\u0641\u0649": {
-             "isolated": "\uFC31",
-             "final": "\uFC7C"
-           },
-           "\u0641\u064A": {
-             "isolated": "\uFC32",
-             "final": "\uFC7D"
-           },
-           "\u0642\u062D": {
-             "isolated": "\uFC33",
-             "initial": "\uFCC2"
-           },
-           "\u0642\u0645": {
-             "isolated": "\uFC34",
-             "initial": "\uFCC3"
-           },
-           "\u0642\u0649": {
-             "isolated": "\uFC35",
-             "final": "\uFC7E"
-           },
-           "\u0642\u064A": {
-             "isolated": "\uFC36",
-             "final": "\uFC7F"
-           },
-           "\u0643\u0627": {
-             "isolated": "\uFC37",
-             "final": "\uFC80"
-           },
-           "\u0643\u062C": {
-             "isolated": "\uFC38",
-             "initial": "\uFCC4"
-           },
-           "\u0643\u062D": {
-             "isolated": "\uFC39",
-             "initial": "\uFCC5"
-           },
-           "\u0643\u062E": {
-             "isolated": "\uFC3A",
-             "initial": "\uFCC6"
-           },
-           "\u0643\u0644": {
-             "isolated": "\uFC3B",
-             "final": "\uFC81",
-             "initial": "\uFCC7",
-             "medial": "\uFCEB"
-           },
-           "\u0643\u0645": {
-             "isolated": "\uFC3C",
-             "final": "\uFC82",
-             "initial": "\uFCC8",
-             "medial": "\uFCEC"
-           },
-           "\u0643\u0649": {
-             "isolated": "\uFC3D",
-             "final": "\uFC83"
-           },
-           "\u0643\u064A": {
-             "isolated": "\uFC3E",
-             "final": "\uFC84"
-           },
-           "\u0644\u062C": {
-             "isolated": "\uFC3F",
-             "initial": "\uFCC9"
-           },
-           "\u0644\u062D": {
-             "isolated": "\uFC40",
-             "initial": "\uFCCA"
-           },
-           "\u0644\u062E": {
-             "isolated": "\uFC41",
-             "initial": "\uFCCB"
-           },
-           "\u0644\u0645": {
-             "isolated": "\uFC42",
-             "final": "\uFC85",
-             "initial": "\uFCCC",
-             "medial": "\uFCED"
-           },
-           "\u0644\u0649": {
-             "isolated": "\uFC43",
-             "final": "\uFC86"
-           },
-           "\u0644\u064A": {
-             "isolated": "\uFC44",
-             "final": "\uFC87"
-           },
-           "\u0645\u062C": {
-             "isolated": "\uFC45",
-             "initial": "\uFCCE"
-           },
-           "\u0645\u062D": {
-             "isolated": "\uFC46",
-             "initial": "\uFCCF"
-           },
-           "\u0645\u062E": {
-             "isolated": "\uFC47",
-             "initial": "\uFCD0"
-           },
-           "\u0645\u0645": {
-             "isolated": "\uFC48",
-             "final": "\uFC89",
-             "initial": "\uFCD1"
-           },
-           "\u0645\u0649": {
-             "isolated": "\uFC49"
-           },
-           "\u0645\u064A": {
-             "isolated": "\uFC4A"
-           },
-           "\u0646\u062C": {
-             "isolated": "\uFC4B",
-             "initial": "\uFCD2"
-           },
-           "\u0646\u062D": {
-             "isolated": "\uFC4C",
-             "initial": "\uFCD3"
-           },
-           "\u0646\u062E": {
-             "isolated": "\uFC4D",
-             "initial": "\uFCD4"
-           },
-           "\u0646\u0645": {
-             "isolated": "\uFC4E",
-             "final": "\uFC8C",
-             "initial": "\uFCD5",
-             "medial": "\uFCEE"
-           },
-           "\u0646\u0649": {
-             "isolated": "\uFC4F",
-             "final": "\uFC8E"
-           },
-           "\u0646\u064A": {
-             "isolated": "\uFC50",
-             "final": "\uFC8F"
-           },
-           "\u0647\u062C": {
-             "isolated": "\uFC51",
-             "initial": "\uFCD7"
-           },
-           "\u0647\u0645": {
-             "isolated": "\uFC52",
-             "initial": "\uFCD8"
-           },
-           "\u0647\u0649": {
-             "isolated": "\uFC53"
-           },
-           "\u0647\u064A": {
-             "isolated": "\uFC54"
-           },
-           "\u064A\u062C": {
-             "isolated": "\uFC55",
-             "initial": "\uFCDA"
-           },
-           "\u064A\u062D": {
-             "isolated": "\uFC56",
-             "initial": "\uFCDB"
-           },
-           "\u064A\u062E": {
-             "isolated": "\uFC57",
-             "initial": "\uFCDC"
-           },
-           "\u064A\u0645": {
-             "isolated": "\uFC58",
-             "final": "\uFC93",
-             "initial": "\uFCDD",
-             "medial": "\uFCF0"
-           },
-           "\u064A\u0649": {
-             "isolated": "\uFC59",
-             "final": "\uFC95"
-           },
-           "\u064A\u064A": {
-             "isolated": "\uFC5A",
-             "final": "\uFC96"
-           },
-           "\u0630\u0670": {
-             "isolated": "\uFC5B"
-           },
-           "\u0631\u0670": {
-             "isolated": "\uFC5C"
-           },
-           "\u0649\u0670": {
-             "isolated": "\uFC5D",
-             "final": "\uFC90"
-           },
-           "\u064C\u0651": {
-             "isolated": "\uFC5E"
-           },
-           "\u064D\u0651": {
-             "isolated": "\uFC5F"
-           },
-           "\u064E\u0651": {
-             "isolated": "\uFC60"
-           },
-           "\u064F\u0651": {
-             "isolated": "\uFC61"
-           },
-           "\u0650\u0651": {
-             "isolated": "\uFC62"
-           },
-           "\u0651\u0670": {
-             "isolated": "\uFC63"
-           },
-           "\u0626\u0631": {
-             "final": "\uFC64"
-           },
-           "\u0626\u0632": {
-             "final": "\uFC65"
-           },
-           "\u0626\u0646": {
-             "final": "\uFC67"
-           },
-           "\u0628\u0631": {
-             "final": "\uFC6A"
-           },
-           "\u0628\u0632": {
-             "final": "\uFC6B"
-           },
-           "\u0628\u0646": {
-             "final": "\uFC6D"
-           },
-           "\u062A\u0631": {
-             "final": "\uFC70"
-           },
-           "\u062A\u0632": {
-             "final": "\uFC71"
-           },
-           "\u062A\u0646": {
-             "final": "\uFC73"
-           },
-           "\u062B\u0631": {
-             "final": "\uFC76"
-           },
-           "\u062B\u0632": {
-             "final": "\uFC77"
-           },
-           "\u062B\u0646": {
-             "final": "\uFC79"
-           },
-           "\u062B\u064A": {
-             "final": "\uFC7B"
-           },
-           "\u0645\u0627": {
-             "final": "\uFC88"
-           },
-           "\u0646\u0631": {
-             "final": "\uFC8A"
-           },
-           "\u0646\u0632": {
-             "final": "\uFC8B"
-           },
-           "\u0646\u0646": {
-             "final": "\uFC8D"
-           },
-           "\u064A\u0631": {
-             "final": "\uFC91"
-           },
-           "\u064A\u0632": {
-             "final": "\uFC92"
-           },
-           "\u064A\u0646": {
-             "final": "\uFC94"
-           },
-           "\u0626\u062E": {
-             "initial": "\uFC99"
-           },
-           "\u0626\u0647": {
-             "initial": "\uFC9B",
-             "medial": "\uFCE0"
-           },
-           "\u0628\u0647": {
-             "initial": "\uFCA0",
-             "medial": "\uFCE2"
-           },
-           "\u062A\u0647": {
-             "initial": "\uFCA5",
-             "medial": "\uFCE4"
-           },
-           "\u0635\u062E": {
-             "initial": "\uFCB2"
-           },
-           "\u0644\u0647": {
-             "initial": "\uFCCD"
-           },
-           "\u0646\u0647": {
-             "initial": "\uFCD6",
-             "medial": "\uFCEF"
-           },
-           "\u0647\u0670": {
-             "initial": "\uFCD9"
-           },
-           "\u064A\u0647": {
-             "initial": "\uFCDE",
-             "medial": "\uFCF1"
-           },
-           "\u062B\u0647": {
-             "medial": "\uFCE6"
-           },
-           "\u0633\u0647": {
-             "medial": "\uFCE8",
-             "initial": "\uFD31"
-           },
-           "\u0634\u0645": {
-             "medial": "\uFCE9",
-             "isolated": "\uFD0C",
-             "final": "\uFD28",
-             "initial": "\uFD30"
-           },
-           "\u0634\u0647": {
-             "medial": "\uFCEA",
-             "initial": "\uFD32"
-           },
-           "\u0640\u064E\u0651": {
-             "medial": "\uFCF2"
-           },
-           "\u0640\u064F\u0651": {
-             "medial": "\uFCF3"
-           },
-           "\u0640\u0650\u0651": {
-             "medial": "\uFCF4"
-           },
-           "\u0637\u0649": {
-             "isolated": "\uFCF5",
-             "final": "\uFD11"
-           },
-           "\u0637\u064A": {
-             "isolated": "\uFCF6",
-             "final": "\uFD12"
-           },
-           "\u0639\u0649": {
-             "isolated": "\uFCF7",
-             "final": "\uFD13"
-           },
-           "\u0639\u064A": {
-             "isolated": "\uFCF8",
-             "final": "\uFD14"
-           },
-           "\u063A\u0649": {
-             "isolated": "\uFCF9",
-             "final": "\uFD15"
-           },
-           "\u063A\u064A": {
-             "isolated": "\uFCFA",
-             "final": "\uFD16"
-           },
-           "\u0633\u0649": {
-             "isolated": "\uFCFB"
-           },
-           "\u0633\u064A": {
-             "isolated": "\uFCFC",
-             "final": "\uFD18"
-           },
-           "\u0634\u0649": {
-             "isolated": "\uFCFD",
-             "final": "\uFD19"
-           },
-           "\u0634\u064A": {
-             "isolated": "\uFCFE",
-             "final": "\uFD1A"
-           },
-           "\u062D\u0649": {
-             "isolated": "\uFCFF",
-             "final": "\uFD1B"
-           },
-           "\u062D\u064A": {
-             "isolated": "\uFD00",
-             "final": "\uFD1C"
-           },
-           "\u062C\u0649": {
-             "isolated": "\uFD01",
-             "final": "\uFD1D"
-           },
-           "\u062C\u064A": {
-             "isolated": "\uFD02",
-             "final": "\uFD1E"
-           },
-           "\u062E\u0649": {
-             "isolated": "\uFD03",
-             "final": "\uFD1F"
-           },
-           "\u062E\u064A": {
-             "isolated": "\uFD04",
-             "final": "\uFD20"
-           },
-           "\u0635\u0649": {
-             "isolated": "\uFD05",
-             "final": "\uFD21"
-           },
-           "\u0635\u064A": {
-             "isolated": "\uFD06",
-             "final": "\uFD22"
-           },
-           "\u0636\u0649": {
-             "isolated": "\uFD07",
-             "final": "\uFD23"
-           },
-           "\u0636\u064A": {
-             "isolated": "\uFD08",
-             "final": "\uFD24"
-           },
-           "\u0634\u062C": {
-             "isolated": "\uFD09",
-             "final": "\uFD25",
-             "initial": "\uFD2D",
-             "medial": "\uFD37"
-           },
-           "\u0634\u062D": {
-             "isolated": "\uFD0A",
-             "final": "\uFD26",
-             "initial": "\uFD2E",
-             "medial": "\uFD38"
-           },
-           "\u0634\u062E": {
-             "isolated": "\uFD0B",
-             "final": "\uFD27",
-             "initial": "\uFD2F",
-             "medial": "\uFD39"
-           },
-           "\u0634\u0631": {
-             "isolated": "\uFD0D",
-             "final": "\uFD29"
-           },
-           "\u0633\u0631": {
-             "isolated": "\uFD0E",
-             "final": "\uFD2A"
-           },
-           "\u0635\u0631": {
-             "isolated": "\uFD0F",
-             "final": "\uFD2B"
-           },
-           "\u0636\u0631": {
-             "isolated": "\uFD10",
-             "final": "\uFD2C"
-           },
-           "\u0633\u0639": {
-             "final": "\uFD17"
-           },
-           "\u062A\u062C\u0645": {
-             "initial": "\uFD50"
-           },
-           "\u062A\u062D\u062C": {
-             "final": "\uFD51",
-             "initial": "\uFD52"
-           },
-           "\u062A\u062D\u0645": {
-             "initial": "\uFD53"
-           },
-           "\u062A\u062E\u0645": {
-             "initial": "\uFD54"
-           },
-           "\u062A\u0645\u062C": {
-             "initial": "\uFD55"
-           },
-           "\u062A\u0645\u062D": {
-             "initial": "\uFD56"
-           },
-           "\u062A\u0645\u062E": {
-             "initial": "\uFD57"
-           },
-           "\u062C\u0645\u062D": {
-             "final": "\uFD58",
-             "initial": "\uFD59"
-           },
-           "\u062D\u0645\u064A": {
-             "final": "\uFD5A"
-           },
-           "\u062D\u0645\u0649": {
-             "final": "\uFD5B"
-           },
-           "\u0633\u062D\u062C": {
-             "initial": "\uFD5C"
-           },
-           "\u0633\u062C\u062D": {
-             "initial": "\uFD5D"
-           },
-           "\u0633\u062C\u0649": {
-             "final": "\uFD5E"
-           },
-           "\u0633\u0645\u062D": {
-             "final": "\uFD5F",
-             "initial": "\uFD60"
-           },
-           "\u0633\u0645\u062C": {
-             "initial": "\uFD61"
-           },
-           "\u0633\u0645\u0645": {
-             "final": "\uFD62",
-             "initial": "\uFD63"
-           },
-           "\u0635\u062D\u062D": {
-             "final": "\uFD64",
-             "initial": "\uFD65"
-           },
-           "\u0635\u0645\u0645": {
-             "final": "\uFD66",
-             "initial": "\uFDC5"
-           },
-           "\u0634\u062D\u0645": {
-             "final": "\uFD67",
-             "initial": "\uFD68"
-           },
-           "\u0634\u062C\u064A": {
-             "final": "\uFD69"
-           },
-           "\u0634\u0645\u062E": {
-             "final": "\uFD6A",
-             "initial": "\uFD6B"
-           },
-           "\u0634\u0645\u0645": {
-             "final": "\uFD6C",
-             "initial": "\uFD6D"
-           },
-           "\u0636\u062D\u0649": {
-             "final": "\uFD6E"
-           },
-           "\u0636\u062E\u0645": {
-             "final": "\uFD6F",
-             "initial": "\uFD70"
-           },
-           "\u0636\u0645\u062D": {
-             "final": "\uFD71"
-           },
-           "\u0637\u0645\u062D": {
-             "initial": "\uFD72"
-           },
-           "\u0637\u0645\u0645": {
-             "initial": "\uFD73"
-           },
-           "\u0637\u0645\u064A": {
-             "final": "\uFD74"
-           },
-           "\u0639\u062C\u0645": {
-             "final": "\uFD75",
-             "initial": "\uFDC4"
-           },
-           "\u0639\u0645\u0645": {
-             "final": "\uFD76",
-             "initial": "\uFD77"
-           },
-           "\u0639\u0645\u0649": {
-             "final": "\uFD78"
-           },
-           "\u063A\u0645\u0645": {
-             "final": "\uFD79"
-           },
-           "\u063A\u0645\u064A": {
-             "final": "\uFD7A"
-           },
-           "\u063A\u0645\u0649": {
-             "final": "\uFD7B"
-           },
-           "\u0641\u062E\u0645": {
-             "final": "\uFD7C",
-             "initial": "\uFD7D"
-           },
-           "\u0642\u0645\u062D": {
-             "final": "\uFD7E",
-             "initial": "\uFDB4"
-           },
-           "\u0642\u0645\u0645": {
-             "final": "\uFD7F"
-           },
-           "\u0644\u062D\u0645": {
-             "final": "\uFD80",
-             "initial": "\uFDB5"
-           },
-           "\u0644\u062D\u064A": {
-             "final": "\uFD81"
-           },
-           "\u0644\u062D\u0649": {
-             "final": "\uFD82"
-           },
-           "\u0644\u062C\u062C": {
-             "initial": "\uFD83",
-             "final": "\uFD84"
-           },
-           "\u0644\u062E\u0645": {
-             "final": "\uFD85",
-             "initial": "\uFD86"
-           },
-           "\u0644\u0645\u062D": {
-             "final": "\uFD87",
-             "initial": "\uFD88"
-           },
-           "\u0645\u062D\u062C": {
-             "initial": "\uFD89"
-           },
-           "\u0645\u062D\u0645": {
-             "initial": "\uFD8A"
-           },
-           "\u0645\u062D\u064A": {
-             "final": "\uFD8B"
-           },
-           "\u0645\u062C\u062D": {
-             "initial": "\uFD8C"
-           },
-           "\u0645\u062C\u0645": {
-             "initial": "\uFD8D"
-           },
-           "\u0645\u062E\u062C": {
-             "initial": "\uFD8E"
-           },
-           "\u0645\u062E\u0645": {
-             "initial": "\uFD8F"
-           },
-           "\u0645\u062C\u062E": {
-             "initial": "\uFD92"
-           },
-           "\u0647\u0645\u062C": {
-             "initial": "\uFD93"
-           },
-           "\u0647\u0645\u0645": {
-             "initial": "\uFD94"
-           },
-           "\u0646\u062D\u0645": {
-             "initial": "\uFD95"
-           },
-           "\u0646\u062D\u0649": {
-             "final": "\uFD96"
-           },
-           "\u0646\u062C\u0645": {
-             "final": "\uFD97",
-             "initial": "\uFD98"
-           },
-           "\u0646\u062C\u0649": {
-             "final": "\uFD99"
-           },
-           "\u0646\u0645\u064A": {
-             "final": "\uFD9A"
-           },
-           "\u0646\u0645\u0649": {
-             "final": "\uFD9B"
-           },
-           "\u064A\u0645\u0645": {
-             "final": "\uFD9C",
-             "initial": "\uFD9D"
-           },
-           "\u0628\u062E\u064A": {
-             "final": "\uFD9E"
-           },
-           "\u062A\u062C\u064A": {
-             "final": "\uFD9F"
-           },
-           "\u062A\u062C\u0649": {
-             "final": "\uFDA0"
-           },
-           "\u062A\u062E\u064A": {
-             "final": "\uFDA1"
-           },
-           "\u062A\u062E\u0649": {
-             "final": "\uFDA2"
-           },
-           "\u062A\u0645\u064A": {
-             "final": "\uFDA3"
-           },
-           "\u062A\u0645\u0649": {
-             "final": "\uFDA4"
-           },
-           "\u062C\u0645\u064A": {
-             "final": "\uFDA5"
-           },
-           "\u062C\u062D\u0649": {
-             "final": "\uFDA6"
-           },
-           "\u062C\u0645\u0649": {
-             "final": "\uFDA7"
-           },
-           "\u0633\u062E\u0649": {
-             "final": "\uFDA8"
-           },
-           "\u0635\u062D\u064A": {
-             "final": "\uFDA9"
-           },
-           "\u0634\u062D\u064A": {
-             "final": "\uFDAA"
-           },
-           "\u0636\u062D\u064A": {
-             "final": "\uFDAB"
-           },
-           "\u0644\u062C\u064A": {
-             "final": "\uFDAC"
-           },
-           "\u0644\u0645\u064A": {
-             "final": "\uFDAD"
-           },
-           "\u064A\u062D\u064A": {
-             "final": "\uFDAE"
-           },
-           "\u064A\u062C\u064A": {
-             "final": "\uFDAF"
-           },
-           "\u064A\u0645\u064A": {
-             "final": "\uFDB0"
-           },
-           "\u0645\u0645\u064A": {
-             "final": "\uFDB1"
-           },
-           "\u0642\u0645\u064A": {
-             "final": "\uFDB2"
-           },
-           "\u0646\u062D\u064A": {
-             "final": "\uFDB3"
-           },
-           "\u0639\u0645\u064A": {
-             "final": "\uFDB6"
-           },
-           "\u0643\u0645\u064A": {
-             "final": "\uFDB7"
-           },
-           "\u0646\u062C\u062D": {
-             "initial": "\uFDB8",
-             "final": "\uFDBD"
-           },
-           "\u0645\u062E\u064A": {
-             "final": "\uFDB9"
-           },
-           "\u0644\u062C\u0645": {
-             "initial": "\uFDBA",
-             "final": "\uFDBC"
-           },
-           "\u0643\u0645\u0645": {
-             "final": "\uFDBB",
-             "initial": "\uFDC3"
-           },
-           "\u062C\u062D\u064A": {
-             "final": "\uFDBE"
-           },
-           "\u062D\u062C\u064A": {
-             "final": "\uFDBF"
-           },
-           "\u0645\u062C\u064A": {
-             "final": "\uFDC0"
-           },
-           "\u0641\u0645\u064A": {
-             "final": "\uFDC1"
-           },
-           "\u0628\u062D\u064A": {
-             "final": "\uFDC2"
-           },
-           "\u0633\u062E\u064A": {
-             "final": "\uFDC6"
-           },
-           "\u0646\u062C\u064A": {
-             "final": "\uFDC7"
-           },
-           "\u0644\u0622": {
-             "isolated": "\uFEF5",
-             "final": "\uFEF6"
-           },
-           "\u0644\u0623": {
-             "isolated": "\uFEF7",
-             "final": "\uFEF8"
-           },
-           "\u0644\u0625": {
-             "isolated": "\uFEF9",
-             "final": "\uFEFA"
-           },
-           "\u0644\u0627": {
-             "isolated": "\uFEFB",
-             "final": "\uFEFC"
-           },
-           "words": {
-             "\u0635\u0644\u06D2": "\uFDF0",
-             "\u0642\u0644\u06D2": "\uFDF1",
-             "\u0627\u0644\u0644\u0647": "\uFDF2",
-             "\u0627\u0643\u0628\u0631": "\uFDF3",
-             "\u0645\u062D\u0645\u062F": "\uFDF4",
-             "\u0635\u0644\u0639\u0645": "\uFDF5",
-             "\u0631\u0633\u0648\u0644": "\uFDF6",
-             "\u0639\u0644\u064A\u0647": "\uFDF7",
-             "\u0648\u0633\u0644\u0645": "\uFDF8",
-             "\u0635\u0644\u0649": "\uFDF9",
-             "\u0635\u0644\u0649\u0627\u0644\u0644\u0647\u0639\u0644\u064A\u0647\u0648\u0633\u0644\u0645": "\uFDFA",
-             "\u062C\u0644\u062C\u0644\u0627\u0644\u0647": "\uFDFB",
-             "\u0631\u06CC\u0627\u0644": "\uFDFC"
-           }
-         };
-         exports["default"] = ligatureReference;
-       });
-
-       var reference = createCommonjsModule(function (module, exports) {
+       var reference = createCommonjsModule(function (module, exports) {
 
          Object.defineProperty(exports, "__esModule", {
            value: true
          addToLineBreakers(0x1EE00, 0x1EEFF);
        });
 
-       var GlyphSplitter_1 = createCommonjsModule(function (module, exports) {
-
-         Object.defineProperty(exports, "__esModule", {
-           value: true
-         });
-
-         function GlyphSplitter(word) {
-           var letters = [];
-           var lastLetter = '';
-           word.split('').forEach(function (letter) {
-             if (isArabic_1.isArabic(letter)) {
-               if (reference.tashkeel.indexOf(letter) > -1) {
-                 letters[letters.length - 1] += letter;
-               } else if (lastLetter.length && (reference.lams.indexOf(lastLetter) === 0 && reference.alefs.indexOf(letter) > -1 || reference.lams.indexOf(lastLetter) > 0 && reference.alefs.indexOf(letter) === 0)) {
-                 // valid LA forms
-                 letters[letters.length - 1] += letter;
-               } else {
-                 letters.push(letter);
-               }
+       function GlyphSplitter(word) {
+         var letters = [];
+         var lastLetter = '';
+         word.split('').forEach(function (letter) {
+           if (isArabic_1.isArabic(letter)) {
+             if (reference.tashkeel.indexOf(letter) > -1) {
+               letters[letters.length - 1] += letter;
+             } else if (lastLetter.length && (reference.lams.indexOf(lastLetter) === 0 && reference.alefs.indexOf(letter) > -1 || reference.lams.indexOf(lastLetter) > 0 && reference.alefs.indexOf(letter) === 0)) {
+               // valid LA forms
+               letters[letters.length - 1] += letter;
              } else {
                letters.push(letter);
              }
+           } else {
+             letters.push(letter);
+           }
 
-             if (reference.tashkeel.indexOf(letter) === -1) {
-               lastLetter = letter;
-             }
-           });
-           return letters;
-         }
+           if (reference.tashkeel.indexOf(letter) === -1) {
+             lastLetter = letter;
+           }
+         });
+         return letters;
+       }
 
-         exports.GlyphSplitter = GlyphSplitter;
+       var GlyphSplitter_2 = GlyphSplitter;
+       var GlyphSplitter_1 = /*#__PURE__*/Object.defineProperty({
+         GlyphSplitter: GlyphSplitter_2
+       }, '__esModule', {
+         value: true
        });
 
-       var BaselineSplitter_1 = createCommonjsModule(function (module, exports) {
-
-         Object.defineProperty(exports, "__esModule", {
-           value: true
-         });
-
-         function BaselineSplitter(word) {
-           var letters = [];
-           var lastLetter = '';
-           word.split('').forEach(function (letter) {
-             if (isArabic_1.isArabic(letter) && isArabic_1.isArabic(lastLetter)) {
-               if (lastLetter.length && reference.tashkeel.indexOf(letter) > -1) {
-                 letters[letters.length - 1] += letter;
-               } else if (reference.lineBreakers.indexOf(lastLetter) > -1) {
-                 letters.push(letter);
-               } else {
-                 letters[letters.length - 1] += letter;
-               }
-             } else {
+       function BaselineSplitter(word) {
+         var letters = [];
+         var lastLetter = '';
+         word.split('').forEach(function (letter) {
+           if (isArabic_1.isArabic(letter) && isArabic_1.isArabic(lastLetter)) {
+             if (lastLetter.length && reference.tashkeel.indexOf(letter) > -1) {
+               letters[letters.length - 1] += letter;
+             } else if (reference.lineBreakers.indexOf(lastLetter) > -1) {
                letters.push(letter);
+             } else {
+               letters[letters.length - 1] += letter;
              }
+           } else {
+             letters.push(letter);
+           }
 
-             if (reference.tashkeel.indexOf(letter) === -1) {
-               // don't allow tashkeel to hide line break
-               lastLetter = letter;
-             }
-           });
-           return letters;
-         }
+           if (reference.tashkeel.indexOf(letter) === -1) {
+             // don't allow tashkeel to hide line break
+             lastLetter = letter;
+           }
+         });
+         return letters;
+       }
 
-         exports.BaselineSplitter = BaselineSplitter;
+       var BaselineSplitter_2 = BaselineSplitter;
+       var BaselineSplitter_1 = /*#__PURE__*/Object.defineProperty({
+         BaselineSplitter: BaselineSplitter_2
+       }, '__esModule', {
+         value: true
        });
 
-       var Normalization = createCommonjsModule(function (module, exports) {
-
-         Object.defineProperty(exports, "__esModule", {
-           value: true
-         });
+       function Normal(word, breakPresentationForm) {
+         // default is to turn initial/isolated/medial/final presentation form to generic
+         if (typeof breakPresentationForm === 'undefined') {
+           breakPresentationForm = true;
+         }
 
-         function Normal(word, breakPresentationForm) {
-           // default is to turn initial/isolated/medial/final presentation form to generic
-           if (typeof breakPresentationForm === 'undefined') {
-             breakPresentationForm = true;
+         var returnable = '';
+         word.split('').forEach(function (letter) {
+           if (!isArabic_1.isArabic(letter)) {
+             returnable += letter;
+             return;
            }
 
-           var returnable = '';
-           word.split('').forEach(function (letter) {
-             if (!isArabic_1.isArabic(letter)) {
-               returnable += letter;
-               return;
-             }
-
-             for (var w = 0; w < reference.letterList.length; w++) {
-               // ok so we are checking this potential lettertron
-               var letterForms = unicodeArabic["default"][reference.letterList[w]];
-               var versions = Object.keys(letterForms);
+           for (var w = 0; w < reference.letterList.length; w++) {
+             // ok so we are checking this potential lettertron
+             var letterForms = unicodeArabic["default"][reference.letterList[w]];
+             var versions = Object.keys(letterForms);
 
-               for (var v = 0; v < versions.length; v++) {
-                 var localVersion = letterForms[versions[v]];
+             for (var v = 0; v < versions.length; v++) {
+               var localVersion = letterForms[versions[v]];
 
-                 if (_typeof(localVersion) === 'object' && typeof localVersion.indexOf === 'undefined') {
-                   // look at this embedded object
-                   var embeddedForms = Object.keys(localVersion);
+               if (_typeof(localVersion) === 'object' && typeof localVersion.indexOf === 'undefined') {
+                 // look at this embedded object
+                 var embeddedForms = Object.keys(localVersion);
 
-                   for (var ef = 0; ef < embeddedForms.length; ef++) {
-                     var form = localVersion[embeddedForms[ef]];
+                 for (var ef = 0; ef < embeddedForms.length; ef++) {
+                   var form = localVersion[embeddedForms[ef]];
 
-                     if (form === letter || _typeof(form) === 'object' && form.indexOf && form.indexOf(letter) > -1) {
-                       // match
-                       // console.log('embedded match');
-                       if (form === letter) {
-                         // match exact
-                         if (breakPresentationForm && localVersion['normal'] && ['isolated', 'initial', 'medial', 'final'].indexOf(embeddedForms[ef]) > -1) {
-                           // replace presentation form
-                           // console.log('keeping normal form of the letter');
-                           if (_typeof(localVersion['normal']) === 'object') {
-                             returnable += localVersion['normal'][0];
-                           } else {
-                             returnable += localVersion['normal'];
-                           }
+                   if (form === letter || _typeof(form) === 'object' && form.indexOf && form.indexOf(letter) > -1) {
+                     // match
+                     // console.log('embedded match');
+                     if (form === letter) {
+                       // match exact
+                       if (breakPresentationForm && localVersion['normal'] && ['isolated', 'initial', 'medial', 'final'].indexOf(embeddedForms[ef]) > -1) {
+                         // replace presentation form
+                         // console.log('keeping normal form of the letter');
+                         if (_typeof(localVersion['normal']) === 'object') {
+                           returnable += localVersion['normal'][0];
+                         } else {
+                           returnable += localVersion['normal'];
+                         }
 
-                           return;
-                         } // console.log('keeping this letter');
+                         return;
+                       } // console.log('keeping this letter');
 
 
-                         returnable += letter;
-                         return;
-                       } else if (_typeof(form) === 'object' && form.indexOf && form.indexOf(letter) > -1) {
-                         // match
-                         returnable += form[0]; // console.log('added the first letter from the same array');
+                       returnable += letter;
+                       return;
+                     } else if (_typeof(form) === 'object' && form.indexOf && form.indexOf(letter) > -1) {
+                       // match
+                       returnable += form[0]; // console.log('added the first letter from the same array');
 
-                         return;
-                       }
+                       return;
                      }
                    }
-                 } else if (localVersion === letter) {
-                   // match exact
-                   if (breakPresentationForm && letterForms['normal'] && ['isolated', 'initial', 'medial', 'final'].indexOf(versions[v]) > -1) {
-                     // replace presentation form
-                     // console.log('keeping normal form of the letter');
-                     if (_typeof(letterForms['normal']) === 'object') {
-                       returnable += letterForms['normal'][0];
-                     } else {
-                       returnable += letterForms['normal'];
-                     }
+                 }
+               } else if (localVersion === letter) {
+                 // match exact
+                 if (breakPresentationForm && letterForms['normal'] && ['isolated', 'initial', 'medial', 'final'].indexOf(versions[v]) > -1) {
+                   // replace presentation form
+                   // console.log('keeping normal form of the letter');
+                   if (_typeof(letterForms['normal']) === 'object') {
+                     returnable += letterForms['normal'][0];
+                   } else {
+                     returnable += letterForms['normal'];
+                   }
 
-                     return;
-                   } // console.log('keeping this letter');
+                   return;
+                 } // console.log('keeping this letter');
 
 
-                   returnable += letter;
-                   return;
-                 } else if (_typeof(localVersion) === 'object' && localVersion.indexOf && localVersion.indexOf(letter) > -1) {
-                   // match
-                   returnable += localVersion[0]; // console.log('added the first letter from the same array');
+                 returnable += letter;
+                 return;
+               } else if (_typeof(localVersion) === 'object' && localVersion.indexOf && localVersion.indexOf(letter) > -1) {
+                 // match
+                 returnable += localVersion[0]; // console.log('added the first letter from the same array');
 
-                   return;
-                 }
+                 return;
                }
-             } // try ligatures
+             }
+           } // try ligatures
 
 
-             for (var v2 = 0; v2 < reference.ligatureList.length; v2++) {
-               var normalForm = reference.ligatureList[v2];
+           for (var v2 = 0; v2 < reference.ligatureList.length; v2++) {
+             var normalForm = reference.ligatureList[v2];
 
-               if (normalForm !== 'words') {
-                 var ligForms = Object.keys(unicodeLigatures["default"][normalForm]);
+             if (normalForm !== 'words') {
+               var ligForms = Object.keys(unicodeLigatures["default"][normalForm]);
 
-                 for (var f = 0; f < ligForms.length; f++) {
-                   if (unicodeLigatures["default"][normalForm][ligForms[f]] === letter) {
-                     returnable += normalForm;
-                     return;
-                   }
+               for (var f = 0; f < ligForms.length; f++) {
+                 if (unicodeLigatures["default"][normalForm][ligForms[f]] === letter) {
+                   returnable += normalForm;
+                   return;
                  }
                }
-             } // try words ligatures
+             }
+           } // try words ligatures
 
 
-             for (var v3 = 0; v3 < reference.ligatureWordList.length; v3++) {
-               var _normalForm = reference.ligatureWordList[v3];
+           for (var v3 = 0; v3 < reference.ligatureWordList.length; v3++) {
+             var _normalForm = reference.ligatureWordList[v3];
 
-               if (unicodeLigatures["default"].words[_normalForm] === letter) {
-                 returnable += _normalForm;
-                 return;
-               }
+             if (unicodeLigatures["default"].words[_normalForm] === letter) {
+               returnable += _normalForm;
+               return;
              }
+           }
 
-             returnable += letter; // console.log('kept the letter')
-           });
-           return returnable;
-         }
+           returnable += letter; // console.log('kept the letter')
+         });
+         return returnable;
+       }
 
-         exports.Normal = Normal;
+       var Normal_1 = Normal;
+       var Normalization = /*#__PURE__*/Object.defineProperty({
+         Normal: Normal_1
+       }, '__esModule', {
+         value: true
        });
 
-       var CharShaper_1 = createCommonjsModule(function (module, exports) {
-
-         Object.defineProperty(exports, "__esModule", {
-           value: true
-         });
+       function CharShaper(letter, form) {
+         if (!isArabic_1.isArabic(letter)) {
+           // fail not Arabic
+           throw new Error('Not Arabic');
+         }
 
-         function CharShaper(letter, form) {
-           if (!isArabic_1.isArabic(letter)) {
-             // fail not Arabic
-             throw new Error('Not Arabic');
-           }
+         if (letter === "\u0621") {
+           // hamza alone
+           return "\u0621";
+         }
 
-           if (letter === "\u0621") {
-             // hamza alone
-             return "\u0621";
-           }
+         for (var w = 0; w < reference.letterList.length; w++) {
+           // ok so we are checking this potential lettertron
+           var letterForms = unicodeArabic["default"][reference.letterList[w]];
+           var versions = Object.keys(letterForms);
 
-           for (var w = 0; w < reference.letterList.length; w++) {
-             // ok so we are checking this potential lettertron
-             var letterForms = unicodeArabic["default"][reference.letterList[w]];
-             var versions = Object.keys(letterForms);
+           for (var v = 0; v < versions.length; v++) {
+             var localVersion = letterForms[versions[v]];
 
-             for (var v = 0; v < versions.length; v++) {
-               var localVersion = letterForms[versions[v]];
+             if (localVersion === letter || _typeof(localVersion) === 'object' && localVersion.indexOf && localVersion.indexOf(letter) > -1) {
+               if (versions.indexOf(form) > -1) {
+                 return letterForms[form];
+               }
+             } else if (_typeof(localVersion) === 'object' && typeof localVersion.indexOf === 'undefined') {
+               // check embedded
+               var embeddedVersions = Object.keys(localVersion);
 
-               if (localVersion === letter || _typeof(localVersion) === 'object' && localVersion.indexOf && localVersion.indexOf(letter) > -1) {
-                 if (versions.indexOf(form) > -1) {
-                   return letterForms[form];
-                 }
-               } else if (_typeof(localVersion) === 'object' && typeof localVersion.indexOf === 'undefined') {
-                 // check embedded
-                 var embeddedVersions = Object.keys(localVersion);
-
-                 for (var ev = 0; ev < embeddedVersions.length; ev++) {
-                   if (localVersion[embeddedVersions[ev]] === letter || _typeof(localVersion[embeddedVersions[ev]]) === 'object' && localVersion[embeddedVersions[ev]].indexOf && localVersion[embeddedVersions[ev]].indexOf(letter) > -1) {
-                     if (embeddedVersions.indexOf(form) > -1) {
-                       return localVersion[form];
-                     }
+               for (var ev = 0; ev < embeddedVersions.length; ev++) {
+                 if (localVersion[embeddedVersions[ev]] === letter || _typeof(localVersion[embeddedVersions[ev]]) === 'object' && localVersion[embeddedVersions[ev]].indexOf && localVersion[embeddedVersions[ev]].indexOf(letter) > -1) {
+                   if (embeddedVersions.indexOf(form) > -1) {
+                     return localVersion[form];
                    }
                  }
                }
              }
            }
          }
+       }
 
-         exports.CharShaper = CharShaper;
+       var CharShaper_2 = CharShaper;
+       var CharShaper_1 = /*#__PURE__*/Object.defineProperty({
+         CharShaper: CharShaper_2
+       }, '__esModule', {
+         value: true
        });
 
-       var WordShaper_1 = createCommonjsModule(function (module, exports) {
+       function WordShaper$1(word) {
+         var state = 'initial';
+         var output = '';
 
-         Object.defineProperty(exports, "__esModule", {
-           value: true
-         });
-
-         function WordShaper(word) {
-           var state = 'initial';
-           var output = '';
-
-           for (var w = 0; w < word.length; w++) {
-             var nextLetter = ' ';
-
-             for (var nxw = w + 1; nxw < word.length; nxw++) {
-               if (!isArabic_1.isArabic(word[nxw])) {
-                 break;
-               }
+         for (var w = 0; w < word.length; w++) {
+           var nextLetter = ' ';
 
-               if (reference.tashkeel.indexOf(word[nxw]) === -1) {
-                 nextLetter = word[nxw];
-                 break;
-               }
+           for (var nxw = w + 1; nxw < word.length; nxw++) {
+             if (!isArabic_1.isArabic(word[nxw])) {
+               break;
              }
 
-             if (!isArabic_1.isArabic(word[w]) || isArabic_1.isMath(word[w])) {
-               // space or other non-Arabic
-               output += word[w];
-               state = 'initial';
-             } else if (reference.tashkeel.indexOf(word[w]) > -1) {
-               // tashkeel - add without changing state
-               output += word[w];
-             } else if (nextLetter === ' ' || // last Arabic letter in this word
-             reference.lineBreakers.indexOf(word[w]) > -1) {
-               // the current letter is known to break lines
-               output += CharShaper_1.CharShaper(word[w], state === 'initial' ? 'isolated' : 'final');
-               state = 'initial';
-             } else if (reference.lams.indexOf(word[w]) > -1 && reference.alefs.indexOf(nextLetter) > -1) {
-               // LA letters - advance an additional letter after this
-               output += unicodeLigatures["default"][word[w] + nextLetter][state === 'initial' ? 'isolated' : 'final'];
+             if (reference.tashkeel.indexOf(word[nxw]) === -1) {
+               nextLetter = word[nxw];
+               break;
+             }
+           }
 
-               while (word[w] !== nextLetter) {
-                 w++;
-               }
+           if (!isArabic_1.isArabic(word[w]) || isArabic_1.isMath(word[w])) {
+             // space or other non-Arabic
+             output += word[w];
+             state = 'initial';
+           } else if (reference.tashkeel.indexOf(word[w]) > -1) {
+             // tashkeel - add without changing state
+             output += word[w];
+           } else if (nextLetter === ' ' || // last Arabic letter in this word
+           reference.lineBreakers.indexOf(word[w]) > -1) {
+             // the current letter is known to break lines
+             output += CharShaper_1.CharShaper(word[w], state === 'initial' ? 'isolated' : 'final');
+             state = 'initial';
+           } else if (reference.lams.indexOf(word[w]) > -1 && reference.alefs.indexOf(nextLetter) > -1) {
+             // LA letters - advance an additional letter after this
+             output += unicodeLigatures["default"][word[w] + nextLetter][state === 'initial' ? 'isolated' : 'final'];
 
-               state = 'initial';
-             } else {
-               output += CharShaper_1.CharShaper(word[w], state);
-               state = 'medial';
+             while (word[w] !== nextLetter) {
+               w++;
              }
-           }
 
-           return output;
+             state = 'initial';
+           } else {
+             output += CharShaper_1.CharShaper(word[w], state);
+             state = 'medial';
+           }
          }
 
-         exports.WordShaper = WordShaper;
-       });
-
-       var ParentLetter_1 = createCommonjsModule(function (module, exports) {
+         return output;
+       }
 
-         Object.defineProperty(exports, "__esModule", {
-           value: true
-         });
+       var WordShaper_2 = WordShaper$1;
+       var WordShaper_1 = /*#__PURE__*/Object.defineProperty({
+         WordShaper: WordShaper_2
+       }, '__esModule', {
+         value: true
+       });
 
-         function ParentLetter(letter) {
-           if (!isArabic_1.isArabic(letter)) {
-             throw new Error('Not an Arabic letter');
-           }
+       function ParentLetter(letter) {
+         if (!isArabic_1.isArabic(letter)) {
+           throw new Error('Not an Arabic letter');
+         }
 
-           for (var w = 0; w < reference.letterList.length; w++) {
-             // ok so we are checking this potential lettertron
-             var letterForms = unicodeArabic["default"][reference.letterList[w]];
-             var versions = Object.keys(letterForms);
+         for (var w = 0; w < reference.letterList.length; w++) {
+           // ok so we are checking this potential lettertron
+           var letterForms = unicodeArabic["default"][reference.letterList[w]];
+           var versions = Object.keys(letterForms);
 
-             for (var v = 0; v < versions.length; v++) {
-               var localVersion = letterForms[versions[v]];
+           for (var v = 0; v < versions.length; v++) {
+             var localVersion = letterForms[versions[v]];
 
-               if (_typeof(localVersion) === 'object' && typeof localVersion.indexOf === 'undefined') {
-                 // look at this embedded object
-                 var embeddedForms = Object.keys(localVersion);
+             if (_typeof(localVersion) === 'object' && typeof localVersion.indexOf === 'undefined') {
+               // look at this embedded object
+               var embeddedForms = Object.keys(localVersion);
 
-                 for (var ef = 0; ef < embeddedForms.length; ef++) {
-                   var form = localVersion[embeddedForms[ef]];
+               for (var ef = 0; ef < embeddedForms.length; ef++) {
+                 var form = localVersion[embeddedForms[ef]];
 
-                   if (form === letter || _typeof(form) === 'object' && form.indexOf && form.indexOf(letter) > -1) {
-                     // match
-                     return localVersion;
-                   }
+                 if (form === letter || _typeof(form) === 'object' && form.indexOf && form.indexOf(letter) > -1) {
+                   // match
+                   return localVersion;
                  }
-               } else if (localVersion === letter || _typeof(localVersion) === 'object' && localVersion.indexOf && localVersion.indexOf(letter) > -1) {
-                 // match
-                 return letterForms;
                }
+             } else if (localVersion === letter || _typeof(localVersion) === 'object' && localVersion.indexOf && localVersion.indexOf(letter) > -1) {
+               // match
+               return letterForms;
              }
-
-             return null;
            }
+
+           return null;
          }
+       }
 
-         exports.ParentLetter = ParentLetter;
+       var ParentLetter_2 = ParentLetter;
 
-         function GrandparentLetter(letter) {
-           if (!isArabic_1.isArabic(letter)) {
-             throw new Error('Not an Arabic letter');
-           }
+       function GrandparentLetter(letter) {
+         if (!isArabic_1.isArabic(letter)) {
+           throw new Error('Not an Arabic letter');
+         }
 
-           for (var w = 0; w < reference.letterList.length; w++) {
-             // ok so we are checking this potential lettertron
-             var letterForms = unicodeArabic["default"][reference.letterList[w]];
-             var versions = Object.keys(letterForms);
+         for (var w = 0; w < reference.letterList.length; w++) {
+           // ok so we are checking this potential lettertron
+           var letterForms = unicodeArabic["default"][reference.letterList[w]];
+           var versions = Object.keys(letterForms);
 
-             for (var v = 0; v < versions.length; v++) {
-               var localVersion = letterForms[versions[v]];
+           for (var v = 0; v < versions.length; v++) {
+             var localVersion = letterForms[versions[v]];
 
-               if (_typeof(localVersion) === 'object' && typeof localVersion.indexOf === 'undefined') {
-                 // look at this embedded object
-                 var embeddedForms = Object.keys(localVersion);
+             if (_typeof(localVersion) === 'object' && typeof localVersion.indexOf === 'undefined') {
+               // look at this embedded object
+               var embeddedForms = Object.keys(localVersion);
 
-                 for (var ef = 0; ef < embeddedForms.length; ef++) {
-                   var form = localVersion[embeddedForms[ef]];
+               for (var ef = 0; ef < embeddedForms.length; ef++) {
+                 var form = localVersion[embeddedForms[ef]];
 
-                   if (form === letter || _typeof(form) === 'object' && form.indexOf && form.indexOf(letter) > -1) {
-                     // match
-                     return letterForms;
-                   }
+                 if (form === letter || _typeof(form) === 'object' && form.indexOf && form.indexOf(letter) > -1) {
+                   // match
+                   return letterForms;
                  }
-               } else if (localVersion === letter || _typeof(localVersion) === 'object' && localVersion.indexOf && localVersion.indexOf(letter) > -1) {
-                 // match
-                 return letterForms;
                }
+             } else if (localVersion === letter || _typeof(localVersion) === 'object' && localVersion.indexOf && localVersion.indexOf(letter) > -1) {
+               // match
+               return letterForms;
              }
-
-             return null;
            }
+
+           return null;
          }
+       }
 
-         exports.GrandparentLetter = GrandparentLetter;
+       var GrandparentLetter_1 = GrandparentLetter;
+       var ParentLetter_1 = /*#__PURE__*/Object.defineProperty({
+         ParentLetter: ParentLetter_2,
+         GrandparentLetter: GrandparentLetter_1
+       }, '__esModule', {
+         value: true
        });
 
-       var lib = createCommonjsModule(function (module, exports) {
-
-         Object.defineProperty(exports, "__esModule", {
-           value: true
-         });
-         exports.isArabic = isArabic_1.isArabic;
-         exports.GlyphSplitter = GlyphSplitter_1.GlyphSplitter;
-         exports.BaselineSplitter = BaselineSplitter_1.BaselineSplitter;
-         exports.Normal = Normalization.Normal;
-         exports.CharShaper = CharShaper_1.CharShaper;
-         exports.WordShaper = WordShaper_1.WordShaper;
-         exports.ParentLetter = ParentLetter_1.ParentLetter;
-         exports.GrandparentLetter = ParentLetter_1.GrandparentLetter;
-       });
+       isArabic_1.isArabic;
+       GlyphSplitter_1.GlyphSplitter;
+       BaselineSplitter_1.BaselineSplitter;
+       Normalization.Normal;
+       CharShaper_1.CharShaper;
+       var WordShaper = WordShaper_1.WordShaper;
+       ParentLetter_1.ParentLetter;
+       ParentLetter_1.GrandparentLetter;
 
        var rtlRegex = /[\u0590-\u05FF\u0600-\u06FF\u0750-\u07BF\u08A0–\u08BF]/;
        function fixRTLTextForSvg(inputText) {
          var hebrewSign = /[\u0591-\u05bd\u05bf\u05c1-\u05c5\u05c7]/; // Arabic word shaping
 
          if (arabicRegex.test(inputText)) {
-           inputText = lib.WordShaper(inputText);
+           inputText = WordShaper(inputText);
          }
 
          for (var n = 0; n < inputText.length; n++) {
        var propertyIsEnumerable = objectPropertyIsEnumerable.f;
 
        // `Object.{ entries, values }` methods implementation
-       var createMethod$5 = function (TO_ENTRIES) {
+       var createMethod$1 = function (TO_ENTRIES) {
          return function (it) {
            var O = toIndexedObject(it);
            var keys = objectKeys(O);
        var objectToArray = {
          // `Object.entries` method
          // https://tc39.es/ecma262/#sec-object.entries
-         entries: createMethod$5(true),
+         entries: createMethod$1(true),
          // `Object.values` method
          // https://tc39.es/ecma262/#sec-object.values
-         values: createMethod$5(false)
+         values: createMethod$1(false)
        };
 
        var $values = objectToArray.values;
          }
        }
 
-       function responseText(response) {
-         if (!response.ok) throw new Error(response.status + " " + response.statusText);
-         return response.text();
-       }
+       var vparse = createCommonjsModule(function (module) {
+         (function (window) {
 
-       function d3_text (input, init) {
-         return fetch(input, init).then(responseText);
-       }
+           function parseVersion(v) {
+             var m = v.replace(/[^0-9.]/g, '').match(/[0-9]*\.|[0-9]+/g) || [];
+             v = {
+               major: +m[0] || 0,
+               minor: +m[1] || 0,
+               patch: +m[2] || 0,
+               build: +m[3] || 0
+             };
+             v.isEmpty = !v.major && !v.minor && !v.patch && !v.build;
+             v.parsed = [v.major, v.minor, v.patch, v.build];
+             v.text = v.parsed.join('.');
+             v.compare = compare;
+             return v;
+           }
 
-       function responseJson(response) {
-         if (!response.ok) throw new Error(response.status + " " + response.statusText);
-         if (response.status === 204 || response.status === 205) return;
-         return response.json();
-       }
+           function compare(v) {
+             if (typeof v === 'string') {
+               v = parseVersion(v);
+             }
 
-       function d3_json (input, init) {
-         return fetch(input, init).then(responseJson);
-       }
+             for (var i = 0; i < 4; i++) {
+               if (this.parsed[i] !== v.parsed[i]) {
+                 return this.parsed[i] > v.parsed[i] ? 1 : -1;
+               }
+             }
 
-       function parser(type) {
-         return function (input, init) {
-           return d3_text(input, init).then(function (text) {
-             return new DOMParser().parseFromString(text, type);
-           });
-         };
-       }
+             return 0;
+           }
+           /* istanbul ignore next */
 
-       var d3_xml = parser("application/xml");
-       var svg = parser("image/svg+xml");
+
+           if (module && 'object' === 'object') {
+             module.exports = parseVersion;
+           } else {
+             window.parseVersion = parseVersion;
+           }
+         })(commonjsGlobal);
+       });
+
+       var name = "iD";
+       var version = "2.20.0";
+       var description = "A friendly editor for OpenStreetMap";
+       var main = "dist/iD.min.js";
+       var repository = "github:openstreetmap/iD";
+       var homepage = "https://github.com/openstreetmap/iD";
+       var bugs = "https://github.com/openstreetmap/iD/issues";
+       var keywords = ["editor","openstreetmap"];
+       var license = "ISC";
+       var scripts = {all:"npm-run-all -s clean build build:legacy dist",build:"npm-run-all -s build:css build:data build:dev","build:css":"node scripts/build_css.js","build:data":"shx mkdir -p dist/data && node scripts/build_data.js","build:dev":"rollup --config config/rollup.config.dev.js","build:legacy":"rollup --config config/rollup.config.legacy.js","build:stats":"rollup --config config/rollup.config.stats.js",clean:"shx rm -f dist/*.js dist/*.map dist/*.css dist/img/*.svg",dist:"npm-run-all -p dist:**","dist:mapillary":"shx mkdir -p dist/mapillary-js && shx cp -R node_modules/mapillary-js/dist/* dist/mapillary-js/","dist:pannellum":"shx mkdir -p dist/pannellum-streetside && shx cp -R node_modules/pannellum/build/* dist/pannellum-streetside/","dist:min:iD":"uglifyjs dist/iD.legacy.js --compress --mangle --output dist/iD.min.js","dist:svg:iD":"svg-sprite --symbol --symbol-dest . --shape-id-generator \"iD-%s\" --symbol-sprite dist/img/iD-sprite.svg \"svg/iD-sprite/**/*.svg\"","dist:svg:community":"svg-sprite --symbol --symbol-dest . --shape-id-generator \"community-%s\" --symbol-sprite dist/img/community-sprite.svg node_modules/osm-community-index/dist/img/*.svg","dist:svg:fa":"svg-sprite --symbol --symbol-dest . --symbol-sprite dist/img/fa-sprite.svg svg/fontawesome/*.svg","dist:svg:maki":"svg-sprite --symbol --symbol-dest . --shape-id-generator \"maki-%s\" --symbol-sprite dist/img/maki-sprite.svg node_modules/@mapbox/maki/icons/*.svg","dist:svg:mapillary:signs":"svg-sprite --symbol --symbol-dest . --symbol-sprite dist/img/mapillary-sprite.svg node_modules/mapillary_sprite_source/package_signs/*.svg","dist:svg:mapillary:objects":"svg-sprite --symbol --symbol-dest . --symbol-sprite dist/img/mapillary-object-sprite.svg node_modules/mapillary_sprite_source/package_objects/*.svg","dist:svg:temaki":"svg-sprite --symbol --symbol-dest . --shape-id-generator \"temaki-%s\" --symbol-sprite dist/img/temaki-sprite.svg node_modules/@ideditor/temaki/icons/*.svg",imagery:"node scripts/update_imagery.js",lint:"eslint scripts test/spec modules","lint:fix":"eslint scripts test/spec modules --fix",start:"npm-run-all -s build start:server",quickstart:"npm-run-all -s build:dev start:server","start:server":"node scripts/server.js",test:"npm-run-all -s lint build:css build:data build:legacy test:spec","test:spec":"phantomjs --web-security=no node_modules/mocha-phantomjs-core/mocha-phantomjs-core.js test/index.html spec",translations:"node scripts/update_locales.js"};
+       var dependencies = {"@ideditor/country-coder":"~5.0.3","@ideditor/location-conflation":"~1.0.2","@mapbox/geojson-area":"^0.2.2","@mapbox/sexagesimal":"1.2.0","@mapbox/togeojson":"0.16.0","@mapbox/vector-tile":"^1.3.1","@turf/bbox-clip":"^6.0.0","abortcontroller-polyfill":"^1.4.0","aes-js":"^3.1.2","alif-toolkit":"^1.2.9","core-js":"^3.6.5",diacritics:"1.3.0","fast-deep-equal":"~3.1.1","fast-json-stable-stringify":"2.1.0","lodash-es":"~4.17.15",marked:"~2.0.0","node-diff3":"2.1.0","osm-auth":"1.1.0",pannellum:"2.5.6","polygon-clipping":"~0.15.1",rbush:"3.0.1","whatwg-fetch":"^3.4.1","which-polygon":"2.2.0"};
+       var devDependencies = {"@babel/core":"^7.11.6","@babel/preset-env":"^7.11.5","@fortawesome/fontawesome-svg-core":"^1.2.32","@fortawesome/free-brands-svg-icons":"~5.15.1","@fortawesome/free-regular-svg-icons":"~5.15.1","@fortawesome/free-solid-svg-icons":"~5.15.1","@ideditor/temaki":"~4.4.0","@mapbox/maki":"^6.0.0","@rollup/plugin-babel":"^5.2.1","@rollup/plugin-commonjs":"^17.0.0","@rollup/plugin-json":"^4.0.1","@rollup/plugin-node-resolve":"~11.2.0",autoprefixer:"^10.0.1",btoa:"^1.2.1",chai:"^4.1.0","cldr-core":"37.0.0","cldr-localenames-full":"37.0.0",colors:"^1.1.2","concat-files":"^0.1.1",d3:"~6.6.0","editor-layer-index":"github:osmlab/editor-layer-index#gh-pages",eslint:"^7.1.0",gaze:"^1.1.3",glob:"^7.1.0",happen:"^0.3.1","js-yaml":"^4.0.0","json-stringify-pretty-compact":"^3.0.0",mapillary_sprite_source:"^1.8.0","mapillary-js":"4.0.0",minimist:"^1.2.3",mocha:"^7.0.1","mocha-phantomjs-core":"^2.1.0","name-suggestion-index":"~6.0","node-fetch":"^2.6.1","npm-run-all":"^4.0.0","object-inspect":"1.10.3","osm-community-index":"~5.1.0","phantomjs-prebuilt":"~2.1.16",postcss:"^8.1.1","postcss-selector-prepend":"^0.5.0",rollup:"~2.52.8","rollup-plugin-includepaths":"~0.2.3","rollup-plugin-progress":"^1.1.1","rollup-plugin-visualizer":"~4.2.0",shelljs:"^0.8.0",shx:"^0.3.0",sinon:"7.5.0","sinon-chai":"^3.3.0",smash:"0.0","static-server":"^2.2.1","svg-sprite":"1.5.0","uglify-js":"~3.13.0",vparse:"~1.1.0"};
+       var engines = {node:">=10"};
+       var browserslist = ["> 0.2%, last 6 major versions, Firefox ESR, IE 11, maintained node versions"];
+       var packageJSON = {
+       name: name,
+       version: version,
+       description: description,
+       main: main,
+       repository: repository,
+       homepage: homepage,
+       bugs: bugs,
+       keywords: keywords,
+       license: license,
+       scripts: scripts,
+       dependencies: dependencies,
+       devDependencies: devDependencies,
+       engines: engines,
+       browserslist: browserslist
+       };
 
        var _mainFileFetcher = coreFileFetcher(); // singleton
        // coreFileFetcher asynchronously fetches data from JSON files
        //
 
        function coreFileFetcher() {
+         var ociVersion = packageJSON.devDependencies['osm-community-index'];
+         var v = vparse(ociVersion);
+         var vMinor = "".concat(v.major, ".").concat(v.minor);
          var _this = {};
          var _inflight = {};
          var _fileMap = {
            'address_formats': 'data/address_formats.min.json',
-           'deprecated': 'data/deprecated.min.json',
-           'discarded': 'data/discarded.min.json',
+           'deprecated': 'https://cdn.jsdelivr.net/npm/@openstreetmap/id-tagging-schema@3/dist/deprecated.min.json',
+           'discarded': 'https://cdn.jsdelivr.net/npm/@openstreetmap/id-tagging-schema@3/dist/discarded.min.json',
            'imagery': 'data/imagery.min.json',
            'intro_graph': 'data/intro_graph.min.json',
            'keepRight': 'data/keepRight.min.json',
            'languages': 'data/languages.min.json',
-           'locales': 'data/locales.min.json',
-           'nsi_brands': 'https://cdn.jsdelivr.net/npm/name-suggestion-index@4/dist/brands.min.json',
-           'nsi_filters': 'https://cdn.jsdelivr.net/npm/name-suggestion-index@4/dist/filters.min.json',
-           'oci_features': 'https://cdn.jsdelivr.net/npm/osm-community-index@2/dist/features.min.json',
-           'oci_resources': 'https://cdn.jsdelivr.net/npm/osm-community-index@2/dist/resources.min.json',
-           'preset_categories': 'data/preset_categories.min.json',
-           'preset_defaults': 'data/preset_defaults.min.json',
-           'preset_fields': 'data/preset_fields.min.json',
-           'preset_presets': 'data/preset_presets.min.json',
+           'locales': 'locales/index.min.json',
+           'oci_defaults': "https://cdn.jsdelivr.net/npm/osm-community-index@".concat(vMinor, "/dist/defaults.min.json"),
+           'oci_features': "https://cdn.jsdelivr.net/npm/osm-community-index@".concat(vMinor, "/dist/featureCollection.min.json"),
+           'oci_resources': "https://cdn.jsdelivr.net/npm/osm-community-index@".concat(vMinor, "/dist/resources.min.json"),
+           'preset_categories': 'https://cdn.jsdelivr.net/npm/@openstreetmap/id-tagging-schema@3/dist/preset_categories.min.json',
+           'preset_defaults': 'https://cdn.jsdelivr.net/npm/@openstreetmap/id-tagging-schema@3/dist/preset_defaults.min.json',
+           'preset_fields': 'https://cdn.jsdelivr.net/npm/@openstreetmap/id-tagging-schema@3/dist/fields.min.json',
+           'preset_presets': 'https://cdn.jsdelivr.net/npm/@openstreetmap/id-tagging-schema@3/dist/presets.min.json',
            'phone_formats': 'data/phone_formats.min.json',
            'qa_data': 'data/qa_data.min.json',
            'shortcuts': 'data/shortcuts.min.json',
            var prom = _inflight[url];
 
            if (!prom) {
-             _inflight[url] = prom = d3_json(url).then(function (result) {
+             _inflight[url] = prom = utilFetchJson(url).then(function (result) {
                delete _inflight[url];
 
                if (!result) {
          return _this;
        }
 
-       var $findIndex$1 = arrayIteration.findIndex;
-
-
-       var FIND_INDEX = 'findIndex';
-       var SKIPS_HOLES$1 = true;
-
-       // Shouldn't skip holes
-       if (FIND_INDEX in []) Array(1)[FIND_INDEX](function () { SKIPS_HOLES$1 = false; });
-
-       // `Array.prototype.findIndex` method
-       // https://tc39.es/ecma262/#sec-array.prototype.findindex
-       _export({ target: 'Array', proto: true, forced: SKIPS_HOLES$1 }, {
-         findIndex: function findIndex(callbackfn /* , that = undefined */) {
-           return $findIndex$1(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
-         }
-       });
-
-       // https://tc39.es/ecma262/#sec-array.prototype-@@unscopables
-       addToUnscopables(FIND_INDEX);
-
-       var $includes$1 = arrayIncludes.includes;
-
-
-       // `Array.prototype.includes` method
-       // https://tc39.es/ecma262/#sec-array.prototype.includes
-       _export({ target: 'Array', proto: true }, {
-         includes: function includes(el /* , fromIndex = 0 */) {
-           return $includes$1(this, el, arguments.length > 1 ? arguments[1] : undefined);
-         }
-       });
-
-       // https://tc39.es/ecma262/#sec-array.prototype-@@unscopables
-       addToUnscopables('includes');
-
-       var notARegexp = function (it) {
-         if (isRegexp(it)) {
-           throw TypeError("The method doesn't accept regular expressions");
-         } return it;
-       };
-
-       var MATCH$2 = wellKnownSymbol('match');
-
-       var correctIsRegexpLogic = function (METHOD_NAME) {
-         var regexp = /./;
-         try {
-           '/./'[METHOD_NAME](regexp);
-         } catch (error1) {
-           try {
-             regexp[MATCH$2] = false;
-             return '/./'[METHOD_NAME](regexp);
-           } catch (error2) { /* empty */ }
-         } return false;
-       };
-
-       // `String.prototype.includes` method
-       // https://tc39.es/ecma262/#sec-string.prototype.includes
-       _export({ target: 'String', proto: true, forced: !correctIsRegexpLogic('includes') }, {
-         includes: function includes(searchString /* , position = 0 */) {
-           return !!~String(requireObjectCoercible(this))
-             .indexOf(notARegexp(searchString), arguments.length > 1 ? arguments[1] : undefined);
-         }
-       });
-
-       var _detected;
-
-       function utilDetect(refresh) {
-         if (_detected && !refresh) return _detected;
-         _detected = {};
-         var ua = navigator.userAgent;
-         var m = null;
-         /* Browser */
-
-         m = ua.match(/(edge)\/?\s*(\.?\d+(\.\d+)*)/i); // Edge
-
-         if (m !== null) {
-           _detected.browser = m[1];
-           _detected.version = m[2];
-         }
-
-         if (!_detected.browser) {
-           m = ua.match(/Trident\/.*rv:([0-9]{1,}[\.0-9]{0,})/i); // IE11
-
-           if (m !== null) {
-             _detected.browser = 'msie';
-             _detected.version = m[1];
-           }
-         }
-
-         if (!_detected.browser) {
-           m = ua.match(/(opr)\/?\s*(\.?\d+(\.\d+)*)/i); // Opera 15+
-
-           if (m !== null) {
-             _detected.browser = 'Opera';
-             _detected.version = m[2];
-           }
-         }
-
-         if (!_detected.browser) {
-           m = ua.match(/(opera|chrome|safari|firefox|msie)\/?\s*(\.?\d+(\.\d+)*)/i);
-
-           if (m !== null) {
-             _detected.browser = m[1];
-             _detected.version = m[2];
-             m = ua.match(/version\/([\.\d]+)/i);
-             if (m !== null) _detected.version = m[1];
-           }
-         }
-
-         if (!_detected.browser) {
-           _detected.browser = navigator.appName;
-           _detected.version = navigator.appVersion;
-         } // keep major.minor version only..
-
-
-         _detected.version = _detected.version.split(/\W/).slice(0, 2).join('.'); // detect other browser capabilities
-         // Legacy Opera has incomplete svg style support. See #715
-
-         _detected.opera = _detected.browser.toLowerCase() === 'opera' && parseFloat(_detected.version) < 15;
-
-         if (_detected.browser.toLowerCase() === 'msie') {
-           _detected.ie = true;
-           _detected.browser = 'Internet Explorer';
-           _detected.support = parseFloat(_detected.version) >= 11;
-         } else {
-           _detected.ie = false;
-           _detected.support = true;
-         }
-
-         _detected.filedrop = window.FileReader && 'ondrop' in window;
-         _detected.download = !(_detected.ie || _detected.browser.toLowerCase() === 'edge');
-         _detected.cssfilters = !(_detected.ie || _detected.browser.toLowerCase() === 'edge');
-         /* Platform */
-
-         if (/Win/.test(ua)) {
-           _detected.os = 'win';
-           _detected.platform = 'Windows';
-         } else if (/Mac/.test(ua)) {
-           _detected.os = 'mac';
-           _detected.platform = 'Macintosh';
-         } else if (/X11/.test(ua) || /Linux/.test(ua)) {
-           _detected.os = 'linux';
-           _detected.platform = 'Linux';
-         } else {
-           _detected.os = 'win';
-           _detected.platform = 'Unknown';
-         }
-
-         _detected.isMobileWebKit = (/\b(iPad|iPhone|iPod)\b/.test(ua) || // HACK: iPadOS 13+ requests desktop sites by default by using a Mac user agent,
-         // so assume any "mac" with multitouch is actually iOS
-         navigator.platform === 'MacIntel' && 'maxTouchPoints' in navigator && navigator.maxTouchPoints > 1) && /WebKit/.test(ua) && !/Edge/.test(ua) && !window.MSStream;
-         /* Locale */
-         // An array of locales requested by the browser in priority order.
-
-         _detected.browserLocales = Array.from(new Set( // remove duplicates
-         [navigator.language].concat(navigator.languages || []).concat([// old property for backwards compatibility
-         navigator.userLanguage]) // remove any undefined values
-         .filter(Boolean)));
-         /* Host */
-
-         var loc = window.top.location;
-         var origin = loc.origin;
-
-         if (!origin) {
-           // for unpatched IE11
-           origin = loc.protocol + '//' + loc.hostname + (loc.port ? ':' + loc.port : '');
-         }
-
-         _detected.host = origin + loc.pathname;
-         return _detected;
-       }
-
-       // `Number.MAX_SAFE_INTEGER` constant
-       // https://tc39.es/ecma262/#sec-number.max_safe_integer
-       _export({ target: 'Number', stat: true }, {
-         MAX_SAFE_INTEGER: 0x1FFFFFFFFFFFFF
-       });
-
-       var getOwnPropertyNames$2 = objectGetOwnPropertyNames.f;
-       var getOwnPropertyDescriptor$3 = objectGetOwnPropertyDescriptor.f;
-       var defineProperty$9 = objectDefineProperty.f;
+       var getOwnPropertyNames = objectGetOwnPropertyNames.f;
+       var getOwnPropertyDescriptor$2 = objectGetOwnPropertyDescriptor.f;
+       var defineProperty = objectDefineProperty.f;
        var trim$2 = stringTrim.trim;
 
        var NUMBER = 'Number';
-       var NativeNumber = global_1[NUMBER];
+       var NativeNumber = global$2[NUMBER];
        var NumberPrototype = NativeNumber.prototype;
 
        // Opera ~12 has broken Object#toString
 
        // `ToNumber` abstract operation
        // https://tc39.es/ecma262/#sec-tonumber
-       var toNumber = function (argument) {
+       var toNumber$1 = function (argument) {
          var it = toPrimitive(argument, false);
          var first, third, radix, maxCode, digits, length, index, code;
          if (typeof it == 'string' && it.length > 2) {
            return dummy instanceof NumberWrapper
              // check on 1..constructor(foo) case
              && (BROKEN_CLASSOF ? fails(function () { NumberPrototype.valueOf.call(dummy); }) : classofRaw(dummy) != NUMBER)
-               ? inheritIfRequired(new NativeNumber(toNumber(it)), dummy, NumberWrapper) : toNumber(it);
+               ? inheritIfRequired(new NativeNumber(toNumber$1(it)), dummy, NumberWrapper) : toNumber$1(it);
          };
-         for (var keys$3 = descriptors ? getOwnPropertyNames$2(NativeNumber) : (
+         for (var keys = descriptors ? getOwnPropertyNames(NativeNumber) : (
            // ES3:
            'MAX_VALUE,MIN_VALUE,NaN,NEGATIVE_INFINITY,POSITIVE_INFINITY,' +
            // ES2015 (in case, if modules with ES2015 Number statics required before):
            'MIN_SAFE_INTEGER,parseFloat,parseInt,isInteger,' +
            // ESNext
            'fromString,range'
-         ).split(','), j$2 = 0, key$1; keys$3.length > j$2; j$2++) {
-           if (has(NativeNumber, key$1 = keys$3[j$2]) && !has(NumberWrapper, key$1)) {
-             defineProperty$9(NumberWrapper, key$1, getOwnPropertyDescriptor$3(NativeNumber, key$1));
+         ).split(','), j = 0, key; keys.length > j; j++) {
+           if (has$1(NativeNumber, key = keys[j]) && !has$1(NumberWrapper, key)) {
+             defineProperty(NumberWrapper, key, getOwnPropertyDescriptor$2(NativeNumber, key));
            }
          }
          NumberWrapper.prototype = NumberPrototype;
          NumberPrototype.constructor = NumberWrapper;
-         redefine(global_1, NUMBER, NumberWrapper);
+         redefine(global$2, NUMBER, NumberWrapper);
        }
 
-       var aesJs = createCommonjsModule(function (module, exports) {
-         /*! MIT License. Copyright 2015-2018 Richard Moore <me@ricmoo.com>. See LICENSE.txt. */
-         (function (root) {
-
-           function checkInt(value) {
-             return parseInt(value) === value;
-           }
+       // `thisNumberValue` abstract operation
+       // https://tc39.es/ecma262/#sec-thisnumbervalue
+       var thisNumberValue = function (value) {
+         if (typeof value != 'number' && classofRaw(value) != 'Number') {
+           throw TypeError('Incorrect invocation');
+         }
+         return +value;
+       };
 
-           function checkInts(arrayish) {
-             if (!checkInt(arrayish.length)) {
-               return false;
-             }
-
-             for (var i = 0; i < arrayish.length; i++) {
-               if (!checkInt(arrayish[i]) || arrayish[i] < 0 || arrayish[i] > 255) {
-                 return false;
-               }
-             }
-
-             return true;
-           }
+       // `String.prototype.repeat` method implementation
+       // https://tc39.es/ecma262/#sec-string.prototype.repeat
+       var stringRepeat = function repeat(count) {
+         var str = String(requireObjectCoercible(this));
+         var result = '';
+         var n = toInteger(count);
+         if (n < 0 || n == Infinity) throw RangeError('Wrong number of repetitions');
+         for (;n > 0; (n >>>= 1) && (str += str)) if (n & 1) result += str;
+         return result;
+       };
 
-           function coerceArray(arg, copy) {
-             // ArrayBuffer view
-             if (arg.buffer && arg.name === 'Uint8Array') {
-               if (copy) {
-                 if (arg.slice) {
-                   arg = arg.slice();
-                 } else {
-                   arg = Array.prototype.slice.call(arg);
-                 }
-               }
+       var nativeToFixed = 1.0.toFixed;
+       var floor = Math.floor;
 
-               return arg;
-             } // It's an array; check it is a valid representation of a byte
+       var pow = function (x, n, acc) {
+         return n === 0 ? acc : n % 2 === 1 ? pow(x, n - 1, acc * x) : pow(x * x, n / 2, acc);
+       };
 
+       var log = function (x) {
+         var n = 0;
+         var x2 = x;
+         while (x2 >= 4096) {
+           n += 12;
+           x2 /= 4096;
+         }
+         while (x2 >= 2) {
+           n += 1;
+           x2 /= 2;
+         } return n;
+       };
 
-             if (Array.isArray(arg)) {
-               if (!checkInts(arg)) {
-                 throw new Error('Array contains invalid value: ' + arg);
-               }
+       var multiply = function (data, n, c) {
+         var index = -1;
+         var c2 = c;
+         while (++index < 6) {
+           c2 += n * data[index];
+           data[index] = c2 % 1e7;
+           c2 = floor(c2 / 1e7);
+         }
+       };
 
-               return new Uint8Array(arg);
-             } // Something else, but behaves like an array (maybe a Buffer? Arguments?)
+       var divide = function (data, n) {
+         var index = 6;
+         var c = 0;
+         while (--index >= 0) {
+           c += data[index];
+           data[index] = floor(c / n);
+           c = (c % n) * 1e7;
+         }
+       };
 
+       var dataToString = function (data) {
+         var index = 6;
+         var s = '';
+         while (--index >= 0) {
+           if (s !== '' || index === 0 || data[index] !== 0) {
+             var t = String(data[index]);
+             s = s === '' ? t : s + stringRepeat.call('0', 7 - t.length) + t;
+           }
+         } return s;
+       };
 
-             if (checkInt(arg.length) && checkInts(arg)) {
-               return new Uint8Array(arg);
-             }
+       var FORCED$4 = nativeToFixed && (
+         0.00008.toFixed(3) !== '0.000' ||
+         0.9.toFixed(0) !== '1' ||
+         1.255.toFixed(2) !== '1.25' ||
+         1000000000000000128.0.toFixed(0) !== '1000000000000000128'
+       ) || !fails(function () {
+         // V8 ~ Android 4.3-
+         nativeToFixed.call({});
+       });
 
-             throw new Error('unsupported array-like object');
-           }
+       // `Number.prototype.toFixed` method
+       // https://tc39.es/ecma262/#sec-number.prototype.tofixed
+       _export({ target: 'Number', proto: true, forced: FORCED$4 }, {
+         toFixed: function toFixed(fractionDigits) {
+           var number = thisNumberValue(this);
+           var fractDigits = toInteger(fractionDigits);
+           var data = [0, 0, 0, 0, 0, 0];
+           var sign = '';
+           var result = '0';
+           var e, z, j, k;
 
-           function createArray(length) {
-             return new Uint8Array(length);
+           if (fractDigits < 0 || fractDigits > 20) throw RangeError('Incorrect fraction digits');
+           // eslint-disable-next-line no-self-compare -- NaN check
+           if (number != number) return 'NaN';
+           if (number <= -1e21 || number >= 1e21) return String(number);
+           if (number < 0) {
+             sign = '-';
+             number = -number;
            }
-
-           function copyArray(sourceArray, targetArray, targetStart, sourceStart, sourceEnd) {
-             if (sourceStart != null || sourceEnd != null) {
-               if (sourceArray.slice) {
-                 sourceArray = sourceArray.slice(sourceStart, sourceEnd);
-               } else {
-                 sourceArray = Array.prototype.slice.call(sourceArray, sourceStart, sourceEnd);
+           if (number > 1e-21) {
+             e = log(number * pow(2, 69, 1)) - 69;
+             z = e < 0 ? number * pow(2, -e, 1) : number / pow(2, e, 1);
+             z *= 0x10000000000000;
+             e = 52 - e;
+             if (e > 0) {
+               multiply(data, 0, z);
+               j = fractDigits;
+               while (j >= 7) {
+                 multiply(data, 1e7, 0);
+                 j -= 7;
+               }
+               multiply(data, pow(10, j, 1), 0);
+               j = e - 1;
+               while (j >= 23) {
+                 divide(data, 1 << 23);
+                 j -= 23;
                }
+               divide(data, 1 << j);
+               multiply(data, 1, 1);
+               divide(data, 2);
+               result = dataToString(data);
+             } else {
+               multiply(data, 0, z);
+               multiply(data, 1 << -e, 0);
+               result = dataToString(data) + stringRepeat.call('0', fractDigits);
              }
-
-             targetArray.set(sourceArray, targetStart);
            }
+           if (fractDigits > 0) {
+             k = result.length;
+             result = sign + (k <= fractDigits
+               ? '0.' + stringRepeat.call('0', fractDigits - k) + result
+               : result.slice(0, k - fractDigits) + '.' + result.slice(k - fractDigits));
+           } else {
+             result = sign + result;
+           } return result;
+         }
+       });
 
-           var convertUtf8 = function () {
-             function toBytes(text) {
-               var result = [],
-                   i = 0;
-               text = encodeURI(text);
+       var globalIsFinite = global$2.isFinite;
 
-               while (i < text.length) {
-                 var c = text.charCodeAt(i++); // if it is a % sign, encode the following 2 bytes as a hex value
+       // `Number.isFinite` method
+       // https://tc39.es/ecma262/#sec-number.isfinite
+       // eslint-disable-next-line es/no-number-isfinite -- safe
+       var numberIsFinite = Number.isFinite || function isFinite(it) {
+         return typeof it == 'number' && globalIsFinite(it);
+       };
 
-                 if (c === 37) {
-                   result.push(parseInt(text.substr(i, 2), 16));
-                   i += 2; // otherwise, just the actual byte
-                 } else {
-                   result.push(c);
-                 }
-               }
+       // `Number.isFinite` method
+       // https://tc39.es/ecma262/#sec-number.isfinite
+       _export({ target: 'Number', stat: true }, { isFinite: numberIsFinite });
 
-               return coerceArray(result);
-             }
+       var fromCharCode = String.fromCharCode;
+       // eslint-disable-next-line es/no-string-fromcodepoint -- required for testing
+       var $fromCodePoint = String.fromCodePoint;
 
-             function fromBytes(bytes) {
-               var result = [],
-                   i = 0;
+       // length should be 1, old FF problem
+       var INCORRECT_LENGTH = !!$fromCodePoint && $fromCodePoint.length != 1;
 
-               while (i < bytes.length) {
-                 var c = bytes[i];
+       // `String.fromCodePoint` method
+       // https://tc39.es/ecma262/#sec-string.fromcodepoint
+       _export({ target: 'String', stat: true, forced: INCORRECT_LENGTH }, {
+         // eslint-disable-next-line no-unused-vars -- required for `.length`
+         fromCodePoint: function fromCodePoint(x) {
+           var elements = [];
+           var length = arguments.length;
+           var i = 0;
+           var code;
+           while (length > i) {
+             code = +arguments[i++];
+             if (toAbsoluteIndex(code, 0x10FFFF) !== code) throw RangeError(code + ' is not a valid code point');
+             elements.push(code < 0x10000
+               ? fromCharCode(code)
+               : fromCharCode(((code -= 0x10000) >> 10) + 0xD800, code % 0x400 + 0xDC00)
+             );
+           } return elements.join('');
+         }
+       });
 
-                 if (c < 128) {
-                   result.push(String.fromCharCode(c));
-                   i++;
-                 } else if (c > 191 && c < 224) {
-                   result.push(String.fromCharCode((c & 0x1f) << 6 | bytes[i + 1] & 0x3f));
-                   i += 2;
-                 } else {
-                   result.push(String.fromCharCode((c & 0x0f) << 12 | (bytes[i + 1] & 0x3f) << 6 | bytes[i + 2] & 0x3f));
-                   i += 3;
-                 }
-               }
+       // @@search logic
+       fixRegexpWellKnownSymbolLogic('search', function (SEARCH, nativeSearch, maybeCallNative) {
+         return [
+           // `String.prototype.search` method
+           // https://tc39.es/ecma262/#sec-string.prototype.search
+           function search(regexp) {
+             var O = requireObjectCoercible(this);
+             var searcher = regexp == undefined ? undefined : regexp[SEARCH];
+             return searcher !== undefined ? searcher.call(regexp, O) : new RegExp(regexp)[SEARCH](String(O));
+           },
+           // `RegExp.prototype[@@search]` method
+           // https://tc39.es/ecma262/#sec-regexp.prototype-@@search
+           function (string) {
+             var res = maybeCallNative(nativeSearch, this, string);
+             if (res.done) return res.value;
 
-               return result.join('');
-             }
+             var rx = anObject(this);
+             var S = String(string);
 
-             return {
-               toBytes: toBytes,
-               fromBytes: fromBytes
-             };
-           }();
+             var previousLastIndex = rx.lastIndex;
+             if (!sameValue(previousLastIndex, 0)) rx.lastIndex = 0;
+             var result = regexpExecAbstract(rx, S);
+             if (!sameValue(rx.lastIndex, previousLastIndex)) rx.lastIndex = previousLastIndex;
+             return result === null ? -1 : result.index;
+           }
+         ];
+       });
 
-           var convertHex = function () {
-             function toBytes(text) {
-               var result = [];
+       var quickselect$1 = createCommonjsModule(function (module, exports) {
+         (function (global, factory) {
+           module.exports = factory() ;
+         })(commonjsGlobal, function () {
 
-               for (var i = 0; i < text.length; i += 2) {
-                 result.push(parseInt(text.substr(i, 2), 16));
-               }
+           function quickselect(arr, k, left, right, compare) {
+             quickselectStep(arr, k, left || 0, right || arr.length - 1, compare || defaultCompare);
+           }
 
-               return result;
-             } // http://ixti.net/development/javascript/2011/11/11/base64-encodedecode-of-utf8-in-browser-with-js.html
+           function quickselectStep(arr, k, left, right, compare) {
+             while (right > left) {
+               if (right - left > 600) {
+                 var n = right - left + 1;
+                 var m = k - left + 1;
+                 var z = Math.log(n);
+                 var s = 0.5 * Math.exp(2 * z / 3);
+                 var sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1);
+                 var newLeft = Math.max(left, Math.floor(k - m * s / n + sd));
+                 var newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd));
+                 quickselectStep(arr, k, newLeft, newRight, compare);
+               }
 
+               var t = arr[k];
+               var i = left;
+               var j = right;
+               swap(arr, left, k);
+               if (compare(arr[right], t) > 0) swap(arr, left, right);
 
-             var Hex = '0123456789abcdef';
+               while (i < j) {
+                 swap(arr, i, j);
+                 i++;
+                 j--;
 
-             function fromBytes(bytes) {
-               var result = [];
+                 while (compare(arr[i], t) < 0) {
+                   i++;
+                 }
 
-               for (var i = 0; i < bytes.length; i++) {
-                 var v = bytes[i];
-                 result.push(Hex[(v & 0xf0) >> 4] + Hex[v & 0x0f]);
+                 while (compare(arr[j], t) > 0) {
+                   j--;
+                 }
                }
 
-               return result.join('');
+               if (compare(arr[left], t) === 0) swap(arr, left, j);else {
+                 j++;
+                 swap(arr, j, right);
+               }
+               if (j <= k) left = j + 1;
+               if (k <= j) right = j - 1;
              }
+           }
 
-             return {
-               toBytes: toBytes,
-               fromBytes: fromBytes
-             };
-           }(); // Number of rounds by keysize
-
+           function swap(arr, i, j) {
+             var tmp = arr[i];
+             arr[i] = arr[j];
+             arr[j] = tmp;
+           }
 
-           var numberOfRounds = {
-             16: 10,
-             24: 12,
-             32: 14
-           }; // Round constant words
+           function defaultCompare(a, b) {
+             return a < b ? -1 : a > b ? 1 : 0;
+           }
 
-           var rcon = [0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36, 0x6c, 0xd8, 0xab, 0x4d, 0x9a, 0x2f, 0x5e, 0xbc, 0x63, 0xc6, 0x97, 0x35, 0x6a, 0xd4, 0xb3, 0x7d, 0xfa, 0xef, 0xc5, 0x91]; // S-box and Inverse S-box (S is for Substitution)
+           return quickselect;
+         });
+       });
 
-           var S = [0x63, 0x7c, 0x77, 0x7b, 0xf2, 0x6b, 0x6f, 0xc5, 0x30, 0x01, 0x67, 0x2b, 0xfe, 0xd7, 0xab, 0x76, 0xca, 0x82, 0xc9, 0x7d, 0xfa, 0x59, 0x47, 0xf0, 0xad, 0xd4, 0xa2, 0xaf, 0x9c, 0xa4, 0x72, 0xc0, 0xb7, 0xfd, 0x93, 0x26, 0x36, 0x3f, 0xf7, 0xcc, 0x34, 0xa5, 0xe5, 0xf1, 0x71, 0xd8, 0x31, 0x15, 0x04, 0xc7, 0x23, 0xc3, 0x18, 0x96, 0x05, 0x9a, 0x07, 0x12, 0x80, 0xe2, 0xeb, 0x27, 0xb2, 0x75, 0x09, 0x83, 0x2c, 0x1a, 0x1b, 0x6e, 0x5a, 0xa0, 0x52, 0x3b, 0xd6, 0xb3, 0x29, 0xe3, 0x2f, 0x84, 0x53, 0xd1, 0x00, 0xed, 0x20, 0xfc, 0xb1, 0x5b, 0x6a, 0xcb, 0xbe, 0x39, 0x4a, 0x4c, 0x58, 0xcf, 0xd0, 0xef, 0xaa, 0xfb, 0x43, 0x4d, 0x33, 0x85, 0x45, 0xf9, 0x02, 0x7f, 0x50, 0x3c, 0x9f, 0xa8, 0x51, 0xa3, 0x40, 0x8f, 0x92, 0x9d, 0x38, 0xf5, 0xbc, 0xb6, 0xda, 0x21, 0x10, 0xff, 0xf3, 0xd2, 0xcd, 0x0c, 0x13, 0xec, 0x5f, 0x97, 0x44, 0x17, 0xc4, 0xa7, 0x7e, 0x3d, 0x64, 0x5d, 0x19, 0x73, 0x60, 0x81, 0x4f, 0xdc, 0x22, 0x2a, 0x90, 0x88, 0x46, 0xee, 0xb8, 0x14, 0xde, 0x5e, 0x0b, 0xdb, 0xe0, 0x32, 0x3a, 0x0a, 0x49, 0x06, 0x24, 0x5c, 0xc2, 0xd3, 0xac, 0x62, 0x91, 0x95, 0xe4, 0x79, 0xe7, 0xc8, 0x37, 0x6d, 0x8d, 0xd5, 0x4e, 0xa9, 0x6c, 0x56, 0xf4, 0xea, 0x65, 0x7a, 0xae, 0x08, 0xba, 0x78, 0x25, 0x2e, 0x1c, 0xa6, 0xb4, 0xc6, 0xe8, 0xdd, 0x74, 0x1f, 0x4b, 0xbd, 0x8b, 0x8a, 0x70, 0x3e, 0xb5, 0x66, 0x48, 0x03, 0xf6, 0x0e, 0x61, 0x35, 0x57, 0xb9, 0x86, 0xc1, 0x1d, 0x9e, 0xe1, 0xf8, 0x98, 0x11, 0x69, 0xd9, 0x8e, 0x94, 0x9b, 0x1e, 0x87, 0xe9, 0xce, 0x55, 0x28, 0xdf, 0x8c, 0xa1, 0x89, 0x0d, 0xbf, 0xe6, 0x42, 0x68, 0x41, 0x99, 0x2d, 0x0f, 0xb0, 0x54, 0xbb, 0x16];
-           var Si = [0x52, 0x09, 0x6a, 0xd5, 0x30, 0x36, 0xa5, 0x38, 0xbf, 0x40, 0xa3, 0x9e, 0x81, 0xf3, 0xd7, 0xfb, 0x7c, 0xe3, 0x39, 0x82, 0x9b, 0x2f, 0xff, 0x87, 0x34, 0x8e, 0x43, 0x44, 0xc4, 0xde, 0xe9, 0xcb, 0x54, 0x7b, 0x94, 0x32, 0xa6, 0xc2, 0x23, 0x3d, 0xee, 0x4c, 0x95, 0x0b, 0x42, 0xfa, 0xc3, 0x4e, 0x08, 0x2e, 0xa1, 0x66, 0x28, 0xd9, 0x24, 0xb2, 0x76, 0x5b, 0xa2, 0x49, 0x6d, 0x8b, 0xd1, 0x25, 0x72, 0xf8, 0xf6, 0x64, 0x86, 0x68, 0x98, 0x16, 0xd4, 0xa4, 0x5c, 0xcc, 0x5d, 0x65, 0xb6, 0x92, 0x6c, 0x70, 0x48, 0x50, 0xfd, 0xed, 0xb9, 0xda, 0x5e, 0x15, 0x46, 0x57, 0xa7, 0x8d, 0x9d, 0x84, 0x90, 0xd8, 0xab, 0x00, 0x8c, 0xbc, 0xd3, 0x0a, 0xf7, 0xe4, 0x58, 0x05, 0xb8, 0xb3, 0x45, 0x06, 0xd0, 0x2c, 0x1e, 0x8f, 0xca, 0x3f, 0x0f, 0x02, 0xc1, 0xaf, 0xbd, 0x03, 0x01, 0x13, 0x8a, 0x6b, 0x3a, 0x91, 0x11, 0x41, 0x4f, 0x67, 0xdc, 0xea, 0x97, 0xf2, 0xcf, 0xce, 0xf0, 0xb4, 0xe6, 0x73, 0x96, 0xac, 0x74, 0x22, 0xe7, 0xad, 0x35, 0x85, 0xe2, 0xf9, 0x37, 0xe8, 0x1c, 0x75, 0xdf, 0x6e, 0x47, 0xf1, 0x1a, 0x71, 0x1d, 0x29, 0xc5, 0x89, 0x6f, 0xb7, 0x62, 0x0e, 0xaa, 0x18, 0xbe, 0x1b, 0xfc, 0x56, 0x3e, 0x4b, 0xc6, 0xd2, 0x79, 0x20, 0x9a, 0xdb, 0xc0, 0xfe, 0x78, 0xcd, 0x5a, 0xf4, 0x1f, 0xdd, 0xa8, 0x33, 0x88, 0x07, 0xc7, 0x31, 0xb1, 0x12, 0x10, 0x59, 0x27, 0x80, 0xec, 0x5f, 0x60, 0x51, 0x7f, 0xa9, 0x19, 0xb5, 0x4a, 0x0d, 0x2d, 0xe5, 0x7a, 0x9f, 0x93, 0xc9, 0x9c, 0xef, 0xa0, 0xe0, 0x3b, 0x4d, 0xae, 0x2a, 0xf5, 0xb0, 0xc8, 0xeb, 0xbb, 0x3c, 0x83, 0x53, 0x99, 0x61, 0x17, 0x2b, 0x04, 0x7e, 0xba, 0x77, 0xd6, 0x26, 0xe1, 0x69, 0x14, 0x63, 0x55, 0x21, 0x0c, 0x7d]; // Transformations for encryption
+       var rbush_1 = rbush;
+       var _default$1 = rbush;
 
-           var T1 = [0xc66363a5, 0xf87c7c84, 0xee777799, 0xf67b7b8d, 0xfff2f20d, 0xd66b6bbd, 0xde6f6fb1, 0x91c5c554, 0x60303050, 0x02010103, 0xce6767a9, 0x562b2b7d, 0xe7fefe19, 0xb5d7d762, 0x4dababe6, 0xec76769a, 0x8fcaca45, 0x1f82829d, 0x89c9c940, 0xfa7d7d87, 0xeffafa15, 0xb25959eb, 0x8e4747c9, 0xfbf0f00b, 0x41adadec, 0xb3d4d467, 0x5fa2a2fd, 0x45afafea, 0x239c9cbf, 0x53a4a4f7, 0xe4727296, 0x9bc0c05b, 0x75b7b7c2, 0xe1fdfd1c, 0x3d9393ae, 0x4c26266a, 0x6c36365a, 0x7e3f3f41, 0xf5f7f702, 0x83cccc4f, 0x6834345c, 0x51a5a5f4, 0xd1e5e534, 0xf9f1f108, 0xe2717193, 0xabd8d873, 0x62313153, 0x2a15153f, 0x0804040c, 0x95c7c752, 0x46232365, 0x9dc3c35e, 0x30181828, 0x379696a1, 0x0a05050f, 0x2f9a9ab5, 0x0e070709, 0x24121236, 0x1b80809b, 0xdfe2e23d, 0xcdebeb26, 0x4e272769, 0x7fb2b2cd, 0xea75759f, 0x1209091b, 0x1d83839e, 0x582c2c74, 0x341a1a2e, 0x361b1b2d, 0xdc6e6eb2, 0xb45a5aee, 0x5ba0a0fb, 0xa45252f6, 0x763b3b4d, 0xb7d6d661, 0x7db3b3ce, 0x5229297b, 0xdde3e33e, 0x5e2f2f71, 0x13848497, 0xa65353f5, 0xb9d1d168, 0x00000000, 0xc1eded2c, 0x40202060, 0xe3fcfc1f, 0x79b1b1c8, 0xb65b5bed, 0xd46a6abe, 0x8dcbcb46, 0x67bebed9, 0x7239394b, 0x944a4ade, 0x984c4cd4, 0xb05858e8, 0x85cfcf4a, 0xbbd0d06b, 0xc5efef2a, 0x4faaaae5, 0xedfbfb16, 0x864343c5, 0x9a4d4dd7, 0x66333355, 0x11858594, 0x8a4545cf, 0xe9f9f910, 0x04020206, 0xfe7f7f81, 0xa05050f0, 0x783c3c44, 0x259f9fba, 0x4ba8a8e3, 0xa25151f3, 0x5da3a3fe, 0x804040c0, 0x058f8f8a, 0x3f9292ad, 0x219d9dbc, 0x70383848, 0xf1f5f504, 0x63bcbcdf, 0x77b6b6c1, 0xafdada75, 0x42212163, 0x20101030, 0xe5ffff1a, 0xfdf3f30e, 0xbfd2d26d, 0x81cdcd4c, 0x180c0c14, 0x26131335, 0xc3ecec2f, 0xbe5f5fe1, 0x359797a2, 0x884444cc, 0x2e171739, 0x93c4c457, 0x55a7a7f2, 0xfc7e7e82, 0x7a3d3d47, 0xc86464ac, 0xba5d5de7, 0x3219192b, 0xe6737395, 0xc06060a0, 0x19818198, 0x9e4f4fd1, 0xa3dcdc7f, 0x44222266, 0x542a2a7e, 0x3b9090ab, 0x0b888883, 0x8c4646ca, 0xc7eeee29, 0x6bb8b8d3, 0x2814143c, 0xa7dede79, 0xbc5e5ee2, 0x160b0b1d, 0xaddbdb76, 0xdbe0e03b, 0x64323256, 0x743a3a4e, 0x140a0a1e, 0x924949db, 0x0c06060a, 0x4824246c, 0xb85c5ce4, 0x9fc2c25d, 0xbdd3d36e, 0x43acacef, 0xc46262a6, 0x399191a8, 0x319595a4, 0xd3e4e437, 0xf279798b, 0xd5e7e732, 0x8bc8c843, 0x6e373759, 0xda6d6db7, 0x018d8d8c, 0xb1d5d564, 0x9c4e4ed2, 0x49a9a9e0, 0xd86c6cb4, 0xac5656fa, 0xf3f4f407, 0xcfeaea25, 0xca6565af, 0xf47a7a8e, 0x47aeaee9, 0x10080818, 0x6fbabad5, 0xf0787888, 0x4a25256f, 0x5c2e2e72, 0x381c1c24, 0x57a6a6f1, 0x73b4b4c7, 0x97c6c651, 0xcbe8e823, 0xa1dddd7c, 0xe874749c, 0x3e1f1f21, 0x964b4bdd, 0x61bdbddc, 0x0d8b8b86, 0x0f8a8a85, 0xe0707090, 0x7c3e3e42, 0x71b5b5c4, 0xcc6666aa, 0x904848d8, 0x06030305, 0xf7f6f601, 0x1c0e0e12, 0xc26161a3, 0x6a35355f, 0xae5757f9, 0x69b9b9d0, 0x17868691, 0x99c1c158, 0x3a1d1d27, 0x279e9eb9, 0xd9e1e138, 0xebf8f813, 0x2b9898b3, 0x22111133, 0xd26969bb, 0xa9d9d970, 0x078e8e89, 0x339494a7, 0x2d9b9bb6, 0x3c1e1e22, 0x15878792, 0xc9e9e920, 0x87cece49, 0xaa5555ff, 0x50282878, 0xa5dfdf7a, 0x038c8c8f, 0x59a1a1f8, 0x09898980, 0x1a0d0d17, 0x65bfbfda, 0xd7e6e631, 0x844242c6, 0xd06868b8, 0x824141c3, 0x299999b0, 0x5a2d2d77, 0x1e0f0f11, 0x7bb0b0cb, 0xa85454fc, 0x6dbbbbd6, 0x2c16163a];
-           var T2 = [0xa5c66363, 0x84f87c7c, 0x99ee7777, 0x8df67b7b, 0x0dfff2f2, 0xbdd66b6b, 0xb1de6f6f, 0x5491c5c5, 0x50603030, 0x03020101, 0xa9ce6767, 0x7d562b2b, 0x19e7fefe, 0x62b5d7d7, 0xe64dabab, 0x9aec7676, 0x458fcaca, 0x9d1f8282, 0x4089c9c9, 0x87fa7d7d, 0x15effafa, 0xebb25959, 0xc98e4747, 0x0bfbf0f0, 0xec41adad, 0x67b3d4d4, 0xfd5fa2a2, 0xea45afaf, 0xbf239c9c, 0xf753a4a4, 0x96e47272, 0x5b9bc0c0, 0xc275b7b7, 0x1ce1fdfd, 0xae3d9393, 0x6a4c2626, 0x5a6c3636, 0x417e3f3f, 0x02f5f7f7, 0x4f83cccc, 0x5c683434, 0xf451a5a5, 0x34d1e5e5, 0x08f9f1f1, 0x93e27171, 0x73abd8d8, 0x53623131, 0x3f2a1515, 0x0c080404, 0x5295c7c7, 0x65462323, 0x5e9dc3c3, 0x28301818, 0xa1379696, 0x0f0a0505, 0xb52f9a9a, 0x090e0707, 0x36241212, 0x9b1b8080, 0x3ddfe2e2, 0x26cdebeb, 0x694e2727, 0xcd7fb2b2, 0x9fea7575, 0x1b120909, 0x9e1d8383, 0x74582c2c, 0x2e341a1a, 0x2d361b1b, 0xb2dc6e6e, 0xeeb45a5a, 0xfb5ba0a0, 0xf6a45252, 0x4d763b3b, 0x61b7d6d6, 0xce7db3b3, 0x7b522929, 0x3edde3e3, 0x715e2f2f, 0x97138484, 0xf5a65353, 0x68b9d1d1, 0x00000000, 0x2cc1eded, 0x60402020, 0x1fe3fcfc, 0xc879b1b1, 0xedb65b5b, 0xbed46a6a, 0x468dcbcb, 0xd967bebe, 0x4b723939, 0xde944a4a, 0xd4984c4c, 0xe8b05858, 0x4a85cfcf, 0x6bbbd0d0, 0x2ac5efef, 0xe54faaaa, 0x16edfbfb, 0xc5864343, 0xd79a4d4d, 0x55663333, 0x94118585, 0xcf8a4545, 0x10e9f9f9, 0x06040202, 0x81fe7f7f, 0xf0a05050, 0x44783c3c, 0xba259f9f, 0xe34ba8a8, 0xf3a25151, 0xfe5da3a3, 0xc0804040, 0x8a058f8f, 0xad3f9292, 0xbc219d9d, 0x48703838, 0x04f1f5f5, 0xdf63bcbc, 0xc177b6b6, 0x75afdada, 0x63422121, 0x30201010, 0x1ae5ffff, 0x0efdf3f3, 0x6dbfd2d2, 0x4c81cdcd, 0x14180c0c, 0x35261313, 0x2fc3ecec, 0xe1be5f5f, 0xa2359797, 0xcc884444, 0x392e1717, 0x5793c4c4, 0xf255a7a7, 0x82fc7e7e, 0x477a3d3d, 0xacc86464, 0xe7ba5d5d, 0x2b321919, 0x95e67373, 0xa0c06060, 0x98198181, 0xd19e4f4f, 0x7fa3dcdc, 0x66442222, 0x7e542a2a, 0xab3b9090, 0x830b8888, 0xca8c4646, 0x29c7eeee, 0xd36bb8b8, 0x3c281414, 0x79a7dede, 0xe2bc5e5e, 0x1d160b0b, 0x76addbdb, 0x3bdbe0e0, 0x56643232, 0x4e743a3a, 0x1e140a0a, 0xdb924949, 0x0a0c0606, 0x6c482424, 0xe4b85c5c, 0x5d9fc2c2, 0x6ebdd3d3, 0xef43acac, 0xa6c46262, 0xa8399191, 0xa4319595, 0x37d3e4e4, 0x8bf27979, 0x32d5e7e7, 0x438bc8c8, 0x596e3737, 0xb7da6d6d, 0x8c018d8d, 0x64b1d5d5, 0xd29c4e4e, 0xe049a9a9, 0xb4d86c6c, 0xfaac5656, 0x07f3f4f4, 0x25cfeaea, 0xafca6565, 0x8ef47a7a, 0xe947aeae, 0x18100808, 0xd56fbaba, 0x88f07878, 0x6f4a2525, 0x725c2e2e, 0x24381c1c, 0xf157a6a6, 0xc773b4b4, 0x5197c6c6, 0x23cbe8e8, 0x7ca1dddd, 0x9ce87474, 0x213e1f1f, 0xdd964b4b, 0xdc61bdbd, 0x860d8b8b, 0x850f8a8a, 0x90e07070, 0x427c3e3e, 0xc471b5b5, 0xaacc6666, 0xd8904848, 0x05060303, 0x01f7f6f6, 0x121c0e0e, 0xa3c26161, 0x5f6a3535, 0xf9ae5757, 0xd069b9b9, 0x91178686, 0x5899c1c1, 0x273a1d1d, 0xb9279e9e, 0x38d9e1e1, 0x13ebf8f8, 0xb32b9898, 0x33221111, 0xbbd26969, 0x70a9d9d9, 0x89078e8e, 0xa7339494, 0xb62d9b9b, 0x223c1e1e, 0x92158787, 0x20c9e9e9, 0x4987cece, 0xffaa5555, 0x78502828, 0x7aa5dfdf, 0x8f038c8c, 0xf859a1a1, 0x80098989, 0x171a0d0d, 0xda65bfbf, 0x31d7e6e6, 0xc6844242, 0xb8d06868, 0xc3824141, 0xb0299999, 0x775a2d2d, 0x111e0f0f, 0xcb7bb0b0, 0xfca85454, 0xd66dbbbb, 0x3a2c1616];
-           var T3 = [0x63a5c663, 0x7c84f87c, 0x7799ee77, 0x7b8df67b, 0xf20dfff2, 0x6bbdd66b, 0x6fb1de6f, 0xc55491c5, 0x30506030, 0x01030201, 0x67a9ce67, 0x2b7d562b, 0xfe19e7fe, 0xd762b5d7, 0xabe64dab, 0x769aec76, 0xca458fca, 0x829d1f82, 0xc94089c9, 0x7d87fa7d, 0xfa15effa, 0x59ebb259, 0x47c98e47, 0xf00bfbf0, 0xadec41ad, 0xd467b3d4, 0xa2fd5fa2, 0xafea45af, 0x9cbf239c, 0xa4f753a4, 0x7296e472, 0xc05b9bc0, 0xb7c275b7, 0xfd1ce1fd, 0x93ae3d93, 0x266a4c26, 0x365a6c36, 0x3f417e3f, 0xf702f5f7, 0xcc4f83cc, 0x345c6834, 0xa5f451a5, 0xe534d1e5, 0xf108f9f1, 0x7193e271, 0xd873abd8, 0x31536231, 0x153f2a15, 0x040c0804, 0xc75295c7, 0x23654623, 0xc35e9dc3, 0x18283018, 0x96a13796, 0x050f0a05, 0x9ab52f9a, 0x07090e07, 0x12362412, 0x809b1b80, 0xe23ddfe2, 0xeb26cdeb, 0x27694e27, 0xb2cd7fb2, 0x759fea75, 0x091b1209, 0x839e1d83, 0x2c74582c, 0x1a2e341a, 0x1b2d361b, 0x6eb2dc6e, 0x5aeeb45a, 0xa0fb5ba0, 0x52f6a452, 0x3b4d763b, 0xd661b7d6, 0xb3ce7db3, 0x297b5229, 0xe33edde3, 0x2f715e2f, 0x84971384, 0x53f5a653, 0xd168b9d1, 0x00000000, 0xed2cc1ed, 0x20604020, 0xfc1fe3fc, 0xb1c879b1, 0x5bedb65b, 0x6abed46a, 0xcb468dcb, 0xbed967be, 0x394b7239, 0x4ade944a, 0x4cd4984c, 0x58e8b058, 0xcf4a85cf, 0xd06bbbd0, 0xef2ac5ef, 0xaae54faa, 0xfb16edfb, 0x43c58643, 0x4dd79a4d, 0x33556633, 0x85941185, 0x45cf8a45, 0xf910e9f9, 0x02060402, 0x7f81fe7f, 0x50f0a050, 0x3c44783c, 0x9fba259f, 0xa8e34ba8, 0x51f3a251, 0xa3fe5da3, 0x40c08040, 0x8f8a058f, 0x92ad3f92, 0x9dbc219d, 0x38487038, 0xf504f1f5, 0xbcdf63bc, 0xb6c177b6, 0xda75afda, 0x21634221, 0x10302010, 0xff1ae5ff, 0xf30efdf3, 0xd26dbfd2, 0xcd4c81cd, 0x0c14180c, 0x13352613, 0xec2fc3ec, 0x5fe1be5f, 0x97a23597, 0x44cc8844, 0x17392e17, 0xc45793c4, 0xa7f255a7, 0x7e82fc7e, 0x3d477a3d, 0x64acc864, 0x5de7ba5d, 0x192b3219, 0x7395e673, 0x60a0c060, 0x81981981, 0x4fd19e4f, 0xdc7fa3dc, 0x22664422, 0x2a7e542a, 0x90ab3b90, 0x88830b88, 0x46ca8c46, 0xee29c7ee, 0xb8d36bb8, 0x143c2814, 0xde79a7de, 0x5ee2bc5e, 0x0b1d160b, 0xdb76addb, 0xe03bdbe0, 0x32566432, 0x3a4e743a, 0x0a1e140a, 0x49db9249, 0x060a0c06, 0x246c4824, 0x5ce4b85c, 0xc25d9fc2, 0xd36ebdd3, 0xacef43ac, 0x62a6c462, 0x91a83991, 0x95a43195, 0xe437d3e4, 0x798bf279, 0xe732d5e7, 0xc8438bc8, 0x37596e37, 0x6db7da6d, 0x8d8c018d, 0xd564b1d5, 0x4ed29c4e, 0xa9e049a9, 0x6cb4d86c, 0x56faac56, 0xf407f3f4, 0xea25cfea, 0x65afca65, 0x7a8ef47a, 0xaee947ae, 0x08181008, 0xbad56fba, 0x7888f078, 0x256f4a25, 0x2e725c2e, 0x1c24381c, 0xa6f157a6, 0xb4c773b4, 0xc65197c6, 0xe823cbe8, 0xdd7ca1dd, 0x749ce874, 0x1f213e1f, 0x4bdd964b, 0xbddc61bd, 0x8b860d8b, 0x8a850f8a, 0x7090e070, 0x3e427c3e, 0xb5c471b5, 0x66aacc66, 0x48d89048, 0x03050603, 0xf601f7f6, 0x0e121c0e, 0x61a3c261, 0x355f6a35, 0x57f9ae57, 0xb9d069b9, 0x86911786, 0xc15899c1, 0x1d273a1d, 0x9eb9279e, 0xe138d9e1, 0xf813ebf8, 0x98b32b98, 0x11332211, 0x69bbd269, 0xd970a9d9, 0x8e89078e, 0x94a73394, 0x9bb62d9b, 0x1e223c1e, 0x87921587, 0xe920c9e9, 0xce4987ce, 0x55ffaa55, 0x28785028, 0xdf7aa5df, 0x8c8f038c, 0xa1f859a1, 0x89800989, 0x0d171a0d, 0xbfda65bf, 0xe631d7e6, 0x42c68442, 0x68b8d068, 0x41c38241, 0x99b02999, 0x2d775a2d, 0x0f111e0f, 0xb0cb7bb0, 0x54fca854, 0xbbd66dbb, 0x163a2c16];
-           var T4 = [0x6363a5c6, 0x7c7c84f8, 0x777799ee, 0x7b7b8df6, 0xf2f20dff, 0x6b6bbdd6, 0x6f6fb1de, 0xc5c55491, 0x30305060, 0x01010302, 0x6767a9ce, 0x2b2b7d56, 0xfefe19e7, 0xd7d762b5, 0xababe64d, 0x76769aec, 0xcaca458f, 0x82829d1f, 0xc9c94089, 0x7d7d87fa, 0xfafa15ef, 0x5959ebb2, 0x4747c98e, 0xf0f00bfb, 0xadadec41, 0xd4d467b3, 0xa2a2fd5f, 0xafafea45, 0x9c9cbf23, 0xa4a4f753, 0x727296e4, 0xc0c05b9b, 0xb7b7c275, 0xfdfd1ce1, 0x9393ae3d, 0x26266a4c, 0x36365a6c, 0x3f3f417e, 0xf7f702f5, 0xcccc4f83, 0x34345c68, 0xa5a5f451, 0xe5e534d1, 0xf1f108f9, 0x717193e2, 0xd8d873ab, 0x31315362, 0x15153f2a, 0x04040c08, 0xc7c75295, 0x23236546, 0xc3c35e9d, 0x18182830, 0x9696a137, 0x05050f0a, 0x9a9ab52f, 0x0707090e, 0x12123624, 0x80809b1b, 0xe2e23ddf, 0xebeb26cd, 0x2727694e, 0xb2b2cd7f, 0x75759fea, 0x09091b12, 0x83839e1d, 0x2c2c7458, 0x1a1a2e34, 0x1b1b2d36, 0x6e6eb2dc, 0x5a5aeeb4, 0xa0a0fb5b, 0x5252f6a4, 0x3b3b4d76, 0xd6d661b7, 0xb3b3ce7d, 0x29297b52, 0xe3e33edd, 0x2f2f715e, 0x84849713, 0x5353f5a6, 0xd1d168b9, 0x00000000, 0xeded2cc1, 0x20206040, 0xfcfc1fe3, 0xb1b1c879, 0x5b5bedb6, 0x6a6abed4, 0xcbcb468d, 0xbebed967, 0x39394b72, 0x4a4ade94, 0x4c4cd498, 0x5858e8b0, 0xcfcf4a85, 0xd0d06bbb, 0xefef2ac5, 0xaaaae54f, 0xfbfb16ed, 0x4343c586, 0x4d4dd79a, 0x33335566, 0x85859411, 0x4545cf8a, 0xf9f910e9, 0x02020604, 0x7f7f81fe, 0x5050f0a0, 0x3c3c4478, 0x9f9fba25, 0xa8a8e34b, 0x5151f3a2, 0xa3a3fe5d, 0x4040c080, 0x8f8f8a05, 0x9292ad3f, 0x9d9dbc21, 0x38384870, 0xf5f504f1, 0xbcbcdf63, 0xb6b6c177, 0xdada75af, 0x21216342, 0x10103020, 0xffff1ae5, 0xf3f30efd, 0xd2d26dbf, 0xcdcd4c81, 0x0c0c1418, 0x13133526, 0xecec2fc3, 0x5f5fe1be, 0x9797a235, 0x4444cc88, 0x1717392e, 0xc4c45793, 0xa7a7f255, 0x7e7e82fc, 0x3d3d477a, 0x6464acc8, 0x5d5de7ba, 0x19192b32, 0x737395e6, 0x6060a0c0, 0x81819819, 0x4f4fd19e, 0xdcdc7fa3, 0x22226644, 0x2a2a7e54, 0x9090ab3b, 0x8888830b, 0x4646ca8c, 0xeeee29c7, 0xb8b8d36b, 0x14143c28, 0xdede79a7, 0x5e5ee2bc, 0x0b0b1d16, 0xdbdb76ad, 0xe0e03bdb, 0x32325664, 0x3a3a4e74, 0x0a0a1e14, 0x4949db92, 0x06060a0c, 0x24246c48, 0x5c5ce4b8, 0xc2c25d9f, 0xd3d36ebd, 0xacacef43, 0x6262a6c4, 0x9191a839, 0x9595a431, 0xe4e437d3, 0x79798bf2, 0xe7e732d5, 0xc8c8438b, 0x3737596e, 0x6d6db7da, 0x8d8d8c01, 0xd5d564b1, 0x4e4ed29c, 0xa9a9e049, 0x6c6cb4d8, 0x5656faac, 0xf4f407f3, 0xeaea25cf, 0x6565afca, 0x7a7a8ef4, 0xaeaee947, 0x08081810, 0xbabad56f, 0x787888f0, 0x25256f4a, 0x2e2e725c, 0x1c1c2438, 0xa6a6f157, 0xb4b4c773, 0xc6c65197, 0xe8e823cb, 0xdddd7ca1, 0x74749ce8, 0x1f1f213e, 0x4b4bdd96, 0xbdbddc61, 0x8b8b860d, 0x8a8a850f, 0x707090e0, 0x3e3e427c, 0xb5b5c471, 0x6666aacc, 0x4848d890, 0x03030506, 0xf6f601f7, 0x0e0e121c, 0x6161a3c2, 0x35355f6a, 0x5757f9ae, 0xb9b9d069, 0x86869117, 0xc1c15899, 0x1d1d273a, 0x9e9eb927, 0xe1e138d9, 0xf8f813eb, 0x9898b32b, 0x11113322, 0x6969bbd2, 0xd9d970a9, 0x8e8e8907, 0x9494a733, 0x9b9bb62d, 0x1e1e223c, 0x87879215, 0xe9e920c9, 0xcece4987, 0x5555ffaa, 0x28287850, 0xdfdf7aa5, 0x8c8c8f03, 0xa1a1f859, 0x89898009, 0x0d0d171a, 0xbfbfda65, 0xe6e631d7, 0x4242c684, 0x6868b8d0, 0x4141c382, 0x9999b029, 0x2d2d775a, 0x0f0f111e, 0xb0b0cb7b, 0x5454fca8, 0xbbbbd66d, 0x16163a2c]; // Transformations for decryption
+       function rbush(maxEntries, format) {
+         if (!(this instanceof rbush)) return new rbush(maxEntries, format); // max entries in a node is 9 by default; min node fill is 40% for best performance
 
-           var T5 = [0x51f4a750, 0x7e416553, 0x1a17a4c3, 0x3a275e96, 0x3bab6bcb, 0x1f9d45f1, 0xacfa58ab, 0x4be30393, 0x2030fa55, 0xad766df6, 0x88cc7691, 0xf5024c25, 0x4fe5d7fc, 0xc52acbd7, 0x26354480, 0xb562a38f, 0xdeb15a49, 0x25ba1b67, 0x45ea0e98, 0x5dfec0e1, 0xc32f7502, 0x814cf012, 0x8d4697a3, 0x6bd3f9c6, 0x038f5fe7, 0x15929c95, 0xbf6d7aeb, 0x955259da, 0xd4be832d, 0x587421d3, 0x49e06929, 0x8ec9c844, 0x75c2896a, 0xf48e7978, 0x99583e6b, 0x27b971dd, 0xbee14fb6, 0xf088ad17, 0xc920ac66, 0x7dce3ab4, 0x63df4a18, 0xe51a3182, 0x97513360, 0x62537f45, 0xb16477e0, 0xbb6bae84, 0xfe81a01c, 0xf9082b94, 0x70486858, 0x8f45fd19, 0x94de6c87, 0x527bf8b7, 0xab73d323, 0x724b02e2, 0xe31f8f57, 0x6655ab2a, 0xb2eb2807, 0x2fb5c203, 0x86c57b9a, 0xd33708a5, 0x302887f2, 0x23bfa5b2, 0x02036aba, 0xed16825c, 0x8acf1c2b, 0xa779b492, 0xf307f2f0, 0x4e69e2a1, 0x65daf4cd, 0x0605bed5, 0xd134621f, 0xc4a6fe8a, 0x342e539d, 0xa2f355a0, 0x058ae132, 0xa4f6eb75, 0x0b83ec39, 0x4060efaa, 0x5e719f06, 0xbd6e1051, 0x3e218af9, 0x96dd063d, 0xdd3e05ae, 0x4de6bd46, 0x91548db5, 0x71c45d05, 0x0406d46f, 0x605015ff, 0x1998fb24, 0xd6bde997, 0x894043cc, 0x67d99e77, 0xb0e842bd, 0x07898b88, 0xe7195b38, 0x79c8eedb, 0xa17c0a47, 0x7c420fe9, 0xf8841ec9, 0x00000000, 0x09808683, 0x322bed48, 0x1e1170ac, 0x6c5a724e, 0xfd0efffb, 0x0f853856, 0x3daed51e, 0x362d3927, 0x0a0fd964, 0x685ca621, 0x9b5b54d1, 0x24362e3a, 0x0c0a67b1, 0x9357e70f, 0xb4ee96d2, 0x1b9b919e, 0x80c0c54f, 0x61dc20a2, 0x5a774b69, 0x1c121a16, 0xe293ba0a, 0xc0a02ae5, 0x3c22e043, 0x121b171d, 0x0e090d0b, 0xf28bc7ad, 0x2db6a8b9, 0x141ea9c8, 0x57f11985, 0xaf75074c, 0xee99ddbb, 0xa37f60fd, 0xf701269f, 0x5c72f5bc, 0x44663bc5, 0x5bfb7e34, 0x8b432976, 0xcb23c6dc, 0xb6edfc68, 0xb8e4f163, 0xd731dcca, 0x42638510, 0x13972240, 0x84c61120, 0x854a247d, 0xd2bb3df8, 0xaef93211, 0xc729a16d, 0x1d9e2f4b, 0xdcb230f3, 0x0d8652ec, 0x77c1e3d0, 0x2bb3166c, 0xa970b999, 0x119448fa, 0x47e96422, 0xa8fc8cc4, 0xa0f03f1a, 0x567d2cd8, 0x223390ef, 0x87494ec7, 0xd938d1c1, 0x8ccaa2fe, 0x98d40b36, 0xa6f581cf, 0xa57ade28, 0xdab78e26, 0x3fadbfa4, 0x2c3a9de4, 0x5078920d, 0x6a5fcc9b, 0x547e4662, 0xf68d13c2, 0x90d8b8e8, 0x2e39f75e, 0x82c3aff5, 0x9f5d80be, 0x69d0937c, 0x6fd52da9, 0xcf2512b3, 0xc8ac993b, 0x10187da7, 0xe89c636e, 0xdb3bbb7b, 0xcd267809, 0x6e5918f4, 0xec9ab701, 0x834f9aa8, 0xe6956e65, 0xaaffe67e, 0x21bccf08, 0xef15e8e6, 0xbae79bd9, 0x4a6f36ce, 0xea9f09d4, 0x29b07cd6, 0x31a4b2af, 0x2a3f2331, 0xc6a59430, 0x35a266c0, 0x744ebc37, 0xfc82caa6, 0xe090d0b0, 0x33a7d815, 0xf104984a, 0x41ecdaf7, 0x7fcd500e, 0x1791f62f, 0x764dd68d, 0x43efb04d, 0xccaa4d54, 0xe49604df, 0x9ed1b5e3, 0x4c6a881b, 0xc12c1fb8, 0x4665517f, 0x9d5eea04, 0x018c355d, 0xfa877473, 0xfb0b412e, 0xb3671d5a, 0x92dbd252, 0xe9105633, 0x6dd64713, 0x9ad7618c, 0x37a10c7a, 0x59f8148e, 0xeb133c89, 0xcea927ee, 0xb761c935, 0xe11ce5ed, 0x7a47b13c, 0x9cd2df59, 0x55f2733f, 0x1814ce79, 0x73c737bf, 0x53f7cdea, 0x5ffdaa5b, 0xdf3d6f14, 0x7844db86, 0xcaaff381, 0xb968c43e, 0x3824342c, 0xc2a3405f, 0x161dc372, 0xbce2250c, 0x283c498b, 0xff0d9541, 0x39a80171, 0x080cb3de, 0xd8b4e49c, 0x6456c190, 0x7bcb8461, 0xd532b670, 0x486c5c74, 0xd0b85742];
-           var T6 = [0x5051f4a7, 0x537e4165, 0xc31a17a4, 0x963a275e, 0xcb3bab6b, 0xf11f9d45, 0xabacfa58, 0x934be303, 0x552030fa, 0xf6ad766d, 0x9188cc76, 0x25f5024c, 0xfc4fe5d7, 0xd7c52acb, 0x80263544, 0x8fb562a3, 0x49deb15a, 0x6725ba1b, 0x9845ea0e, 0xe15dfec0, 0x02c32f75, 0x12814cf0, 0xa38d4697, 0xc66bd3f9, 0xe7038f5f, 0x9515929c, 0xebbf6d7a, 0xda955259, 0x2dd4be83, 0xd3587421, 0x2949e069, 0x448ec9c8, 0x6a75c289, 0x78f48e79, 0x6b99583e, 0xdd27b971, 0xb6bee14f, 0x17f088ad, 0x66c920ac, 0xb47dce3a, 0x1863df4a, 0x82e51a31, 0x60975133, 0x4562537f, 0xe0b16477, 0x84bb6bae, 0x1cfe81a0, 0x94f9082b, 0x58704868, 0x198f45fd, 0x8794de6c, 0xb7527bf8, 0x23ab73d3, 0xe2724b02, 0x57e31f8f, 0x2a6655ab, 0x07b2eb28, 0x032fb5c2, 0x9a86c57b, 0xa5d33708, 0xf2302887, 0xb223bfa5, 0xba02036a, 0x5ced1682, 0x2b8acf1c, 0x92a779b4, 0xf0f307f2, 0xa14e69e2, 0xcd65daf4, 0xd50605be, 0x1fd13462, 0x8ac4a6fe, 0x9d342e53, 0xa0a2f355, 0x32058ae1, 0x75a4f6eb, 0x390b83ec, 0xaa4060ef, 0x065e719f, 0x51bd6e10, 0xf93e218a, 0x3d96dd06, 0xaedd3e05, 0x464de6bd, 0xb591548d, 0x0571c45d, 0x6f0406d4, 0xff605015, 0x241998fb, 0x97d6bde9, 0xcc894043, 0x7767d99e, 0xbdb0e842, 0x8807898b, 0x38e7195b, 0xdb79c8ee, 0x47a17c0a, 0xe97c420f, 0xc9f8841e, 0x00000000, 0x83098086, 0x48322bed, 0xac1e1170, 0x4e6c5a72, 0xfbfd0eff, 0x560f8538, 0x1e3daed5, 0x27362d39, 0x640a0fd9, 0x21685ca6, 0xd19b5b54, 0x3a24362e, 0xb10c0a67, 0x0f9357e7, 0xd2b4ee96, 0x9e1b9b91, 0x4f80c0c5, 0xa261dc20, 0x695a774b, 0x161c121a, 0x0ae293ba, 0xe5c0a02a, 0x433c22e0, 0x1d121b17, 0x0b0e090d, 0xadf28bc7, 0xb92db6a8, 0xc8141ea9, 0x8557f119, 0x4caf7507, 0xbbee99dd, 0xfda37f60, 0x9ff70126, 0xbc5c72f5, 0xc544663b, 0x345bfb7e, 0x768b4329, 0xdccb23c6, 0x68b6edfc, 0x63b8e4f1, 0xcad731dc, 0x10426385, 0x40139722, 0x2084c611, 0x7d854a24, 0xf8d2bb3d, 0x11aef932, 0x6dc729a1, 0x4b1d9e2f, 0xf3dcb230, 0xec0d8652, 0xd077c1e3, 0x6c2bb316, 0x99a970b9, 0xfa119448, 0x2247e964, 0xc4a8fc8c, 0x1aa0f03f, 0xd8567d2c, 0xef223390, 0xc787494e, 0xc1d938d1, 0xfe8ccaa2, 0x3698d40b, 0xcfa6f581, 0x28a57ade, 0x26dab78e, 0xa43fadbf, 0xe42c3a9d, 0x0d507892, 0x9b6a5fcc, 0x62547e46, 0xc2f68d13, 0xe890d8b8, 0x5e2e39f7, 0xf582c3af, 0xbe9f5d80, 0x7c69d093, 0xa96fd52d, 0xb3cf2512, 0x3bc8ac99, 0xa710187d, 0x6ee89c63, 0x7bdb3bbb, 0x09cd2678, 0xf46e5918, 0x01ec9ab7, 0xa8834f9a, 0x65e6956e, 0x7eaaffe6, 0x0821bccf, 0xe6ef15e8, 0xd9bae79b, 0xce4a6f36, 0xd4ea9f09, 0xd629b07c, 0xaf31a4b2, 0x312a3f23, 0x30c6a594, 0xc035a266, 0x37744ebc, 0xa6fc82ca, 0xb0e090d0, 0x1533a7d8, 0x4af10498, 0xf741ecda, 0x0e7fcd50, 0x2f1791f6, 0x8d764dd6, 0x4d43efb0, 0x54ccaa4d, 0xdfe49604, 0xe39ed1b5, 0x1b4c6a88, 0xb8c12c1f, 0x7f466551, 0x049d5eea, 0x5d018c35, 0x73fa8774, 0x2efb0b41, 0x5ab3671d, 0x5292dbd2, 0x33e91056, 0x136dd647, 0x8c9ad761, 0x7a37a10c, 0x8e59f814, 0x89eb133c, 0xeecea927, 0x35b761c9, 0xede11ce5, 0x3c7a47b1, 0x599cd2df, 0x3f55f273, 0x791814ce, 0xbf73c737, 0xea53f7cd, 0x5b5ffdaa, 0x14df3d6f, 0x867844db, 0x81caaff3, 0x3eb968c4, 0x2c382434, 0x5fc2a340, 0x72161dc3, 0x0cbce225, 0x8b283c49, 0x41ff0d95, 0x7139a801, 0xde080cb3, 0x9cd8b4e4, 0x906456c1, 0x617bcb84, 0x70d532b6, 0x74486c5c, 0x42d0b857];
-           var T7 = [0xa75051f4, 0x65537e41, 0xa4c31a17, 0x5e963a27, 0x6bcb3bab, 0x45f11f9d, 0x58abacfa, 0x03934be3, 0xfa552030, 0x6df6ad76, 0x769188cc, 0x4c25f502, 0xd7fc4fe5, 0xcbd7c52a, 0x44802635, 0xa38fb562, 0x5a49deb1, 0x1b6725ba, 0x0e9845ea, 0xc0e15dfe, 0x7502c32f, 0xf012814c, 0x97a38d46, 0xf9c66bd3, 0x5fe7038f, 0x9c951592, 0x7aebbf6d, 0x59da9552, 0x832dd4be, 0x21d35874, 0x692949e0, 0xc8448ec9, 0x896a75c2, 0x7978f48e, 0x3e6b9958, 0x71dd27b9, 0x4fb6bee1, 0xad17f088, 0xac66c920, 0x3ab47dce, 0x4a1863df, 0x3182e51a, 0x33609751, 0x7f456253, 0x77e0b164, 0xae84bb6b, 0xa01cfe81, 0x2b94f908, 0x68587048, 0xfd198f45, 0x6c8794de, 0xf8b7527b, 0xd323ab73, 0x02e2724b, 0x8f57e31f, 0xab2a6655, 0x2807b2eb, 0xc2032fb5, 0x7b9a86c5, 0x08a5d337, 0x87f23028, 0xa5b223bf, 0x6aba0203, 0x825ced16, 0x1c2b8acf, 0xb492a779, 0xf2f0f307, 0xe2a14e69, 0xf4cd65da, 0xbed50605, 0x621fd134, 0xfe8ac4a6, 0x539d342e, 0x55a0a2f3, 0xe132058a, 0xeb75a4f6, 0xec390b83, 0xefaa4060, 0x9f065e71, 0x1051bd6e, 0x8af93e21, 0x063d96dd, 0x05aedd3e, 0xbd464de6, 0x8db59154, 0x5d0571c4, 0xd46f0406, 0x15ff6050, 0xfb241998, 0xe997d6bd, 0x43cc8940, 0x9e7767d9, 0x42bdb0e8, 0x8b880789, 0x5b38e719, 0xeedb79c8, 0x0a47a17c, 0x0fe97c42, 0x1ec9f884, 0x00000000, 0x86830980, 0xed48322b, 0x70ac1e11, 0x724e6c5a, 0xfffbfd0e, 0x38560f85, 0xd51e3dae, 0x3927362d, 0xd9640a0f, 0xa621685c, 0x54d19b5b, 0x2e3a2436, 0x67b10c0a, 0xe70f9357, 0x96d2b4ee, 0x919e1b9b, 0xc54f80c0, 0x20a261dc, 0x4b695a77, 0x1a161c12, 0xba0ae293, 0x2ae5c0a0, 0xe0433c22, 0x171d121b, 0x0d0b0e09, 0xc7adf28b, 0xa8b92db6, 0xa9c8141e, 0x198557f1, 0x074caf75, 0xddbbee99, 0x60fda37f, 0x269ff701, 0xf5bc5c72, 0x3bc54466, 0x7e345bfb, 0x29768b43, 0xc6dccb23, 0xfc68b6ed, 0xf163b8e4, 0xdccad731, 0x85104263, 0x22401397, 0x112084c6, 0x247d854a, 0x3df8d2bb, 0x3211aef9, 0xa16dc729, 0x2f4b1d9e, 0x30f3dcb2, 0x52ec0d86, 0xe3d077c1, 0x166c2bb3, 0xb999a970, 0x48fa1194, 0x642247e9, 0x8cc4a8fc, 0x3f1aa0f0, 0x2cd8567d, 0x90ef2233, 0x4ec78749, 0xd1c1d938, 0xa2fe8cca, 0x0b3698d4, 0x81cfa6f5, 0xde28a57a, 0x8e26dab7, 0xbfa43fad, 0x9de42c3a, 0x920d5078, 0xcc9b6a5f, 0x4662547e, 0x13c2f68d, 0xb8e890d8, 0xf75e2e39, 0xaff582c3, 0x80be9f5d, 0x937c69d0, 0x2da96fd5, 0x12b3cf25, 0x993bc8ac, 0x7da71018, 0x636ee89c, 0xbb7bdb3b, 0x7809cd26, 0x18f46e59, 0xb701ec9a, 0x9aa8834f, 0x6e65e695, 0xe67eaaff, 0xcf0821bc, 0xe8e6ef15, 0x9bd9bae7, 0x36ce4a6f, 0x09d4ea9f, 0x7cd629b0, 0xb2af31a4, 0x23312a3f, 0x9430c6a5, 0x66c035a2, 0xbc37744e, 0xcaa6fc82, 0xd0b0e090, 0xd81533a7, 0x984af104, 0xdaf741ec, 0x500e7fcd, 0xf62f1791, 0xd68d764d, 0xb04d43ef, 0x4d54ccaa, 0x04dfe496, 0xb5e39ed1, 0x881b4c6a, 0x1fb8c12c, 0x517f4665, 0xea049d5e, 0x355d018c, 0x7473fa87, 0x412efb0b, 0x1d5ab367, 0xd25292db, 0x5633e910, 0x47136dd6, 0x618c9ad7, 0x0c7a37a1, 0x148e59f8, 0x3c89eb13, 0x27eecea9, 0xc935b761, 0xe5ede11c, 0xb13c7a47, 0xdf599cd2, 0x733f55f2, 0xce791814, 0x37bf73c7, 0xcdea53f7, 0xaa5b5ffd, 0x6f14df3d, 0xdb867844, 0xf381caaf, 0xc43eb968, 0x342c3824, 0x405fc2a3, 0xc372161d, 0x250cbce2, 0x498b283c, 0x9541ff0d, 0x017139a8, 0xb3de080c, 0xe49cd8b4, 0xc1906456, 0x84617bcb, 0xb670d532, 0x5c74486c, 0x5742d0b8];
-           var T8 = [0xf4a75051, 0x4165537e, 0x17a4c31a, 0x275e963a, 0xab6bcb3b, 0x9d45f11f, 0xfa58abac, 0xe303934b, 0x30fa5520, 0x766df6ad, 0xcc769188, 0x024c25f5, 0xe5d7fc4f, 0x2acbd7c5, 0x35448026, 0x62a38fb5, 0xb15a49de, 0xba1b6725, 0xea0e9845, 0xfec0e15d, 0x2f7502c3, 0x4cf01281, 0x4697a38d, 0xd3f9c66b, 0x8f5fe703, 0x929c9515, 0x6d7aebbf, 0x5259da95, 0xbe832dd4, 0x7421d358, 0xe0692949, 0xc9c8448e, 0xc2896a75, 0x8e7978f4, 0x583e6b99, 0xb971dd27, 0xe14fb6be, 0x88ad17f0, 0x20ac66c9, 0xce3ab47d, 0xdf4a1863, 0x1a3182e5, 0x51336097, 0x537f4562, 0x6477e0b1, 0x6bae84bb, 0x81a01cfe, 0x082b94f9, 0x48685870, 0x45fd198f, 0xde6c8794, 0x7bf8b752, 0x73d323ab, 0x4b02e272, 0x1f8f57e3, 0x55ab2a66, 0xeb2807b2, 0xb5c2032f, 0xc57b9a86, 0x3708a5d3, 0x2887f230, 0xbfa5b223, 0x036aba02, 0x16825ced, 0xcf1c2b8a, 0x79b492a7, 0x07f2f0f3, 0x69e2a14e, 0xdaf4cd65, 0x05bed506, 0x34621fd1, 0xa6fe8ac4, 0x2e539d34, 0xf355a0a2, 0x8ae13205, 0xf6eb75a4, 0x83ec390b, 0x60efaa40, 0x719f065e, 0x6e1051bd, 0x218af93e, 0xdd063d96, 0x3e05aedd, 0xe6bd464d, 0x548db591, 0xc45d0571, 0x06d46f04, 0x5015ff60, 0x98fb2419, 0xbde997d6, 0x4043cc89, 0xd99e7767, 0xe842bdb0, 0x898b8807, 0x195b38e7, 0xc8eedb79, 0x7c0a47a1, 0x420fe97c, 0x841ec9f8, 0x00000000, 0x80868309, 0x2bed4832, 0x1170ac1e, 0x5a724e6c, 0x0efffbfd, 0x8538560f, 0xaed51e3d, 0x2d392736, 0x0fd9640a, 0x5ca62168, 0x5b54d19b, 0x362e3a24, 0x0a67b10c, 0x57e70f93, 0xee96d2b4, 0x9b919e1b, 0xc0c54f80, 0xdc20a261, 0x774b695a, 0x121a161c, 0x93ba0ae2, 0xa02ae5c0, 0x22e0433c, 0x1b171d12, 0x090d0b0e, 0x8bc7adf2, 0xb6a8b92d, 0x1ea9c814, 0xf1198557, 0x75074caf, 0x99ddbbee, 0x7f60fda3, 0x01269ff7, 0x72f5bc5c, 0x663bc544, 0xfb7e345b, 0x4329768b, 0x23c6dccb, 0xedfc68b6, 0xe4f163b8, 0x31dccad7, 0x63851042, 0x97224013, 0xc6112084, 0x4a247d85, 0xbb3df8d2, 0xf93211ae, 0x29a16dc7, 0x9e2f4b1d, 0xb230f3dc, 0x8652ec0d, 0xc1e3d077, 0xb3166c2b, 0x70b999a9, 0x9448fa11, 0xe9642247, 0xfc8cc4a8, 0xf03f1aa0, 0x7d2cd856, 0x3390ef22, 0x494ec787, 0x38d1c1d9, 0xcaa2fe8c, 0xd40b3698, 0xf581cfa6, 0x7ade28a5, 0xb78e26da, 0xadbfa43f, 0x3a9de42c, 0x78920d50, 0x5fcc9b6a, 0x7e466254, 0x8d13c2f6, 0xd8b8e890, 0x39f75e2e, 0xc3aff582, 0x5d80be9f, 0xd0937c69, 0xd52da96f, 0x2512b3cf, 0xac993bc8, 0x187da710, 0x9c636ee8, 0x3bbb7bdb, 0x267809cd, 0x5918f46e, 0x9ab701ec, 0x4f9aa883, 0x956e65e6, 0xffe67eaa, 0xbccf0821, 0x15e8e6ef, 0xe79bd9ba, 0x6f36ce4a, 0x9f09d4ea, 0xb07cd629, 0xa4b2af31, 0x3f23312a, 0xa59430c6, 0xa266c035, 0x4ebc3774, 0x82caa6fc, 0x90d0b0e0, 0xa7d81533, 0x04984af1, 0xecdaf741, 0xcd500e7f, 0x91f62f17, 0x4dd68d76, 0xefb04d43, 0xaa4d54cc, 0x9604dfe4, 0xd1b5e39e, 0x6a881b4c, 0x2c1fb8c1, 0x65517f46, 0x5eea049d, 0x8c355d01, 0x877473fa, 0x0b412efb, 0x671d5ab3, 0xdbd25292, 0x105633e9, 0xd647136d, 0xd7618c9a, 0xa10c7a37, 0xf8148e59, 0x133c89eb, 0xa927eece, 0x61c935b7, 0x1ce5ede1, 0x47b13c7a, 0xd2df599c, 0xf2733f55, 0x14ce7918, 0xc737bf73, 0xf7cdea53, 0xfdaa5b5f, 0x3d6f14df, 0x44db8678, 0xaff381ca, 0x68c43eb9, 0x24342c38, 0xa3405fc2, 0x1dc37216, 0xe2250cbc, 0x3c498b28, 0x0d9541ff, 0xa8017139, 0x0cb3de08, 0xb4e49cd8, 0x56c19064, 0xcb84617b, 0x32b670d5, 0x6c5c7448, 0xb85742d0]; // Transformations for decryption key expansion
+         this._maxEntries = Math.max(4, maxEntries || 9);
+         this._minEntries = Math.max(2, Math.ceil(this._maxEntries * 0.4));
 
-           var U1 = [0x00000000, 0x0e090d0b, 0x1c121a16, 0x121b171d, 0x3824342c, 0x362d3927, 0x24362e3a, 0x2a3f2331, 0x70486858, 0x7e416553, 0x6c5a724e, 0x62537f45, 0x486c5c74, 0x4665517f, 0x547e4662, 0x5a774b69, 0xe090d0b0, 0xee99ddbb, 0xfc82caa6, 0xf28bc7ad, 0xd8b4e49c, 0xd6bde997, 0xc4a6fe8a, 0xcaaff381, 0x90d8b8e8, 0x9ed1b5e3, 0x8ccaa2fe, 0x82c3aff5, 0xa8fc8cc4, 0xa6f581cf, 0xb4ee96d2, 0xbae79bd9, 0xdb3bbb7b, 0xd532b670, 0xc729a16d, 0xc920ac66, 0xe31f8f57, 0xed16825c, 0xff0d9541, 0xf104984a, 0xab73d323, 0xa57ade28, 0xb761c935, 0xb968c43e, 0x9357e70f, 0x9d5eea04, 0x8f45fd19, 0x814cf012, 0x3bab6bcb, 0x35a266c0, 0x27b971dd, 0x29b07cd6, 0x038f5fe7, 0x0d8652ec, 0x1f9d45f1, 0x119448fa, 0x4be30393, 0x45ea0e98, 0x57f11985, 0x59f8148e, 0x73c737bf, 0x7dce3ab4, 0x6fd52da9, 0x61dc20a2, 0xad766df6, 0xa37f60fd, 0xb16477e0, 0xbf6d7aeb, 0x955259da, 0x9b5b54d1, 0x894043cc, 0x87494ec7, 0xdd3e05ae, 0xd33708a5, 0xc12c1fb8, 0xcf2512b3, 0xe51a3182, 0xeb133c89, 0xf9082b94, 0xf701269f, 0x4de6bd46, 0x43efb04d, 0x51f4a750, 0x5ffdaa5b, 0x75c2896a, 0x7bcb8461, 0x69d0937c, 0x67d99e77, 0x3daed51e, 0x33a7d815, 0x21bccf08, 0x2fb5c203, 0x058ae132, 0x0b83ec39, 0x1998fb24, 0x1791f62f, 0x764dd68d, 0x7844db86, 0x6a5fcc9b, 0x6456c190, 0x4e69e2a1, 0x4060efaa, 0x527bf8b7, 0x5c72f5bc, 0x0605bed5, 0x080cb3de, 0x1a17a4c3, 0x141ea9c8, 0x3e218af9, 0x302887f2, 0x223390ef, 0x2c3a9de4, 0x96dd063d, 0x98d40b36, 0x8acf1c2b, 0x84c61120, 0xaef93211, 0xa0f03f1a, 0xb2eb2807, 0xbce2250c, 0xe6956e65, 0xe89c636e, 0xfa877473, 0xf48e7978, 0xdeb15a49, 0xd0b85742, 0xc2a3405f, 0xccaa4d54, 0x41ecdaf7, 0x4fe5d7fc, 0x5dfec0e1, 0x53f7cdea, 0x79c8eedb, 0x77c1e3d0, 0x65daf4cd, 0x6bd3f9c6, 0x31a4b2af, 0x3fadbfa4, 0x2db6a8b9, 0x23bfa5b2, 0x09808683, 0x07898b88, 0x15929c95, 0x1b9b919e, 0xa17c0a47, 0xaf75074c, 0xbd6e1051, 0xb3671d5a, 0x99583e6b, 0x97513360, 0x854a247d, 0x8b432976, 0xd134621f, 0xdf3d6f14, 0xcd267809, 0xc32f7502, 0xe9105633, 0xe7195b38, 0xf5024c25, 0xfb0b412e, 0x9ad7618c, 0x94de6c87, 0x86c57b9a, 0x88cc7691, 0xa2f355a0, 0xacfa58ab, 0xbee14fb6, 0xb0e842bd, 0xea9f09d4, 0xe49604df, 0xf68d13c2, 0xf8841ec9, 0xd2bb3df8, 0xdcb230f3, 0xcea927ee, 0xc0a02ae5, 0x7a47b13c, 0x744ebc37, 0x6655ab2a, 0x685ca621, 0x42638510, 0x4c6a881b, 0x5e719f06, 0x5078920d, 0x0a0fd964, 0x0406d46f, 0x161dc372, 0x1814ce79, 0x322bed48, 0x3c22e043, 0x2e39f75e, 0x2030fa55, 0xec9ab701, 0xe293ba0a, 0xf088ad17, 0xfe81a01c, 0xd4be832d, 0xdab78e26, 0xc8ac993b, 0xc6a59430, 0x9cd2df59, 0x92dbd252, 0x80c0c54f, 0x8ec9c844, 0xa4f6eb75, 0xaaffe67e, 0xb8e4f163, 0xb6edfc68, 0x0c0a67b1, 0x02036aba, 0x10187da7, 0x1e1170ac, 0x342e539d, 0x3a275e96, 0x283c498b, 0x26354480, 0x7c420fe9, 0x724b02e2, 0x605015ff, 0x6e5918f4, 0x44663bc5, 0x4a6f36ce, 0x587421d3, 0x567d2cd8, 0x37a10c7a, 0x39a80171, 0x2bb3166c, 0x25ba1b67, 0x0f853856, 0x018c355d, 0x13972240, 0x1d9e2f4b, 0x47e96422, 0x49e06929, 0x5bfb7e34, 0x55f2733f, 0x7fcd500e, 0x71c45d05, 0x63df4a18, 0x6dd64713, 0xd731dcca, 0xd938d1c1, 0xcb23c6dc, 0xc52acbd7, 0xef15e8e6, 0xe11ce5ed, 0xf307f2f0, 0xfd0efffb, 0xa779b492, 0xa970b999, 0xbb6bae84, 0xb562a38f, 0x9f5d80be, 0x91548db5, 0x834f9aa8, 0x8d4697a3];
-           var U2 = [0x00000000, 0x0b0e090d, 0x161c121a, 0x1d121b17, 0x2c382434, 0x27362d39, 0x3a24362e, 0x312a3f23, 0x58704868, 0x537e4165, 0x4e6c5a72, 0x4562537f, 0x74486c5c, 0x7f466551, 0x62547e46, 0x695a774b, 0xb0e090d0, 0xbbee99dd, 0xa6fc82ca, 0xadf28bc7, 0x9cd8b4e4, 0x97d6bde9, 0x8ac4a6fe, 0x81caaff3, 0xe890d8b8, 0xe39ed1b5, 0xfe8ccaa2, 0xf582c3af, 0xc4a8fc8c, 0xcfa6f581, 0xd2b4ee96, 0xd9bae79b, 0x7bdb3bbb, 0x70d532b6, 0x6dc729a1, 0x66c920ac, 0x57e31f8f, 0x5ced1682, 0x41ff0d95, 0x4af10498, 0x23ab73d3, 0x28a57ade, 0x35b761c9, 0x3eb968c4, 0x0f9357e7, 0x049d5eea, 0x198f45fd, 0x12814cf0, 0xcb3bab6b, 0xc035a266, 0xdd27b971, 0xd629b07c, 0xe7038f5f, 0xec0d8652, 0xf11f9d45, 0xfa119448, 0x934be303, 0x9845ea0e, 0x8557f119, 0x8e59f814, 0xbf73c737, 0xb47dce3a, 0xa96fd52d, 0xa261dc20, 0xf6ad766d, 0xfda37f60, 0xe0b16477, 0xebbf6d7a, 0xda955259, 0xd19b5b54, 0xcc894043, 0xc787494e, 0xaedd3e05, 0xa5d33708, 0xb8c12c1f, 0xb3cf2512, 0x82e51a31, 0x89eb133c, 0x94f9082b, 0x9ff70126, 0x464de6bd, 0x4d43efb0, 0x5051f4a7, 0x5b5ffdaa, 0x6a75c289, 0x617bcb84, 0x7c69d093, 0x7767d99e, 0x1e3daed5, 0x1533a7d8, 0x0821bccf, 0x032fb5c2, 0x32058ae1, 0x390b83ec, 0x241998fb, 0x2f1791f6, 0x8d764dd6, 0x867844db, 0x9b6a5fcc, 0x906456c1, 0xa14e69e2, 0xaa4060ef, 0xb7527bf8, 0xbc5c72f5, 0xd50605be, 0xde080cb3, 0xc31a17a4, 0xc8141ea9, 0xf93e218a, 0xf2302887, 0xef223390, 0xe42c3a9d, 0x3d96dd06, 0x3698d40b, 0x2b8acf1c, 0x2084c611, 0x11aef932, 0x1aa0f03f, 0x07b2eb28, 0x0cbce225, 0x65e6956e, 0x6ee89c63, 0x73fa8774, 0x78f48e79, 0x49deb15a, 0x42d0b857, 0x5fc2a340, 0x54ccaa4d, 0xf741ecda, 0xfc4fe5d7, 0xe15dfec0, 0xea53f7cd, 0xdb79c8ee, 0xd077c1e3, 0xcd65daf4, 0xc66bd3f9, 0xaf31a4b2, 0xa43fadbf, 0xb92db6a8, 0xb223bfa5, 0x83098086, 0x8807898b, 0x9515929c, 0x9e1b9b91, 0x47a17c0a, 0x4caf7507, 0x51bd6e10, 0x5ab3671d, 0x6b99583e, 0x60975133, 0x7d854a24, 0x768b4329, 0x1fd13462, 0x14df3d6f, 0x09cd2678, 0x02c32f75, 0x33e91056, 0x38e7195b, 0x25f5024c, 0x2efb0b41, 0x8c9ad761, 0x8794de6c, 0x9a86c57b, 0x9188cc76, 0xa0a2f355, 0xabacfa58, 0xb6bee14f, 0xbdb0e842, 0xd4ea9f09, 0xdfe49604, 0xc2f68d13, 0xc9f8841e, 0xf8d2bb3d, 0xf3dcb230, 0xeecea927, 0xe5c0a02a, 0x3c7a47b1, 0x37744ebc, 0x2a6655ab, 0x21685ca6, 0x10426385, 0x1b4c6a88, 0x065e719f, 0x0d507892, 0x640a0fd9, 0x6f0406d4, 0x72161dc3, 0x791814ce, 0x48322bed, 0x433c22e0, 0x5e2e39f7, 0x552030fa, 0x01ec9ab7, 0x0ae293ba, 0x17f088ad, 0x1cfe81a0, 0x2dd4be83, 0x26dab78e, 0x3bc8ac99, 0x30c6a594, 0x599cd2df, 0x5292dbd2, 0x4f80c0c5, 0x448ec9c8, 0x75a4f6eb, 0x7eaaffe6, 0x63b8e4f1, 0x68b6edfc, 0xb10c0a67, 0xba02036a, 0xa710187d, 0xac1e1170, 0x9d342e53, 0x963a275e, 0x8b283c49, 0x80263544, 0xe97c420f, 0xe2724b02, 0xff605015, 0xf46e5918, 0xc544663b, 0xce4a6f36, 0xd3587421, 0xd8567d2c, 0x7a37a10c, 0x7139a801, 0x6c2bb316, 0x6725ba1b, 0x560f8538, 0x5d018c35, 0x40139722, 0x4b1d9e2f, 0x2247e964, 0x2949e069, 0x345bfb7e, 0x3f55f273, 0x0e7fcd50, 0x0571c45d, 0x1863df4a, 0x136dd647, 0xcad731dc, 0xc1d938d1, 0xdccb23c6, 0xd7c52acb, 0xe6ef15e8, 0xede11ce5, 0xf0f307f2, 0xfbfd0eff, 0x92a779b4, 0x99a970b9, 0x84bb6bae, 0x8fb562a3, 0xbe9f5d80, 0xb591548d, 0xa8834f9a, 0xa38d4697];
-           var U3 = [0x00000000, 0x0d0b0e09, 0x1a161c12, 0x171d121b, 0x342c3824, 0x3927362d, 0x2e3a2436, 0x23312a3f, 0x68587048, 0x65537e41, 0x724e6c5a, 0x7f456253, 0x5c74486c, 0x517f4665, 0x4662547e, 0x4b695a77, 0xd0b0e090, 0xddbbee99, 0xcaa6fc82, 0xc7adf28b, 0xe49cd8b4, 0xe997d6bd, 0xfe8ac4a6, 0xf381caaf, 0xb8e890d8, 0xb5e39ed1, 0xa2fe8cca, 0xaff582c3, 0x8cc4a8fc, 0x81cfa6f5, 0x96d2b4ee, 0x9bd9bae7, 0xbb7bdb3b, 0xb670d532, 0xa16dc729, 0xac66c920, 0x8f57e31f, 0x825ced16, 0x9541ff0d, 0x984af104, 0xd323ab73, 0xde28a57a, 0xc935b761, 0xc43eb968, 0xe70f9357, 0xea049d5e, 0xfd198f45, 0xf012814c, 0x6bcb3bab, 0x66c035a2, 0x71dd27b9, 0x7cd629b0, 0x5fe7038f, 0x52ec0d86, 0x45f11f9d, 0x48fa1194, 0x03934be3, 0x0e9845ea, 0x198557f1, 0x148e59f8, 0x37bf73c7, 0x3ab47dce, 0x2da96fd5, 0x20a261dc, 0x6df6ad76, 0x60fda37f, 0x77e0b164, 0x7aebbf6d, 0x59da9552, 0x54d19b5b, 0x43cc8940, 0x4ec78749, 0x05aedd3e, 0x08a5d337, 0x1fb8c12c, 0x12b3cf25, 0x3182e51a, 0x3c89eb13, 0x2b94f908, 0x269ff701, 0xbd464de6, 0xb04d43ef, 0xa75051f4, 0xaa5b5ffd, 0x896a75c2, 0x84617bcb, 0x937c69d0, 0x9e7767d9, 0xd51e3dae, 0xd81533a7, 0xcf0821bc, 0xc2032fb5, 0xe132058a, 0xec390b83, 0xfb241998, 0xf62f1791, 0xd68d764d, 0xdb867844, 0xcc9b6a5f, 0xc1906456, 0xe2a14e69, 0xefaa4060, 0xf8b7527b, 0xf5bc5c72, 0xbed50605, 0xb3de080c, 0xa4c31a17, 0xa9c8141e, 0x8af93e21, 0x87f23028, 0x90ef2233, 0x9de42c3a, 0x063d96dd, 0x0b3698d4, 0x1c2b8acf, 0x112084c6, 0x3211aef9, 0x3f1aa0f0, 0x2807b2eb, 0x250cbce2, 0x6e65e695, 0x636ee89c, 0x7473fa87, 0x7978f48e, 0x5a49deb1, 0x5742d0b8, 0x405fc2a3, 0x4d54ccaa, 0xdaf741ec, 0xd7fc4fe5, 0xc0e15dfe, 0xcdea53f7, 0xeedb79c8, 0xe3d077c1, 0xf4cd65da, 0xf9c66bd3, 0xb2af31a4, 0xbfa43fad, 0xa8b92db6, 0xa5b223bf, 0x86830980, 0x8b880789, 0x9c951592, 0x919e1b9b, 0x0a47a17c, 0x074caf75, 0x1051bd6e, 0x1d5ab367, 0x3e6b9958, 0x33609751, 0x247d854a, 0x29768b43, 0x621fd134, 0x6f14df3d, 0x7809cd26, 0x7502c32f, 0x5633e910, 0x5b38e719, 0x4c25f502, 0x412efb0b, 0x618c9ad7, 0x6c8794de, 0x7b9a86c5, 0x769188cc, 0x55a0a2f3, 0x58abacfa, 0x4fb6bee1, 0x42bdb0e8, 0x09d4ea9f, 0x04dfe496, 0x13c2f68d, 0x1ec9f884, 0x3df8d2bb, 0x30f3dcb2, 0x27eecea9, 0x2ae5c0a0, 0xb13c7a47, 0xbc37744e, 0xab2a6655, 0xa621685c, 0x85104263, 0x881b4c6a, 0x9f065e71, 0x920d5078, 0xd9640a0f, 0xd46f0406, 0xc372161d, 0xce791814, 0xed48322b, 0xe0433c22, 0xf75e2e39, 0xfa552030, 0xb701ec9a, 0xba0ae293, 0xad17f088, 0xa01cfe81, 0x832dd4be, 0x8e26dab7, 0x993bc8ac, 0x9430c6a5, 0xdf599cd2, 0xd25292db, 0xc54f80c0, 0xc8448ec9, 0xeb75a4f6, 0xe67eaaff, 0xf163b8e4, 0xfc68b6ed, 0x67b10c0a, 0x6aba0203, 0x7da71018, 0x70ac1e11, 0x539d342e, 0x5e963a27, 0x498b283c, 0x44802635, 0x0fe97c42, 0x02e2724b, 0x15ff6050, 0x18f46e59, 0x3bc54466, 0x36ce4a6f, 0x21d35874, 0x2cd8567d, 0x0c7a37a1, 0x017139a8, 0x166c2bb3, 0x1b6725ba, 0x38560f85, 0x355d018c, 0x22401397, 0x2f4b1d9e, 0x642247e9, 0x692949e0, 0x7e345bfb, 0x733f55f2, 0x500e7fcd, 0x5d0571c4, 0x4a1863df, 0x47136dd6, 0xdccad731, 0xd1c1d938, 0xc6dccb23, 0xcbd7c52a, 0xe8e6ef15, 0xe5ede11c, 0xf2f0f307, 0xfffbfd0e, 0xb492a779, 0xb999a970, 0xae84bb6b, 0xa38fb562, 0x80be9f5d, 0x8db59154, 0x9aa8834f, 0x97a38d46];
-           var U4 = [0x00000000, 0x090d0b0e, 0x121a161c, 0x1b171d12, 0x24342c38, 0x2d392736, 0x362e3a24, 0x3f23312a, 0x48685870, 0x4165537e, 0x5a724e6c, 0x537f4562, 0x6c5c7448, 0x65517f46, 0x7e466254, 0x774b695a, 0x90d0b0e0, 0x99ddbbee, 0x82caa6fc, 0x8bc7adf2, 0xb4e49cd8, 0xbde997d6, 0xa6fe8ac4, 0xaff381ca, 0xd8b8e890, 0xd1b5e39e, 0xcaa2fe8c, 0xc3aff582, 0xfc8cc4a8, 0xf581cfa6, 0xee96d2b4, 0xe79bd9ba, 0x3bbb7bdb, 0x32b670d5, 0x29a16dc7, 0x20ac66c9, 0x1f8f57e3, 0x16825ced, 0x0d9541ff, 0x04984af1, 0x73d323ab, 0x7ade28a5, 0x61c935b7, 0x68c43eb9, 0x57e70f93, 0x5eea049d, 0x45fd198f, 0x4cf01281, 0xab6bcb3b, 0xa266c035, 0xb971dd27, 0xb07cd629, 0x8f5fe703, 0x8652ec0d, 0x9d45f11f, 0x9448fa11, 0xe303934b, 0xea0e9845, 0xf1198557, 0xf8148e59, 0xc737bf73, 0xce3ab47d, 0xd52da96f, 0xdc20a261, 0x766df6ad, 0x7f60fda3, 0x6477e0b1, 0x6d7aebbf, 0x5259da95, 0x5b54d19b, 0x4043cc89, 0x494ec787, 0x3e05aedd, 0x3708a5d3, 0x2c1fb8c1, 0x2512b3cf, 0x1a3182e5, 0x133c89eb, 0x082b94f9, 0x01269ff7, 0xe6bd464d, 0xefb04d43, 0xf4a75051, 0xfdaa5b5f, 0xc2896a75, 0xcb84617b, 0xd0937c69, 0xd99e7767, 0xaed51e3d, 0xa7d81533, 0xbccf0821, 0xb5c2032f, 0x8ae13205, 0x83ec390b, 0x98fb2419, 0x91f62f17, 0x4dd68d76, 0x44db8678, 0x5fcc9b6a, 0x56c19064, 0x69e2a14e, 0x60efaa40, 0x7bf8b752, 0x72f5bc5c, 0x05bed506, 0x0cb3de08, 0x17a4c31a, 0x1ea9c814, 0x218af93e, 0x2887f230, 0x3390ef22, 0x3a9de42c, 0xdd063d96, 0xd40b3698, 0xcf1c2b8a, 0xc6112084, 0xf93211ae, 0xf03f1aa0, 0xeb2807b2, 0xe2250cbc, 0x956e65e6, 0x9c636ee8, 0x877473fa, 0x8e7978f4, 0xb15a49de, 0xb85742d0, 0xa3405fc2, 0xaa4d54cc, 0xecdaf741, 0xe5d7fc4f, 0xfec0e15d, 0xf7cdea53, 0xc8eedb79, 0xc1e3d077, 0xdaf4cd65, 0xd3f9c66b, 0xa4b2af31, 0xadbfa43f, 0xb6a8b92d, 0xbfa5b223, 0x80868309, 0x898b8807, 0x929c9515, 0x9b919e1b, 0x7c0a47a1, 0x75074caf, 0x6e1051bd, 0x671d5ab3, 0x583e6b99, 0x51336097, 0x4a247d85, 0x4329768b, 0x34621fd1, 0x3d6f14df, 0x267809cd, 0x2f7502c3, 0x105633e9, 0x195b38e7, 0x024c25f5, 0x0b412efb, 0xd7618c9a, 0xde6c8794, 0xc57b9a86, 0xcc769188, 0xf355a0a2, 0xfa58abac, 0xe14fb6be, 0xe842bdb0, 0x9f09d4ea, 0x9604dfe4, 0x8d13c2f6, 0x841ec9f8, 0xbb3df8d2, 0xb230f3dc, 0xa927eece, 0xa02ae5c0, 0x47b13c7a, 0x4ebc3774, 0x55ab2a66, 0x5ca62168, 0x63851042, 0x6a881b4c, 0x719f065e, 0x78920d50, 0x0fd9640a, 0x06d46f04, 0x1dc37216, 0x14ce7918, 0x2bed4832, 0x22e0433c, 0x39f75e2e, 0x30fa5520, 0x9ab701ec, 0x93ba0ae2, 0x88ad17f0, 0x81a01cfe, 0xbe832dd4, 0xb78e26da, 0xac993bc8, 0xa59430c6, 0xd2df599c, 0xdbd25292, 0xc0c54f80, 0xc9c8448e, 0xf6eb75a4, 0xffe67eaa, 0xe4f163b8, 0xedfc68b6, 0x0a67b10c, 0x036aba02, 0x187da710, 0x1170ac1e, 0x2e539d34, 0x275e963a, 0x3c498b28, 0x35448026, 0x420fe97c, 0x4b02e272, 0x5015ff60, 0x5918f46e, 0x663bc544, 0x6f36ce4a, 0x7421d358, 0x7d2cd856, 0xa10c7a37, 0xa8017139, 0xb3166c2b, 0xba1b6725, 0x8538560f, 0x8c355d01, 0x97224013, 0x9e2f4b1d, 0xe9642247, 0xe0692949, 0xfb7e345b, 0xf2733f55, 0xcd500e7f, 0xc45d0571, 0xdf4a1863, 0xd647136d, 0x31dccad7, 0x38d1c1d9, 0x23c6dccb, 0x2acbd7c5, 0x15e8e6ef, 0x1ce5ede1, 0x07f2f0f3, 0x0efffbfd, 0x79b492a7, 0x70b999a9, 0x6bae84bb, 0x62a38fb5, 0x5d80be9f, 0x548db591, 0x4f9aa883, 0x4697a38d];
+         if (format) {
+           this._initFormat(format);
+         }
 
-           function convertToInt32(bytes) {
-             var result = [];
+         this.clear();
+       }
 
-             for (var i = 0; i < bytes.length; i += 4) {
-               result.push(bytes[i] << 24 | bytes[i + 1] << 16 | bytes[i + 2] << 8 | bytes[i + 3]);
-             }
+       rbush.prototype = {
+         all: function all() {
+           return this._all(this.data, []);
+         },
+         search: function search(bbox) {
+           var node = this.data,
+               result = [],
+               toBBox = this.toBBox;
+           if (!intersects$1(bbox, node)) return result;
+           var nodesToSearch = [],
+               i,
+               len,
+               child,
+               childBBox;
 
-             return result;
-           }
+           while (node) {
+             for (i = 0, len = node.children.length; i < len; i++) {
+               child = node.children[i];
+               childBBox = node.leaf ? toBBox(child) : child;
 
-           var AES = function AES(key) {
-             if (!(this instanceof AES)) {
-               throw Error('AES must be instanitated with `new`');
+               if (intersects$1(bbox, childBBox)) {
+                 if (node.leaf) result.push(child);else if (contains$1(bbox, childBBox)) this._all(child, result);else nodesToSearch.push(child);
+               }
              }
 
-             Object.defineProperty(this, 'key', {
-               value: coerceArray(key, true)
-             });
-
-             this._prepare();
-           };
-
-           AES.prototype._prepare = function () {
-             var rounds = numberOfRounds[this.key.length];
+             node = nodesToSearch.pop();
+           }
 
-             if (rounds == null) {
-               throw new Error('invalid key size (must be 16, 24 or 32 bytes)');
-             } // encryption round keys
+           return result;
+         },
+         collides: function collides(bbox) {
+           var node = this.data,
+               toBBox = this.toBBox;
+           if (!intersects$1(bbox, node)) return false;
+           var nodesToSearch = [],
+               i,
+               len,
+               child,
+               childBBox;
 
+           while (node) {
+             for (i = 0, len = node.children.length; i < len; i++) {
+               child = node.children[i];
+               childBBox = node.leaf ? toBBox(child) : child;
 
-             this._Ke = []; // decryption round keys
+               if (intersects$1(bbox, childBBox)) {
+                 if (node.leaf || contains$1(bbox, childBBox)) return true;
+                 nodesToSearch.push(child);
+               }
+             }
 
-             this._Kd = [];
+             node = nodesToSearch.pop();
+           }
 
-             for (var i = 0; i <= rounds; i++) {
-               this._Ke.push([0, 0, 0, 0]);
+           return false;
+         },
+         load: function load(data) {
+           if (!(data && data.length)) return this;
 
-               this._Kd.push([0, 0, 0, 0]);
+           if (data.length < this._minEntries) {
+             for (var i = 0, len = data.length; i < len; i++) {
+               this.insert(data[i]);
              }
 
-             var roundKeyCount = (rounds + 1) * 4;
-             var KC = this.key.length / 4; // convert the key into ints
-
-             var tk = convertToInt32(this.key); // copy values into round key arrays
+             return this;
+           } // recursively build the tree with the given data from scratch using OMT algorithm
 
-             var index;
 
-             for (var i = 0; i < KC; i++) {
-               index = i >> 2;
-               this._Ke[index][i % 4] = tk[i];
-               this._Kd[rounds - index][i % 4] = tk[i];
-             } // key expansion (fips-197 section 5.2)
+           var node = this._build(data.slice(), 0, data.length - 1, 0);
 
+           if (!this.data.children.length) {
+             // save as is if tree is empty
+             this.data = node;
+           } else if (this.data.height === node.height) {
+             // split root if trees have the same height
+             this._splitRoot(this.data, node);
+           } else {
+             if (this.data.height < node.height) {
+               // swap trees if inserted one is bigger
+               var tmpNode = this.data;
+               this.data = node;
+               node = tmpNode;
+             } // insert the small tree into the large tree at appropriate level
 
-             var rconpointer = 0;
-             var t = KC,
-                 tt;
 
-             while (t < roundKeyCount) {
-               tt = tk[KC - 1];
-               tk[0] ^= S[tt >> 16 & 0xFF] << 24 ^ S[tt >> 8 & 0xFF] << 16 ^ S[tt & 0xFF] << 8 ^ S[tt >> 24 & 0xFF] ^ rcon[rconpointer] << 24;
-               rconpointer += 1; // key expansion (for non-256 bit)
+             this._insert(node, this.data.height - node.height - 1, true);
+           }
 
-               if (KC != 8) {
-                 for (var i = 1; i < KC; i++) {
-                   tk[i] ^= tk[i - 1];
-                 } // key expansion for 256-bit keys is "slightly different" (fips-197)
+           return this;
+         },
+         insert: function insert(item) {
+           if (item) this._insert(item, this.data.height - 1);
+           return this;
+         },
+         clear: function clear() {
+           this.data = createNode$1([]);
+           return this;
+         },
+         remove: function remove(item, equalsFn) {
+           if (!item) return this;
+           var node = this.data,
+               bbox = this.toBBox(item),
+               path = [],
+               indexes = [],
+               i,
+               parent,
+               index,
+               goingUp; // depth-first iterative tree traversal
 
-               } else {
-                 for (var i = 1; i < KC / 2; i++) {
-                   tk[i] ^= tk[i - 1];
-                 }
+           while (node || path.length) {
+             if (!node) {
+               // go up
+               node = path.pop();
+               parent = path[path.length - 1];
+               i = indexes.pop();
+               goingUp = true;
+             }
 
-                 tt = tk[KC / 2 - 1];
-                 tk[KC / 2] ^= S[tt & 0xFF] ^ S[tt >> 8 & 0xFF] << 8 ^ S[tt >> 16 & 0xFF] << 16 ^ S[tt >> 24 & 0xFF] << 24;
+             if (node.leaf) {
+               // check current node
+               index = findItem$1(item, node.children, equalsFn);
 
-                 for (var i = KC / 2 + 1; i < KC; i++) {
-                   tk[i] ^= tk[i - 1];
-                 }
-               } // copy values into round key arrays
+               if (index !== -1) {
+                 // item found, remove the item and condense tree upwards
+                 node.children.splice(index, 1);
+                 path.push(node);
 
+                 this._condense(path);
 
-               var i = 0,
-                   r,
-                   c;
+                 return this;
+               }
+             }
 
-               while (i < KC && t < roundKeyCount) {
-                 r = t >> 2;
-                 c = t % 4;
-                 this._Ke[r][c] = tk[i];
-                 this._Kd[rounds - r][c] = tk[i++];
-                 t++;
-               }
-             } // inverse-cipher-ify the decryption round key (fips-197 section 5.3)
-
-
-             for (var r = 1; r < rounds; r++) {
-               for (var c = 0; c < 4; c++) {
-                 tt = this._Kd[r][c];
-                 this._Kd[r][c] = U1[tt >> 24 & 0xFF] ^ U2[tt >> 16 & 0xFF] ^ U3[tt >> 8 & 0xFF] ^ U4[tt & 0xFF];
-               }
-             }
-           };
+             if (!goingUp && !node.leaf && contains$1(node, bbox)) {
+               // go down
+               path.push(node);
+               indexes.push(i);
+               i = 0;
+               parent = node;
+               node = node.children[0];
+             } else if (parent) {
+               // go right
+               i++;
+               node = parent.children[i];
+               goingUp = false;
+             } else node = null; // nothing found
 
-           AES.prototype.encrypt = function (plaintext) {
-             if (plaintext.length != 16) {
-               throw new Error('invalid plaintext size (must be 16 bytes)');
-             }
+           }
 
-             var rounds = this._Ke.length - 1;
-             var a = [0, 0, 0, 0]; // convert plaintext to (ints ^ key)
+           return this;
+         },
+         toBBox: function toBBox(item) {
+           return item;
+         },
+         compareMinX: compareNodeMinX$1,
+         compareMinY: compareNodeMinY$1,
+         toJSON: function toJSON() {
+           return this.data;
+         },
+         fromJSON: function fromJSON(data) {
+           this.data = data;
+           return this;
+         },
+         _all: function _all(node, result) {
+           var nodesToSearch = [];
 
-             var t = convertToInt32(plaintext);
+           while (node) {
+             if (node.leaf) result.push.apply(result, node.children);else nodesToSearch.push.apply(nodesToSearch, node.children);
+             node = nodesToSearch.pop();
+           }
 
-             for (var i = 0; i < 4; i++) {
-               t[i] ^= this._Ke[0][i];
-             } // apply round transforms
+           return result;
+         },
+         _build: function _build(items, left, right, height) {
+           var N = right - left + 1,
+               M = this._maxEntries,
+               node;
 
+           if (N <= M) {
+             // reached leaf level; return leaf
+             node = createNode$1(items.slice(left, right + 1));
+             calcBBox$1(node, this.toBBox);
+             return node;
+           }
 
-             for (var r = 1; r < rounds; r++) {
-               for (var i = 0; i < 4; i++) {
-                 a[i] = T1[t[i] >> 24 & 0xff] ^ T2[t[(i + 1) % 4] >> 16 & 0xff] ^ T3[t[(i + 2) % 4] >> 8 & 0xff] ^ T4[t[(i + 3) % 4] & 0xff] ^ this._Ke[r][i];
-               }
+           if (!height) {
+             // target height of the bulk-loaded tree
+             height = Math.ceil(Math.log(N) / Math.log(M)); // target number of root entries to maximize storage utilization
 
-               t = a.slice();
-             } // the last round is special
+             M = Math.ceil(N / Math.pow(M, height - 1));
+           }
 
+           node = createNode$1([]);
+           node.leaf = false;
+           node.height = height; // split the items into M mostly square tiles
 
-             var result = createArray(16),
-                 tt;
+           var N2 = Math.ceil(N / M),
+               N1 = N2 * Math.ceil(Math.sqrt(M)),
+               i,
+               j,
+               right2,
+               right3;
+           multiSelect$1(items, left, right, N1, this.compareMinX);
 
-             for (var i = 0; i < 4; i++) {
-               tt = this._Ke[rounds][i];
-               result[4 * i] = (S[t[i] >> 24 & 0xff] ^ tt >> 24) & 0xff;
-               result[4 * i + 1] = (S[t[(i + 1) % 4] >> 16 & 0xff] ^ tt >> 16) & 0xff;
-               result[4 * i + 2] = (S[t[(i + 2) % 4] >> 8 & 0xff] ^ tt >> 8) & 0xff;
-               result[4 * i + 3] = (S[t[(i + 3) % 4] & 0xff] ^ tt) & 0xff;
-             }
+           for (i = left; i <= right; i += N1) {
+             right2 = Math.min(i + N1 - 1, right);
+             multiSelect$1(items, i, right2, N2, this.compareMinY);
 
-             return result;
-           };
+             for (j = i; j <= right2; j += N2) {
+               right3 = Math.min(j + N2 - 1, right2); // pack each entry recursively
 
-           AES.prototype.decrypt = function (ciphertext) {
-             if (ciphertext.length != 16) {
-               throw new Error('invalid ciphertext size (must be 16 bytes)');
+               node.children.push(this._build(items, j, right3, height - 1));
              }
+           }
 
-             var rounds = this._Kd.length - 1;
-             var a = [0, 0, 0, 0]; // convert plaintext to (ints ^ key)
-
-             var t = convertToInt32(ciphertext);
+           calcBBox$1(node, this.toBBox);
+           return node;
+         },
+         _chooseSubtree: function _chooseSubtree(bbox, node, level, path) {
+           var i, len, child, targetNode, area, enlargement, minArea, minEnlargement;
 
-             for (var i = 0; i < 4; i++) {
-               t[i] ^= this._Kd[0][i];
-             } // apply round transforms
+           while (true) {
+             path.push(node);
+             if (node.leaf || path.length - 1 === level) break;
+             minArea = minEnlargement = Infinity;
 
+             for (i = 0, len = node.children.length; i < len; i++) {
+               child = node.children[i];
+               area = bboxArea$1(child);
+               enlargement = enlargedArea$1(bbox, child) - area; // choose entry with the least area enlargement
 
-             for (var r = 1; r < rounds; r++) {
-               for (var i = 0; i < 4; i++) {
-                 a[i] = T5[t[i] >> 24 & 0xff] ^ T6[t[(i + 3) % 4] >> 16 & 0xff] ^ T7[t[(i + 2) % 4] >> 8 & 0xff] ^ T8[t[(i + 1) % 4] & 0xff] ^ this._Kd[r][i];
+               if (enlargement < minEnlargement) {
+                 minEnlargement = enlargement;
+                 minArea = area < minArea ? area : minArea;
+                 targetNode = child;
+               } else if (enlargement === minEnlargement) {
+                 // otherwise choose one with the smallest area
+                 if (area < minArea) {
+                   minArea = area;
+                   targetNode = child;
+                 }
                }
-
-               t = a.slice();
-             } // the last round is special
-
-
-             var result = createArray(16),
-                 tt;
-
-             for (var i = 0; i < 4; i++) {
-               tt = this._Kd[rounds][i];
-               result[4 * i] = (Si[t[i] >> 24 & 0xff] ^ tt >> 24) & 0xff;
-               result[4 * i + 1] = (Si[t[(i + 3) % 4] >> 16 & 0xff] ^ tt >> 16) & 0xff;
-               result[4 * i + 2] = (Si[t[(i + 2) % 4] >> 8 & 0xff] ^ tt >> 8) & 0xff;
-               result[4 * i + 3] = (Si[t[(i + 1) % 4] & 0xff] ^ tt) & 0xff;
              }
 
-             return result;
-           };
-           /**
-            *  Mode Of Operation - Electonic Codebook (ECB)
-            */
+             node = targetNode || node.children[0];
+           }
 
+           return node;
+         },
+         _insert: function _insert(item, level, isNode) {
+           var toBBox = this.toBBox,
+               bbox = isNode ? item : toBBox(item),
+               insertPath = []; // find the best node for accommodating the item, saving all nodes along the path too
 
-           var ModeOfOperationECB = function ModeOfOperationECB(key) {
-             if (!(this instanceof ModeOfOperationECB)) {
-               throw Error('AES must be instanitated with `new`');
-             }
+           var node = this._chooseSubtree(bbox, this.data, level, insertPath); // put the item into the node
 
-             this.description = "Electronic Code Block";
-             this.name = "ecb";
-             this._aes = new AES(key);
-           };
 
-           ModeOfOperationECB.prototype.encrypt = function (plaintext) {
-             plaintext = coerceArray(plaintext);
+           node.children.push(item);
+           extend$2(node, bbox); // split on node overflow; propagate upwards if necessary
 
-             if (plaintext.length % 16 !== 0) {
-               throw new Error('invalid plaintext size (must be multiple of 16 bytes)');
-             }
+           while (level >= 0) {
+             if (insertPath[level].children.length > this._maxEntries) {
+               this._split(insertPath, level);
 
-             var ciphertext = createArray(plaintext.length);
-             var block = createArray(16);
+               level--;
+             } else break;
+           } // adjust bboxes along the insertion path
 
-             for (var i = 0; i < plaintext.length; i += 16) {
-               copyArray(plaintext, block, 0, i, i + 16);
-               block = this._aes.encrypt(block);
-               copyArray(block, ciphertext, i);
-             }
 
-             return ciphertext;
-           };
+           this._adjustParentBBoxes(bbox, insertPath, level);
+         },
+         // split overflowed node into two
+         _split: function _split(insertPath, level) {
+           var node = insertPath[level],
+               M = node.children.length,
+               m = this._minEntries;
 
-           ModeOfOperationECB.prototype.decrypt = function (ciphertext) {
-             ciphertext = coerceArray(ciphertext);
+           this._chooseSplitAxis(node, m, M);
 
-             if (ciphertext.length % 16 !== 0) {
-               throw new Error('invalid ciphertext size (must be multiple of 16 bytes)');
-             }
+           var splitIndex = this._chooseSplitIndex(node, m, M);
 
-             var plaintext = createArray(ciphertext.length);
-             var block = createArray(16);
+           var newNode = createNode$1(node.children.splice(splitIndex, node.children.length - splitIndex));
+           newNode.height = node.height;
+           newNode.leaf = node.leaf;
+           calcBBox$1(node, this.toBBox);
+           calcBBox$1(newNode, this.toBBox);
+           if (level) insertPath[level - 1].children.push(newNode);else this._splitRoot(node, newNode);
+         },
+         _splitRoot: function _splitRoot(node, newNode) {
+           // split root node
+           this.data = createNode$1([node, newNode]);
+           this.data.height = node.height + 1;
+           this.data.leaf = false;
+           calcBBox$1(this.data, this.toBBox);
+         },
+         _chooseSplitIndex: function _chooseSplitIndex(node, m, M) {
+           var i, bbox1, bbox2, overlap, area, minOverlap, minArea, index;
+           minOverlap = minArea = Infinity;
 
-             for (var i = 0; i < ciphertext.length; i += 16) {
-               copyArray(ciphertext, block, 0, i, i + 16);
-               block = this._aes.decrypt(block);
-               copyArray(block, plaintext, i);
+           for (i = m; i <= M - m; i++) {
+             bbox1 = distBBox$1(node, 0, i, this.toBBox);
+             bbox2 = distBBox$1(node, i, M, this.toBBox);
+             overlap = intersectionArea$1(bbox1, bbox2);
+             area = bboxArea$1(bbox1) + bboxArea$1(bbox2); // choose distribution with minimum overlap
+
+             if (overlap < minOverlap) {
+               minOverlap = overlap;
+               index = i;
+               minArea = area < minArea ? area : minArea;
+             } else if (overlap === minOverlap) {
+               // otherwise choose distribution with minimum area
+               if (area < minArea) {
+                 minArea = area;
+                 index = i;
+               }
              }
+           }
 
-             return plaintext;
-           };
-           /**
-            *  Mode Of Operation - Cipher Block Chaining (CBC)
-            */
+           return index;
+         },
+         // sorts node children by the best axis for split
+         _chooseSplitAxis: function _chooseSplitAxis(node, m, M) {
+           var compareMinX = node.leaf ? this.compareMinX : compareNodeMinX$1,
+               compareMinY = node.leaf ? this.compareMinY : compareNodeMinY$1,
+               xMargin = this._allDistMargin(node, m, M, compareMinX),
+               yMargin = this._allDistMargin(node, m, M, compareMinY); // if total distributions margin value is minimal for x, sort by minX,
+           // otherwise it's already sorted by minY
 
 
-           var ModeOfOperationCBC = function ModeOfOperationCBC(key, iv) {
-             if (!(this instanceof ModeOfOperationCBC)) {
-               throw Error('AES must be instanitated with `new`');
-             }
+           if (xMargin < yMargin) node.children.sort(compareMinX);
+         },
+         // total margin of all possible split distributions where each node is at least m full
+         _allDistMargin: function _allDistMargin(node, m, M, compare) {
+           node.children.sort(compare);
+           var toBBox = this.toBBox,
+               leftBBox = distBBox$1(node, 0, m, toBBox),
+               rightBBox = distBBox$1(node, M - m, M, toBBox),
+               margin = bboxMargin$1(leftBBox) + bboxMargin$1(rightBBox),
+               i,
+               child;
 
-             this.description = "Cipher Block Chaining";
-             this.name = "cbc";
+           for (i = m; i < M - m; i++) {
+             child = node.children[i];
+             extend$2(leftBBox, node.leaf ? toBBox(child) : child);
+             margin += bboxMargin$1(leftBBox);
+           }
 
-             if (!iv) {
-               iv = createArray(16);
-             } else if (iv.length != 16) {
-               throw new Error('invalid initialation vector size (must be 16 bytes)');
-             }
+           for (i = M - m - 1; i >= m; i--) {
+             child = node.children[i];
+             extend$2(rightBBox, node.leaf ? toBBox(child) : child);
+             margin += bboxMargin$1(rightBBox);
+           }
 
-             this._lastCipherblock = coerceArray(iv, true);
-             this._aes = new AES(key);
-           };
+           return margin;
+         },
+         _adjustParentBBoxes: function _adjustParentBBoxes(bbox, path, level) {
+           // adjust bboxes along the given tree path
+           for (var i = level; i >= 0; i--) {
+             extend$2(path[i], bbox);
+           }
+         },
+         _condense: function _condense(path) {
+           // go through the path, removing empty nodes and updating bboxes
+           for (var i = path.length - 1, siblings; i >= 0; i--) {
+             if (path[i].children.length === 0) {
+               if (i > 0) {
+                 siblings = path[i - 1].children;
+                 siblings.splice(siblings.indexOf(path[i]), 1);
+               } else this.clear();
+             } else calcBBox$1(path[i], this.toBBox);
+           }
+         },
+         _initFormat: function _initFormat(format) {
+           // data format (minX, minY, maxX, maxY accessors)
+           // uses eval-type function compilation instead of just accepting a toBBox function
+           // because the algorithms are very sensitive to sorting functions performance,
+           // so they should be dead simple and without inner calls
+           var compareArr = ['return a', ' - b', ';'];
+           this.compareMinX = new Function('a', 'b', compareArr.join(format[0]));
+           this.compareMinY = new Function('a', 'b', compareArr.join(format[1]));
+           this.toBBox = new Function('a', 'return {minX: a' + format[0] + ', minY: a' + format[1] + ', maxX: a' + format[2] + ', maxY: a' + format[3] + '};');
+         }
+       };
 
-           ModeOfOperationCBC.prototype.encrypt = function (plaintext) {
-             plaintext = coerceArray(plaintext);
+       function findItem$1(item, items, equalsFn) {
+         if (!equalsFn) return items.indexOf(item);
 
-             if (plaintext.length % 16 !== 0) {
-               throw new Error('invalid plaintext size (must be multiple of 16 bytes)');
-             }
+         for (var i = 0; i < items.length; i++) {
+           if (equalsFn(item, items[i])) return i;
+         }
 
-             var ciphertext = createArray(plaintext.length);
-             var block = createArray(16);
+         return -1;
+       } // calculate node's bbox from bboxes of its children
 
-             for (var i = 0; i < plaintext.length; i += 16) {
-               copyArray(plaintext, block, 0, i, i + 16);
 
-               for (var j = 0; j < 16; j++) {
-                 block[j] ^= this._lastCipherblock[j];
-               }
+       function calcBBox$1(node, toBBox) {
+         distBBox$1(node, 0, node.children.length, toBBox, node);
+       } // min bounding rectangle of node children from k to p-1
 
-               this._lastCipherblock = this._aes.encrypt(block);
-               copyArray(this._lastCipherblock, ciphertext, i);
-             }
 
-             return ciphertext;
-           };
+       function distBBox$1(node, k, p, toBBox, destNode) {
+         if (!destNode) destNode = createNode$1(null);
+         destNode.minX = Infinity;
+         destNode.minY = Infinity;
+         destNode.maxX = -Infinity;
+         destNode.maxY = -Infinity;
 
-           ModeOfOperationCBC.prototype.decrypt = function (ciphertext) {
-             ciphertext = coerceArray(ciphertext);
+         for (var i = k, child; i < p; i++) {
+           child = node.children[i];
+           extend$2(destNode, node.leaf ? toBBox(child) : child);
+         }
 
-             if (ciphertext.length % 16 !== 0) {
-               throw new Error('invalid ciphertext size (must be multiple of 16 bytes)');
-             }
+         return destNode;
+       }
 
-             var plaintext = createArray(ciphertext.length);
-             var block = createArray(16);
+       function extend$2(a, b) {
+         a.minX = Math.min(a.minX, b.minX);
+         a.minY = Math.min(a.minY, b.minY);
+         a.maxX = Math.max(a.maxX, b.maxX);
+         a.maxY = Math.max(a.maxY, b.maxY);
+         return a;
+       }
 
-             for (var i = 0; i < ciphertext.length; i += 16) {
-               copyArray(ciphertext, block, 0, i, i + 16);
-               block = this._aes.decrypt(block);
+       function compareNodeMinX$1(a, b) {
+         return a.minX - b.minX;
+       }
 
-               for (var j = 0; j < 16; j++) {
-                 plaintext[i + j] = block[j] ^ this._lastCipherblock[j];
-               }
+       function compareNodeMinY$1(a, b) {
+         return a.minY - b.minY;
+       }
 
-               copyArray(ciphertext, this._lastCipherblock, 0, i, i + 16);
-             }
+       function bboxArea$1(a) {
+         return (a.maxX - a.minX) * (a.maxY - a.minY);
+       }
 
-             return plaintext;
-           };
-           /**
-            *  Mode Of Operation - Cipher Feedback (CFB)
-            */
+       function bboxMargin$1(a) {
+         return a.maxX - a.minX + (a.maxY - a.minY);
+       }
 
+       function enlargedArea$1(a, b) {
+         return (Math.max(b.maxX, a.maxX) - Math.min(b.minX, a.minX)) * (Math.max(b.maxY, a.maxY) - Math.min(b.minY, a.minY));
+       }
 
-           var ModeOfOperationCFB = function ModeOfOperationCFB(key, iv, segmentSize) {
-             if (!(this instanceof ModeOfOperationCFB)) {
-               throw Error('AES must be instanitated with `new`');
-             }
+       function intersectionArea$1(a, b) {
+         var minX = Math.max(a.minX, b.minX),
+             minY = Math.max(a.minY, b.minY),
+             maxX = Math.min(a.maxX, b.maxX),
+             maxY = Math.min(a.maxY, b.maxY);
+         return Math.max(0, maxX - minX) * Math.max(0, maxY - minY);
+       }
 
-             this.description = "Cipher Feedback";
-             this.name = "cfb";
+       function contains$1(a, b) {
+         return a.minX <= b.minX && a.minY <= b.minY && b.maxX <= a.maxX && b.maxY <= a.maxY;
+       }
 
-             if (!iv) {
-               iv = createArray(16);
-             } else if (iv.length != 16) {
-               throw new Error('invalid initialation vector size (must be 16 size)');
-             }
+       function intersects$1(a, b) {
+         return b.minX <= a.maxX && b.minY <= a.maxY && b.maxX >= a.minX && b.maxY >= a.minY;
+       }
 
-             if (!segmentSize) {
-               segmentSize = 1;
-             }
+       function createNode$1(children) {
+         return {
+           children: children,
+           height: 1,
+           leaf: true,
+           minX: Infinity,
+           minY: Infinity,
+           maxX: -Infinity,
+           maxY: -Infinity
+         };
+       } // sort an array so that items come in groups of n unsorted items, with groups sorted between each other;
+       // combines selection algorithm with binary divide & conquer approach
 
-             this.segmentSize = segmentSize;
-             this._shiftRegister = coerceArray(iv, true);
-             this._aes = new AES(key);
-           };
 
-           ModeOfOperationCFB.prototype.encrypt = function (plaintext) {
-             if (plaintext.length % this.segmentSize != 0) {
-               throw new Error('invalid plaintext size (must be segmentSize bytes)');
-             }
+       function multiSelect$1(arr, left, right, n, compare) {
+         var stack = [left, right],
+             mid;
 
-             var encrypted = coerceArray(plaintext, true);
-             var xorSegment;
+         while (stack.length) {
+           right = stack.pop();
+           left = stack.pop();
+           if (right - left <= n) continue;
+           mid = left + Math.ceil((right - left) / n / 2) * n;
+           quickselect$1(arr, mid, left, right, compare);
+           stack.push(left, mid, mid, right);
+         }
+       }
+       rbush_1["default"] = _default$1;
 
-             for (var i = 0; i < encrypted.length; i += this.segmentSize) {
-               xorSegment = this._aes.encrypt(this._shiftRegister);
+       var lineclip_1 = lineclip$1;
+       lineclip$1.polyline = lineclip$1;
+       lineclip$1.polygon = polygonclip$1; // Cohen-Sutherland line clippign algorithm, adapted to efficiently
+       // handle polylines rather than just segments
 
-               for (var j = 0; j < this.segmentSize; j++) {
-                 encrypted[i + j] ^= xorSegment[j];
-               } // Shift the register
+       function lineclip$1(points, bbox, result) {
+         var len = points.length,
+             codeA = bitCode$1(points[0], bbox),
+             part = [],
+             i,
+             a,
+             b,
+             codeB,
+             lastCode;
+         if (!result) result = [];
 
+         for (i = 1; i < len; i++) {
+           a = points[i - 1];
+           b = points[i];
+           codeB = lastCode = bitCode$1(b, bbox);
 
-               copyArray(this._shiftRegister, this._shiftRegister, 0, this.segmentSize);
-               copyArray(encrypted, this._shiftRegister, 16 - this.segmentSize, i, i + this.segmentSize);
-             }
+           while (true) {
+             if (!(codeA | codeB)) {
+               // accept
+               part.push(a);
 
-             return encrypted;
-           };
+               if (codeB !== lastCode) {
+                 // segment went outside
+                 part.push(b);
 
-           ModeOfOperationCFB.prototype.decrypt = function (ciphertext) {
-             if (ciphertext.length % this.segmentSize != 0) {
-               throw new Error('invalid ciphertext size (must be segmentSize bytes)');
-             }
-
-             var plaintext = coerceArray(ciphertext, true);
-             var xorSegment;
-
-             for (var i = 0; i < plaintext.length; i += this.segmentSize) {
-               xorSegment = this._aes.encrypt(this._shiftRegister);
-
-               for (var j = 0; j < this.segmentSize; j++) {
-                 plaintext[i + j] ^= xorSegment[j];
-               } // Shift the register
-
-
-               copyArray(this._shiftRegister, this._shiftRegister, 0, this.segmentSize);
-               copyArray(ciphertext, this._shiftRegister, 16 - this.segmentSize, i, i + this.segmentSize);
-             }
-
-             return plaintext;
-           };
-           /**
-            *  Mode Of Operation - Output Feedback (OFB)
-            */
-
-
-           var ModeOfOperationOFB = function ModeOfOperationOFB(key, iv) {
-             if (!(this instanceof ModeOfOperationOFB)) {
-               throw Error('AES must be instanitated with `new`');
-             }
-
-             this.description = "Output Feedback";
-             this.name = "ofb";
-
-             if (!iv) {
-               iv = createArray(16);
-             } else if (iv.length != 16) {
-               throw new Error('invalid initialation vector size (must be 16 bytes)');
-             }
-
-             this._lastPrecipher = coerceArray(iv, true);
-             this._lastPrecipherIndex = 16;
-             this._aes = new AES(key);
-           };
-
-           ModeOfOperationOFB.prototype.encrypt = function (plaintext) {
-             var encrypted = coerceArray(plaintext, true);
-
-             for (var i = 0; i < encrypted.length; i++) {
-               if (this._lastPrecipherIndex === 16) {
-                 this._lastPrecipher = this._aes.encrypt(this._lastPrecipher);
-                 this._lastPrecipherIndex = 0;
+                 if (i < len - 1) {
+                   // start a new line
+                   result.push(part);
+                   part = [];
+                 }
+               } else if (i === len - 1) {
+                 part.push(b);
                }
 
-               encrypted[i] ^= this._lastPrecipher[this._lastPrecipherIndex++];
-             }
-
-             return encrypted;
-           }; // Decryption is symetric
-
-
-           ModeOfOperationOFB.prototype.decrypt = ModeOfOperationOFB.prototype.encrypt;
-           /**
-            *  Counter object for CTR common mode of operation
-            */
-
-           var Counter = function Counter(initialValue) {
-             if (!(this instanceof Counter)) {
-               throw Error('Counter must be instanitated with `new`');
-             } // We allow 0, but anything false-ish uses the default 1
-
-
-             if (initialValue !== 0 && !initialValue) {
-               initialValue = 1;
-             }
-
-             if (typeof initialValue === 'number') {
-               this._counter = createArray(16);
-               this.setValue(initialValue);
+               break;
+             } else if (codeA & codeB) {
+               // trivial reject
+               break;
+             } else if (codeA) {
+               // a outside, intersect with clip edge
+               a = intersect$1(a, b, codeA, bbox);
+               codeA = bitCode$1(a, bbox);
              } else {
-               this.setBytes(initialValue);
-             }
-           };
-
-           Counter.prototype.setValue = function (value) {
-             if (typeof value !== 'number' || parseInt(value) != value) {
-               throw new Error('invalid counter value (must be an integer)');
-             } // We cannot safely handle numbers beyond the safe range for integers
-
-
-             if (value > Number.MAX_SAFE_INTEGER) {
-               throw new Error('integer value out of safe range');
-             }
-
-             for (var index = 15; index >= 0; --index) {
-               this._counter[index] = value % 256;
-               value = parseInt(value / 256);
-             }
-           };
-
-           Counter.prototype.setBytes = function (bytes) {
-             bytes = coerceArray(bytes, true);
-
-             if (bytes.length != 16) {
-               throw new Error('invalid counter bytes size (must be 16 bytes)');
-             }
-
-             this._counter = bytes;
-           };
-
-           Counter.prototype.increment = function () {
-             for (var i = 15; i >= 0; i--) {
-               if (this._counter[i] === 255) {
-                 this._counter[i] = 0;
-               } else {
-                 this._counter[i]++;
-                 break;
-               }
-             }
-           };
-           /**
-            *  Mode Of Operation - Counter (CTR)
-            */
-
-
-           var ModeOfOperationCTR = function ModeOfOperationCTR(key, counter) {
-             if (!(this instanceof ModeOfOperationCTR)) {
-               throw Error('AES must be instanitated with `new`');
-             }
-
-             this.description = "Counter";
-             this.name = "ctr";
-
-             if (!(counter instanceof Counter)) {
-               counter = new Counter(counter);
-             }
-
-             this._counter = counter;
-             this._remainingCounter = null;
-             this._remainingCounterIndex = 16;
-             this._aes = new AES(key);
-           };
-
-           ModeOfOperationCTR.prototype.encrypt = function (plaintext) {
-             var encrypted = coerceArray(plaintext, true);
-
-             for (var i = 0; i < encrypted.length; i++) {
-               if (this._remainingCounterIndex === 16) {
-                 this._remainingCounter = this._aes.encrypt(this._counter._counter);
-                 this._remainingCounterIndex = 0;
-
-                 this._counter.increment();
-               }
-
-               encrypted[i] ^= this._remainingCounter[this._remainingCounterIndex++];
-             }
-
-             return encrypted;
-           }; // Decryption is symetric
-
-
-           ModeOfOperationCTR.prototype.decrypt = ModeOfOperationCTR.prototype.encrypt; ///////////////////////
-           // Padding
-           // See:https://tools.ietf.org/html/rfc2315
-
-           function pkcs7pad(data) {
-             data = coerceArray(data, true);
-             var padder = 16 - data.length % 16;
-             var result = createArray(data.length + padder);
-             copyArray(data, result);
-
-             for (var i = data.length; i < result.length; i++) {
-               result[i] = padder;
+               // b outside
+               b = intersect$1(a, b, codeB, bbox);
+               codeB = bitCode$1(b, bbox);
              }
-
-             return result;
            }
 
-           function pkcs7strip(data) {
-             data = coerceArray(data, true);
-
-             if (data.length < 16) {
-               throw new Error('PKCS#7 invalid length');
-             }
-
-             var padder = data[data.length - 1];
+           codeA = lastCode;
+         }
 
-             if (padder > 16) {
-               throw new Error('PKCS#7 padding byte out of range');
-             }
+         if (part.length) result.push(part);
+         return result;
+       } // Sutherland-Hodgeman polygon clipping algorithm
 
-             var length = data.length - padder;
 
-             for (var i = 0; i < padder; i++) {
-               if (data[length + i] !== padder) {
-                 throw new Error('PKCS#7 invalid padding byte');
-               }
-             }
+       function polygonclip$1(points, bbox) {
+         var result, edge, prev, prevInside, i, p, inside; // clip against each side of the clip rectangle
 
-             var result = createArray(length);
-             copyArray(data, result, 0, 0, length);
-             return result;
-           } ///////////////////////
-           // Exporting
-           // The block cipher
+         for (edge = 1; edge <= 8; edge *= 2) {
+           result = [];
+           prev = points[points.length - 1];
+           prevInside = !(bitCode$1(prev, bbox) & edge);
 
+           for (i = 0; i < points.length; i++) {
+             p = points[i];
+             inside = !(bitCode$1(p, bbox) & edge); // if segment goes through the clip window, add an intersection
 
-           var aesjs = {
-             AES: AES,
-             Counter: Counter,
-             ModeOfOperation: {
-               ecb: ModeOfOperationECB,
-               cbc: ModeOfOperationCBC,
-               cfb: ModeOfOperationCFB,
-               ofb: ModeOfOperationOFB,
-               ctr: ModeOfOperationCTR
-             },
-             utils: {
-               hex: convertHex,
-               utf8: convertUtf8
-             },
-             padding: {
-               pkcs7: {
-                 pad: pkcs7pad,
-                 strip: pkcs7strip
-               }
-             },
-             _arrayTest: {
-               coerceArray: coerceArray,
-               createArray: createArray,
-               copyArray: copyArray
-             }
-           }; // node.js
+             if (inside !== prevInside) result.push(intersect$1(prev, p, edge, bbox));
+             if (inside) result.push(p); // add a point if it's inside
 
-           {
-             module.exports = aesjs; // RequireJS/AMD
-             // http://www.requirejs.org/docs/api.html
-             // https://github.com/amdjs/amdjs-api/wiki/AMD
+             prev = p;
+             prevInside = inside;
            }
-         })();
-       });
-
-       // We can use keys that are 128 bits (16 bytes), 192 bits (24 bytes) or 256 bits (32 bytes).
-       // To generate a random key:  window.crypto.getRandomValues(new Uint8Array(16));
-       // This default signing key is built into iD and can be used to mask/unmask sensitive values.
-
-       var DEFAULT_128 = [250, 157, 60, 79, 142, 134, 229, 129, 138, 126, 210, 129, 29, 71, 160, 208];
-       function utilAesEncrypt(text, key) {
-         key = key || DEFAULT_128;
-         var textBytes = aesJs.utils.utf8.toBytes(text);
-         var aesCtr = new aesJs.ModeOfOperation.ctr(key);
-         var encryptedBytes = aesCtr.encrypt(textBytes);
-         var encryptedHex = aesJs.utils.hex.fromBytes(encryptedBytes);
-         return encryptedHex;
-       }
-       function utilAesDecrypt(encryptedHex, key) {
-         key = key || DEFAULT_128;
-         var encryptedBytes = aesJs.utils.hex.toBytes(encryptedHex);
-         var aesCtr = new aesJs.ModeOfOperation.ctr(key);
-         var decryptedBytes = aesCtr.decrypt(encryptedBytes);
-         var text = aesJs.utils.utf8.fromBytes(decryptedBytes);
-         return text;
-       }
-
-       function utilCleanTags(tags) {
-         var out = {};
-
-         for (var k in tags) {
-           if (!k) continue;
-           var v = tags[k];
 
-           if (v !== undefined) {
-             out[k] = cleanValue(k, v);
-           }
+           points = result;
+           if (!points.length) break;
          }
 
-         return out;
+         return result;
+       } // intersect a segment against one of the 4 lines that make up the bbox
 
-         function cleanValue(k, v) {
-           function keepSpaces(k) {
-             return /_hours|_times|:conditional$/.test(k);
-           }
 
-           function skip(k) {
-             return /^(description|note|fixme)$/.test(k);
-           }
+       function intersect$1(a, b, edge, bbox) {
+         return edge & 8 ? [a[0] + (b[0] - a[0]) * (bbox[3] - a[1]) / (b[1] - a[1]), bbox[3]] : // top
+         edge & 4 ? [a[0] + (b[0] - a[0]) * (bbox[1] - a[1]) / (b[1] - a[1]), bbox[1]] : // bottom
+         edge & 2 ? [bbox[2], a[1] + (b[1] - a[1]) * (bbox[2] - a[0]) / (b[0] - a[0])] : // right
+         edge & 1 ? [bbox[0], a[1] + (b[1] - a[1]) * (bbox[0] - a[0]) / (b[0] - a[0])] : // left
+         null;
+       } // bit code reflects the point position relative to the bbox:
+       //         left  mid  right
+       //    top  1001  1000  1010
+       //    mid  0001  0000  0010
+       // bottom  0101  0100  0110
 
-           if (skip(k)) return v;
-           var cleaned = v.split(';').map(function (s) {
-             return s.trim();
-           }).join(keepSpaces(k) ? '; ' : ';'); // The code below is not intended to validate websites and emails.
-           // It is only intended to prevent obvious copy-paste errors. (#2323)
-           // clean website- and email-like tags
 
-           if (k.indexOf('website') !== -1 || k.indexOf('email') !== -1 || cleaned.indexOf('http') === 0) {
-             cleaned = cleaned.replace(/[\u200B-\u200F\uFEFF]/g, ''); // strip LRM and other zero width chars
-           }
+       function bitCode$1(p, bbox) {
+         var code = 0;
+         if (p[0] < bbox[0]) code |= 1; // left
+         else if (p[0] > bbox[2]) code |= 2; // right
 
-           return cleaned;
-         }
+         if (p[1] < bbox[1]) code |= 4; // bottom
+         else if (p[1] > bbox[3]) code |= 8; // top
+
+         return code;
        }
 
-       // Like selection.property('value', ...), but avoids no-op value sets,
-       // which can result in layout/repaint thrashing in some situations.
-       function utilGetSetValue(selection, value) {
-         function d3_selection_value(value) {
-           function valueNull() {
-             delete this.value;
-           }
+       var whichPolygon_1 = whichPolygon;
 
-           function valueConstant() {
-             if (this.value !== value) {
-               this.value = value;
-             }
-           }
+       function whichPolygon(data) {
+         var bboxes = [];
 
-           function valueFunction() {
-             var x = value.apply(this, arguments);
+         for (var i = 0; i < data.features.length; i++) {
+           var feature = data.features[i];
+           var coords = feature.geometry.coordinates;
 
-             if (x === null || x === undefined) {
-               delete this.value;
-             } else if (this.value !== x) {
-               this.value = x;
+           if (feature.geometry.type === 'Polygon') {
+             bboxes.push(treeItem(coords, feature.properties));
+           } else if (feature.geometry.type === 'MultiPolygon') {
+             for (var j = 0; j < coords.length; j++) {
+               bboxes.push(treeItem(coords[j], feature.properties));
              }
            }
-
-           return value === null || value === undefined ? valueNull : typeof value === 'function' ? valueFunction : valueConstant;
-         }
-
-         if (arguments.length === 1) {
-           return selection.property('value');
          }
 
-         return selection.each(d3_selection_value(value));
-       }
-
-       function utilKeybinding(namespace) {
-         var _keybindings = {};
+         var tree = rbush_1().load(bboxes);
 
-         function testBindings(d3_event, isCapturing) {
-           var didMatch = false;
-           var bindings = Object.keys(_keybindings).map(function (id) {
-             return _keybindings[id];
+         function query(p, multi) {
+           var output = [],
+               result = tree.search({
+             minX: p[0],
+             minY: p[1],
+             maxX: p[0],
+             maxY: p[1]
            });
-           var i, binding; // Most key shortcuts will accept either lower or uppercase ('h' or 'H'),
-           // so we don't strictly match on the shift key, but we prioritize
-           // shifted keybindings first, and fallback to unshifted only if no match.
-           // (This lets us differentiate between '←'/'⇧←' or '⌘Z'/'⌘⇧Z')
-           // priority match shifted keybindings first
-
-           for (i = 0; i < bindings.length; i++) {
-             binding = bindings[i];
-             if (!binding.event.modifiers.shiftKey) continue; // no shift
-
-             if (!!binding.capture !== isCapturing) continue;
-
-             if (matches(d3_event, binding, true)) {
-               binding.callback(d3_event);
-               didMatch = true; // match a max of one binding per event
 
-               break;
+           for (var i = 0; i < result.length; i++) {
+             if (insidePolygon(result[i].coords, p)) {
+               if (multi) output.push(result[i].props);else return result[i].props;
              }
            }
 
-           if (didMatch) return; // then unshifted keybindings
+           return multi && output.length ? output : null;
+         }
 
-           for (i = 0; i < bindings.length; i++) {
-             binding = bindings[i];
-             if (binding.event.modifiers.shiftKey) continue; // shift
+         query.tree = tree;
 
-             if (!!binding.capture !== isCapturing) continue;
+         query.bbox = function queryBBox(bbox) {
+           var output = [];
+           var result = tree.search({
+             minX: bbox[0],
+             minY: bbox[1],
+             maxX: bbox[2],
+             maxY: bbox[3]
+           });
 
-             if (matches(d3_event, binding, false)) {
-               binding.callback(d3_event);
-               break;
+           for (var i = 0; i < result.length; i++) {
+             if (polygonIntersectsBBox(result[i].coords, bbox)) {
+               output.push(result[i].props);
              }
            }
 
-           function matches(d3_event, binding, testShift) {
-             var event = d3_event;
-             var isMatch = false;
-             var tryKeyCode = true; // Prefer a match on `KeyboardEvent.key`
+           return output;
+         };
 
-             if (event.key !== undefined) {
-               tryKeyCode = event.key.charCodeAt(0) > 255; // outside ISO-Latin-1
+         return query;
+       }
 
-               isMatch = true;
+       function polygonIntersectsBBox(polygon, bbox) {
+         var bboxCenter = [(bbox[0] + bbox[2]) / 2, (bbox[1] + bbox[3]) / 2];
+         if (insidePolygon(polygon, bboxCenter)) return true;
 
-               if (binding.event.key === undefined) {
-                 isMatch = false;
-               } else if (Array.isArray(binding.event.key)) {
-                 if (binding.event.key.map(function (s) {
-                   return s.toLowerCase();
-                 }).indexOf(event.key.toLowerCase()) === -1) isMatch = false;
-               } else {
-                 if (event.key.toLowerCase() !== binding.event.key.toLowerCase()) isMatch = false;
-               }
-             } // Fallback match on `KeyboardEvent.keyCode`, can happen if:
-             // - browser doesn't support `KeyboardEvent.key`
-             // - `KeyboardEvent.key` is outside ISO-Latin-1 range (cyrillic?)
+         for (var i = 0; i < polygon.length; i++) {
+           if (lineclip_1(polygon[i], bbox).length > 0) return true;
+         }
 
+         return false;
+       } // ray casting algorithm for detecting if point is in polygon
 
-             if (!isMatch && tryKeyCode) {
-               isMatch = event.keyCode === binding.event.keyCode;
-             }
 
-             if (!isMatch) return false; // test modifier keys
+       function insidePolygon(rings, p) {
+         var inside = false;
 
-             if (!(event.ctrlKey && event.altKey)) {
-               // if both are set, assume AltGr and skip it - #4096
-               if (event.ctrlKey !== binding.event.modifiers.ctrlKey) return false;
-               if (event.altKey !== binding.event.modifiers.altKey) return false;
-             }
+         for (var i = 0, len = rings.length; i < len; i++) {
+           var ring = rings[i];
 
-             if (event.metaKey !== binding.event.modifiers.metaKey) return false;
-             if (testShift && event.shiftKey !== binding.event.modifiers.shiftKey) return false;
-             return true;
+           for (var j = 0, len2 = ring.length, k = len2 - 1; j < len2; k = j++) {
+             if (rayIntersect(p, ring[j], ring[k])) inside = !inside;
            }
          }
 
-         function capture(d3_event) {
-           testBindings(d3_event, true);
-         }
+         return inside;
+       }
 
-         function bubble(d3_event) {
-           var tagName = select(d3_event.target).node().tagName;
+       function rayIntersect(p, p1, p2) {
+         return p1[1] > p[1] !== p2[1] > p[1] && p[0] < (p2[0] - p1[0]) * (p[1] - p1[1]) / (p2[1] - p1[1]) + p1[0];
+       }
 
-           if (tagName === 'INPUT' || tagName === 'SELECT' || tagName === 'TEXTAREA') {
-             return;
-           }
+       function treeItem(coords, props) {
+         var item = {
+           minX: Infinity,
+           minY: Infinity,
+           maxX: -Infinity,
+           maxY: -Infinity,
+           coords: coords,
+           props: props
+         };
 
-           testBindings(d3_event, false);
+         for (var i = 0; i < coords[0].length; i++) {
+           var p = coords[0][i];
+           item.minX = Math.min(item.minX, p[0]);
+           item.minY = Math.min(item.minY, p[1]);
+           item.maxX = Math.max(item.maxX, p[0]);
+           item.maxY = Math.max(item.maxY, p[1]);
          }
 
-         function keybinding(selection) {
-           selection = selection || select(document);
-           selection.on('keydown.capture.' + namespace, capture, true);
-           selection.on('keydown.bubble.' + namespace, bubble, false);
-           return keybinding;
-         } // was: keybinding.off()
-
-
-         keybinding.unbind = function (selection) {
-           _keybindings = [];
-           selection = selection || select(document);
-           selection.on('keydown.capture.' + namespace, null);
-           selection.on('keydown.bubble.' + namespace, null);
-           return keybinding;
-         };
-
-         keybinding.clear = function () {
-           _keybindings = {};
-           return keybinding;
-         }; // Remove one or more keycode bindings.
-
-
-         keybinding.off = function (codes, capture) {
-           var arr = utilArrayUniq([].concat(codes));
-
-           for (var i = 0; i < arr.length; i++) {
-             var id = arr[i] + (capture ? '-capture' : '-bubble');
-             delete _keybindings[id];
-           }
-
-           return keybinding;
-         }; // Add one or more keycode bindings.
-
-
-         keybinding.on = function (codes, callback, capture) {
-           if (typeof callback !== 'function') {
-             return keybinding.off(codes, capture);
-           }
-
-           var arr = utilArrayUniq([].concat(codes));
-
-           for (var i = 0; i < arr.length; i++) {
-             var id = arr[i] + (capture ? '-capture' : '-bubble');
-             var binding = {
-               id: id,
-               capture: capture,
-               callback: callback,
-               event: {
-                 key: undefined,
-                 // preferred
-                 keyCode: 0,
-                 // fallback
-                 modifiers: {
-                   shiftKey: false,
-                   ctrlKey: false,
-                   altKey: false,
-                   metaKey: false
-                 }
-               }
-             };
-
-             if (_keybindings[id]) {
-               console.warn('warning: duplicate keybinding for "' + id + '"'); // eslint-disable-line no-console
-             }
-
-             _keybindings[id] = binding;
-             var matches = arr[i].toLowerCase().match(/(?:(?:[^+⇧⌃⌥⌘])+|[⇧⌃⌥⌘]|\+\+|^\+$)/g);
-
-             for (var j = 0; j < matches.length; j++) {
-               // Normalise matching errors
-               if (matches[j] === '++') matches[j] = '+';
-
-               if (matches[j] in utilKeybinding.modifierCodes) {
-                 var prop = utilKeybinding.modifierProperties[utilKeybinding.modifierCodes[matches[j]]];
-                 binding.event.modifiers[prop] = true;
-               } else {
-                 binding.event.key = utilKeybinding.keys[matches[j]] || matches[j];
-
-                 if (matches[j] in utilKeybinding.keyCodes) {
-                   binding.event.keyCode = utilKeybinding.keyCodes[matches[j]];
-                 }
-               }
-             }
-           }
-
-           return keybinding;
-         };
-
-         return keybinding;
-       }
-       /*
-        * See https://github.com/keithamus/jwerty
-        */
-
-       utilKeybinding.modifierCodes = {
-         // Shift key, ⇧
-         '⇧': 16,
-         shift: 16,
-         // CTRL key, on Mac: ⌃
-         '⌃': 17,
-         ctrl: 17,
-         // ALT key, on Mac: ⌥ (Alt)
-         '⌥': 18,
-         alt: 18,
-         option: 18,
-         // META, on Mac: ⌘ (CMD), on Windows (Win), on Linux (Super)
-         '⌘': 91,
-         meta: 91,
-         cmd: 91,
-         'super': 91,
-         win: 91
-       };
-       utilKeybinding.modifierProperties = {
-         16: 'shiftKey',
-         17: 'ctrlKey',
-         18: 'altKey',
-         91: 'metaKey'
-       };
-       utilKeybinding.plusKeys = ['plus', 'ffplus', '=', 'ffequals', '≠', '±'];
-       utilKeybinding.minusKeys = ['_', '-', 'ffminus', 'dash', '–', '—'];
-       utilKeybinding.keys = {
-         // Backspace key, on Mac: ⌫ (Backspace)
-         '⌫': 'Backspace',
-         backspace: 'Backspace',
-         // Tab Key, on Mac: ⇥ (Tab), on Windows ⇥⇥
-         '⇥': 'Tab',
-         '⇆': 'Tab',
-         tab: 'Tab',
-         // Return key, ↩
-         '↩': 'Enter',
-         '↵': 'Enter',
-         '⏎': 'Enter',
-         'return': 'Enter',
-         enter: 'Enter',
-         '⌅': 'Enter',
-         // Pause/Break key
-         'pause': 'Pause',
-         'pause-break': 'Pause',
-         // Caps Lock key, ⇪
-         '⇪': 'CapsLock',
-         caps: 'CapsLock',
-         'caps-lock': 'CapsLock',
-         // Escape key, on Mac: ⎋, on Windows: Esc
-         '⎋': ['Escape', 'Esc'],
-         escape: ['Escape', 'Esc'],
-         esc: ['Escape', 'Esc'],
-         // Space key
-         space: [' ', 'Spacebar'],
-         // Page-Up key, or pgup, on Mac: ↖
-         '↖': 'PageUp',
-         pgup: 'PageUp',
-         'page-up': 'PageUp',
-         // Page-Down key, or pgdown, on Mac: ↘
-         '↘': 'PageDown',
-         pgdown: 'PageDown',
-         'page-down': 'PageDown',
-         // END key, on Mac: ⇟
-         '⇟': 'End',
-         end: 'End',
-         // HOME key, on Mac: ⇞
-         '⇞': 'Home',
-         home: 'Home',
-         // Insert key, or ins
-         ins: 'Insert',
-         insert: 'Insert',
-         // Delete key, on Mac: ⌦ (Delete)
-         '⌦': ['Delete', 'Del'],
-         del: ['Delete', 'Del'],
-         'delete': ['Delete', 'Del'],
-         // Left Arrow Key, or ←
-         '←': ['ArrowLeft', 'Left'],
-         left: ['ArrowLeft', 'Left'],
-         'arrow-left': ['ArrowLeft', 'Left'],
-         // Up Arrow Key, or ↑
-         '↑': ['ArrowUp', 'Up'],
-         up: ['ArrowUp', 'Up'],
-         'arrow-up': ['ArrowUp', 'Up'],
-         // Right Arrow Key, or →
-         '→': ['ArrowRight', 'Right'],
-         right: ['ArrowRight', 'Right'],
-         'arrow-right': ['ArrowRight', 'Right'],
-         // Up Arrow Key, or ↓
-         '↓': ['ArrowDown', 'Down'],
-         down: ['ArrowDown', 'Down'],
-         'arrow-down': ['ArrowDown', 'Down'],
-         // odities, stuff for backward compatibility (browsers and code):
-         // Num-Multiply, or *
-         '*': ['*', 'Multiply'],
-         star: ['*', 'Multiply'],
-         asterisk: ['*', 'Multiply'],
-         multiply: ['*', 'Multiply'],
-         // Num-Plus or +
-         '+': ['+', 'Add'],
-         'plus': ['+', 'Add'],
-         // Num-Subtract, or -
-         '-': ['-', 'Subtract'],
-         subtract: ['-', 'Subtract'],
-         'dash': ['-', 'Subtract'],
-         // Semicolon
-         semicolon: ';',
-         // = or equals
-         equals: '=',
-         // Comma, or ,
-         comma: ',',
-         // Period, or ., or full-stop
-         period: '.',
-         'full-stop': '.',
-         // Slash, or /, or forward-slash
-         slash: '/',
-         'forward-slash': '/',
-         // Tick, or `, or back-quote
-         tick: '`',
-         'back-quote': '`',
-         // Open bracket, or [
-         'open-bracket': '[',
-         // Back slash, or \
-         'back-slash': '\\',
-         // Close backet, or ]
-         'close-bracket': ']',
-         // Apostrophe, or Quote, or '
-         quote: '\'',
-         apostrophe: '\'',
-         // NUMPAD 0-9
-         'num-0': '0',
-         'num-1': '1',
-         'num-2': '2',
-         'num-3': '3',
-         'num-4': '4',
-         'num-5': '5',
-         'num-6': '6',
-         'num-7': '7',
-         'num-8': '8',
-         'num-9': '9',
-         // F1-F25
-         f1: 'F1',
-         f2: 'F2',
-         f3: 'F3',
-         f4: 'F4',
-         f5: 'F5',
-         f6: 'F6',
-         f7: 'F7',
-         f8: 'F8',
-         f9: 'F9',
-         f10: 'F10',
-         f11: 'F11',
-         f12: 'F12',
-         f13: 'F13',
-         f14: 'F14',
-         f15: 'F15',
-         f16: 'F16',
-         f17: 'F17',
-         f18: 'F18',
-         f19: 'F19',
-         f20: 'F20',
-         f21: 'F21',
-         f22: 'F22',
-         f23: 'F23',
-         f24: 'F24',
-         f25: 'F25'
-       };
-       utilKeybinding.keyCodes = {
-         // Backspace key, on Mac: ⌫ (Backspace)
-         '⌫': 8,
-         backspace: 8,
-         // Tab Key, on Mac: ⇥ (Tab), on Windows ⇥⇥
-         '⇥': 9,
-         '⇆': 9,
-         tab: 9,
-         // Return key, ↩
-         '↩': 13,
-         '↵': 13,
-         '⏎': 13,
-         'return': 13,
-         enter: 13,
-         '⌅': 13,
-         // Pause/Break key
-         'pause': 19,
-         'pause-break': 19,
-         // Caps Lock key, ⇪
-         '⇪': 20,
-         caps: 20,
-         'caps-lock': 20,
-         // Escape key, on Mac: ⎋, on Windows: Esc
-         '⎋': 27,
-         escape: 27,
-         esc: 27,
-         // Space key
-         space: 32,
-         // Page-Up key, or pgup, on Mac: ↖
-         '↖': 33,
-         pgup: 33,
-         'page-up': 33,
-         // Page-Down key, or pgdown, on Mac: ↘
-         '↘': 34,
-         pgdown: 34,
-         'page-down': 34,
-         // END key, on Mac: ⇟
-         '⇟': 35,
-         end: 35,
-         // HOME key, on Mac: ⇞
-         '⇞': 36,
-         home: 36,
-         // Insert key, or ins
-         ins: 45,
-         insert: 45,
-         // Delete key, on Mac: ⌦ (Delete)
-         '⌦': 46,
-         del: 46,
-         'delete': 46,
-         // Left Arrow Key, or ←
-         '←': 37,
-         left: 37,
-         'arrow-left': 37,
-         // Up Arrow Key, or ↑
-         '↑': 38,
-         up: 38,
-         'arrow-up': 38,
-         // Right Arrow Key, or →
-         '→': 39,
-         right: 39,
-         'arrow-right': 39,
-         // Up Arrow Key, or ↓
-         '↓': 40,
-         down: 40,
-         'arrow-down': 40,
-         // odities, printing characters that come out wrong:
-         // Firefox Equals
-         'ffequals': 61,
-         // Num-Multiply, or *
-         '*': 106,
-         star: 106,
-         asterisk: 106,
-         multiply: 106,
-         // Num-Plus or +
-         '+': 107,
-         'plus': 107,
-         // Num-Subtract, or -
-         '-': 109,
-         subtract: 109,
-         // Firefox Plus
-         'ffplus': 171,
-         // Firefox Minus
-         'ffminus': 173,
-         // Semicolon
-         ';': 186,
-         semicolon: 186,
-         // = or equals
-         '=': 187,
-         'equals': 187,
-         // Comma, or ,
-         ',': 188,
-         comma: 188,
-         // Dash / Underscore key
-         'dash': 189,
-         // Period, or ., or full-stop
-         '.': 190,
-         period: 190,
-         'full-stop': 190,
-         // Slash, or /, or forward-slash
-         '/': 191,
-         slash: 191,
-         'forward-slash': 191,
-         // Tick, or `, or back-quote
-         '`': 192,
-         tick: 192,
-         'back-quote': 192,
-         // Open bracket, or [
-         '[': 219,
-         'open-bracket': 219,
-         // Back slash, or \
-         '\\': 220,
-         'back-slash': 220,
-         // Close backet, or ]
-         ']': 221,
-         'close-bracket': 221,
-         // Apostrophe, or Quote, or '
-         '\'': 222,
-         quote: 222,
-         apostrophe: 222
-       }; // NUMPAD 0-9
-
-       var i$1 = 95,
-           n = 0;
-
-       while (++i$1 < 106) {
-         utilKeybinding.keyCodes['num-' + n] = i$1;
-         ++n;
-       } // 0-9
-
-
-       i$1 = 47;
-       n = 0;
-
-       while (++i$1 < 58) {
-         utilKeybinding.keyCodes[n] = i$1;
-         ++n;
-       } // F1-F25
-
-
-       i$1 = 111;
-       n = 1;
-
-       while (++i$1 < 136) {
-         utilKeybinding.keyCodes['f' + n] = i$1;
-         ++n;
-       } // a-z
-
-
-       i$1 = 64;
-
-       while (++i$1 < 91) {
-         utilKeybinding.keyCodes[String.fromCharCode(i$1).toLowerCase()] = i$1;
-       }
-
-       function utilObjectOmit(obj, omitKeys) {
-         return Object.keys(obj).reduce(function (result, key) {
-           if (omitKeys.indexOf(key) === -1) {
-             result[key] = obj[key]; // keep
-           }
-
-           return result;
-         }, {});
+         return item;
        }
 
-       // Copies a variable number of methods from source to target.
-       function utilRebind(target, source) {
-         var i = 1,
-             n = arguments.length,
-             method;
-
-         while (++i < n) {
-           target[method = arguments[i]] = d3_rebind(target, source, source[method]);
+       var type = "FeatureCollection";
+       var features = [{
+         type: "Feature",
+         properties: {
+           wikidata: "Q21",
+           nameEn: "England",
+           aliases: ["GB-ENG"],
+           country: "GB",
+           groups: ["Q23666", "Q3336843", "154", "150", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["44"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-6.03913, 51.13217], [-7.74976, 48.64773], [1.17405, 50.74239], [2.18458, 51.52087], [2.56575, 51.85301], [0.792, 57.56437], [-2.30613, 55.62698], [-2.17058, 55.45916], [-2.6095, 55.28488], [-2.63532, 55.19452], [-3.02906, 55.04606], [-3.09361, 54.94924], [-3.38407, 54.94278], [-4.1819, 54.57861], [-3.5082, 53.54318], [-3.08228, 53.25526], [-3.03675, 53.25092], [-2.92329, 53.19383], [-2.92022, 53.17685], [-2.98598, 53.15589], [-2.90649, 53.10964], [-2.87469, 53.12337], [-2.89131, 53.09374], [-2.83133, 52.99184], [-2.7251, 52.98389], [-2.72221, 52.92969], [-2.80549, 52.89428], [-2.85897, 52.94487], [-2.92401, 52.93836], [-2.97243, 52.9651], [-3.13576, 52.895], [-3.15744, 52.84947], [-3.16105, 52.79599], [-3.08734, 52.77504], [-3.01001, 52.76636], [-2.95581, 52.71794], [-3.01724, 52.72083], [-3.04398, 52.65435], [-3.13648, 52.58208], [-3.12926, 52.5286], [-3.09746, 52.53077], [-3.08662, 52.54811], [-3.00929, 52.57774], [-2.99701, 52.551], [-3.03603, 52.49969], [-3.13359, 52.49174], [-3.22971, 52.45344], [-3.22754, 52.42526], [-3.04687, 52.34504], [-2.95364, 52.3501], [-2.99701, 52.323], [-3.00785, 52.2753], [-3.09289, 52.20546], [-3.12638, 52.08114], [-2.97111, 51.90456], [-2.8818, 51.93196], [-2.78742, 51.88833], [-2.74277, 51.84367], [-2.66234, 51.83555], [-2.66336, 51.59504], [-3.20563, 51.31615], [-6.03913, 51.13217]]]]
          }
-
-         return target;
-       } // Method is assumed to be a standard D3 getter-setter:
-       // If passed with no arguments, gets the value.
-       // If passed with arguments, sets the value and returns the target.
-
-       function d3_rebind(target, source, method) {
-         return function () {
-           var value = method.apply(source, arguments);
-           return value === source ? target : value;
-         };
-       }
-
-       // A per-domain session mutex backed by a cookie and dead man's
-       // switch. If the session crashes, the mutex will auto-release
-       // after 5 seconds.
-       // This accepts a string and returns an object that complies with utilSessionMutexType
-       function utilSessionMutex(name) {
-         var mutex = {};
-         var intervalID;
-
-         function renew() {
-           var expires = new Date();
-           expires.setSeconds(expires.getSeconds() + 5);
-           document.cookie = name + '=1; expires=' + expires.toUTCString() + '; sameSite=strict';
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q22",
+           nameEn: "Scotland",
+           aliases: ["GB-SCT"],
+           country: "GB",
+           groups: ["Q23666", "Q3336843", "154", "150", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["44"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[0.792, 57.56437], [-0.3751, 61.32236], [-14.78497, 57.60709], [-6.82333, 55.83103], [-4.69044, 54.3629], [-3.38407, 54.94278], [-3.09361, 54.94924], [-3.02906, 55.04606], [-2.63532, 55.19452], [-2.6095, 55.28488], [-2.17058, 55.45916], [-2.30613, 55.62698], [0.792, 57.56437]]]]
          }
-
-         mutex.lock = function () {
-           if (intervalID) return true;
-           var cookie = document.cookie.replace(new RegExp('(?:(?:^|.*;)\\s*' + name + '\\s*\\=\\s*([^;]*).*$)|^.*$'), '$1');
-           if (cookie) return false;
-           renew();
-           intervalID = window.setInterval(renew, 4000);
-           return true;
-         };
-
-         mutex.unlock = function () {
-           if (!intervalID) return;
-           document.cookie = name + '=; expires=Thu, 01 Jan 1970 00:00:00 GMT; sameSite=strict';
-           clearInterval(intervalID);
-           intervalID = null;
-         };
-
-         mutex.locked = function () {
-           return !!intervalID;
-         };
-
-         return mutex;
-       }
-
-       function utilTiler() {
-         var _size = [256, 256];
-         var _scale = 256;
-         var _tileSize = 256;
-         var _zoomExtent = [0, 20];
-         var _translate = [_size[0] / 2, _size[1] / 2];
-         var _margin = 0;
-         var _skipNullIsland = false;
-
-         function clamp(num, min, max) {
-           return Math.max(min, Math.min(num, max));
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q25",
+           nameEn: "Wales",
+           aliases: ["GB-WLS"],
+           country: "GB",
+           groups: ["Q23666", "Q3336843", "154", "150", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["44"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-3.5082, 53.54318], [-5.37267, 53.63269], [-6.03913, 51.13217], [-3.20563, 51.31615], [-2.66336, 51.59504], [-2.66234, 51.83555], [-2.74277, 51.84367], [-2.78742, 51.88833], [-2.8818, 51.93196], [-2.97111, 51.90456], [-3.12638, 52.08114], [-3.09289, 52.20546], [-3.00785, 52.2753], [-2.99701, 52.323], [-2.95364, 52.3501], [-3.04687, 52.34504], [-3.22754, 52.42526], [-3.22971, 52.45344], [-3.13359, 52.49174], [-3.03603, 52.49969], [-2.99701, 52.551], [-3.00929, 52.57774], [-3.08662, 52.54811], [-3.09746, 52.53077], [-3.12926, 52.5286], [-3.13648, 52.58208], [-3.04398, 52.65435], [-3.01724, 52.72083], [-2.95581, 52.71794], [-3.01001, 52.76636], [-3.08734, 52.77504], [-3.16105, 52.79599], [-3.15744, 52.84947], [-3.13576, 52.895], [-2.97243, 52.9651], [-2.92401, 52.93836], [-2.85897, 52.94487], [-2.80549, 52.89428], [-2.72221, 52.92969], [-2.7251, 52.98389], [-2.83133, 52.99184], [-2.89131, 53.09374], [-2.87469, 53.12337], [-2.90649, 53.10964], [-2.98598, 53.15589], [-2.92022, 53.17685], [-2.92329, 53.19383], [-3.03675, 53.25092], [-3.08228, 53.25526], [-3.5082, 53.54318]]]]
          }
-
-         function nearNullIsland(tile) {
-           var x = tile[0];
-           var y = tile[1];
-           var z = tile[2];
-
-           if (z >= 7) {
-             var center = Math.pow(2, z - 1);
-             var width = Math.pow(2, z - 6);
-             var min = center - width / 2;
-             var max = center + width / 2 - 1;
-             return x >= min && x <= max && y >= min && y <= max;
-           }
-
-           return false;
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q26",
+           nameEn: "Northern Ireland",
+           aliases: ["GB-NIR"],
+           country: "GB",
+           groups: ["Q22890", "Q3336843", "154", "150", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["44"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-6.34755, 55.49206], [-7.2471, 55.06933], [-7.34464, 55.04688], [-7.4033, 55.00391], [-7.40004, 54.94498], [-7.44404, 54.9403], [-7.4473, 54.87003], [-7.47626, 54.83084], [-7.54508, 54.79401], [-7.54671, 54.74606], [-7.64449, 54.75265], [-7.75041, 54.7103], [-7.83352, 54.73854], [-7.93293, 54.66603], [-7.70315, 54.62077], [-7.8596, 54.53671], [-7.99812, 54.54427], [-8.04538, 54.48941], [-8.179, 54.46763], [-8.04555, 54.36292], [-7.87101, 54.29299], [-7.8596, 54.21779], [-7.81397, 54.20159], [-7.69501, 54.20731], [-7.55812, 54.12239], [-7.4799, 54.12239], [-7.44567, 54.1539], [-7.32834, 54.11475], [-7.30553, 54.11869], [-7.34005, 54.14698], [-7.29157, 54.17191], [-7.28017, 54.16714], [-7.29687, 54.1354], [-7.29493, 54.12013], [-7.26316, 54.13863], [-7.25012, 54.20063], [-7.14908, 54.22732], [-7.19145, 54.31296], [-7.02034, 54.4212], [-6.87775, 54.34682], [-6.85179, 54.29176], [-6.81583, 54.22791], [-6.74575, 54.18788], [-6.70175, 54.20218], [-6.6382, 54.17071], [-6.66264, 54.0666], [-6.62842, 54.03503], [-6.47849, 54.06947], [-6.36605, 54.07234], [-6.36279, 54.11248], [-6.32694, 54.09337], [-6.29003, 54.11278], [-6.26218, 54.09785], [-5.83481, 53.87749], [-4.69044, 54.3629], [-6.34755, 55.49206]]]]
          }
-
-         function tiler() {
-           var z = geoScaleToZoom(_scale / (2 * Math.PI), _tileSize);
-           var z0 = clamp(Math.round(z), _zoomExtent[0], _zoomExtent[1]);
-           var tileMin = 0;
-           var tileMax = Math.pow(2, z0) - 1;
-           var log2ts = Math.log(_tileSize) * Math.LOG2E;
-           var k = Math.pow(2, z - z0 + log2ts);
-           var origin = [(_translate[0] - _scale / 2) / k, (_translate[1] - _scale / 2) / k];
-           var cols = range(clamp(Math.floor(-origin[0]) - _margin, tileMin, tileMax + 1), clamp(Math.ceil(_size[0] / k - origin[0]) + _margin, tileMin, tileMax + 1));
-           var rows = range(clamp(Math.floor(-origin[1]) - _margin, tileMin, tileMax + 1), clamp(Math.ceil(_size[1] / k - origin[1]) + _margin, tileMin, tileMax + 1));
-           var tiles = [];
-
-           for (var i = 0; i < rows.length; i++) {
-             var y = rows[i];
-
-             for (var j = 0; j < cols.length; j++) {
-               var x = cols[j];
-
-               if (i >= _margin && i <= rows.length - _margin && j >= _margin && j <= cols.length - _margin) {
-                 tiles.unshift([x, y, z0]); // tiles in view at beginning
-               } else {
-                 tiles.push([x, y, z0]); // tiles in margin at the end
-               }
-             }
-           }
-
-           tiles.translate = origin;
-           tiles.scale = k;
-           return tiles;
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q35",
+           nameEn: "Denmark",
+           country: "DK",
+           groups: ["EU", "154", "150", "UN"],
+           callingCodes: ["45"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[12.16597, 56.60205], [10.40861, 58.38489], [7.28637, 57.35913], [8.02459, 55.09613], [8.45719, 55.06747], [8.55769, 54.91837], [8.63979, 54.91069], [8.76387, 54.8948], [8.81178, 54.90518], [8.92795, 54.90452], [9.04629, 54.87249], [9.14275, 54.87421], [9.20571, 54.85841], [9.24631, 54.84726], [9.23445, 54.83432], [9.2474, 54.8112], [9.32771, 54.80602], [9.33849, 54.80233], [9.36496, 54.81749], [9.38532, 54.83968], [9.41213, 54.84254], [9.43155, 54.82586], [9.4659, 54.83131], [9.58937, 54.88785], [9.62734, 54.88057], [9.61187, 54.85548], [9.73563, 54.8247], [9.89314, 54.84171], [10.16755, 54.73883], [10.31111, 54.65968], [11.00303, 54.63689], [11.90309, 54.38543], [12.85844, 54.82438], [13.93395, 54.84044], [15.36991, 54.73263], [15.79951, 55.54655], [14.89259, 55.5623], [14.28399, 55.1553], [12.84405, 55.13257], [12.60345, 55.42675], [12.88472, 55.63369], [12.6372, 55.91371], [12.65312, 56.04345], [12.07466, 56.29488], [12.16597, 56.60205]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q55",
+           nameEn: "Netherlands",
+           country: "NL",
+           groups: ["EU", "155", "150", "UN"],
+           callingCodes: ["31"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[5.45168, 54.20039], [2.56575, 51.85301], [3.36263, 51.37112], [3.38696, 51.33436], [3.35847, 51.31572], [3.38289, 51.27331], [3.41704, 51.25933], [3.43488, 51.24135], [3.52698, 51.2458], [3.51502, 51.28697], [3.58939, 51.30064], [3.78999, 51.25766], [3.78783, 51.2151], [3.90125, 51.20371], [3.97889, 51.22537], [4.01957, 51.24504], [4.05165, 51.24171], [4.16721, 51.29348], [4.24024, 51.35371], [4.21923, 51.37443], [4.33265, 51.37687], [4.34086, 51.35738], [4.39292, 51.35547], [4.43777, 51.36989], [4.38064, 51.41965], [4.39747, 51.43316], [4.38122, 51.44905], [4.47736, 51.4778], [4.5388, 51.48184], [4.54675, 51.47265], [4.52846, 51.45002], [4.53521, 51.4243], [4.57489, 51.4324], [4.65442, 51.42352], [4.72935, 51.48424], [4.74578, 51.48937], [4.77321, 51.50529], [4.78803, 51.50284], [4.84139, 51.4799], [4.82409, 51.44736], [4.82946, 51.4213], [4.78314, 51.43319], [4.76577, 51.43046], [4.77229, 51.41337], [4.78941, 51.41102], [4.84988, 51.41502], [4.90016, 51.41404], [4.92152, 51.39487], [5.00393, 51.44406], [5.0106, 51.47167], [5.03281, 51.48679], [5.04774, 51.47022], [5.07891, 51.4715], [5.10456, 51.43163], [5.07102, 51.39469], [5.13105, 51.34791], [5.13377, 51.31592], [5.16222, 51.31035], [5.2002, 51.32243], [5.24244, 51.30495], [5.22542, 51.26888], [5.23814, 51.26064], [5.26461, 51.26693], [5.29716, 51.26104], [5.33886, 51.26314], [5.347, 51.27502], [5.41672, 51.26248], [5.4407, 51.28169], [5.46519, 51.2849], [5.48476, 51.30053], [5.515, 51.29462], [5.5569, 51.26544], [5.5603, 51.22249], [5.65145, 51.19788], [5.65528, 51.18736], [5.70344, 51.1829], [5.74617, 51.18928], [5.77735, 51.17845], [5.77697, 51.1522], [5.82564, 51.16753], [5.85508, 51.14445], [5.80798, 51.11661], [5.8109, 51.10861], [5.83226, 51.10585], [5.82921, 51.09328], [5.79903, 51.09371], [5.79835, 51.05834], [5.77258, 51.06196], [5.75961, 51.03113], [5.77688, 51.02483], [5.76242, 50.99703], [5.71864, 50.96092], [5.72875, 50.95428], [5.74752, 50.96202], [5.75927, 50.95601], [5.74644, 50.94723], [5.72545, 50.92312], [5.72644, 50.91167], [5.71626, 50.90796], [5.69858, 50.91046], [5.67886, 50.88142], [5.64504, 50.87107], [5.64009, 50.84742], [5.65259, 50.82309], [5.70118, 50.80764], [5.68995, 50.79641], [5.70107, 50.7827], [5.68091, 50.75804], [5.69469, 50.75529], [5.72216, 50.76398], [5.73904, 50.75674], [5.74356, 50.7691], [5.76533, 50.78159], [5.77513, 50.78308], [5.80673, 50.7558], [5.84548, 50.76542], [5.84888, 50.75448], [5.88734, 50.77092], [5.89129, 50.75125], [5.89132, 50.75124], [5.95942, 50.7622], [5.97545, 50.75441], [6.01976, 50.75398], [6.02624, 50.77453], [5.97497, 50.79992], [5.98404, 50.80988], [6.00462, 50.80065], [6.02328, 50.81694], [6.01921, 50.84435], [6.05623, 50.8572], [6.05702, 50.85179], [6.07431, 50.84674], [6.07693, 50.86025], [6.08805, 50.87223], [6.07486, 50.89307], [6.09297, 50.92066], [6.01615, 50.93367], [6.02697, 50.98303], [5.95282, 50.98728], [5.90296, 50.97356], [5.90493, 51.00198], [5.87849, 51.01969], [5.86735, 51.05182], [5.9134, 51.06736], [5.9541, 51.03496], [5.98292, 51.07469], [6.16706, 51.15677], [6.17384, 51.19589], [6.07889, 51.17038], [6.07889, 51.24432], [6.16977, 51.33169], [6.22674, 51.36135], [6.22641, 51.39948], [6.20654, 51.40049], [6.21724, 51.48568], [6.18017, 51.54096], [6.09055, 51.60564], [6.11759, 51.65609], [6.02767, 51.6742], [6.04091, 51.71821], [5.95003, 51.7493], [5.98665, 51.76944], [5.94568, 51.82786], [5.99848, 51.83195], [6.06705, 51.86136], [6.10337, 51.84829], [6.16902, 51.84094], [6.11551, 51.89769], [6.15349, 51.90439], [6.21443, 51.86801], [6.29872, 51.86801], [6.30593, 51.84998], [6.40704, 51.82771], [6.38815, 51.87257], [6.47179, 51.85395], [6.50231, 51.86313], [6.58556, 51.89386], [6.68386, 51.91861], [6.72319, 51.89518], [6.82357, 51.96711], [6.83035, 51.9905], [6.68128, 52.05052], [6.76117, 52.11895], [6.83984, 52.11728], [6.97189, 52.20329], [6.9897, 52.2271], [7.03729, 52.22695], [7.06365, 52.23789], [7.02703, 52.27941], [7.07044, 52.37805], [7.03417, 52.40237], [6.99041, 52.47235], [6.94293, 52.43597], [6.69507, 52.488], [6.71641, 52.62905], [6.77307, 52.65375], [7.04557, 52.63318], [7.07253, 52.81083], [7.21694, 53.00742], [7.17898, 53.13817], [7.22681, 53.18165], [7.21679, 53.20058], [7.19052, 53.31866], [7.00198, 53.32672], [6.91025, 53.44221], [5.45168, 54.20039]], [[4.93295, 51.44945], [4.95244, 51.45207], [4.9524, 51.45014], [4.93909, 51.44632], [4.93295, 51.44945]], [[4.91493, 51.4353], [4.91935, 51.43634], [4.92227, 51.44252], [4.91811, 51.44621], [4.92287, 51.44741], [4.92811, 51.4437], [4.92566, 51.44273], [4.92815, 51.43856], [4.92879, 51.44161], [4.93544, 51.44634], [4.94025, 51.44193], [4.93416, 51.44185], [4.93471, 51.43861], [4.94265, 51.44003], [4.93986, 51.43064], [4.92952, 51.42984], [4.92652, 51.43329], [4.91493, 51.4353]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q782",
+           nameEn: "Hawaii",
+           aliases: ["US-HI"],
+           country: "US",
+           groups: ["Q35657", "061", "009", "UN"],
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["1"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-177.8563, 29.18961], [-179.49839, 27.86265], [-151.6784, 9.55515], [-154.05867, 45.51124], [-177.5224, 27.7635], [-177.8563, 29.18961]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q797",
+           nameEn: "Alaska",
+           aliases: ["US-AK"],
+           country: "US",
+           groups: ["Q35657", "021", "003", "019", "UN"],
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["1"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[169.34848, 52.47228], [180, 51.0171], [179.84401, 55.10087], [169.34848, 52.47228]]], [[[-168.95635, 65.98512], [-169.03888, 65.48473], [-172.76104, 63.77445], [-179.55295, 57.62081], [-179.55295, 50.81807], [-133.92876, 54.62289], [-130.61931, 54.70835], [-130.64499, 54.76912], [-130.44184, 54.85377], [-130.27203, 54.97174], [-130.18765, 55.07744], [-130.08035, 55.21556], [-129.97513, 55.28029], [-130.15373, 55.74895], [-130.00857, 55.91344], [-130.00093, 56.00325], [-130.10173, 56.12178], [-130.33965, 56.10849], [-130.77769, 56.36185], [-131.8271, 56.62247], [-133.38523, 58.42773], [-133.84645, 58.73543], [-134.27175, 58.8634], [-134.48059, 59.13231], [-134.55699, 59.1297], [-134.7047, 59.2458], [-135.00267, 59.28745], [-135.03069, 59.56208], [-135.48007, 59.79937], [-136.31566, 59.59083], [-136.22381, 59.55526], [-136.33727, 59.44466], [-136.47323, 59.46617], [-136.52365, 59.16752], [-136.82619, 59.16198], [-137.4925, 58.89415], [-137.60623, 59.24465], [-138.62145, 59.76431], [-138.71149, 59.90728], [-139.05365, 59.99655], [-139.20603, 60.08896], [-139.05831, 60.35205], [-139.68991, 60.33693], [-139.98024, 60.18027], [-140.45648, 60.30919], [-140.5227, 60.22077], [-141.00116, 60.30648], [-140.97446, 84.39275], [-168.25765, 71.99091], [-168.95635, 65.98512]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q3492",
+           nameEn: "Sumatra",
+           aliases: ["ID-SM"],
+           country: "ID",
+           groups: ["035", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["62"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[109.82788, 2.86812], [110.90339, 7.52694], [105.01437, 3.24936], [104.56723, 1.44271], [104.34728, 1.33529], [104.12282, 1.27714], [104.03085, 1.26954], [103.74084, 1.12902], [103.66049, 1.18825], [103.56591, 1.19719], [103.03657, 1.30383], [96.11174, 6.69841], [74.28481, -3.17525], [102.92489, -8.17146], [106.32259, -5.50116], [106.38511, -5.16715], [109.17017, -4.07401], [109.3962, -2.07276], [108.50935, -2.01066], [107.94791, 1.06924], [109.82788, 2.86812]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q3757",
+           nameEn: "Java",
+           aliases: ["ID-JW"],
+           country: "ID",
+           groups: ["035", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["62"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[109.17017, -4.07401], [106.38511, -5.16715], [106.32259, -5.50116], [102.92489, -8.17146], [116.22542, -10.49172], [114.39575, -8.2889], [114.42235, -8.09762], [114.92859, -7.49253], [116.33992, -7.56171], [116.58433, -5.30385], [109.17017, -4.07401]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q3795",
+           nameEn: "Kalimantan",
+           aliases: ["ID-KA"],
+           country: "ID",
+           groups: ["Q36117", "035", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["62"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[120.02464, 2.83703], [118.06469, 4.16638], [117.67641, 4.16535], [117.47313, 4.18857], [117.25801, 4.35108], [115.90217, 4.37708], [115.58276, 3.93499], [115.53713, 3.14776], [115.11343, 2.82879], [115.1721, 2.49671], [114.80706, 2.21665], [114.80706, 1.92351], [114.57892, 1.5], [114.03788, 1.44787], [113.64677, 1.23933], [113.01448, 1.42832], [113.021, 1.57819], [112.48648, 1.56516], [112.2127, 1.44135], [112.15679, 1.17004], [111.94553, 1.12016], [111.82846, 0.99349], [111.55434, 0.97864], [111.22979, 1.08326], [110.62374, 0.873], [110.49182, 0.88088], [110.35354, 0.98869], [109.66397, 1.60425], [109.66397, 1.79972], [109.57923, 1.80624], [109.53794, 1.91771], [109.62558, 1.99182], [109.82788, 2.86812], [107.94791, 1.06924], [108.50935, -2.01066], [109.3962, -2.07276], [109.17017, -4.07401], [116.58433, -5.30385], [120.02464, 2.83703]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q3803",
+           nameEn: "Lesser Sunda Islands",
+           aliases: ["ID-NU"],
+           country: "ID",
+           groups: ["035", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["62"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[116.96967, -8.01483], [114.92859, -7.49253], [114.42235, -8.09762], [114.39575, -8.2889], [116.22542, -10.49172], [122.14954, -11.52517], [125.68138, -9.85176], [125.09025, -9.46406], [124.97892, -9.19281], [125.04044, -9.17093], [125.09434, -9.19669], [125.18907, -9.16434], [125.18632, -9.03142], [125.11764, -8.96359], [124.97742, -9.08128], [124.94011, -8.85617], [124.46701, -9.13002], [124.45971, -9.30263], [124.38554, -9.3582], [124.35258, -9.43002], [124.3535, -9.48493], [124.28115, -9.50453], [124.28115, -9.42189], [124.21247, -9.36904], [124.14517, -9.42324], [124.10539, -9.41206], [124.04286, -9.34243], [124.04628, -9.22671], [124.33472, -9.11416], [124.92337, -8.75859], [125.87688, -7.49892], [116.96967, -8.01483]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q3812",
+           nameEn: "Sulawesi",
+           aliases: ["ID-SL"],
+           country: "ID",
+           groups: ["035", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["62"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[128.34321, 3.90322], [126.69413, 6.02692], [119.56457, 0.90759], [116.58433, -5.30385], [116.33992, -7.56171], [116.96967, -8.01483], [125.87688, -7.49892], [123.78965, -0.86805], [128.34321, 3.90322]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q3827",
+           nameEn: "Maluku Islands",
+           aliases: ["ID-ML"],
+           country: "ID",
+           groups: ["035", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["62"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[129.63187, 2.21409], [128.34321, 3.90322], [123.78965, -0.86805], [125.87688, -7.49892], [125.58506, -7.95311], [125.87691, -8.31789], [127.42116, -8.22471], [127.55165, -9.05052], [135.49042, -9.2276], [135.35517, -5.01442], [132.8312, -4.70282], [130.8468, -2.61103], [128.40647, -2.30349], [129.71519, -0.24692], [129.63187, 2.21409]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q3845",
+           nameEn: "Western New Guinea",
+           aliases: ["ID-PP"],
+           country: "ID",
+           groups: ["035", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["62"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[135.49042, -9.2276], [141.01842, -9.35091], [141.01763, -6.90181], [140.90448, -6.85033], [140.85295, -6.72996], [140.99813, -6.3233], [141.02352, 0.08993], [129.63187, 2.21409], [129.71519, -0.24692], [128.40647, -2.30349], [130.8468, -2.61103], [132.8312, -4.70282], [135.35517, -5.01442], [135.49042, -9.2276]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q5765",
+           nameEn: "Balearic Islands",
+           aliases: ["ES-IB"],
+           country: "ES",
+           groups: ["EU", "039", "150", "UN"],
+           callingCodes: ["34 971"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-2.27707, 35.35051], [5.10072, 39.89531], [3.75438, 42.33445], [-2.27707, 35.35051]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q5823",
+           nameEn: "Ceuta",
+           aliases: ["ES-CE"],
+           country: "ES",
+           groups: ["EA", "EU", "015", "002", "UN"],
+           level: "subterritory",
+           callingCodes: ["34"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-5.38491, 35.92591], [-5.37338, 35.88417], [-5.35844, 35.87375], [-5.34379, 35.8711], [-5.21179, 35.90091], [-5.38491, 35.92591]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q5831",
+           nameEn: "Melilla",
+           aliases: ["ES-ML"],
+           country: "ES",
+           groups: ["EA", "EU", "015", "002", "UN"],
+           level: "subterritory",
+           callingCodes: ["34"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-2.91909, 35.33927], [-2.96038, 35.31609], [-2.96648, 35.30475], [-2.96978, 35.29459], [-2.97035, 35.28852], [-2.96507, 35.28801], [-2.96826, 35.28296], [-2.96516, 35.27967], [-2.95431, 35.2728], [-2.95065, 35.26576], [-2.93893, 35.26737], [-2.92272, 35.27509], [-2.91909, 35.33927]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q7835",
+           nameEn: "Crimea",
+           country: "RU",
+           groups: ["151", "150", "UN"],
+           level: "subterritory",
+           callingCodes: ["7"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[33.5, 44], [36.4883, 45.0488], [36.475, 45.2411], [36.5049, 45.3136], [36.6545, 45.3417], [36.6645, 45.4514], [35.0498, 45.7683], [34.9601, 45.7563], [34.7991, 45.8101], [34.8015, 45.9005], [34.7548, 45.907], [34.6668, 45.9714], [34.6086, 45.9935], [34.5589, 45.9935], [34.5201, 45.951], [34.4873, 45.9427], [34.4415, 45.9599], [34.4122, 46.0025], [34.3391, 46.0611], [34.2511, 46.0532], [34.181, 46.068], [34.1293, 46.1049], [34.0731, 46.1177], [34.0527, 46.1084], [33.9155, 46.1594], [33.8523, 46.1986], [33.7972, 46.2048], [33.7405, 46.1855], [33.646, 46.2303], [33.6152, 46.2261], [33.6385, 46.1415], [33.6147, 46.1356], [33.5732, 46.1032], [33.5909, 46.0601], [33.5597, 46.0307], [31.5, 45.5], [33.5, 44]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q12837",
+           nameEn: "Iberia",
+           level: "sharedLandform"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q14056",
+           nameEn: "Jan Mayen",
+           aliases: ["NO-22"],
+           country: "NO",
+           groups: ["SJ", "154", "150", "UN"],
+           level: "subterritory"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-9.18243, 72.23144], [-10.71459, 70.09565], [-5.93364, 70.76368], [-9.18243, 72.23144]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q19188",
+           nameEn: "Mainland China",
+           country: "CN",
+           groups: ["030", "142", "UN"],
+           callingCodes: ["86"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[125.6131, 53.07229], [125.17522, 53.20225], [124.46078, 53.21881], [123.86158, 53.49391], [123.26989, 53.54843], [122.85966, 53.47395], [122.35063, 53.49565], [121.39213, 53.31888], [120.85633, 53.28499], [120.0451, 52.7359], [120.04049, 52.58773], [120.46454, 52.63811], [120.71673, 52.54099], [120.61346, 52.32447], [120.77337, 52.20805], [120.65907, 51.93544], [120.10963, 51.671], [119.13553, 50.37412], [119.38598, 50.35162], [119.27996, 50.13348], [119.11003, 50.00276], [118.61623, 49.93809], [117.82343, 49.52696], [117.48208, 49.62324], [117.27597, 49.62544], [116.71193, 49.83813], [116.03781, 48.87014], [116.06565, 48.81716], [115.78876, 48.51781], [115.811, 48.25699], [115.52082, 48.15367], [115.57128, 47.91988], [115.94296, 47.67741], [116.21879, 47.88505], [116.4465, 47.83662], [116.67405, 47.89039], [116.9723, 47.87285], [117.37875, 47.63627], [117.50181, 47.77216], [117.80196, 48.01661], [118.03676, 48.00982], [118.11009, 48.04], [118.22677, 48.03853], [118.29654, 48.00246], [118.55766, 47.99277], [118.7564, 47.76947], [119.12343, 47.66458], [119.13995, 47.53997], [119.35892, 47.48104], [119.31964, 47.42617], [119.54918, 47.29505], [119.56019, 47.24874], [119.62403, 47.24575], [119.71209, 47.19192], [119.85518, 46.92196], [119.91242, 46.90091], [119.89261, 46.66423], [119.80455, 46.67631], [119.77373, 46.62947], [119.68127, 46.59015], [119.65265, 46.62342], [119.42827, 46.63783], [119.32827, 46.61433], [119.24978, 46.64761], [119.10448, 46.65516], [119.00541, 46.74273], [118.92616, 46.72765], [118.89974, 46.77139], [118.8337, 46.77742], [118.78747, 46.68689], [118.30534, 46.73519], [117.69554, 46.50991], [117.60748, 46.59771], [117.41782, 46.57862], [117.36609, 46.36335], [116.83166, 46.38637], [116.75551, 46.33083], [116.58612, 46.30211], [116.26678, 45.96479], [116.24012, 45.8778], [116.27366, 45.78637], [116.16989, 45.68603], [115.60329, 45.44717], [114.94546, 45.37377], [114.74612, 45.43585], [114.54801, 45.38337], [114.5166, 45.27189], [113.70918, 44.72891], [112.74662, 44.86297], [112.4164, 45.06858], [111.98695, 45.09074], [111.76275, 44.98032], [111.40498, 44.3461], [111.96289, 43.81596], [111.93776, 43.68709], [111.79758, 43.6637], [111.59087, 43.51207], [111.0149, 43.3289], [110.4327, 42.78293], [110.08401, 42.6411], [109.89402, 42.63111], [109.452, 42.44842], [109.00679, 42.45302], [108.84489, 42.40246], [107.57258, 42.40898], [107.49681, 42.46221], [107.29755, 42.41395], [107.24774, 42.36107], [106.76517, 42.28741], [105.0123, 41.63188], [104.51667, 41.66113], [104.52258, 41.8706], [103.92804, 41.78246], [102.72403, 42.14675], [102.07645, 42.22519], [101.80515, 42.50074], [100.84979, 42.67087], [100.33297, 42.68231], [99.50671, 42.56535], [97.1777, 42.7964], [96.37926, 42.72055], [96.35658, 42.90363], [95.89543, 43.2528], [95.52594, 43.99353], [95.32891, 44.02407], [95.39772, 44.2805], [95.01191, 44.25274], [94.71959, 44.35284], [94.10003, 44.71016], [93.51161, 44.95964], [91.64048, 45.07408], [90.89169, 45.19667], [90.65114, 45.49314], [90.70907, 45.73437], [91.03026, 46.04194], [90.99672, 46.14207], [90.89639, 46.30711], [91.07696, 46.57315], [91.0147, 46.58171], [91.03649, 46.72916], [90.84035, 46.99525], [90.76108, 46.99399], [90.48542, 47.30438], [90.48854, 47.41826], [90.33598, 47.68303], [90.10871, 47.7375], [90.06512, 47.88177], [89.76624, 47.82745], [89.55453, 48.0423], [89.0711, 47.98528], [88.93186, 48.10263], [88.8011, 48.11302], [88.58316, 48.21893], [88.58939, 48.34531], [87.96361, 48.58478], [88.0788, 48.71436], [87.73822, 48.89582], [87.88171, 48.95853], [87.81333, 49.17354], [87.48983, 49.13794], [87.478, 49.07403], [87.28386, 49.11626], [86.87238, 49.12432], [86.73568, 48.99918], [86.75343, 48.70331], [86.38069, 48.46064], [85.73581, 48.3939], [85.5169, 48.05493], [85.61067, 47.49753], [85.69696, 47.2898], [85.54294, 47.06171], [85.22443, 47.04816], [84.93995, 46.87399], [84.73077, 47.01394], [83.92184, 46.98912], [83.04622, 47.19053], [82.21792, 45.56619], [82.58474, 45.40027], [82.51374, 45.1755], [81.73278, 45.3504], [80.11169, 45.03352], [79.8987, 44.89957], [80.38384, 44.63073], [80.40229, 44.23319], [80.40031, 44.10986], [80.75156, 43.44948], [80.69718, 43.32589], [80.77771, 43.30065], [80.78817, 43.14235], [80.62913, 43.141], [80.3735, 43.01557], [80.58999, 42.9011], [80.38169, 42.83142], [80.26886, 42.8366], [80.16892, 42.61137], [80.26841, 42.23797], [80.17807, 42.21166], [80.17842, 42.03211], [79.92977, 42.04113], [78.3732, 41.39603], [78.15757, 41.38565], [78.12873, 41.23091], [77.81287, 41.14307], [77.76206, 41.01574], [77.52723, 41.00227], [77.3693, 41.0375], [77.28004, 41.0033], [76.99302, 41.0696], [76.75681, 40.95354], [76.5261, 40.46114], [76.33659, 40.3482], [75.96168, 40.38064], [75.91361, 40.2948], [75.69663, 40.28642], [75.5854, 40.66874], [75.22834, 40.45382], [75.08243, 40.43945], [74.82013, 40.52197], [74.78168, 40.44886], [74.85996, 40.32857], [74.69875, 40.34668], [74.35063, 40.09742], [74.25533, 40.13191], [73.97049, 40.04378], [73.83006, 39.76136], [73.9051, 39.75073], [73.92354, 39.69565], [73.94683, 39.60733], [73.87018, 39.47879], [73.59831, 39.46425], [73.59241, 39.40843], [73.5004, 39.38402], [73.55396, 39.3543], [73.54572, 39.27567], [73.60638, 39.24534], [73.75823, 39.023], [73.81728, 39.04007], [73.82964, 38.91517], [73.7445, 38.93867], [73.7033, 38.84782], [73.80656, 38.66449], [73.79806, 38.61106], [73.97933, 38.52945], [74.17022, 38.65504], [74.51217, 38.47034], [74.69619, 38.42947], [74.69894, 38.22155], [74.80331, 38.19889], [74.82665, 38.07359], [74.9063, 38.03033], [74.92416, 37.83428], [75.00935, 37.77486], [74.8912, 37.67576], [74.94338, 37.55501], [75.06011, 37.52779], [75.15899, 37.41443], [75.09719, 37.37297], [75.12328, 37.31839], [74.88887, 37.23275], [74.80605, 37.21565], [74.49981, 37.24518], [74.56453, 37.03023], [75.13839, 37.02622], [75.40481, 36.95382], [75.45562, 36.71971], [75.72737, 36.7529], [75.92391, 36.56986], [76.0324, 36.41198], [76.00906, 36.17511], [75.93028, 36.13136], [76.15325, 35.9264], [76.14913, 35.82848], [76.33453, 35.84296], [76.50961, 35.8908], [76.77323, 35.66062], [76.84539, 35.67356], [76.96624, 35.5932], [77.44277, 35.46132], [77.70232, 35.46244], [77.80532, 35.52058], [78.11664, 35.48022], [78.03466, 35.3785], [78.00033, 35.23954], [78.22692, 34.88771], [78.18435, 34.7998], [78.27781, 34.61484], [78.54964, 34.57283], [78.56475, 34.50835], [78.74465, 34.45174], [79.05364, 34.32482], [78.99802, 34.3027], [78.91769, 34.15452], [78.66225, 34.08858], [78.65657, 34.03195], [78.73367, 34.01121], [78.77349, 33.73871], [78.67599, 33.66445], [78.73636, 33.56521], [79.15252, 33.17156], [79.14016, 33.02545], [79.46562, 32.69668], [79.26768, 32.53277], [79.13174, 32.47766], [79.0979, 32.38051], [78.99322, 32.37948], [78.96713, 32.33655], [78.7831, 32.46873], [78.73916, 32.69438], [78.38897, 32.53938], [78.4645, 32.45367], [78.49609, 32.2762], [78.68754, 32.10256], [78.74404, 32.00384], [78.78036, 31.99478], [78.69933, 31.78723], [78.84516, 31.60631], [78.71032, 31.50197], [78.77898, 31.31209], [78.89344, 31.30481], [79.01931, 31.42817], [79.14016, 31.43403], [79.30694, 31.17357], [79.59884, 30.93943], [79.93255, 30.88288], [80.20721, 30.58541], [80.54504, 30.44936], [80.83343, 30.32023], [81.03953, 30.20059], [81.12842, 30.01395], [81.24362, 30.0126], [81.29032, 30.08806], [81.2623, 30.14596], [81.33355, 30.15303], [81.39928, 30.21862], [81.41018, 30.42153], [81.5459, 30.37688], [81.62033, 30.44703], [81.99082, 30.33423], [82.10135, 30.35439], [82.10757, 30.23745], [82.19475, 30.16884], [82.16984, 30.0692], [82.38622, 30.02608], [82.5341, 29.9735], [82.73024, 29.81695], [83.07116, 29.61957], [83.28131, 29.56813], [83.44787, 29.30513], [83.63156, 29.16249], [83.82303, 29.30513], [83.97559, 29.33091], [84.18107, 29.23451], [84.24801, 29.02783], [84.2231, 28.89571], [84.47528, 28.74023], [84.62317, 28.73887], [84.85511, 28.58041], [85.06059, 28.68562], [85.19135, 28.62825], [85.18668, 28.54076], [85.10729, 28.34092], [85.38127, 28.28336], [85.4233, 28.32996], [85.59765, 28.30529], [85.60854, 28.25045], [85.69105, 28.38475], [85.71907, 28.38064], [85.74864, 28.23126], [85.84672, 28.18187], [85.90743, 28.05144], [85.97813, 27.99023], [85.94946, 27.9401], [86.06309, 27.90021], [86.12069, 27.93047], [86.08333, 28.02121], [86.088, 28.09264], [86.18607, 28.17364], [86.22966, 27.9786], [86.42736, 27.91122], [86.51609, 27.96623], [86.56265, 28.09569], [86.74181, 28.10638], [86.75582, 28.04182], [87.03757, 27.94835], [87.11696, 27.84104], [87.56996, 27.84517], [87.72718, 27.80938], [87.82681, 27.95248], [88.13378, 27.88015], [88.1278, 27.95417], [88.25332, 27.9478], [88.54858, 28.06057], [88.63235, 28.12356], [88.83559, 28.01936], [88.88091, 27.85192], [88.77517, 27.45415], [88.82981, 27.38814], [88.91901, 27.32483], [88.93678, 27.33777], [88.96947, 27.30319], [89.00216, 27.32532], [88.95355, 27.4106], [88.97213, 27.51671], [89.0582, 27.60985], [89.12825, 27.62502], [89.59525, 28.16433], [89.79762, 28.23979], [90.13387, 28.19178], [90.58842, 28.02838], [90.69894, 28.07784], [91.20019, 27.98715], [91.25779, 28.07509], [91.46327, 28.0064], [91.48973, 27.93903], [91.5629, 27.84823], [91.6469, 27.76358], [91.84722, 27.76325], [91.87057, 27.7195], [92.27432, 27.89077], [92.32101, 27.79363], [92.42538, 27.80092], [92.7275, 27.98662], [92.73025, 28.05814], [92.65472, 28.07632], [92.67486, 28.15018], [92.93075, 28.25671], [93.14635, 28.37035], [93.18069, 28.50319], [93.44621, 28.67189], [93.72797, 28.68821], [94.35897, 29.01965], [94.2752, 29.11687], [94.69318, 29.31739], [94.81353, 29.17804], [95.0978, 29.14446], [95.11291, 29.09527], [95.2214, 29.10727], [95.26122, 29.07727], [95.3038, 29.13847], [95.41091, 29.13007], [95.50842, 29.13487], [95.72086, 29.20797], [95.75149, 29.32063], [95.84899, 29.31464], [96.05361, 29.38167], [96.31316, 29.18643], [96.18682, 29.11087], [96.20467, 29.02325], [96.3626, 29.10607], [96.61391, 28.72742], [96.40929, 28.51526], [96.48895, 28.42955], [96.6455, 28.61657], [96.85561, 28.4875], [96.88445, 28.39452], [96.98882, 28.32564], [97.1289, 28.3619], [97.34547, 28.21385], [97.41729, 28.29783], [97.47085, 28.2688], [97.50518, 28.49716], [97.56835, 28.55628], [97.70705, 28.5056], [97.79632, 28.33168], [97.90069, 28.3776], [98.15337, 28.12114], [98.13964, 27.9478], [98.32641, 27.51385], [98.42529, 27.55404], [98.43353, 27.67086], [98.69582, 27.56499], [98.7333, 26.85615], [98.77547, 26.61994], [98.72741, 26.36183], [98.67797, 26.24487], [98.7329, 26.17218], [98.66884, 26.09165], [98.63128, 26.15492], [98.57085, 26.11547], [98.60763, 26.01512], [98.70818, 25.86241], [98.63128, 25.79937], [98.54064, 25.85129], [98.40606, 25.61129], [98.31268, 25.55307], [98.25774, 25.6051], [98.16848, 25.62739], [98.18084, 25.56298], [98.12591, 25.50722], [98.14925, 25.41547], [97.92541, 25.20815], [97.83614, 25.2715], [97.77023, 25.11492], [97.72216, 25.08508], [97.72903, 24.91332], [97.79949, 24.85655], [97.76481, 24.8289], [97.73127, 24.83015], [97.70181, 24.84557], [97.64354, 24.79171], [97.56648, 24.76475], [97.56383, 24.75535], [97.5542, 24.74943], [97.54675, 24.74202], [97.56525, 24.72838], [97.56286, 24.54535], [97.52757, 24.43748], [97.60029, 24.4401], [97.66998, 24.45288], [97.7098, 24.35658], [97.65624, 24.33781], [97.66723, 24.30027], [97.71941, 24.29652], [97.76799, 24.26365], [97.72998, 24.2302], [97.72799, 24.18883], [97.75305, 24.16902], [97.72903, 24.12606], [97.62363, 24.00506], [97.5247, 23.94032], [97.64667, 23.84574], [97.72302, 23.89288], [97.79456, 23.94836], [97.79416, 23.95663], [97.84328, 23.97603], [97.86545, 23.97723], [97.88811, 23.97446], [97.8955, 23.97758], [97.89676, 23.97931], [97.89683, 23.98389], [97.88814, 23.98605], [97.88414, 23.99405], [97.88616, 24.00463], [97.90998, 24.02094], [97.93951, 24.01953], [97.98691, 24.03897], [97.99583, 24.04932], [98.04709, 24.07616], [98.05302, 24.07408], [98.05671, 24.07961], [98.0607, 24.07812], [98.06703, 24.08028], [98.07806, 24.07988], [98.20666, 24.11406], [98.54476, 24.13119], [98.59256, 24.08371], [98.85319, 24.13042], [98.87998, 24.15624], [98.89632, 24.10612], [98.67797, 23.9644], [98.68209, 23.80492], [98.79607, 23.77947], [98.82933, 23.72921], [98.81775, 23.694], [98.88396, 23.59555], [98.80294, 23.5345], [98.82877, 23.47908], [98.87683, 23.48995], [98.92104, 23.36946], [98.87573, 23.33038], [98.93958, 23.31414], [98.92515, 23.29535], [98.88597, 23.18656], [99.05975, 23.16382], [99.04601, 23.12215], [99.25741, 23.09025], [99.34127, 23.13099], [99.52214, 23.08218], [99.54218, 22.90014], [99.43537, 22.94086], [99.45654, 22.85726], [99.31243, 22.73893], [99.38247, 22.57544], [99.37972, 22.50188], [99.28771, 22.4105], [99.17318, 22.18025], [99.19176, 22.16983], [99.1552, 22.15874], [99.33166, 22.09656], [99.47585, 22.13345], [99.85351, 22.04183], [99.96612, 22.05965], [99.99084, 21.97053], [99.94003, 21.82782], [99.98654, 21.71064], [100.04956, 21.66843], [100.12679, 21.70539], [100.17486, 21.65306], [100.10757, 21.59945], [100.12542, 21.50365], [100.1625, 21.48704], [100.18447, 21.51898], [100.25863, 21.47043], [100.35201, 21.53176], [100.42892, 21.54325], [100.4811, 21.46148], [100.57861, 21.45637], [100.72143, 21.51898], [100.87265, 21.67396], [101.11744, 21.77659], [101.15156, 21.56129], [101.2124, 21.56422], [101.19349, 21.41959], [101.26912, 21.36482], [101.2229, 21.23271], [101.29326, 21.17254], [101.54563, 21.25668], [101.6068, 21.23329], [101.59491, 21.18621], [101.60886, 21.17947], [101.66977, 21.20004], [101.70548, 21.14911], [101.7622, 21.14813], [101.79266, 21.19025], [101.76745, 21.21571], [101.83887, 21.20983], [101.84412, 21.25291], [101.74014, 21.30967], [101.74224, 21.48276], [101.7727, 21.51794], [101.7475, 21.5873], [101.80001, 21.57461], [101.83257, 21.61562], [101.74555, 21.72852], [101.7791, 21.83019], [101.62566, 21.96574], [101.57525, 22.13026], [101.60675, 22.13513], [101.53638, 22.24794], [101.56789, 22.28876], [101.61306, 22.27515], [101.68973, 22.46843], [101.7685, 22.50337], [101.86828, 22.38397], [101.90714, 22.38688], [101.91344, 22.44417], [101.98487, 22.42766], [102.03633, 22.46164], [102.1245, 22.43372], [102.14099, 22.40092], [102.16621, 22.43336], [102.26428, 22.41321], [102.25339, 22.4607], [102.41061, 22.64184], [102.38415, 22.67919], [102.42618, 22.69212], [102.46665, 22.77108], [102.51802, 22.77969], [102.57095, 22.7036], [102.60675, 22.73376], [102.8636, 22.60735], [102.9321, 22.48659], [103.0722, 22.44775], [103.07843, 22.50097], [103.17961, 22.55705], [103.15782, 22.59873], [103.18895, 22.64471], [103.28079, 22.68063], [103.32282, 22.8127], [103.43179, 22.75816], [103.43646, 22.70648], [103.52675, 22.59155], [103.57812, 22.65764], [103.56255, 22.69499], [103.64506, 22.79979], [103.87904, 22.56683], [103.93286, 22.52703], [103.94513, 22.52553], [103.95191, 22.5134], [103.96352, 22.50584], [103.96783, 22.51173], [103.97384, 22.50634], [103.99247, 22.51958], [104.01088, 22.51823], [104.03734, 22.72945], [104.11384, 22.80363], [104.27084, 22.8457], [104.25683, 22.76534], [104.35593, 22.69353], [104.47225, 22.75813], [104.58122, 22.85571], [104.60457, 22.81841], [104.65283, 22.83419], [104.72755, 22.81984], [104.77114, 22.90017], [104.84942, 22.93631], [104.86765, 22.95178], [104.8334, 23.01484], [104.79478, 23.12934], [104.87382, 23.12854], [104.87992, 23.17141], [104.91435, 23.18666], [104.9486, 23.17235], [104.96532, 23.20463], [104.98712, 23.19176], [105.07002, 23.26248], [105.11672, 23.25247], [105.17276, 23.28679], [105.22569, 23.27249], [105.32376, 23.39684], [105.40782, 23.28107], [105.42805, 23.30824], [105.49966, 23.20669], [105.56037, 23.16806], [105.57594, 23.075], [105.72382, 23.06641], [105.8726, 22.92756], [105.90119, 22.94168], [105.99568, 22.94178], [106.00179, 22.99049], [106.19705, 22.98475], [106.27022, 22.87722], [106.34961, 22.86718], [106.49749, 22.91164], [106.51306, 22.94891], [106.55976, 22.92311], [106.60179, 22.92884], [106.6516, 22.86862], [106.6734, 22.89587], [106.71387, 22.88296], [106.71128, 22.85982], [106.78422, 22.81532], [106.81271, 22.8226], [106.83685, 22.8098], [106.82404, 22.7881], [106.76293, 22.73491], [106.72321, 22.63606], [106.71698, 22.58432], [106.65316, 22.5757], [106.61269, 22.60301], [106.58395, 22.474], [106.55665, 22.46498], [106.57221, 22.37], [106.55976, 22.34841], [106.6516, 22.33977], [106.69986, 22.22309], [106.67495, 22.1885], [106.6983, 22.15102], [106.70142, 22.02409], [106.68274, 21.99811], [106.69276, 21.96013], [106.72551, 21.97923], [106.74345, 22.00965], [106.81038, 21.97934], [106.9178, 21.97357], [106.92714, 21.93459], [106.97228, 21.92592], [106.99252, 21.95191], [107.05634, 21.92303], [107.06101, 21.88982], [107.00964, 21.85948], [107.02615, 21.81981], [107.10771, 21.79879], [107.20734, 21.71493], [107.24625, 21.7077], [107.29296, 21.74674], [107.35834, 21.6672], [107.35989, 21.60063], [107.38636, 21.59774], [107.41593, 21.64839], [107.47197, 21.6672], [107.49532, 21.62958], [107.49065, 21.59774], [107.54047, 21.5934], [107.56537, 21.61945], [107.66967, 21.60787], [107.80355, 21.66141], [107.86114, 21.65128], [107.90006, 21.5905], [107.92652, 21.58906], [107.95232, 21.5388], [107.96774, 21.53601], [107.97074, 21.54072], [107.97383, 21.53961], [107.97932, 21.54503], [108.02926, 21.54997], [108.0569, 21.53604], [108.10003, 21.47338], [108.00365, 17.98159], [111.60491, 13.57105], [118.41371, 24.06775], [118.11703, 24.39734], [118.28244, 24.51231], [118.35291, 24.51645], [118.42453, 24.54644], [118.56434, 24.49266], [120.49232, 25.22863], [121.03532, 26.8787], [123.5458, 31.01942], [122.29378, 31.76513], [122.80525, 33.30571], [123.85601, 37.49093], [123.90497, 38.79949], [124.17532, 39.8232], [124.23201, 39.9248], [124.35029, 39.95639], [124.37089, 40.03004], [124.3322, 40.05573], [124.38556, 40.11047], [124.40719, 40.13655], [124.86913, 40.45387], [125.71172, 40.85223], [125.76869, 40.87908], [126.00335, 40.92835], [126.242, 41.15454], [126.53189, 41.35206], [126.60631, 41.65565], [126.90729, 41.79955], [127.17841, 41.59714], [127.29712, 41.49473], [127.92943, 41.44291], [128.02633, 41.42103], [128.03311, 41.39232], [128.12967, 41.37931], [128.18546, 41.41279], [128.20061, 41.40895], [128.30716, 41.60322], [128.15119, 41.74568], [128.04487, 42.01769], [128.94007, 42.03537], [128.96068, 42.06657], [129.15178, 42.17224], [129.22285, 42.26491], [129.22423, 42.3553], [129.28541, 42.41574], [129.42882, 42.44702], [129.54701, 42.37254], [129.60482, 42.44461], [129.72541, 42.43739], [129.75294, 42.59409], [129.77183, 42.69435], [129.7835, 42.76521], [129.80719, 42.79218], [129.83277, 42.86746], [129.85261, 42.96494], [129.8865, 43.00395], [129.95082, 43.01051], [129.96409, 42.97306], [130.12957, 42.98361], [130.09764, 42.91425], [130.26095, 42.9027], [130.23068, 42.80125], [130.2385, 42.71127], [130.41826, 42.6011], [130.44361, 42.54849], [130.50123, 42.61636], [130.55143, 42.52158], [130.62107, 42.58413], [130.56576, 42.68925], [130.40213, 42.70788], [130.44361, 42.76205], [130.66524, 42.84753], [131.02438, 42.86518], [131.02668, 42.91246], [131.135, 42.94114], [131.10274, 43.04734], [131.20414, 43.13654], [131.19031, 43.21385], [131.30324, 43.39498], [131.29402, 43.46695], [131.19492, 43.53047], [131.21105, 43.82383], [131.26176, 43.94011], [131.23583, 43.96085], [131.25484, 44.03131], [131.30365, 44.04262], [131.1108, 44.70266], [130.95639, 44.85154], [131.48415, 44.99513], [131.68466, 45.12374], [131.66852, 45.2196], [131.76532, 45.22609], [131.86903, 45.33636], [131.99417, 45.2567], [132.83978, 45.05916], [132.96373, 45.0212], [133.12293, 45.1332], [133.09279, 45.25693], [133.19419, 45.51913], [133.41083, 45.57723], [133.48457, 45.86203], [133.60442, 45.90053], [133.67569, 45.9759], [133.72695, 46.05576], [133.68047, 46.14697], [133.88097, 46.25066], [133.91496, 46.4274], [133.84104, 46.46681], [134.03538, 46.75668], [134.20016, 47.33458], [134.50898, 47.4812], [134.7671, 47.72051], [134.55508, 47.98651], [134.67098, 48.1564], [134.75328, 48.36763], [134.49516, 48.42884], [132.66989, 47.96491], [132.57309, 47.71741], [131.90448, 47.68011], [131.2635, 47.73325], [131.09871, 47.6852], [130.95985, 47.6957], [130.90915, 47.90623], [130.65103, 48.10052], [130.84462, 48.30942], [130.52147, 48.61745], [130.66946, 48.88251], [130.43232, 48.90844], [130.2355, 48.86741], [129.85416, 49.11067], [129.67598, 49.29596], [129.50685, 49.42398], [129.40398, 49.44194], [129.35317, 49.3481], [129.23232, 49.40353], [129.11153, 49.36813], [128.72896, 49.58676], [127.83476, 49.5748], [127.53516, 49.84306], [127.49299, 50.01251], [127.60515, 50.23503], [127.37384, 50.28393], [127.36009, 50.43787], [127.28765, 50.46585], [127.36335, 50.58306], [127.28165, 50.72075], [127.14586, 50.91152], [126.93135, 51.0841], [126.90369, 51.3238], [126.68349, 51.70607], [126.44606, 51.98254], [126.558, 52.13738], [125.6131, 53.07229]], [[113.56865, 22.20973], [113.57123, 22.20416], [113.60504, 22.20464], [113.63011, 22.10782], [113.57191, 22.07696], [113.54839, 22.10909], [113.54942, 22.14519], [113.54093, 22.15497], [113.52659, 22.18271], [113.53552, 22.20607], [113.53301, 22.21235], [113.53591, 22.21369], [113.54093, 22.21314], [113.54333, 22.21688], [113.5508, 22.21672], [113.56865, 22.20973]], [[114.50148, 22.15017], [113.92195, 22.13873], [113.83338, 22.1826], [113.81621, 22.2163], [113.86771, 22.42972], [114.03113, 22.5065], [114.05438, 22.5026], [114.05729, 22.51104], [114.06272, 22.51617], [114.07267, 22.51855], [114.07817, 22.52997], [114.08606, 22.53276], [114.09048, 22.53716], [114.09692, 22.53435], [114.1034, 22.5352], [114.11181, 22.52878], [114.11656, 22.53415], [114.12665, 22.54003], [114.13823, 22.54319], [114.1482, 22.54091], [114.15123, 22.55163], [114.1597, 22.56041], [114.17247, 22.55944], [114.18338, 22.55444], [114.20655, 22.55706], [114.22185, 22.55343], [114.22888, 22.5436], [114.25154, 22.55977], [114.44998, 22.55977], [114.50148, 22.15017]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q22890",
+           nameEn: "Ireland",
+           level: "sharedLandform"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q23666",
+           nameEn: "Great Britain",
+           country: "GB",
+           level: "sharedLandform"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q23681",
+           nameEn: "Northern Cyprus",
+           groups: ["Q644636", "145", "142"],
+           driveSide: "left",
+           callingCodes: ["90 392"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[33.67678, 35.03866], [33.67742, 35.05963], [33.68474, 35.06602], [33.69095, 35.06237], [33.70861, 35.07644], [33.7161, 35.07279], [33.70209, 35.04882], [33.71482, 35.03722], [33.73824, 35.05321], [33.76106, 35.04253], [33.78581, 35.05104], [33.82067, 35.07826], [33.84168, 35.06823], [33.8541, 35.07201], [33.87479, 35.08881], [33.87097, 35.09389], [33.87622, 35.10457], [33.87224, 35.12293], [33.88561, 35.12449], [33.88943, 35.12007], [33.88737, 35.11408], [33.89853, 35.11377], [33.91789, 35.08688], [33.91299, 35.07579], [33.90247, 35.07686], [33.89485, 35.06873], [33.88367, 35.07877], [33.85261, 35.0574], [33.8355, 35.05777], [33.82051, 35.0667], [33.8012, 35.04786], [33.81524, 35.04192], [33.83055, 35.02865], [33.82875, 35.01685], [33.84045, 35.00616], [33.85216, 35.00579], [33.85891, 35.001], [33.85621, 34.98956], [33.83505, 34.98108], [33.84811, 34.97075], [33.86432, 34.97592], [33.90075, 34.96623], [33.98684, 34.76642], [35.48515, 34.70851], [35.51152, 36.10954], [32.82353, 35.70297], [32.46489, 35.48584], [32.60361, 35.16647], [32.64864, 35.19967], [32.70947, 35.18328], [32.70779, 35.14127], [32.85733, 35.07742], [32.86406, 35.1043], [32.94471, 35.09422], [33.01192, 35.15639], [33.08249, 35.17319], [33.11105, 35.15639], [33.15138, 35.19504], [33.27068, 35.16815], [33.3072, 35.16816], [33.31955, 35.18096], [33.35056, 35.18328], [33.34964, 35.17803], [33.35596, 35.17942], [33.35612, 35.17402], [33.36569, 35.17479], [33.3717, 35.1788], [33.37248, 35.18698], [33.38575, 35.2018], [33.4076, 35.20062], [33.41675, 35.16325], [33.46813, 35.10564], [33.48136, 35.0636], [33.47825, 35.04103], [33.45178, 35.02078], [33.45256, 35.00288], [33.47666, 35.00701], [33.48915, 35.06594], [33.53975, 35.08151], [33.57478, 35.06049], [33.567, 35.04803], [33.59658, 35.03635], [33.61215, 35.0527], [33.63765, 35.03869], [33.67678, 35.03866]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q25231",
+           nameEn: "Svalbard",
+           aliases: ["NO-21"],
+           country: "NO",
+           groups: ["SJ", "154", "150", "UN"],
+           level: "subterritory",
+           callingCodes: ["47 79"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-7.49892, 77.24208], [32.07813, 72.01005], [36.85549, 84.09565], [-7.49892, 77.24208]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q25263",
+           nameEn: "Azores",
+           aliases: ["PT-20"],
+           country: "PT",
+           groups: ["Q3320166", "Q2914565", "Q105472", "EU", "039", "150", "UN"],
+           callingCodes: ["351"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-23.12984, 40.26428], [-36.43765, 41.39418], [-22.54767, 33.34416], [-23.12984, 40.26428]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q25359",
+           nameEn: "Navassa Island",
+           aliases: ["UM-76"],
+           country: "US",
+           groups: ["UM", "Q1352230", "029", "003", "419", "019", "UN"],
+           level: "subterritory",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-74.7289, 18.71009], [-75.71816, 18.46438], [-74.76465, 18.06252], [-74.7289, 18.71009]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q25396",
+           nameEn: "Bonaire",
+           aliases: ["BQ-BO", "NL-BQ1"],
+           country: "NL",
+           groups: ["Q1451600", "BQ", "029", "003", "419", "019", "UN"],
+           level: "subterritory",
+           callingCodes: ["599 7"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-67.89186, 12.4116], [-68.90012, 12.62309], [-68.33524, 11.78151], [-68.01417, 11.77722], [-67.89186, 12.4116]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q25528",
+           nameEn: "Saba",
+           aliases: ["BQ-SA", "NL-BQ2"],
+           country: "NL",
+           groups: ["Q1451600", "BQ", "029", "003", "419", "019", "UN"],
+           level: "subterritory",
+           callingCodes: ["599 4"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-63.07669, 17.79659], [-63.81314, 17.95045], [-63.22932, 17.32592], [-63.07669, 17.79659]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q26180",
+           nameEn: "Sint Eustatius",
+           aliases: ["BQ-SE", "NL-BQ3"],
+           country: "NL",
+           groups: ["Q1451600", "BQ", "029", "003", "419", "019", "UN"],
+           level: "subterritory",
+           callingCodes: ["599 3"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-63.07669, 17.79659], [-63.34999, 16.94218], [-62.76692, 17.64353], [-63.07669, 17.79659]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q26253",
+           nameEn: "Madeira",
+           aliases: ["PT-30"],
+           country: "PT",
+           groups: ["Q3320166", "Q2914565", "Q105472", "EU", "039", "150", "UN"],
+           callingCodes: ["351"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-19.30302, 33.65304], [-16.04789, 29.65076], [-11.68307, 33.12333], [-19.30302, 33.65304]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q26927",
+           nameEn: "Lakshadweep",
+           aliases: ["IN-LD"],
+           country: "IN",
+           groups: ["034", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["91"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[67.64074, 11.57295], [76.59015, 5.591], [72.67494, 13.58102], [67.64074, 11.57295]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q27329",
+           nameEn: "Asian Russia",
+           country: "RU",
+           groups: ["142", "UN"],
+           callingCodes: ["7"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-179.99933, 64.74703], [-172.76104, 63.77445], [-169.03888, 65.48473], [-168.95635, 65.98512], [-168.25765, 71.99091], [-179.9843, 71.90735], [-179.99933, 64.74703]]], [[[59.99809, 51.98263], [60.19925, 51.99173], [60.48915, 52.15175], [60.72581, 52.15538], [60.78201, 52.22067], [61.05417, 52.35096], [60.98021, 52.50068], [60.84709, 52.52228], [60.84118, 52.63912], [60.71693, 52.66245], [60.71989, 52.75923], [61.05842, 52.92217], [61.23462, 53.03227], [62.0422, 52.96105], [62.12799, 52.99133], [62.14574, 53.09626], [61.19024, 53.30536], [61.14291, 53.41481], [61.29082, 53.50992], [61.37957, 53.45887], [61.57185, 53.50112], [61.55706, 53.57144], [60.90626, 53.62937], [61.22574, 53.80268], [61.14283, 53.90063], [60.99796, 53.93699], [61.26863, 53.92797], [61.3706, 54.08464], [61.47603, 54.08048], [61.56941, 53.95703], [61.65318, 54.02445], [62.03913, 53.94768], [62.00966, 54.04134], [62.38535, 54.03961], [62.45931, 53.90737], [62.56876, 53.94047], [62.58651, 54.05871], [63.80604, 54.27079], [63.91224, 54.20013], [64.02715, 54.22679], [63.97686, 54.29763], [64.97216, 54.4212], [65.11033, 54.33028], [65.24663, 54.35721], [65.20174, 54.55216], [68.21308, 54.98645], [68.26661, 55.09226], [68.19206, 55.18823], [68.90865, 55.38148], [69.34224, 55.36344], [69.74917, 55.35545], [70.19179, 55.1476], [70.76493, 55.3027], [70.96009, 55.10558], [71.08288, 54.71253], [71.24185, 54.64965], [71.08706, 54.33376], [71.10379, 54.13326], [71.96141, 54.17736], [72.17477, 54.36303], [72.43415, 53.92685], [72.71026, 54.1161], [73.37963, 53.96132], [73.74778, 54.07194], [73.68921, 53.86522], [73.25412, 53.61532], [73.39218, 53.44623], [75.07405, 53.80831], [75.43398, 53.98652], [75.3668, 54.07439], [76.91052, 54.4677], [76.82266, 54.1798], [76.44076, 54.16017], [76.54243, 53.99329], [77.90383, 53.29807], [79.11255, 52.01171], [80.08138, 50.77658], [80.4127, 50.95581], [80.44819, 51.20855], [80.80318, 51.28262], [81.16999, 51.15662], [81.06091, 50.94833], [81.41248, 50.97524], [81.46581, 50.77658], [81.94999, 50.79307], [82.55443, 50.75412], [83.14607, 51.00796], [83.8442, 50.87375], [84.29385, 50.27257], [84.99198, 50.06793], [85.24047, 49.60239], [86.18709, 49.50259], [86.63674, 49.80136], [86.79056, 49.74787], [86.61307, 49.60239], [86.82606, 49.51796], [87.03071, 49.25142], [87.31465, 49.23603], [87.28386, 49.11626], [87.478, 49.07403], [87.48983, 49.13794], [87.81333, 49.17354], [87.98977, 49.18147], [88.15543, 49.30314], [88.17223, 49.46934], [88.42449, 49.48821], [88.82499, 49.44808], [89.70687, 49.72535], [89.59711, 49.90851], [91.86048, 50.73734], [92.07173, 50.69585], [92.44714, 50.78762], [93.01109, 50.79001], [92.99595, 50.63183], [94.30823, 50.57498], [94.39258, 50.22193], [94.49477, 50.17832], [94.6121, 50.04239], [94.97166, 50.04725], [95.02465, 49.96941], [95.74757, 49.97915], [95.80056, 50.04239], [96.97388, 49.88413], [97.24639, 49.74737], [97.56811, 49.84265], [97.56432, 49.92801], [97.76871, 49.99861], [97.85197, 49.91339], [98.29481, 50.33561], [98.31373, 50.4996], [98.06393, 50.61262], [97.9693, 50.78044], [98.01472, 50.86652], [97.83305, 51.00248], [98.05257, 51.46696], [98.22053, 51.46579], [98.33222, 51.71832], [98.74142, 51.8637], [98.87768, 52.14563], [99.27888, 51.96876], [99.75578, 51.90108], [99.89203, 51.74903], [100.61116, 51.73028], [101.39085, 51.45753], [101.5044, 51.50467], [102.14032, 51.35566], [102.32194, 50.67982], [102.71178, 50.38873], [103.70343, 50.13952], [105.32528, 50.4648], [106.05562, 50.40582], [106.07865, 50.33474], [106.47156, 50.31909], [106.49628, 50.32436], [106.51122, 50.34408], [106.58373, 50.34044], [106.80326, 50.30177], [107.00007, 50.1977], [107.1174, 50.04239], [107.36407, 49.97612], [107.96116, 49.93191], [107.95387, 49.66659], [108.27937, 49.53167], [108.53969, 49.32325], [109.18017, 49.34709], [109.51325, 49.22859], [110.24373, 49.16676], [110.39891, 49.25083], [110.64493, 49.1816], [113.02647, 49.60772], [113.20216, 49.83356], [114.325, 50.28098], [114.9703, 50.19254], [115.26068, 49.97367], [115.73602, 49.87688], [116.22402, 50.04477], [116.62502, 49.92919], [116.71193, 49.83813], [117.27597, 49.62544], [117.48208, 49.62324], [117.82343, 49.52696], [118.61623, 49.93809], [119.11003, 50.00276], [119.27996, 50.13348], [119.38598, 50.35162], [119.13553, 50.37412], [120.10963, 51.671], [120.65907, 51.93544], [120.77337, 52.20805], [120.61346, 52.32447], [120.71673, 52.54099], [120.46454, 52.63811], [120.04049, 52.58773], [120.0451, 52.7359], [120.85633, 53.28499], [121.39213, 53.31888], [122.35063, 53.49565], [122.85966, 53.47395], [123.26989, 53.54843], [123.86158, 53.49391], [124.46078, 53.21881], [125.17522, 53.20225], [125.6131, 53.07229], [126.558, 52.13738], [126.44606, 51.98254], [126.68349, 51.70607], [126.90369, 51.3238], [126.93135, 51.0841], [127.14586, 50.91152], [127.28165, 50.72075], [127.36335, 50.58306], [127.28765, 50.46585], [127.36009, 50.43787], [127.37384, 50.28393], [127.60515, 50.23503], [127.49299, 50.01251], [127.53516, 49.84306], [127.83476, 49.5748], [128.72896, 49.58676], [129.11153, 49.36813], [129.23232, 49.40353], [129.35317, 49.3481], [129.40398, 49.44194], [129.50685, 49.42398], [129.67598, 49.29596], [129.85416, 49.11067], [130.2355, 48.86741], [130.43232, 48.90844], [130.66946, 48.88251], [130.52147, 48.61745], [130.84462, 48.30942], [130.65103, 48.10052], [130.90915, 47.90623], [130.95985, 47.6957], [131.09871, 47.6852], [131.2635, 47.73325], [131.90448, 47.68011], [132.57309, 47.71741], [132.66989, 47.96491], [134.49516, 48.42884], [134.75328, 48.36763], [134.67098, 48.1564], [134.55508, 47.98651], [134.7671, 47.72051], [134.50898, 47.4812], [134.20016, 47.33458], [134.03538, 46.75668], [133.84104, 46.46681], [133.91496, 46.4274], [133.88097, 46.25066], [133.68047, 46.14697], [133.72695, 46.05576], [133.67569, 45.9759], [133.60442, 45.90053], [133.48457, 45.86203], [133.41083, 45.57723], [133.19419, 45.51913], [133.09279, 45.25693], [133.12293, 45.1332], [132.96373, 45.0212], [132.83978, 45.05916], [131.99417, 45.2567], [131.86903, 45.33636], [131.76532, 45.22609], [131.66852, 45.2196], [131.68466, 45.12374], [131.48415, 44.99513], [130.95639, 44.85154], [131.1108, 44.70266], [131.30365, 44.04262], [131.25484, 44.03131], [131.23583, 43.96085], [131.26176, 43.94011], [131.21105, 43.82383], [131.19492, 43.53047], [131.29402, 43.46695], [131.30324, 43.39498], [131.19031, 43.21385], [131.20414, 43.13654], [131.10274, 43.04734], [131.135, 42.94114], [131.02668, 42.91246], [131.02438, 42.86518], [130.66524, 42.84753], [130.44361, 42.76205], [130.40213, 42.70788], [130.56576, 42.68925], [130.62107, 42.58413], [130.55143, 42.52158], [130.56835, 42.43281], [130.60805, 42.4317], [130.64181, 42.41422], [130.66367, 42.38024], [130.65022, 42.32281], [131.95041, 41.5445], [140.9182, 45.92937], [145.82343, 44.571], [145.23667, 43.76813], [153.94307, 38.42848], [180, 62.52334], [180, 71.53642], [155.31937, 81.93282], [76.13964, 83.37843], [64.18965, 69.94255], [66.1708, 67.61252], [61.98014, 65.72191], [60.74386, 64.95767], [59.63945, 64.78384], [59.80579, 64.13948], [59.24834, 63.01859], [59.61398, 62.44915], [59.36223, 61.3882], [59.50685, 60.91162], [58.3853, 59.487], [59.15636, 59.14682], [59.40376, 58.45822], [58.71104, 58.07475], [58.81412, 57.71602], [58.13789, 57.68097], [58.07604, 57.08308], [57.28024, 56.87898], [57.51527, 56.08729], [59.28419, 56.15739], [59.49035, 55.60486], [58.81825, 55.03378], [57.25137, 55.26262], [57.14829, 54.84204], [57.95234, 54.39672], [59.95217, 54.85853], [59.70487, 54.14846], [58.94336, 53.953], [58.79644, 52.43392], [59.22409, 52.28437], [59.25033, 52.46803], [60.17516, 52.39457], [60.17253, 52.25814], [59.91279, 52.06924], [59.99809, 51.98263]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q34366",
+           nameEn: "Tasmania",
+           aliases: ["AU-TAS"],
+           country: "AU",
+           groups: ["053", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["61"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[123.64533, -39.13605], [159.69067, -56.28945], [159.74028, -39.1978], [123.64533, -39.13605]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q34497",
+           nameEn: "Saint Helena",
+           aliases: ["SH-HL"],
+           country: "GB",
+           groups: ["SH", "BOTS", "011", "202", "002", "UN"],
+           level: "subterritory",
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["290"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-8.3824, -13.9131], [-6.17428, -19.07236], [-3.29308, -15.22647], [-8.3824, -13.9131]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q35657",
+           nameEn: "US States",
+           country: "US",
+           level: "subcountryGroup"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q36117",
+           nameEn: "Borneo",
+           level: "sharedLandform"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q36678",
+           nameEn: "West Bank",
+           country: "PS",
+           groups: ["145", "142"],
+           callingCodes: ["970"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[35.47672, 31.49578], [35.55941, 31.76535], [35.52758, 31.9131], [35.54375, 31.96587], [35.52012, 32.04076], [35.57111, 32.21877], [35.55807, 32.38674], [35.42078, 32.41562], [35.41048, 32.43706], [35.41598, 32.45593], [35.42034, 32.46009], [35.40224, 32.50136], [35.35212, 32.52047], [35.30685, 32.51024], [35.29306, 32.50947], [35.25049, 32.52453], [35.2244, 32.55289], [35.15937, 32.50466], [35.10882, 32.4757], [35.10024, 32.47856], [35.09236, 32.47614], [35.08564, 32.46948], [35.07059, 32.4585], [35.05423, 32.41754], [35.05311, 32.4024], [35.0421, 32.38242], [35.05142, 32.3667], [35.04243, 32.35008], [35.01772, 32.33863], [35.01119, 32.28684], [35.02939, 32.2671], [35.01841, 32.23981], [34.98885, 32.20758], [34.95703, 32.19522], [34.96009, 32.17503], [34.99039, 32.14626], [34.98507, 32.12606], [34.99437, 32.10962], [34.9863, 32.09551], [35.00261, 32.027], [34.98682, 31.96935], [35.00124, 31.93264], [35.03489, 31.92448], [35.03978, 31.89276], [35.03489, 31.85919], [34.99712, 31.85569], [34.9724, 31.83352], [35.01978, 31.82944], [35.05617, 31.85685], [35.07677, 31.85627], [35.14174, 31.81325], [35.18603, 31.80901], [35.18169, 31.82542], [35.19461, 31.82687], [35.21469, 31.81835], [35.216, 31.83894], [35.21128, 31.863], [35.20381, 31.86716], [35.20673, 31.88151], [35.20791, 31.8821], [35.20945, 31.8815], [35.21016, 31.88237], [35.21276, 31.88153], [35.2136, 31.88241], [35.22014, 31.88264], [35.22294, 31.87889], [35.22567, 31.86745], [35.22817, 31.8638], [35.2249, 31.85433], [35.2304, 31.84222], [35.24816, 31.8458], [35.25753, 31.8387], [35.251, 31.83085], [35.26404, 31.82567], [35.25573, 31.81362], [35.26058, 31.79064], [35.25225, 31.7678], [35.26319, 31.74846], [35.25182, 31.73945], [35.24981, 31.72543], [35.2438, 31.7201], [35.24315, 31.71244], [35.23972, 31.70896], [35.22392, 31.71899], [35.21937, 31.71578], [35.20538, 31.72388], [35.18023, 31.72067], [35.16478, 31.73242], [35.15474, 31.73352], [35.15119, 31.73634], [35.13931, 31.73012], [35.12933, 31.7325], [35.11895, 31.71454], [35.10782, 31.71594], [35.08226, 31.69107], [35.00879, 31.65426], [34.95249, 31.59813], [34.9415, 31.55601], [34.94356, 31.50743], [34.93258, 31.47816], [34.89756, 31.43891], [34.87833, 31.39321], [34.88932, 31.37093], [34.92571, 31.34337], [35.02459, 31.35979], [35.13033, 31.3551], [35.22921, 31.37445], [35.39675, 31.49572], [35.47672, 31.49578]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q37362",
+           nameEn: "Akrotiri and Dhekelia",
+           aliases: ["SBA"],
+           country: "GB"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q38095",
+           nameEn: "Gal\xE1pagos Islands",
+           aliases: ["EC-W"],
+           country: "EC",
+           groups: ["005", "419", "019", "UN"],
+           callingCodes: ["593"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-93.12365, 2.64343], [-92.46744, -2.52874], [-87.07749, -0.8849], [-93.12365, 2.64343]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q39760",
+           nameEn: "Gaza Strip",
+           country: "PS",
+           groups: ["145", "142"],
+           callingCodes: ["970"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[34.052, 31.46619], [34.21853, 31.32363], [34.23572, 31.2966], [34.24012, 31.29591], [34.26742, 31.21998], [34.29417, 31.24194], [34.36523, 31.28963], [34.37381, 31.30598], [34.36505, 31.36404], [34.40077, 31.40926], [34.48892, 31.48365], [34.56797, 31.54197], [34.48681, 31.59711], [34.29262, 31.70393], [34.052, 31.46619]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q40888",
+           nameEn: "Andaman and Nicobar Islands",
+           aliases: ["IN-AN"],
+           country: "IN",
+           groups: ["034", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["91"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[94.42132, 5.96581], [94.6371, 13.81803], [86.7822, 13.41052], [94.42132, 5.96581]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q41684",
+           nameEn: "Stewart Island",
+           country: "NZ",
+           groups: ["053", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["64"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[166.59185, -47.61313], [169.70504, -47.56021], [167.52103, -46.41337], [166.59185, -47.61313]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q43296",
+           nameEn: "Wake Island",
+           aliases: ["WK", "WAK", "WKUM", "872", "UM-79"],
+           country: "US",
+           groups: ["UM", "Q1352230", "057", "009", "UN"],
+           level: "subterritory",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["1"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[167.34779, 18.97692], [166.67967, 20.14834], [165.82549, 18.97692], [167.34779, 18.97692]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q46275",
+           nameEn: "New Zealand Subantarctic Islands",
+           country: "NZ",
+           groups: ["Q851132", "053", "009", "UN"],
+           driveSide: "left"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[164.30551, -47.88072], [161.96603, -56.07661], [179.49541, -50.04657], [179.49541, -47.2902], [169.91032, -47.66283], [164.30551, -47.88072]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q46395",
+           nameEn: "British Overseas Territories",
+           aliases: ["BOTS", "UKOTS"],
+           country: "GB",
+           level: "subcountryGroup"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q46772",
+           nameEn: "Kerguelen Islands",
+           country: "FR",
+           groups: ["TF", "Q1451600", "014", "202", "002", "UN"],
+           level: "subterritory"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[61.9216, -49.39746], [70.67507, -51.14192], [74.25129, -45.45074], [61.9216, -49.39746]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q46879",
+           nameEn: "Baker Island",
+           aliases: ["UM-81"],
+           country: "US",
+           groups: ["UM", "Q1352230", "061", "009", "UN"],
+           level: "subterritory",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-175.33482, -1.40631], [-175.31323, 0.5442], [-177.91421, 0.39582], [-175.33482, -1.40631]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q47863",
+           nameEn: "Midway Atoll",
+           aliases: ["MI", "MID", "MIUM", "488", "UM-71"],
+           country: "US",
+           groups: ["UM", "Q1352230", "061", "009", "UN"],
+           level: "subterritory",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["1"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-176.29741, 29.09786], [-177.77531, 29.29793], [-177.5224, 27.7635], [-176.29741, 29.09786]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q62218",
+           nameEn: "Jarvis Island",
+           aliases: ["UM-86"],
+           country: "US",
+           groups: ["UM", "Q1352230", "061", "009", "UN"],
+           level: "subterritory",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-160.42921, -1.4364], [-159.12443, 0.19975], [-160.38779, 0.30331], [-160.42921, -1.4364]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q105472",
+           nameEn: "Macaronesia",
+           level: "sharedLandform"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q114935",
+           nameEn: "Kermadec Islands",
+           country: "NZ",
+           groups: ["Q851132", "053", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["64"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-174.40891, -29.09438], [-180, -24.21376], [-179.96512, -35.00791], [-174.40891, -29.09438]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q115459",
+           nameEn: "Chatham Islands",
+           aliases: ["NZ-CIT"],
+           country: "NZ",
+           groups: ["Q851132", "053", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["64"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-179.93224, -45.18423], [-172.47015, -45.17912], [-176.30998, -41.38382], [-179.93224, -45.18423]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q118863",
+           nameEn: "North Island",
+           country: "NZ",
+           groups: ["053", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["64"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[179.49541, -47.2902], [179.49541, -36.79303], [174.17679, -32.62487], [170.27492, -36.38133], [174.58663, -40.80446], [174.46634, -41.55028], [179.49541, -47.2902]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q120755",
+           nameEn: "South Island",
+           country: "NZ",
+           groups: ["053", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["64"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[169.70504, -47.56021], [179.49541, -47.2902], [174.46634, -41.55028], [174.58663, -40.80446], [170.27492, -36.38133], [166.56976, -39.94841], [164.8365, -46.0205], [167.52103, -46.41337], [169.70504, -47.56021]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q123076",
+           nameEn: "Palmyra Atoll",
+           aliases: ["UM-95"],
+           country: "US",
+           groups: ["UM", "Q1352230", "061", "009", "UN"],
+           level: "subterritory",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["1"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-161.06795, 5.2462], [-161.0731, 7.1291], [-163.24478, 5.24198], [-161.06795, 5.2462]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q130574",
+           nameEn: "Chafarinas Islands",
+           country: "ES",
+           groups: ["EU", "Q191011", "015", "002", "UN"],
+           level: "subterritory"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-2.40316, 35.16893], [-2.43262, 35.20652], [-2.45965, 35.16527], [-2.40316, 35.16893]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q130895",
+           nameEn: "Kingman Reef",
+           aliases: ["UM-89"],
+           country: "US",
+           groups: ["UM", "Q1352230", "061", "009", "UN"],
+           level: "subterritory",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-161.0731, 7.1291], [-163.16627, 7.15036], [-163.24478, 5.24198], [-161.0731, 7.1291]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q131008",
+           nameEn: "Johnston Atoll",
+           aliases: ["JT", "JTN", "JTUM", "396", "UM-67"],
+           country: "US",
+           groups: ["UM", "Q1352230", "061", "009", "UN"],
+           level: "subterritory",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["1"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-170.65691, 16.57199], [-168.87689, 16.01159], [-169.2329, 17.4933], [-170.65691, 16.57199]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q131305",
+           nameEn: "Howland Island",
+           aliases: ["UM-84"],
+           country: "US",
+           groups: ["UM", "Q1352230", "061", "009", "UN"],
+           level: "subterritory",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-177.91421, 0.39582], [-175.31323, 0.5442], [-176.74464, 2.28109], [-177.91421, 0.39582]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q133888",
+           nameEn: "Ashmore and Cartier Islands",
+           country: "AU",
+           groups: ["053", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["61"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[123.7463, -11.1783], [120.6877, -13.59408], [125.29076, -12.33139], [123.7463, -11.1783]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q153732",
+           nameEn: "Mariana Islands",
+           level: "sharedLandform"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q172216",
+           nameEn: "Coral Sea Islands",
+           country: "AU",
+           groups: ["053", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["61"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[159.77159, -28.41151], [156.73836, -14.50464], [145.2855, -9.62524], [147.69992, -17.5933], [152.93188, -20.92631], [154.02855, -24.43238], [159.77159, -28.41151]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q179313",
+           nameEn: "Alderney",
+           country: "GB",
+           groups: ["GG", "830", "Q185086", "154", "150", "UN"],
+           level: "subterritory",
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["44 01481"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-2.36485, 49.48223], [-2.09454, 49.46288], [-2.02963, 49.91866], [-2.49556, 49.79012], [-2.36485, 49.48223]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q185086",
+           nameEn: "Crown Dependencies",
+           country: "GB",
+           level: "subcountryGroup"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q190571",
+           nameEn: "Scattered Islands",
+           country: "FR",
+           groups: ["TF", "Q1451600", "014", "202", "002", "UN"],
+           level: "subterritory"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[53.53458, -16.36909], [54.96649, -16.28353], [54.61476, -15.02273], [53.53458, -16.36909]]], [[[38.55969, -20.75596], [40.68027, -23.38889], [43.52893, -15.62903], [38.55969, -20.75596]]], [[[47.03092, -11.05648], [47.11593, -12.08552], [47.96702, -11.46447], [47.03092, -11.05648]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q191011",
+           nameEn: "Plazas de soberan\xEDa",
+           country: "ES"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q191146",
+           nameEn: "Pe\xF1\xF3n de V\xE9lez de la Gomera",
+           country: "ES",
+           groups: ["EU", "Q191011", "015", "002", "UN"],
+           level: "subterritory"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-4.30191, 35.17419], [-4.30112, 35.17058], [-4.29436, 35.17149], [-4.30191, 35.17419]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q201698",
+           nameEn: "Crozet Islands",
+           country: "FR",
+           groups: ["TF", "Q1451600", "014", "202", "002", "UN"],
+           level: "subterritory"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[55.03425, -43.65017], [46.31615, -46.28749], [54.5587, -47.93013], [55.03425, -43.65017]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q578170",
+           nameEn: "Contiguous United States",
+           aliases: ["CONUS"],
+           country: "US",
+           groups: ["Q35657", "021", "003", "019", "UN"],
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["1"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-97.13927, 25.96583], [-96.92418, 25.97377], [-80.57035, 24.0565], [-78.91214, 27.76553], [-61.98255, 37.34815], [-67.16117, 44.20069], [-66.93432, 44.82597], [-66.96824, 44.83078], [-66.98249, 44.87071], [-66.96824, 44.90965], [-67.0216, 44.95333], [-67.11316, 45.11176], [-67.15965, 45.16179], [-67.19603, 45.16771], [-67.20349, 45.1722], [-67.22751, 45.16344], [-67.27039, 45.1934], [-67.29748, 45.18173], [-67.29754, 45.14865], [-67.34927, 45.122], [-67.48201, 45.27351], [-67.42394, 45.37969], [-67.50578, 45.48971], [-67.42144, 45.50584], [-67.43815, 45.59162], [-67.6049, 45.60725], [-67.80705, 45.69528], [-67.80653, 45.80022], [-67.75654, 45.82324], [-67.80961, 45.87531], [-67.75196, 45.91814], [-67.78111, 45.9392], [-67.78578, 47.06473], [-67.87993, 47.10377], [-67.94843, 47.1925], [-68.23244, 47.35712], [-68.37458, 47.35851], [-68.38332, 47.28723], [-68.57914, 47.28431], [-68.60575, 47.24659], [-68.70125, 47.24399], [-68.89222, 47.1807], [-69.05039, 47.2456], [-69.05073, 47.30076], [-69.05148, 47.42012], [-69.22119, 47.46461], [-69.99966, 46.69543], [-70.05812, 46.41768], [-70.18547, 46.35357], [-70.29078, 46.18832], [-70.23855, 46.1453], [-70.31025, 45.96424], [-70.24694, 45.95138], [-70.25976, 45.89675], [-70.41523, 45.79497], [-70.38934, 45.73215], [-70.54019, 45.67291], [-70.68516, 45.56964], [-70.72651, 45.49771], [-70.62518, 45.42286], [-70.65383, 45.37592], [-70.78372, 45.43269], [-70.82638, 45.39828], [-70.80236, 45.37444], [-70.84816, 45.22698], [-70.89864, 45.2398], [-70.91169, 45.29849], [-70.95193, 45.33895], [-71.0107, 45.34819], [-71.01866, 45.31573], [-71.08364, 45.30623], [-71.14568, 45.24128], [-71.19723, 45.25438], [-71.22338, 45.25184], [-71.29371, 45.29996], [-71.37133, 45.24624], [-71.44252, 45.2361], [-71.40364, 45.21382], [-71.42778, 45.12624], [-71.48735, 45.07784], [-71.50067, 45.01357], [-73.35025, 45.00942], [-74.32699, 44.99029], [-74.66689, 45.00646], [-74.8447, 45.00606], [-74.99101, 44.98051], [-75.01363, 44.95608], [-75.2193, 44.87821], [-75.41441, 44.76614], [-75.76813, 44.51537], [-75.8217, 44.43176], [-75.95947, 44.34463], [-76.00018, 44.34896], [-76.16285, 44.28262], [-76.1664, 44.23051], [-76.244, 44.19643], [-76.31222, 44.19894], [-76.35324, 44.13493], [-76.43859, 44.09393], [-76.79706, 43.63099], [-79.25796, 43.54052], [-79.06921, 43.26183], [-79.05512, 43.25375], [-79.05544, 43.21224], [-79.05002, 43.20133], [-79.05384, 43.17418], [-79.04652, 43.16396], [-79.0427, 43.13934], [-79.06881, 43.12029], [-79.05671, 43.10937], [-79.07486, 43.07845], [-79.01055, 43.06659], [-78.99941, 43.05612], [-79.02424, 43.01983], [-79.02074, 42.98444], [-78.98126, 42.97], [-78.96312, 42.95509], [-78.93224, 42.95229], [-78.90905, 42.93022], [-78.90712, 42.89733], [-78.93684, 42.82887], [-82.67862, 41.67615], [-83.11184, 41.95671], [-83.14962, 42.04089], [-83.12724, 42.2376], [-83.09837, 42.28877], [-83.07837, 42.30978], [-83.02253, 42.33045], [-82.82964, 42.37355], [-82.64242, 42.55594], [-82.58873, 42.54984], [-82.57583, 42.5718], [-82.51858, 42.611], [-82.51063, 42.66025], [-82.46613, 42.76615], [-82.4826, 42.8068], [-82.45331, 42.93139], [-82.4253, 42.95423], [-82.4146, 42.97626], [-82.42469, 42.992], [-82.48419, 45.30225], [-83.59589, 45.82131], [-83.43746, 45.99749], [-83.57017, 46.105], [-83.83329, 46.12169], [-83.90453, 46.05922], [-83.95399, 46.05634], [-84.1096, 46.23987], [-84.09756, 46.25512], [-84.11615, 46.2681], [-84.11254, 46.32329], [-84.13451, 46.39218], [-84.11196, 46.50248], [-84.12885, 46.53068], [-84.17723, 46.52753], [-84.1945, 46.54061], [-84.2264, 46.53337], [-84.26351, 46.49508], [-84.29893, 46.49127], [-84.34174, 46.50683], [-84.42101, 46.49853], [-84.4481, 46.48972], [-84.47607, 46.45225], [-84.55635, 46.45974], [-84.85871, 46.88881], [-88.37033, 48.30586], [-89.48837, 48.01412], [-89.57972, 48.00023], [-89.77248, 48.02607], [-89.89974, 47.98109], [-90.07418, 48.11043], [-90.56312, 48.09488], [-90.56444, 48.12184], [-90.75045, 48.09143], [-90.87588, 48.2484], [-91.08016, 48.18096], [-91.25025, 48.08522], [-91.43248, 48.04912], [-91.45829, 48.07454], [-91.58025, 48.04339], [-91.55649, 48.10611], [-91.70451, 48.11805], [-91.71231, 48.19875], [-91.86125, 48.21278], [-91.98929, 48.25409], [-92.05339, 48.35958], [-92.14732, 48.36578], [-92.202, 48.35252], [-92.26662, 48.35651], [-92.30939, 48.31251], [-92.27167, 48.25046], [-92.37185, 48.22259], [-92.48147, 48.36609], [-92.45588, 48.40624], [-92.50712, 48.44921], [-92.65606, 48.43471], [-92.71323, 48.46081], [-92.69927, 48.49573], [-92.62747, 48.50278], [-92.6342, 48.54133], [-92.7287, 48.54005], [-92.94973, 48.60866], [-93.25391, 48.64266], [-93.33946, 48.62787], [-93.3712, 48.60599], [-93.39758, 48.60364], [-93.40693, 48.60948], [-93.44472, 48.59147], [-93.47022, 48.54357], [-93.66382, 48.51845], [-93.79267, 48.51631], [-93.80939, 48.52439], [-93.80676, 48.58232], [-93.83288, 48.62745], [-93.85769, 48.63284], [-94.23215, 48.65202], [-94.25104, 48.65729], [-94.25172, 48.68404], [-94.27153, 48.70232], [-94.4174, 48.71049], [-94.44258, 48.69223], [-94.53826, 48.70216], [-94.54885, 48.71543], [-94.58903, 48.71803], [-94.69335, 48.77883], [-94.69669, 48.80918], [-94.70486, 48.82365], [-94.70087, 48.8339], [-94.687, 48.84077], [-94.75017, 49.09931], [-94.77355, 49.11998], [-94.82487, 49.29483], [-94.8159, 49.32299], [-94.85381, 49.32492], [-94.95681, 49.37035], [-94.99532, 49.36579], [-95.01419, 49.35647], [-95.05825, 49.35311], [-95.12903, 49.37056], [-95.15357, 49.384], [-95.15355, 48.9996], [-123.32163, 49.00419], [-123.0093, 48.83186], [-123.0093, 48.76586], [-123.26565, 48.6959], [-123.15614, 48.35395], [-123.50039, 48.21223], [-125.03842, 48.53282], [-133.98258, 38.06389], [-118.48109, 32.5991], [-117.1243, 32.53427], [-115.88053, 32.63624], [-114.71871, 32.71894], [-114.76736, 32.64094], [-114.80584, 32.62028], [-114.81141, 32.55543], [-114.79524, 32.55731], [-114.82011, 32.49609], [-111.07523, 31.33232], [-108.20979, 31.33316], [-108.20899, 31.78534], [-106.529, 31.784], [-106.52266, 31.77509], [-106.51251, 31.76922], [-106.50962, 31.76155], [-106.50111, 31.75714], [-106.48815, 31.74769], [-106.47298, 31.75054], [-106.46726, 31.75998], [-106.45244, 31.76523], [-106.43419, 31.75478], [-106.41773, 31.75196], [-106.38003, 31.73151], [-106.3718, 31.71165], [-106.34864, 31.69663], [-106.33419, 31.66303], [-106.30305, 31.62154], [-106.28084, 31.56173], [-106.24612, 31.54193], [-106.23711, 31.51262], [-106.20346, 31.46305], [-106.09025, 31.40569], [-106.00363, 31.39181], [-104.77674, 30.4236], [-104.5171, 29.64671], [-104.3969, 29.57105], [-104.39363, 29.55396], [-104.37752, 29.54255], [-103.15787, 28.93865], [-102.60596, 29.8192], [-101.47277, 29.7744], [-101.05686, 29.44738], [-101.01128, 29.36947], [-100.96725, 29.3477], [-100.94579, 29.34523], [-100.94056, 29.33371], [-100.87982, 29.296], [-100.79696, 29.24688], [-100.67294, 29.09744], [-100.63689, 28.90812], [-100.59809, 28.88197], [-100.52313, 28.75598], [-100.5075, 28.74066], [-100.51222, 28.70679], [-100.50029, 28.66117], [-99.55409, 27.61314], [-99.51478, 27.55836], [-99.52955, 27.49747], [-99.50208, 27.50021], [-99.48045, 27.49016], [-99.482, 27.47128], [-99.49744, 27.43746], [-99.53573, 27.30926], [-99.08477, 26.39849], [-99.03053, 26.41249], [-99.00546, 26.3925], [-98.35126, 26.15129], [-98.30491, 26.10475], [-98.27075, 26.09457], [-98.24603, 26.07191], [-97.97017, 26.05232], [-97.95155, 26.0625], [-97.66511, 26.01708], [-97.52025, 25.88518], [-97.49828, 25.89877], [-97.45669, 25.86874], [-97.42511, 25.83969], [-97.37332, 25.83854], [-97.35946, 25.92189], [-97.13927, 25.96583]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q620634",
+           nameEn: "Bir Tawil",
+           groups: ["015", "002"],
+           level: "territory"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[33.17563, 22.00405], [33.57251, 21.72406], [33.99686, 21.76784], [34.0765, 22.00501], [33.17563, 22.00405]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q639185",
+           nameEn: "Peros Banhos",
+           country: "GB",
+           groups: ["IO", "BOTS", "014", "202", "002", "UN"],
+           level: "subterritory"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[72.12587, -4.02588], [70.1848, -6.37445], [72.09518, -5.61768], [72.12587, -4.02588]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q644636",
+           nameEn: "Cyprus",
+           level: "sharedLandform"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q851132",
+           nameEn: "New Zealand Outlying Islands",
+           country: "NZ",
+           level: "subcountryGroup"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q875134",
+           nameEn: "European Russia",
+           country: "RU",
+           groups: ["151", "150", "UN"],
+           callingCodes: ["7"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[18.57853, 55.25302], [19.64312, 54.45423], [19.8038, 54.44203], [20.63871, 54.3706], [21.41123, 54.32395], [22.79705, 54.36264], [22.7253, 54.41732], [22.70208, 54.45312], [22.67788, 54.532], [22.71293, 54.56454], [22.68021, 54.58486], [22.7522, 54.63525], [22.74225, 54.64339], [22.75467, 54.6483], [22.73397, 54.66604], [22.73631, 54.72952], [22.87317, 54.79492], [22.85083, 54.88711], [22.76422, 54.92521], [22.68723, 54.9811], [22.65451, 54.97037], [22.60075, 55.01863], [22.58907, 55.07085], [22.47688, 55.04408], [22.31562, 55.0655], [22.14267, 55.05345], [22.11697, 55.02131], [22.06087, 55.02935], [22.02582, 55.05078], [22.03984, 55.07888], [21.99543, 55.08691], [21.96505, 55.07353], [21.85521, 55.09493], [21.64954, 55.1791], [21.55605, 55.20311], [21.51095, 55.18507], [21.46766, 55.21115], [21.38446, 55.29348], [21.35465, 55.28427], [21.26425, 55.24456], [20.95181, 55.27994], [20.60454, 55.40986], [18.57853, 55.25302]]], [[[26.32936, 60.00121], [26.90044, 59.63819], [27.85643, 59.58538], [28.04187, 59.47017], [28.19061, 59.39962], [28.21137, 59.38058], [28.20537, 59.36491], [28.19284, 59.35791], [28.14215, 59.28934], [28.00689, 59.28351], [27.90911, 59.24353], [27.87978, 59.18097], [27.80482, 59.1116], [27.74429, 58.98351], [27.36366, 58.78381], [27.55489, 58.39525], [27.48541, 58.22615], [27.62393, 58.09462], [27.67282, 57.92627], [27.81841, 57.89244], [27.78526, 57.83963], [27.56689, 57.83356], [27.50171, 57.78842], [27.52615, 57.72843], [27.3746, 57.66834], [27.40393, 57.62125], [27.31919, 57.57672], [27.34698, 57.52242], [27.56832, 57.53728], [27.52453, 57.42826], [27.86101, 57.29402], [27.66511, 56.83921], [27.86101, 56.88204], [28.04768, 56.59004], [28.13526, 56.57989], [28.10069, 56.524], [28.19057, 56.44637], [28.16599, 56.37806], [28.23716, 56.27588], [28.15217, 56.16964], [28.30571, 56.06035], [28.36888, 56.05805], [28.37987, 56.11399], [28.43068, 56.09407], [28.5529, 56.11705], [28.68337, 56.10173], [28.63668, 56.07262], [28.73418, 55.97131], [29.08299, 56.03427], [29.21717, 55.98971], [29.44692, 55.95978], [29.3604, 55.75862], [29.51283, 55.70294], [29.61446, 55.77716], [29.80672, 55.79569], [29.97975, 55.87281], [30.12136, 55.8358], [30.27776, 55.86819], [30.30987, 55.83592], [30.48257, 55.81066], [30.51346, 55.78982], [30.51037, 55.76568], [30.63344, 55.73079], [30.67464, 55.64176], [30.72957, 55.66268], [30.7845, 55.58514], [30.86003, 55.63169], [30.93419, 55.6185], [30.95204, 55.50667], [30.90123, 55.46621], [30.93144, 55.3914], [30.8257, 55.3313], [30.81946, 55.27931], [30.87944, 55.28223], [30.97369, 55.17134], [31.02071, 55.06167], [31.00972, 55.02783], [30.94243, 55.03964], [30.9081, 55.02232], [30.95754, 54.98609], [30.93144, 54.9585], [30.81759, 54.94064], [30.8264, 54.90062], [30.75165, 54.80699], [30.95479, 54.74346], [30.97127, 54.71967], [31.0262, 54.70698], [30.98226, 54.68872], [30.99187, 54.67046], [31.19339, 54.66947], [31.21399, 54.63113], [31.08543, 54.50361], [31.22945, 54.46585], [31.3177, 54.34067], [31.30791, 54.25315], [31.57002, 54.14535], [31.89599, 54.0837], [31.88744, 54.03653], [31.85019, 53.91801], [31.77028, 53.80015], [31.89137, 53.78099], [32.12621, 53.81586], [32.36663, 53.7166], [32.45717, 53.74039], [32.50112, 53.68594], [32.40499, 53.6656], [32.47777, 53.5548], [32.74968, 53.45597], [32.73257, 53.33494], [32.51725, 53.28431], [32.40773, 53.18856], [32.15368, 53.07594], [31.82373, 53.10042], [31.787, 53.18033], [31.62496, 53.22886], [31.56316, 53.19432], [31.40523, 53.21406], [31.36403, 53.13504], [31.3915, 53.09712], [31.33519, 53.08805], [31.32283, 53.04101], [31.24147, 53.031], [31.35667, 52.97854], [31.592, 52.79011], [31.57277, 52.71613], [31.50406, 52.69707], [31.63869, 52.55361], [31.56316, 52.51518], [31.61397, 52.48843], [31.62084, 52.33849], [31.57971, 52.32146], [31.70735, 52.26711], [31.6895, 52.1973], [31.77877, 52.18636], [31.7822, 52.11406], [31.81722, 52.09955], [31.85018, 52.11305], [31.96141, 52.08015], [31.92159, 52.05144], [32.08813, 52.03319], [32.23331, 52.08085], [32.2777, 52.10266], [32.34044, 52.1434], [32.33083, 52.23685], [32.38988, 52.24946], [32.3528, 52.32842], [32.54781, 52.32423], [32.69475, 52.25535], [32.85405, 52.27888], [32.89937, 52.2461], [33.18913, 52.3754], [33.51323, 52.35779], [33.48027, 52.31499], [33.55718, 52.30324], [33.78789, 52.37204], [34.05239, 52.20132], [34.11199, 52.14087], [34.09413, 52.00835], [34.41136, 51.82793], [34.42922, 51.72852], [34.07765, 51.67065], [34.17599, 51.63253], [34.30562, 51.5205], [34.22048, 51.4187], [34.33446, 51.363], [34.23009, 51.26429], [34.31661, 51.23936], [34.38802, 51.2746], [34.6613, 51.25053], [34.6874, 51.18], [34.82472, 51.17483], [34.97304, 51.2342], [35.14058, 51.23162], [35.12685, 51.16191], [35.20375, 51.04723], [35.31774, 51.08434], [35.40837, 51.04119], [35.32598, 50.94524], [35.39307, 50.92145], [35.41367, 50.80227], [35.47704, 50.77274], [35.48116, 50.66405], [35.39464, 50.64751], [35.47463, 50.49247], [35.58003, 50.45117], [35.61711, 50.35707], [35.73659, 50.35489], [35.80388, 50.41356], [35.8926, 50.43829], [36.06893, 50.45205], [36.20763, 50.3943], [36.30101, 50.29088], [36.47817, 50.31457], [36.58371, 50.28563], [36.56655, 50.2413], [36.64571, 50.218], [36.69377, 50.26982], [36.91762, 50.34963], [37.08468, 50.34935], [37.48204, 50.46079], [37.47243, 50.36277], [37.62486, 50.29966], [37.62879, 50.24481], [37.61113, 50.21976], [37.75807, 50.07896], [37.79515, 50.08425], [37.90776, 50.04194], [38.02999, 49.94482], [38.02999, 49.90592], [38.21675, 49.98104], [38.18517, 50.08161], [38.32524, 50.08866], [38.35408, 50.00664], [38.65688, 49.97176], [38.68677, 50.00904], [38.73311, 49.90238], [38.90477, 49.86787], [38.9391, 49.79524], [39.1808, 49.88911], [39.27968, 49.75976], [39.44496, 49.76067], [39.59142, 49.73758], [39.65047, 49.61761], [39.84548, 49.56064], [40.13249, 49.61672], [40.16683, 49.56865], [40.03636, 49.52321], [40.03087, 49.45452], [40.1141, 49.38798], [40.14912, 49.37681], [40.18331, 49.34996], [40.22176, 49.25683], [40.01988, 49.1761], [39.93437, 49.05709], [39.6836, 49.05121], [39.6683, 48.99454], [39.71353, 48.98959], [39.72649, 48.9754], [39.74874, 48.98675], [39.78368, 48.91596], [39.98967, 48.86901], [40.03636, 48.91957], [40.08168, 48.87443], [39.97182, 48.79398], [39.79466, 48.83739], [39.73104, 48.7325], [39.71765, 48.68673], [39.67226, 48.59368], [39.79764, 48.58668], [39.84548, 48.57821], [39.86196, 48.46633], [39.88794, 48.44226], [39.94847, 48.35055], [39.84136, 48.33321], [39.84273, 48.30947], [39.90041, 48.3049], [39.91465, 48.26743], [39.95248, 48.29972], [39.9693, 48.29904], [39.97325, 48.31399], [39.99241, 48.31768], [40.00752, 48.22445], [39.94847, 48.22811], [39.83724, 48.06501], [39.88256, 48.04482], [39.77544, 48.04206], [39.82213, 47.96396], [39.73935, 47.82876], [38.87979, 47.87719], [38.79628, 47.81109], [38.76379, 47.69346], [38.35062, 47.61631], [38.28679, 47.53552], [38.28954, 47.39255], [38.22225, 47.30788], [38.33074, 47.30508], [38.32112, 47.2585], [38.23049, 47.2324], [38.22955, 47.12069], [38.3384, 46.98085], [38.12112, 46.86078], [37.62608, 46.82615], [35.23066, 45.79231], [35.04991, 45.76827], [36.6645, 45.4514], [36.6545, 45.3417], [36.5049, 45.3136], [36.475, 45.2411], [36.4883, 45.0488], [33.5943, 44.03313], [39.81147, 43.06294], [40.0078, 43.38551], [40.00853, 43.40578], [40.01552, 43.42025], [40.01007, 43.42411], [40.03312, 43.44262], [40.04445, 43.47776], [40.10657, 43.57344], [40.65957, 43.56212], [41.64935, 43.22331], [42.40563, 43.23226], [42.66667, 43.13917], [42.75889, 43.19651], [43.03322, 43.08883], [43.0419, 43.02413], [43.81453, 42.74297], [43.73119, 42.62043], [43.95517, 42.55396], [44.54202, 42.75699], [44.70002, 42.74679], [44.80941, 42.61277], [44.88754, 42.74934], [45.15318, 42.70598], [45.36501, 42.55268], [45.78692, 42.48358], [45.61676, 42.20768], [46.42738, 41.91323], [46.5332, 41.87389], [46.58924, 41.80547], [46.75269, 41.8623], [46.8134, 41.76252], [47.00955, 41.63583], [46.99554, 41.59743], [47.03757, 41.55434], [47.10762, 41.59044], [47.34579, 41.27884], [47.49004, 41.26366], [47.54504, 41.20275], [47.62288, 41.22969], [47.75831, 41.19455], [47.87973, 41.21798], [48.07587, 41.49957], [48.22064, 41.51472], [48.2878, 41.56221], [48.40277, 41.60441], [48.42301, 41.65444], [48.55078, 41.77917], [48.5867, 41.84306], [48.80971, 41.95365], [49.2134, 44.84989], [49.88945, 46.04554], [49.32259, 46.26944], [49.16518, 46.38542], [48.54988, 46.56267], [48.51142, 46.69268], [49.01136, 46.72716], [48.52326, 47.4102], [48.45173, 47.40818], [48.15348, 47.74545], [47.64973, 47.76559], [47.41689, 47.83687], [47.38731, 47.68176], [47.12107, 47.83687], [47.11516, 48.27188], [46.49011, 48.43019], [46.78392, 48.95352], [47.00857, 49.04921], [47.04658, 49.19834], [46.78398, 49.34026], [46.9078, 49.86707], [47.18319, 49.93721], [47.34589, 50.09308], [47.30448, 50.30894], [47.58551, 50.47867], [48.10044, 50.09242], [48.24519, 49.86099], [48.42564, 49.82283], [48.68352, 49.89546], [48.90782, 50.02281], [48.57946, 50.63278], [48.86936, 50.61589], [49.12673, 50.78639], [49.41959, 50.85927], [49.39001, 51.09396], [49.76866, 51.11067], [49.97277, 51.2405], [50.26859, 51.28677], [50.59695, 51.61859], [51.26254, 51.68466], [51.301, 51.48799], [51.77431, 51.49536], [51.8246, 51.67916], [52.36119, 51.74161], [52.54329, 51.48444], [53.46165, 51.49445], [53.69299, 51.23466], [54.12248, 51.11542], [54.46331, 50.85554], [54.41894, 50.61214], [54.55797, 50.52006], [54.71476, 50.61214], [54.56685, 51.01958], [54.72067, 51.03261], [55.67774, 50.54508], [56.11398, 50.7471], [56.17906, 50.93204], [57.17302, 51.11253], [57.44221, 50.88354], [57.74986, 50.93017], [57.75578, 51.13852], [58.3208, 51.15151], [58.87974, 50.70852], [59.48928, 50.64216], [59.51886, 50.49937], [59.81172, 50.54451], [60.01288, 50.8163], [60.17262, 50.83312], [60.31914, 50.67705], [60.81833, 50.6629], [61.4431, 50.80679], [61.56889, 51.23679], [61.6813, 51.25716], [61.55114, 51.32746], [61.50677, 51.40687], [60.95655, 51.48615], [60.92401, 51.61124], [60.5424, 51.61675], [60.36787, 51.66815], [60.50986, 51.7964], [60.09867, 51.87135], [59.99809, 51.98263], [59.91279, 52.06924], [60.17253, 52.25814], [60.17516, 52.39457], [59.25033, 52.46803], [59.22409, 52.28437], [58.79644, 52.43392], [58.94336, 53.953], [59.70487, 54.14846], [59.95217, 54.85853], [57.95234, 54.39672], [57.14829, 54.84204], [57.25137, 55.26262], [58.81825, 55.03378], [59.49035, 55.60486], [59.28419, 56.15739], [57.51527, 56.08729], [57.28024, 56.87898], [58.07604, 57.08308], [58.13789, 57.68097], [58.81412, 57.71602], [58.71104, 58.07475], [59.40376, 58.45822], [59.15636, 59.14682], [58.3853, 59.487], [59.50685, 60.91162], [59.36223, 61.3882], [59.61398, 62.44915], [59.24834, 63.01859], [59.80579, 64.13948], [59.63945, 64.78384], [60.74386, 64.95767], [61.98014, 65.72191], [66.1708, 67.61252], [64.18965, 69.94255], [76.13964, 83.37843], [36.85549, 84.09565], [32.07813, 72.01005], [31.59909, 70.16571], [30.84095, 69.80584], [30.95011, 69.54699], [30.52662, 69.54699], [30.16363, 69.65244], [29.97205, 69.41623], [29.27631, 69.2811], [29.26623, 69.13794], [29.0444, 69.0119], [28.91738, 69.04774], [28.45957, 68.91417], [28.78224, 68.86696], [28.43941, 68.53366], [28.62982, 68.19816], [29.34179, 68.06655], [29.66955, 67.79872], [30.02041, 67.67523], [29.91155, 67.51507], [28.9839, 66.94139], [29.91155, 66.13863], [30.16363, 65.66935], [29.97205, 65.70256], [29.74013, 65.64025], [29.84096, 65.56945], [29.68972, 65.31803], [29.61914, 65.23791], [29.8813, 65.22101], [29.84096, 65.1109], [29.61914, 65.05993], [29.68972, 64.80789], [30.05271, 64.79072], [30.12329, 64.64862], [30.01238, 64.57513], [30.06279, 64.35782], [30.4762, 64.25728], [30.55687, 64.09036], [30.25437, 63.83364], [29.98213, 63.75795], [30.49637, 63.46666], [31.23244, 63.22239], [31.29294, 63.09035], [31.58535, 62.91642], [31.38369, 62.66284], [31.10136, 62.43042], [29.01829, 61.17448], [28.82816, 61.1233], [28.47974, 60.93365], [27.77352, 60.52722], [27.71177, 60.3893], [27.44953, 60.22766], [26.32936, 60.00121]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q1083368",
+           nameEn: "Mainland Finland",
+           country: "FI",
+           groups: ["EU", "154", "150", "UN"],
+           callingCodes: ["358"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[29.12697, 69.69193], [28.36883, 69.81658], [28.32849, 69.88605], [27.97558, 69.99671], [27.95542, 70.0965], [27.57226, 70.06215], [27.05802, 69.92069], [26.64461, 69.96565], [26.40261, 69.91377], [25.96904, 69.68397], [25.69679, 69.27039], [25.75729, 68.99383], [25.61613, 68.89602], [25.42455, 68.90328], [25.12206, 68.78684], [25.10189, 68.63307], [24.93048, 68.61102], [24.90023, 68.55579], [24.74898, 68.65143], [24.18432, 68.73936], [24.02299, 68.81601], [23.781, 68.84514], [23.68017, 68.70276], [23.13064, 68.64684], [22.53321, 68.74393], [22.38367, 68.71561], [22.27276, 68.89514], [21.63833, 69.27485], [21.27827, 69.31281], [21.00732, 69.22755], [20.98641, 69.18809], [21.11099, 69.10291], [21.05775, 69.0356], [20.72171, 69.11874], [20.55258, 69.06069], [20.78802, 69.03087], [20.91658, 68.96764], [20.85104, 68.93142], [20.90649, 68.89696], [21.03001, 68.88969], [22.00429, 68.50692], [22.73028, 68.40881], [23.10336, 68.26551], [23.15377, 68.14759], [23.26469, 68.15134], [23.40081, 68.05545], [23.65793, 67.9497], [23.45627, 67.85297], [23.54701, 67.59306], [23.39577, 67.46974], [23.75372, 67.43688], [23.75372, 67.29914], [23.54701, 67.25435], [23.58735, 67.20752], [23.56214, 67.17038], [23.98563, 66.84149], [23.98059, 66.79585], [23.89488, 66.772], [23.85959, 66.56434], [23.63776, 66.43568], [23.67591, 66.3862], [23.64982, 66.30603], [23.71339, 66.21299], [23.90497, 66.15802], [24.15791, 65.85385], [24.14798, 65.83466], [24.15107, 65.81427], [24.14112, 65.39731], [20.15877, 63.06556], [19.23413, 60.61414], [20.96741, 60.71528], [21.15143, 60.54555], [21.08159, 60.20167], [21.02509, 60.12142], [21.35468, 59.67511], [20.5104, 59.15546], [26.32936, 60.00121], [27.44953, 60.22766], [27.71177, 60.3893], [27.77352, 60.52722], [28.47974, 60.93365], [28.82816, 61.1233], [29.01829, 61.17448], [31.10136, 62.43042], [31.38369, 62.66284], [31.58535, 62.91642], [31.29294, 63.09035], [31.23244, 63.22239], [30.49637, 63.46666], [29.98213, 63.75795], [30.25437, 63.83364], [30.55687, 64.09036], [30.4762, 64.25728], [30.06279, 64.35782], [30.01238, 64.57513], [30.12329, 64.64862], [30.05271, 64.79072], [29.68972, 64.80789], [29.61914, 65.05993], [29.84096, 65.1109], [29.8813, 65.22101], [29.61914, 65.23791], [29.68972, 65.31803], [29.84096, 65.56945], [29.74013, 65.64025], [29.97205, 65.70256], [30.16363, 65.66935], [29.91155, 66.13863], [28.9839, 66.94139], [29.91155, 67.51507], [30.02041, 67.67523], [29.66955, 67.79872], [29.34179, 68.06655], [28.62982, 68.19816], [28.43941, 68.53366], [28.78224, 68.86696], [28.45957, 68.91417], [28.91738, 69.04774], [28.81248, 69.11997], [28.8629, 69.22395], [29.31664, 69.47994], [29.12697, 69.69193]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q1184963",
+           nameEn: "Alhucemas Islands",
+           country: "ES",
+           groups: ["EU", "Q191011", "015", "002", "UN"],
+           level: "subterritory"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-3.90602, 35.21494], [-3.88372, 35.20767], [-3.89343, 35.22728], [-3.90602, 35.21494]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q1298289",
+           nameEn: "Egmont Islands",
+           country: "GB",
+           groups: ["IO", "BOTS", "014", "202", "002", "UN"],
+           level: "subterritory"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[70.1848, -6.37445], [70.67958, -8.2663], [72.17991, -6.68509], [70.1848, -6.37445]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q1352230",
+           nameEn: "US Territories",
+           country: "US",
+           level: "subcountryGroup"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q1451600",
+           nameEn: "Overseas Countries and Territories of the EU",
+           aliases: ["OCT"],
+           level: "subunion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q1544253",
+           nameEn: "Great Chagos Bank",
+           country: "GB",
+           groups: ["IO", "BOTS", "014", "202", "002", "UN"],
+           level: "subterritory"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[70.1848, -6.37445], [72.17991, -6.68509], [73.20573, -5.20727], [70.1848, -6.37445]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q1585511",
+           nameEn: "Salomon Atoll",
+           country: "GB",
+           groups: ["IO", "BOTS", "014", "202", "002", "UN"],
+           level: "subterritory"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[72.09518, -5.61768], [73.20573, -5.20727], [72.12587, -4.02588], [72.09518, -5.61768]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q1681727",
+           nameEn: "Saint-Paul and Amsterdam",
+           country: "FR",
+           groups: ["TF", "Q1451600", "014", "202", "002", "UN"],
+           level: "subterritory"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[76.31747, -42.16264], [80.15867, -36.04977], [71.22311, -38.75287], [76.31747, -42.16264]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q1901211",
+           nameEn: "East Malaysia",
+           country: "MY",
+           groups: ["Q36117", "035", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["60"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[110.90339, 7.52694], [109.82788, 2.86812], [109.62558, 1.99182], [109.53794, 1.91771], [109.57923, 1.80624], [109.66397, 1.79972], [109.66397, 1.60425], [110.35354, 0.98869], [110.49182, 0.88088], [110.62374, 0.873], [111.22979, 1.08326], [111.55434, 0.97864], [111.82846, 0.99349], [111.94553, 1.12016], [112.15679, 1.17004], [112.2127, 1.44135], [112.48648, 1.56516], [113.021, 1.57819], [113.01448, 1.42832], [113.64677, 1.23933], [114.03788, 1.44787], [114.57892, 1.5], [114.80706, 1.92351], [114.80706, 2.21665], [115.1721, 2.49671], [115.11343, 2.82879], [115.53713, 3.14776], [115.58276, 3.93499], [115.90217, 4.37708], [117.25801, 4.35108], [117.47313, 4.18857], [117.67641, 4.16535], [118.06469, 4.16638], [118.93936, 4.09009], [119.52945, 5.35672], [117.98544, 6.27477], [117.93857, 6.89845], [117.17735, 7.52841], [116.79524, 7.43869], [115.02521, 5.35005], [115.16236, 5.01011], [115.15092, 4.87604], [115.20737, 4.8256], [115.27819, 4.63661], [115.2851, 4.42295], [115.36346, 4.33563], [115.31275, 4.30806], [115.09978, 4.39123], [115.07737, 4.53418], [115.04064, 4.63706], [115.02278, 4.74137], [115.02955, 4.82087], [115.05038, 4.90275], [114.99417, 4.88201], [114.96982, 4.81146], [114.88841, 4.81905], [114.8266, 4.75062], [114.77303, 4.72871], [114.83189, 4.42387], [114.88039, 4.4257], [114.78539, 4.12205], [114.64211, 4.00694], [114.49922, 4.13108], [114.4416, 4.27588], [114.32176, 4.2552], [114.32176, 4.34942], [114.26876, 4.49878], [114.15813, 4.57], [114.07448, 4.58441], [114.10166, 4.76112], [110.90339, 7.52694]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q1973345",
+           nameEn: "Peninsular Malaysia",
+           country: "MY",
+           groups: ["035", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["60"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[102.46318, 7.22462], [102.09086, 6.23546], [102.08127, 6.22679], [102.07732, 6.193], [102.09182, 6.14161], [102.01835, 6.05407], [101.99209, 6.04075], [101.97114, 6.01992], [101.9714, 6.00575], [101.94712, 5.98421], [101.92819, 5.85511], [101.91776, 5.84269], [101.89188, 5.8386], [101.80144, 5.74505], [101.75074, 5.79091], [101.69773, 5.75881], [101.58019, 5.93534], [101.25524, 5.78633], [101.25755, 5.71065], [101.14062, 5.61613], [100.98815, 5.79464], [101.02708, 5.91013], [101.087, 5.9193], [101.12388, 6.11411], [101.06165, 6.14161], [101.12618, 6.19431], [101.10313, 6.25617], [100.85884, 6.24929], [100.81045, 6.45086], [100.74822, 6.46231], [100.74361, 6.50811], [100.66986, 6.45086], [100.43027, 6.52389], [100.42351, 6.51762], [100.41791, 6.5189], [100.41152, 6.52299], [100.35413, 6.54932], [100.31929, 6.65413], [100.32607, 6.65933], [100.32671, 6.66526], [100.31884, 6.66423], [100.31618, 6.66781], [100.30828, 6.66462], [100.29651, 6.68439], [100.19511, 6.72559], [100.12, 6.42105], [100.0756, 6.4045], [99.91873, 6.50233], [99.50117, 6.44501], [99.31854, 5.99868], [99.75778, 3.86466], [103.03657, 1.30383], [103.56591, 1.19719], [103.62738, 1.35255], [103.67468, 1.43166], [103.7219, 1.46108], [103.74161, 1.4502], [103.76395, 1.45183], [103.81181, 1.47953], [103.86383, 1.46288], [103.89565, 1.42841], [103.93384, 1.42926], [104.00131, 1.42405], [104.02277, 1.4438], [104.04622, 1.44691], [104.07348, 1.43322], [104.08871, 1.42015], [104.09162, 1.39694], [104.08072, 1.35998], [104.12282, 1.27714], [104.34728, 1.33529], [104.56723, 1.44271], [105.01437, 3.24936], [102.46318, 7.22462]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q2093907",
+           nameEn: "Three Kings Islands",
+           country: "NZ",
+           groups: ["Q851132", "053", "009", "UN"],
+           driveSide: "left"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[174.17679, -32.62487], [170.93268, -32.97889], [171.97383, -34.64644], [174.17679, -32.62487]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q2298216",
+           nameEn: "Solander Islands",
+           country: "NZ",
+           groups: ["Q851132", "053", "009", "UN"],
+           driveSide: "left"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[167.39068, -46.49187], [166.5534, -46.39484], [166.84561, -46.84889], [167.39068, -46.49187]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q2872203",
+           nameEn: "Mainland Australia",
+           country: "AU",
+           groups: ["053", "009", "UN"],
+           level: "subcountryGroup",
+           driveSide: "left",
+           callingCodes: ["61"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[88.16419, -23.49578], [123.64533, -39.13605], [159.74028, -39.1978], [159.76765, -29.76946], [154.02855, -24.43238], [152.93188, -20.92631], [147.69992, -17.5933], [145.2855, -9.62524], [143.87386, -9.02382], [143.29772, -9.33993], [142.48658, -9.36754], [142.19246, -9.15378], [141.88934, -9.36111], [141.01842, -9.35091], [135.49042, -9.2276], [127.55165, -9.05052], [125.29076, -12.33139], [88.16419, -23.49578]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q2914565",
+           nameEn: "Autonomous Regions of Portugal",
+           country: "PT",
+           level: "subcountryGroup"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q2915956",
+           nameEn: "Mainland Portugal",
+           country: "PT",
+           groups: ["Q12837", "EU", "039", "150", "UN"],
+           level: "subcountryGroup",
+           callingCodes: ["351"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-10.39881, 36.12218], [-7.37282, 36.96896], [-7.39769, 37.16868], [-7.41133, 37.20314], [-7.41854, 37.23813], [-7.43227, 37.25152], [-7.43974, 37.38913], [-7.46878, 37.47127], [-7.51759, 37.56119], [-7.41981, 37.75729], [-7.33441, 37.81193], [-7.27314, 37.90145], [-7.24544, 37.98884], [-7.12648, 38.00296], [-7.10366, 38.04404], [-7.05966, 38.01966], [-7.00375, 38.01914], [-6.93418, 38.21454], [-7.09389, 38.17227], [-7.15581, 38.27597], [-7.32529, 38.44336], [-7.265, 38.61674], [-7.26174, 38.72107], [-7.03848, 38.87221], [-7.051, 38.907], [-6.95211, 39.0243], [-6.97004, 39.07619], [-7.04011, 39.11919], [-7.10692, 39.10275], [-7.14929, 39.11287], [-7.12811, 39.17101], [-7.23566, 39.20132], [-7.23403, 39.27579], [-7.3149, 39.34857], [-7.2927, 39.45847], [-7.49477, 39.58794], [-7.54121, 39.66717], [-7.33507, 39.64569], [-7.24707, 39.66576], [-7.01613, 39.66877], [-6.97492, 39.81488], [-6.91463, 39.86618], [-6.86737, 40.01986], [-6.94233, 40.10716], [-7.00589, 40.12087], [-7.02544, 40.18564], [-7.00426, 40.23169], [-6.86085, 40.26776], [-6.86085, 40.2976], [-6.80218, 40.33239], [-6.78426, 40.36468], [-6.84618, 40.42177], [-6.84944, 40.46394], [-6.7973, 40.51723], [-6.80218, 40.55067], [-6.84292, 40.56801], [-6.79567, 40.65955], [-6.82826, 40.74603], [-6.82337, 40.84472], [-6.79892, 40.84842], [-6.80707, 40.88047], [-6.84292, 40.89771], [-6.8527, 40.93958], [-6.9357, 41.02888], [-6.913, 41.03922], [-6.88843, 41.03027], [-6.84781, 41.02692], [-6.80942, 41.03629], [-6.79241, 41.05397], [-6.75655, 41.10187], [-6.77319, 41.13049], [-6.69711, 41.1858], [-6.68286, 41.21641], [-6.65046, 41.24725], [-6.55937, 41.24417], [-6.38551, 41.35274], [-6.38553, 41.38655], [-6.3306, 41.37677], [-6.26777, 41.48796], [-6.19128, 41.57638], [-6.29863, 41.66432], [-6.44204, 41.68258], [-6.49907, 41.65823], [-6.54633, 41.68623], [-6.56426, 41.74219], [-6.51374, 41.8758], [-6.56752, 41.88429], [-6.5447, 41.94371], [-6.58544, 41.96674], [-6.61967, 41.94008], [-6.75004, 41.94129], [-6.76959, 41.98734], [-6.81196, 41.99097], [-6.82174, 41.94493], [-6.94396, 41.94403], [-6.95537, 41.96553], [-6.98144, 41.9728], [-7.01078, 41.94977], [-7.07596, 41.94977], [-7.08574, 41.97401], [-7.14115, 41.98855], [-7.18549, 41.97515], [-7.18677, 41.88793], [-7.32366, 41.8406], [-7.37092, 41.85031], [-7.42864, 41.80589], [-7.42854, 41.83262], [-7.44759, 41.84451], [-7.45566, 41.86488], [-7.49803, 41.87095], [-7.52737, 41.83939], [-7.62188, 41.83089], [-7.58603, 41.87944], [-7.65774, 41.88308], [-7.69848, 41.90977], [-7.84188, 41.88065], [-7.88055, 41.84571], [-7.88751, 41.92553], [-7.90707, 41.92432], [-7.92336, 41.8758], [-7.9804, 41.87337], [-8.01136, 41.83453], [-8.0961, 41.81024], [-8.16455, 41.81753], [-8.16944, 41.87944], [-8.19551, 41.87459], [-8.2185, 41.91237], [-8.16232, 41.9828], [-8.08796, 42.01398], [-8.08847, 42.05767], [-8.11729, 42.08537], [-8.18178, 42.06436], [-8.19406, 42.12141], [-8.18947, 42.13853], [-8.1986, 42.15402], [-8.22406, 42.1328], [-8.24681, 42.13993], [-8.2732, 42.12396], [-8.29809, 42.106], [-8.32161, 42.10218], [-8.33912, 42.08358], [-8.36353, 42.09065], [-8.38323, 42.07683], [-8.40143, 42.08052], [-8.42512, 42.07199], [-8.44123, 42.08218], [-8.48185, 42.0811], [-8.52837, 42.07658], [-8.5252, 42.06264], [-8.54563, 42.0537], [-8.58086, 42.05147], [-8.59493, 42.05708], [-8.63791, 42.04691], [-8.64626, 42.03668], [-8.65832, 42.02972], [-8.6681, 41.99703], [-8.69071, 41.98862], [-8.7478, 41.96282], [-8.74606, 41.9469], [-8.75712, 41.92833], [-8.81794, 41.90375], [-8.87157, 41.86488], [-11.19304, 41.83075], [-10.39881, 36.12218]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q3311985",
+           nameEn: "Guernsey",
+           country: "GB",
+           groups: ["GG", "830", "Q185086", "154", "150", "UN"],
+           level: "subterritory",
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["44 01481"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-2.49556, 49.79012], [-3.28154, 49.57329], [-2.65349, 49.15373], [-2.36485, 49.48223], [-2.49556, 49.79012]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q3320166",
+           nameEn: "Outermost Regions of the EU",
+           aliases: ["OMR"],
+           level: "subunion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q3336843",
+           nameEn: "Countries of the United Kingdom",
+           country: "GB",
+           level: "subcountryGroup"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q6736667",
+           nameEn: "Mainland India",
+           country: "IN",
+           groups: ["034", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["91"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[89.08044, 21.41871], [89.07114, 22.15335], [88.9367, 22.58527], [88.94614, 22.66941], [88.9151, 22.75228], [88.96713, 22.83346], [88.87063, 22.95235], [88.88327, 23.03885], [88.86377, 23.08759], [88.99148, 23.21134], [88.71133, 23.2492], [88.79254, 23.46028], [88.79351, 23.50535], [88.74841, 23.47361], [88.56507, 23.64044], [88.58087, 23.87105], [88.66189, 23.87607], [88.73743, 23.91751], [88.6976, 24.14703], [88.74841, 24.1959], [88.68801, 24.31464], [88.50934, 24.32474], [88.12296, 24.51301], [88.08786, 24.63232], [88.00683, 24.66477], [88.15515, 24.85806], [88.14004, 24.93529], [88.21832, 24.96642], [88.27325, 24.88796], [88.33917, 24.86803], [88.46277, 25.07468], [88.44766, 25.20149], [88.94067, 25.18534], [89.00463, 25.26583], [89.01105, 25.30303], [88.85278, 25.34679], [88.81296, 25.51546], [88.677, 25.46959], [88.4559, 25.59227], [88.45103, 25.66245], [88.242, 25.80811], [88.13138, 25.78773], [88.08804, 25.91334], [88.16581, 26.0238], [88.1844, 26.14417], [88.34757, 26.22216], [88.35153, 26.29123], [88.51649, 26.35923], [88.48749, 26.45855], [88.36938, 26.48683], [88.35153, 26.45241], [88.33093, 26.48929], [88.41196, 26.63837], [88.4298, 26.54489], [88.62144, 26.46783], [88.69485, 26.38353], [88.67837, 26.26291], [88.78961, 26.31093], [88.85004, 26.23211], [89.05328, 26.2469], [88.91321, 26.37984], [88.92357, 26.40711], [88.95612, 26.4564], [89.08899, 26.38845], [89.15869, 26.13708], [89.35953, 26.0077], [89.53515, 26.00382], [89.57101, 25.9682], [89.63968, 26.22595], [89.70201, 26.15138], [89.73581, 26.15818], [89.77865, 26.08387], [89.77728, 26.04254], [89.86592, 25.93115], [89.80585, 25.82489], [89.84388, 25.70042], [89.86129, 25.61714], [89.81208, 25.37244], [89.84086, 25.31854], [89.83371, 25.29548], [89.87629, 25.28337], [89.90478, 25.31038], [90.1155, 25.22686], [90.40034, 25.1534], [90.65042, 25.17788], [90.87427, 25.15799], [91.25517, 25.20677], [91.63648, 25.12846], [92.0316, 25.1834], [92.33957, 25.07593], [92.39147, 25.01471], [92.49887, 24.88796], [92.38626, 24.86055], [92.25854, 24.9191], [92.15796, 24.54435], [92.11662, 24.38997], [91.96603, 24.3799], [91.89258, 24.14674], [91.82596, 24.22345], [91.76004, 24.23848], [91.73257, 24.14703], [91.65292, 24.22095], [91.63782, 24.1132], [91.55542, 24.08687], [91.37414, 24.10693], [91.35741, 23.99072], [91.29587, 24.0041], [91.22308, 23.89616], [91.25192, 23.83463], [91.15579, 23.6599], [91.28293, 23.37538], [91.36453, 23.06612], [91.40848, 23.07117], [91.4035, 23.27522], [91.46615, 23.2328], [91.54993, 23.01051], [91.61571, 22.93929], [91.7324, 23.00043], [91.81634, 23.08001], [91.76417, 23.26619], [91.84789, 23.42235], [91.95642, 23.47361], [91.95093, 23.73284], [92.04706, 23.64229], [92.15417, 23.73409], [92.26541, 23.70392], [92.38214, 23.28705], [92.37665, 22.9435], [92.5181, 22.71441], [92.60029, 22.1522], [92.56616, 22.13554], [92.60949, 21.97638], [92.67532, 22.03547], [92.70416, 22.16017], [92.86208, 22.05456], [92.89504, 21.95143], [92.93899, 22.02656], [92.99804, 21.98964], [92.99255, 22.05965], [93.04885, 22.20595], [93.15734, 22.18687], [93.14224, 22.24535], [93.19991, 22.25425], [93.18206, 22.43716], [93.13537, 22.45873], [93.11477, 22.54374], [93.134, 22.59573], [93.09417, 22.69459], [93.134, 22.92498], [93.12988, 23.05772], [93.2878, 23.00464], [93.38478, 23.13698], [93.36862, 23.35426], [93.38781, 23.36139], [93.39981, 23.38828], [93.38805, 23.4728], [93.43475, 23.68299], [93.3908, 23.7622], [93.3908, 23.92925], [93.36059, 23.93176], [93.32351, 24.04468], [93.34735, 24.10151], [93.41415, 24.07854], [93.46633, 23.97067], [93.50616, 23.94432], [93.62871, 24.00922], [93.75952, 24.0003], [93.80279, 23.92549], [93.92089, 23.95812], [94.14081, 23.83333], [94.30215, 24.23752], [94.32362, 24.27692], [94.45279, 24.56656], [94.50729, 24.59281], [94.5526, 24.70764], [94.60204, 24.70889], [94.73937, 25.00545], [94.74212, 25.13606], [94.57458, 25.20318], [94.68032, 25.47003], [94.80117, 25.49359], [95.18556, 26.07338], [95.11428, 26.1019], [95.12801, 26.38397], [95.05798, 26.45408], [95.23513, 26.68499], [95.30339, 26.65372], [95.437, 26.7083], [95.81603, 27.01335], [95.93002, 27.04149], [96.04949, 27.19428], [96.15591, 27.24572], [96.40779, 27.29818], [96.55761, 27.29928], [96.73888, 27.36638], [96.88445, 27.25046], [96.85287, 27.2065], [96.89132, 27.17474], [97.14675, 27.09041], [97.17422, 27.14052], [96.91431, 27.45752], [96.90112, 27.62149], [97.29919, 27.92233], [97.35824, 27.87256], [97.38845, 28.01329], [97.35412, 28.06663], [97.31292, 28.06784], [97.34547, 28.21385], [97.1289, 28.3619], [96.98882, 28.32564], [96.88445, 28.39452], [96.85561, 28.4875], [96.6455, 28.61657], [96.48895, 28.42955], [96.40929, 28.51526], [96.61391, 28.72742], [96.3626, 29.10607], [96.20467, 29.02325], [96.18682, 29.11087], [96.31316, 29.18643], [96.05361, 29.38167], [95.84899, 29.31464], [95.75149, 29.32063], [95.72086, 29.20797], [95.50842, 29.13487], [95.41091, 29.13007], [95.3038, 29.13847], [95.26122, 29.07727], [95.2214, 29.10727], [95.11291, 29.09527], [95.0978, 29.14446], [94.81353, 29.17804], [94.69318, 29.31739], [94.2752, 29.11687], [94.35897, 29.01965], [93.72797, 28.68821], [93.44621, 28.67189], [93.18069, 28.50319], [93.14635, 28.37035], [92.93075, 28.25671], [92.67486, 28.15018], [92.65472, 28.07632], [92.73025, 28.05814], [92.7275, 27.98662], [92.42538, 27.80092], [92.32101, 27.79363], [92.27432, 27.89077], [91.87057, 27.7195], [91.84722, 27.76325], [91.6469, 27.76358], [91.55819, 27.6144], [91.65007, 27.48287], [92.01132, 27.47352], [92.12019, 27.27829], [92.04702, 27.26861], [92.03457, 27.07334], [92.11863, 26.893], [92.05523, 26.8692], [91.83181, 26.87318], [91.50067, 26.79223], [90.67715, 26.77215], [90.48504, 26.8594], [90.39271, 26.90704], [90.30402, 26.85098], [90.04535, 26.72422], [89.86124, 26.73307], [89.63369, 26.74402], [89.42349, 26.83727], [89.3901, 26.84225], [89.38319, 26.85963], [89.37913, 26.86224], [89.1926, 26.81329], [89.12825, 26.81661], [89.09554, 26.89089], [88.95807, 26.92668], [88.92301, 26.99286], [88.8714, 26.97488], [88.86984, 27.10937], [88.74219, 27.144], [88.91901, 27.32483], [88.82981, 27.38814], [88.77517, 27.45415], [88.88091, 27.85192], [88.83559, 28.01936], [88.63235, 28.12356], [88.54858, 28.06057], [88.25332, 27.9478], [88.1278, 27.95417], [88.13378, 27.88015], [88.1973, 27.85067], [88.19107, 27.79285], [88.04008, 27.49223], [88.07277, 27.43007], [88.01646, 27.21612], [88.01587, 27.21388], [87.9887, 27.11045], [88.11719, 26.98758], [88.13422, 26.98705], [88.12302, 26.95324], [88.19107, 26.75516], [88.1659, 26.68177], [88.16452, 26.64111], [88.09963, 26.54195], [88.09414, 26.43732], [88.00895, 26.36029], [87.90115, 26.44923], [87.89085, 26.48565], [87.84193, 26.43663], [87.7918, 26.46737], [87.76004, 26.40711], [87.67893, 26.43501], [87.66803, 26.40294], [87.59175, 26.38342], [87.55274, 26.40596], [87.51571, 26.43106], [87.46566, 26.44058], [87.37314, 26.40815], [87.34568, 26.34787], [87.26568, 26.37294], [87.26587, 26.40592], [87.24682, 26.4143], [87.18863, 26.40558], [87.14751, 26.40542], [87.09147, 26.45039], [87.0707, 26.58571], [87.04691, 26.58685], [87.01559, 26.53228], [86.95912, 26.52076], [86.94543, 26.52076], [86.82898, 26.43919], [86.76797, 26.45892], [86.74025, 26.42386], [86.69124, 26.45169], [86.62686, 26.46891], [86.61313, 26.48658], [86.57073, 26.49825], [86.54258, 26.53819], [86.49726, 26.54218], [86.31564, 26.61925], [86.26235, 26.61886], [86.22513, 26.58863], [86.13596, 26.60651], [86.02729, 26.66756], [85.8492, 26.56667], [85.85126, 26.60866], [85.83126, 26.61134], [85.76907, 26.63076], [85.72315, 26.67471], [85.73483, 26.79613], [85.66239, 26.84822], [85.61621, 26.86721], [85.59461, 26.85161], [85.5757, 26.85955], [85.56471, 26.84133], [85.47752, 26.79292], [85.34302, 26.74954], [85.21159, 26.75933], [85.18046, 26.80519], [85.19291, 26.86909], [85.15883, 26.86966], [85.02635, 26.85381], [85.05592, 26.88991], [85.00536, 26.89523], [84.97186, 26.9149], [84.96687, 26.95599], [84.85754, 26.98984], [84.82913, 27.01989], [84.793, 26.9968], [84.64496, 27.04669], [84.69166, 27.21294], [84.62161, 27.33885], [84.29315, 27.39], [84.25735, 27.44941], [84.21376, 27.45218], [84.10791, 27.52399], [84.02229, 27.43836], [83.93306, 27.44939], [83.86182, 27.4241], [83.85595, 27.35797], [83.61288, 27.47013], [83.39495, 27.4798], [83.38872, 27.39276], [83.35136, 27.33885], [83.29999, 27.32778], [83.2673, 27.36235], [83.27197, 27.38309], [83.19413, 27.45632], [82.94938, 27.46036], [82.93261, 27.50328], [82.74119, 27.49838], [82.70378, 27.72122], [82.46405, 27.6716], [82.06554, 27.92222], [81.97214, 27.93322], [81.91223, 27.84995], [81.47867, 28.08303], [81.48179, 28.12148], [81.38683, 28.17638], [81.32923, 28.13521], [81.19847, 28.36284], [81.03471, 28.40054], [80.55142, 28.69182], [80.50575, 28.6706], [80.52443, 28.54897], [80.44504, 28.63098], [80.37188, 28.63371], [80.12125, 28.82346], [80.06957, 28.82763], [80.05743, 28.91479], [80.18085, 29.13649], [80.23178, 29.11626], [80.26602, 29.13938], [80.24112, 29.21414], [80.28626, 29.20327], [80.31428, 29.30784], [80.24322, 29.44299], [80.37939, 29.57098], [80.41858, 29.63581], [80.38428, 29.68513], [80.36803, 29.73865], [80.41554, 29.79451], [80.43458, 29.80466], [80.48997, 29.79566], [80.56247, 29.86661], [80.57179, 29.91422], [80.60226, 29.95732], [80.67076, 29.95732], [80.8778, 30.13384], [80.86673, 30.17321], [80.91143, 30.22173], [80.92547, 30.17193], [81.03953, 30.20059], [80.83343, 30.32023], [80.54504, 30.44936], [80.20721, 30.58541], [79.93255, 30.88288], [79.59884, 30.93943], [79.30694, 31.17357], [79.14016, 31.43403], [79.01931, 31.42817], [78.89344, 31.30481], [78.77898, 31.31209], [78.71032, 31.50197], [78.84516, 31.60631], [78.69933, 31.78723], [78.78036, 31.99478], [78.74404, 32.00384], [78.68754, 32.10256], [78.49609, 32.2762], [78.4645, 32.45367], [78.38897, 32.53938], [78.73916, 32.69438], [78.7831, 32.46873], [78.96713, 32.33655], [78.99322, 32.37948], [79.0979, 32.38051], [79.13174, 32.47766], [79.26768, 32.53277], [79.46562, 32.69668], [79.14016, 33.02545], [79.15252, 33.17156], [78.73636, 33.56521], [78.67599, 33.66445], [78.77349, 33.73871], [78.73367, 34.01121], [78.65657, 34.03195], [78.66225, 34.08858], [78.91769, 34.15452], [78.99802, 34.3027], [79.05364, 34.32482], [78.74465, 34.45174], [78.56475, 34.50835], [78.54964, 34.57283], [78.27781, 34.61484], [78.18435, 34.7998], [78.22692, 34.88771], [78.00033, 35.23954], [78.03466, 35.3785], [78.11664, 35.48022], [77.80532, 35.52058], [77.70232, 35.46244], [77.44277, 35.46132], [76.96624, 35.5932], [76.84539, 35.67356], [76.77323, 35.66062], [76.75475, 35.52617], [76.85088, 35.39754], [76.93465, 35.39866], [77.11796, 35.05419], [76.99251, 34.93349], [76.87193, 34.96906], [76.74514, 34.92488], [76.74377, 34.84039], [76.67648, 34.76371], [76.47186, 34.78965], [76.15463, 34.6429], [76.04614, 34.67566], [75.75438, 34.51827], [75.38009, 34.55021], [75.01479, 34.64629], [74.6663, 34.703], [74.58083, 34.77386], [74.31239, 34.79626], [74.12897, 34.70073], [73.96423, 34.68244], [73.93401, 34.63386], [73.93951, 34.57169], [73.89419, 34.54568], [73.88732, 34.48911], [73.74999, 34.3781], [73.74862, 34.34183], [73.8475, 34.32935], [73.90517, 34.35317], [73.98208, 34.2522], [73.90677, 34.10504], [73.88732, 34.05105], [73.91341, 34.01235], [74.21554, 34.03853], [74.25262, 34.01577], [74.26086, 33.92237], [74.14001, 33.83002], [74.05898, 33.82089], [74.00891, 33.75437], [73.96423, 33.73071], [73.98968, 33.66155], [73.97367, 33.64061], [74.03576, 33.56718], [74.10115, 33.56392], [74.18121, 33.4745], [74.17983, 33.3679], [74.08782, 33.26232], [74.01366, 33.25199], [74.02144, 33.18908], [74.15374, 33.13477], [74.17571, 33.07495], [74.31854, 33.02891], [74.34875, 32.97823], [74.31227, 32.92795], [74.41467, 32.90563], [74.45312, 32.77755], [74.6289, 32.75561], [74.64675, 32.82604], [74.7113, 32.84219], [74.65345, 32.71225], [74.69542, 32.66792], [74.64424, 32.60985], [74.65251, 32.56416], [74.67431, 32.56676], [74.68362, 32.49298], [74.84725, 32.49075], [74.97634, 32.45367], [75.03265, 32.49538], [75.28259, 32.36556], [75.38046, 32.26836], [75.25649, 32.10187], [75.00793, 32.03786], [74.9269, 32.0658], [74.86236, 32.04485], [74.79919, 31.95983], [74.58907, 31.87824], [74.47771, 31.72227], [74.57498, 31.60382], [74.61517, 31.55698], [74.59319, 31.50197], [74.64713, 31.45605], [74.59773, 31.4136], [74.53223, 31.30321], [74.51629, 31.13829], [74.56023, 31.08303], [74.60281, 31.10419], [74.60006, 31.13711], [74.6852, 31.12771], [74.67971, 31.05479], [74.5616, 31.04153], [73.88993, 30.36305], [73.95736, 30.28466], [73.97225, 30.19829], [73.80299, 30.06969], [73.58665, 30.01848], [73.3962, 29.94707], [73.28094, 29.56646], [73.05886, 29.1878], [73.01337, 29.16422], [72.94272, 29.02487], [72.40402, 28.78283], [72.29495, 28.66367], [72.20329, 28.3869], [71.9244, 28.11555], [71.89921, 27.96035], [70.79054, 27.68423], [70.60927, 28.02178], [70.37307, 28.01208], [70.12502, 27.8057], [70.03136, 27.56627], [69.58519, 27.18109], [69.50904, 26.74892], [69.88555, 26.56836], [70.05584, 26.60398], [70.17532, 26.55362], [70.17532, 26.24118], [70.08193, 26.08094], [70.0985, 25.93238], [70.2687, 25.71156], [70.37444, 25.67443], [70.53649, 25.68928], [70.60378, 25.71898], [70.67382, 25.68186], [70.66695, 25.39314], [70.89148, 25.15064], [70.94002, 24.92843], [71.09405, 24.69017], [70.97594, 24.60904], [71.00341, 24.46038], [71.12838, 24.42662], [71.04461, 24.34657], [70.94985, 24.3791], [70.85784, 24.30903], [70.88393, 24.27398], [70.71502, 24.23517], [70.57906, 24.27774], [70.5667, 24.43787], [70.11712, 24.30915], [70.03428, 24.172], [69.73335, 24.17007], [69.59579, 24.29777], [69.29778, 24.28712], [69.19341, 24.25646], [69.07806, 24.29777], [68.97781, 24.26021], [68.90914, 24.33156], [68.7416, 24.31904], [68.74643, 23.97027], [68.39339, 23.96838], [68.20763, 23.85849], [68.11329, 23.53945], [76.59015, 5.591], [79.50447, 8.91876], [79.42124, 9.80115], [80.48418, 10.20786], [89.08044, 21.41871]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q9143535",
+           nameEn: "Akrotiri",
+           country: "GB",
+           groups: ["Q644636", "Q37362", "BOTS", "145", "142", "UN"],
+           level: "subterritory",
+           driveSide: "left",
+           callingCodes: ["357"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[32.86014, 34.70585], [32.82717, 34.70622], [32.79433, 34.67883], [32.76136, 34.68318], [32.75515, 34.64985], [32.74412, 34.43926], [33.26744, 34.49942], [33.0138, 34.64424], [32.96968, 34.64046], [32.96718, 34.63446], [32.95891, 34.62919], [32.95323, 34.64075], [32.95471, 34.64528], [32.94976, 34.65204], [32.94796, 34.6587], [32.95325, 34.66462], [32.97079, 34.66112], [32.97736, 34.65277], [32.99014, 34.65518], [32.98668, 34.67268], [32.99135, 34.68061], [32.95539, 34.68471], [32.94683, 34.67907], [32.94379, 34.67111], [32.93693, 34.67027], [32.93449, 34.66241], [32.92807, 34.66736], [32.93043, 34.67091], [32.91398, 34.67343], [32.9068, 34.66102], [32.86167, 34.68734], [32.86014, 34.70585]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q9206745",
+           nameEn: "Dhekelia",
+           country: "GB",
+           groups: ["Q644636", "Q37362", "BOTS", "145", "142", "UN"],
+           level: "subterritory",
+           driveSide: "left",
+           callingCodes: ["357"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[33.70575, 34.97947], [33.83531, 34.73974], [33.98684, 34.76642], [33.90075, 34.96623], [33.86432, 34.97592], [33.84811, 34.97075], [33.83505, 34.98108], [33.85621, 34.98956], [33.85891, 35.001], [33.85216, 35.00579], [33.84045, 35.00616], [33.82875, 35.01685], [33.83055, 35.02865], [33.81524, 35.04192], [33.8012, 35.04786], [33.82051, 35.0667], [33.8355, 35.05777], [33.85261, 35.0574], [33.88367, 35.07877], [33.89485, 35.06873], [33.90247, 35.07686], [33.91299, 35.07579], [33.91789, 35.08688], [33.89853, 35.11377], [33.88737, 35.11408], [33.88943, 35.12007], [33.88561, 35.12449], [33.87224, 35.12293], [33.87622, 35.10457], [33.87097, 35.09389], [33.87479, 35.08881], [33.8541, 35.07201], [33.84168, 35.06823], [33.82067, 35.07826], [33.78581, 35.05104], [33.76106, 35.04253], [33.73824, 35.05321], [33.71482, 35.03722], [33.70209, 35.04882], [33.7161, 35.07279], [33.70861, 35.07644], [33.69095, 35.06237], [33.68474, 35.06602], [33.67742, 35.05963], [33.67678, 35.03866], [33.69938, 35.03123], [33.69731, 35.01754], [33.71514, 35.00294], [33.70639, 34.99303], [33.70575, 34.97947]], [[33.77312, 34.9976], [33.77553, 34.99518], [33.78516, 34.99582], [33.79191, 34.98914], [33.78917, 34.98854], [33.78571, 34.98951], [33.78318, 34.98699], [33.78149, 34.98854], [33.77843, 34.988], [33.7778, 34.98981], [33.76738, 34.99188], [33.76605, 34.99543], [33.75682, 34.99916], [33.75994, 35.00113], [33.77312, 34.9976]], [[33.74144, 35.01053], [33.7343, 35.01178], [33.73781, 35.02181], [33.74265, 35.02329], [33.74983, 35.02274], [33.7492, 35.01319], [33.74144, 35.01053]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q16390686",
+           nameEn: "Peninsular Spain",
+           country: "ES",
+           groups: ["Q12837", "EU", "039", "150", "UN"],
+           callingCodes: ["34"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[3.75438, 42.33445], [3.17156, 42.43545], [3.11379, 42.43646], [3.10027, 42.42621], [3.08167, 42.42748], [3.03734, 42.47363], [2.96518, 42.46692], [2.94283, 42.48174], [2.92107, 42.4573], [2.88413, 42.45938], [2.86983, 42.46843], [2.85675, 42.45444], [2.84335, 42.45724], [2.77464, 42.41046], [2.75497, 42.42578], [2.72056, 42.42298], [2.65311, 42.38771], [2.6747, 42.33974], [2.57934, 42.35808], [2.55516, 42.35351], [2.54382, 42.33406], [2.48457, 42.33933], [2.43508, 42.37568], [2.43299, 42.39423], [2.38504, 42.39977], [2.25551, 42.43757], [2.20578, 42.41633], [2.16599, 42.42314], [2.12789, 42.41291], [2.11621, 42.38393], [2.06241, 42.35906], [2.00488, 42.35399], [1.96482, 42.37787], [1.9574, 42.42401], [1.94084, 42.43039], [1.94061, 42.43333], [1.94292, 42.44316], [1.93663, 42.45439], [1.88853, 42.4501], [1.83037, 42.48395], [1.76335, 42.48863], [1.72515, 42.50338], [1.70571, 42.48867], [1.66826, 42.50779], [1.65674, 42.47125], [1.58933, 42.46275], [1.57953, 42.44957], [1.55937, 42.45808], [1.55073, 42.43299], [1.5127, 42.42959], [1.44529, 42.43724], [1.43838, 42.47848], [1.41648, 42.48315], [1.46661, 42.50949], [1.44759, 42.54431], [1.41245, 42.53539], [1.4234, 42.55959], [1.44529, 42.56722], [1.42512, 42.58292], [1.44197, 42.60217], [1.35562, 42.71944], [1.15928, 42.71407], [1.0804, 42.78569], [0.98292, 42.78754], [0.96166, 42.80629], [0.93089, 42.79154], [0.711, 42.86372], [0.66121, 42.84021], [0.65421, 42.75872], [0.67873, 42.69458], [0.40214, 42.69779], [0.36251, 42.72282], [0.29407, 42.67431], [0.25336, 42.7174], [0.17569, 42.73424], [-0.02468, 42.68513], [-0.10519, 42.72761], [-0.16141, 42.79535], [-0.17939, 42.78974], [-0.3122, 42.84788], [-0.38833, 42.80132], [-0.41319, 42.80776], [-0.44334, 42.79939], [-0.50863, 42.82713], [-0.55497, 42.77846], [-0.67637, 42.88303], [-0.69837, 42.87945], [-0.72608, 42.89318], [-0.73422, 42.91228], [-0.72037, 42.92541], [-0.75478, 42.96916], [-0.81652, 42.95166], [-0.97133, 42.96239], [-1.00963, 42.99279], [-1.10333, 43.0059], [-1.22881, 43.05534], [-1.25244, 43.04164], [-1.30531, 43.06859], [-1.30052, 43.09581], [-1.27118, 43.11961], [-1.32209, 43.1127], [-1.34419, 43.09665], [-1.35272, 43.02658], [-1.44067, 43.047], [-1.47555, 43.08372], [-1.41562, 43.12815], [-1.3758, 43.24511], [-1.40942, 43.27272], [-1.45289, 43.27049], [-1.50992, 43.29481], [-1.55963, 43.28828], [-1.57674, 43.25269], [-1.61341, 43.25269], [-1.63052, 43.28591], [-1.62481, 43.30726], [-1.69407, 43.31378], [-1.73074, 43.29481], [-1.7397, 43.32979], [-1.75079, 43.3317], [-1.75334, 43.34107], [-1.77068, 43.34396], [-1.78714, 43.35476], [-1.78332, 43.36399], [-1.79319, 43.37497], [-1.77289, 43.38957], [-1.81005, 43.59738], [-10.14298, 44.17365], [-11.19304, 41.83075], [-8.87157, 41.86488], [-8.81794, 41.90375], [-8.75712, 41.92833], [-8.74606, 41.9469], [-8.7478, 41.96282], [-8.69071, 41.98862], [-8.6681, 41.99703], [-8.65832, 42.02972], [-8.64626, 42.03668], [-8.63791, 42.04691], [-8.59493, 42.05708], [-8.58086, 42.05147], [-8.54563, 42.0537], [-8.5252, 42.06264], [-8.52837, 42.07658], [-8.48185, 42.0811], [-8.44123, 42.08218], [-8.42512, 42.07199], [-8.40143, 42.08052], [-8.38323, 42.07683], [-8.36353, 42.09065], [-8.33912, 42.08358], [-8.32161, 42.10218], [-8.29809, 42.106], [-8.2732, 42.12396], [-8.24681, 42.13993], [-8.22406, 42.1328], [-8.1986, 42.15402], [-8.18947, 42.13853], [-8.19406, 42.12141], [-8.18178, 42.06436], [-8.11729, 42.08537], [-8.08847, 42.05767], [-8.08796, 42.01398], [-8.16232, 41.9828], [-8.2185, 41.91237], [-8.19551, 41.87459], [-8.16944, 41.87944], [-8.16455, 41.81753], [-8.0961, 41.81024], [-8.01136, 41.83453], [-7.9804, 41.87337], [-7.92336, 41.8758], [-7.90707, 41.92432], [-7.88751, 41.92553], [-7.88055, 41.84571], [-7.84188, 41.88065], [-7.69848, 41.90977], [-7.65774, 41.88308], [-7.58603, 41.87944], [-7.62188, 41.83089], [-7.52737, 41.83939], [-7.49803, 41.87095], [-7.45566, 41.86488], [-7.44759, 41.84451], [-7.42854, 41.83262], [-7.42864, 41.80589], [-7.37092, 41.85031], [-7.32366, 41.8406], [-7.18677, 41.88793], [-7.18549, 41.97515], [-7.14115, 41.98855], [-7.08574, 41.97401], [-7.07596, 41.94977], [-7.01078, 41.94977], [-6.98144, 41.9728], [-6.95537, 41.96553], [-6.94396, 41.94403], [-6.82174, 41.94493], [-6.81196, 41.99097], [-6.76959, 41.98734], [-6.75004, 41.94129], [-6.61967, 41.94008], [-6.58544, 41.96674], [-6.5447, 41.94371], [-6.56752, 41.88429], [-6.51374, 41.8758], [-6.56426, 41.74219], [-6.54633, 41.68623], [-6.49907, 41.65823], [-6.44204, 41.68258], [-6.29863, 41.66432], [-6.19128, 41.57638], [-6.26777, 41.48796], [-6.3306, 41.37677], [-6.38553, 41.38655], [-6.38551, 41.35274], [-6.55937, 41.24417], [-6.65046, 41.24725], [-6.68286, 41.21641], [-6.69711, 41.1858], [-6.77319, 41.13049], [-6.75655, 41.10187], [-6.79241, 41.05397], [-6.80942, 41.03629], [-6.84781, 41.02692], [-6.88843, 41.03027], [-6.913, 41.03922], [-6.9357, 41.02888], [-6.8527, 40.93958], [-6.84292, 40.89771], [-6.80707, 40.88047], [-6.79892, 40.84842], [-6.82337, 40.84472], [-6.82826, 40.74603], [-6.79567, 40.65955], [-6.84292, 40.56801], [-6.80218, 40.55067], [-6.7973, 40.51723], [-6.84944, 40.46394], [-6.84618, 40.42177], [-6.78426, 40.36468], [-6.80218, 40.33239], [-6.86085, 40.2976], [-6.86085, 40.26776], [-7.00426, 40.23169], [-7.02544, 40.18564], [-7.00589, 40.12087], [-6.94233, 40.10716], [-6.86737, 40.01986], [-6.91463, 39.86618], [-6.97492, 39.81488], [-7.01613, 39.66877], [-7.24707, 39.66576], [-7.33507, 39.64569], [-7.54121, 39.66717], [-7.49477, 39.58794], [-7.2927, 39.45847], [-7.3149, 39.34857], [-7.23403, 39.27579], [-7.23566, 39.20132], [-7.12811, 39.17101], [-7.14929, 39.11287], [-7.10692, 39.10275], [-7.04011, 39.11919], [-6.97004, 39.07619], [-6.95211, 39.0243], [-7.051, 38.907], [-7.03848, 38.87221], [-7.26174, 38.72107], [-7.265, 38.61674], [-7.32529, 38.44336], [-7.15581, 38.27597], [-7.09389, 38.17227], [-6.93418, 38.21454], [-7.00375, 38.01914], [-7.05966, 38.01966], [-7.10366, 38.04404], [-7.12648, 38.00296], [-7.24544, 37.98884], [-7.27314, 37.90145], [-7.33441, 37.81193], [-7.41981, 37.75729], [-7.51759, 37.56119], [-7.46878, 37.47127], [-7.43974, 37.38913], [-7.43227, 37.25152], [-7.41854, 37.23813], [-7.41133, 37.20314], [-7.39769, 37.16868], [-7.37282, 36.96896], [-7.2725, 35.73269], [-5.10878, 36.05227], [-2.27707, 35.35051], [3.75438, 42.33445]], [[-5.27801, 36.14942], [-5.34064, 36.03744], [-5.40526, 36.15488], [-5.34536, 36.15501], [-5.33822, 36.15272], [-5.27801, 36.14942]]], [[[1.99838, 42.44682], [2.01564, 42.45171], [1.99216, 42.46208], [1.98579, 42.47486], [1.99766, 42.4858], [1.98916, 42.49351], [1.98022, 42.49569], [1.97697, 42.48568], [1.97227, 42.48487], [1.97003, 42.48081], [1.96215, 42.47854], [1.95606, 42.45785], [1.96125, 42.45364], [1.98378, 42.44697], [1.99838, 42.44682]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q98059339",
+           nameEn: "Mainland Norway",
+           country: "NO",
+           groups: ["154", "150", "UN"],
+           callingCodes: ["47"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[10.40861, 58.38489], [10.64958, 58.89391], [11.08911, 58.98745], [11.15367, 59.07862], [11.34459, 59.11672], [11.4601, 58.99022], [11.45199, 58.89604], [11.65732, 58.90177], [11.8213, 59.24985], [11.69297, 59.59442], [11.92112, 59.69531], [11.87121, 59.86039], [12.15641, 59.8926], [12.36317, 59.99259], [12.52003, 60.13846], [12.59133, 60.50559], [12.2277, 61.02442], [12.69115, 61.06584], [12.86939, 61.35427], [12.57707, 61.56547], [12.40595, 61.57226], [12.14746, 61.7147], [12.29187, 62.25699], [12.07085, 62.6297], [12.19919, 63.00104], [11.98529, 63.27487], [12.19919, 63.47935], [12.14928, 63.59373], [12.74105, 64.02171], [13.23411, 64.09087], [13.98222, 64.00953], [14.16051, 64.18725], [14.11117, 64.46674], [13.64276, 64.58402], [14.50926, 65.31786], [14.53778, 66.12399], [15.05113, 66.15572], [15.49318, 66.28509], [15.37197, 66.48217], [16.35589, 67.06419], [16.39154, 67.21653], [16.09922, 67.4364], [16.12774, 67.52106], [16.38441, 67.52923], [16.7409, 67.91037], [17.30416, 68.11591], [17.90787, 67.96537], [18.13836, 68.20874], [18.1241, 68.53721], [18.39503, 68.58672], [18.63032, 68.50849], [18.97255, 68.52416], [19.93508, 68.35911], [20.22027, 68.48759], [19.95647, 68.55546], [20.22027, 68.67246], [20.33435, 68.80174], [20.28444, 68.93283], [20.0695, 69.04469], [20.55258, 69.06069], [20.72171, 69.11874], [21.05775, 69.0356], [21.11099, 69.10291], [20.98641, 69.18809], [21.00732, 69.22755], [21.27827, 69.31281], [21.63833, 69.27485], [22.27276, 68.89514], [22.38367, 68.71561], [22.53321, 68.74393], [23.13064, 68.64684], [23.68017, 68.70276], [23.781, 68.84514], [24.02299, 68.81601], [24.18432, 68.73936], [24.74898, 68.65143], [24.90023, 68.55579], [24.93048, 68.61102], [25.10189, 68.63307], [25.12206, 68.78684], [25.42455, 68.90328], [25.61613, 68.89602], [25.75729, 68.99383], [25.69679, 69.27039], [25.96904, 69.68397], [26.40261, 69.91377], [26.64461, 69.96565], [27.05802, 69.92069], [27.57226, 70.06215], [27.95542, 70.0965], [27.97558, 69.99671], [28.32849, 69.88605], [28.36883, 69.81658], [29.12697, 69.69193], [29.31664, 69.47994], [28.8629, 69.22395], [28.81248, 69.11997], [28.91738, 69.04774], [29.0444, 69.0119], [29.26623, 69.13794], [29.27631, 69.2811], [29.97205, 69.41623], [30.16363, 69.65244], [30.52662, 69.54699], [30.95011, 69.54699], [30.84095, 69.80584], [31.59909, 70.16571], [32.07813, 72.01005], [-11.60274, 67.73467], [7.28637, 57.35913], [10.40861, 58.38489]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           wikidata: "Q98543636",
+           nameEn: "Mainland Ecuador",
+           country: "EC",
+           groups: ["005", "419", "019", "UN"],
+           callingCodes: ["593"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-84.52388, -3.36941], [-80.30602, -3.39149], [-80.20647, -3.431], [-80.24123, -3.46124], [-80.24586, -3.48677], [-80.23651, -3.48652], [-80.22629, -3.501], [-80.20535, -3.51667], [-80.21642, -3.5888], [-80.19848, -3.59249], [-80.18741, -3.63994], [-80.19926, -3.68894], [-80.13232, -3.90317], [-80.46386, -4.01342], [-80.4822, -4.05477], [-80.45023, -4.20938], [-80.32114, -4.21323], [-80.46386, -4.41516], [-80.39256, -4.48269], [-80.13945, -4.29786], [-79.79722, -4.47558], [-79.59402, -4.46848], [-79.26248, -4.95167], [-79.1162, -4.97774], [-79.01659, -5.01481], [-78.85149, -4.66795], [-78.68394, -4.60754], [-78.34362, -3.38633], [-78.24589, -3.39907], [-78.22642, -3.51113], [-78.14324, -3.47653], [-78.19369, -3.36431], [-77.94147, -3.05454], [-76.6324, -2.58397], [-76.05203, -2.12179], [-75.57429, -1.55961], [-75.3872, -0.9374], [-75.22862, -0.95588], [-75.22862, -0.60048], [-75.53615, -0.19213], [-75.60169, -0.18708], [-75.61997, -0.10012], [-75.40192, -0.17196], [-75.25764, -0.11943], [-75.82927, 0.09578], [-76.23441, 0.42294], [-76.41215, 0.38228], [-76.4094, 0.24015], [-76.89177, 0.24736], [-77.52001, 0.40782], [-77.49984, 0.64476], [-77.67815, 0.73863], [-77.66416, 0.81604], [-77.68613, 0.83029], [-77.7148, 0.85003], [-77.85677, 0.80197], [-78.42749, 1.15389], [-78.87137, 1.47457], [-82.12561, 4.00341], [-84.52388, -3.36941]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "001",
+           wikidata: "Q2",
+           nameEn: "World",
+           aliases: ["Earth", "Planet"],
+           level: "world"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "002",
+           wikidata: "Q15",
+           nameEn: "Africa",
+           level: "region"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "003",
+           wikidata: "Q49",
+           nameEn: "North America",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "005",
+           wikidata: "Q18",
+           nameEn: "South America",
+           level: "intermediateRegion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "009",
+           wikidata: "Q538",
+           nameEn: "Oceania",
+           level: "region"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "011",
+           wikidata: "Q4412",
+           nameEn: "Western Africa",
+           level: "intermediateRegion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "013",
+           wikidata: "Q27611",
+           nameEn: "Central America",
+           level: "intermediateRegion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "014",
+           wikidata: "Q27407",
+           nameEn: "Eastern Africa",
+           level: "intermediateRegion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "015",
+           wikidata: "Q27381",
+           nameEn: "Northern Africa",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "017",
+           wikidata: "Q27433",
+           nameEn: "Middle Africa",
+           level: "intermediateRegion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "018",
+           wikidata: "Q27394",
+           nameEn: "Southern Africa",
+           level: "intermediateRegion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "019",
+           wikidata: "Q828",
+           nameEn: "Americas",
+           level: "region"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "021",
+           wikidata: "Q2017699",
+           nameEn: "Northern America",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "029",
+           wikidata: "Q664609",
+           nameEn: "Caribbean",
+           level: "intermediateRegion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "030",
+           wikidata: "Q27231",
+           nameEn: "Eastern Asia",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "034",
+           wikidata: "Q771405",
+           nameEn: "Southern Asia",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "035",
+           wikidata: "Q11708",
+           nameEn: "South-eastern Asia",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "039",
+           wikidata: "Q27449",
+           nameEn: "Southern Europe",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "053",
+           wikidata: "Q45256",
+           nameEn: "Australia and New Zealand",
+           aliases: ["Australasia"],
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "054",
+           wikidata: "Q37394",
+           nameEn: "Melanesia",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "057",
+           wikidata: "Q3359409",
+           nameEn: "Micronesia",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "061",
+           wikidata: "Q35942",
+           nameEn: "Polynesia",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "142",
+           wikidata: "Q48",
+           nameEn: "Asia",
+           level: "region"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "143",
+           wikidata: "Q27275",
+           nameEn: "Central Asia",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "145",
+           wikidata: "Q27293",
+           nameEn: "Western Asia",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "150",
+           wikidata: "Q46",
+           nameEn: "Europe",
+           level: "region"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "151",
+           wikidata: "Q27468",
+           nameEn: "Eastern Europe",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "154",
+           wikidata: "Q27479",
+           nameEn: "Northern Europe",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "155",
+           wikidata: "Q27496",
+           nameEn: "Western Europe",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "202",
+           wikidata: "Q132959",
+           nameEn: "Sub-Saharan Africa",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "419",
+           wikidata: "Q72829598",
+           nameEn: "Latin America and the Caribbean",
+           level: "subregion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "680",
+           wikidata: "Q3405693",
+           nameEn: "Sark",
+           country: "GB",
+           groups: ["GG", "830", "Q185086", "154", "150", "UN"],
+           level: "subterritory",
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["44 01481"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-2.36485, 49.48223], [-2.65349, 49.15373], [-2.09454, 49.46288], [-2.36485, 49.48223]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           m49: "830",
+           wikidata: "Q42314",
+           nameEn: "Channel Islands",
+           level: "intermediateRegion"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AC",
+           iso1A3: "ASC",
+           wikidata: "Q46197",
+           nameEn: "Ascension Island",
+           aliases: ["SH-AC"],
+           country: "GB",
+           groups: ["SH", "BOTS", "011", "202", "002", "UN"],
+           isoStatus: "excRes",
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["247"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-14.82771, -8.70814], [-13.33271, -8.07391], [-14.91926, -6.63386], [-14.82771, -8.70814]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AD",
+           iso1A3: "AND",
+           iso1N3: "020",
+           wikidata: "Q228",
+           nameEn: "Andorra",
+           groups: ["Q12837", "039", "150", "UN"],
+           callingCodes: ["376"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[1.72515, 42.50338], [1.73683, 42.55492], [1.7858, 42.57698], [1.72588, 42.59098], [1.73452, 42.61515], [1.68267, 42.62533], [1.6625, 42.61982], [1.63485, 42.62957], [1.60085, 42.62703], [1.55418, 42.65669], [1.50867, 42.64483], [1.48043, 42.65203], [1.46718, 42.63296], [1.47986, 42.61346], [1.44197, 42.60217], [1.42512, 42.58292], [1.44529, 42.56722], [1.4234, 42.55959], [1.41245, 42.53539], [1.44759, 42.54431], [1.46661, 42.50949], [1.41648, 42.48315], [1.43838, 42.47848], [1.44529, 42.43724], [1.5127, 42.42959], [1.55073, 42.43299], [1.55937, 42.45808], [1.57953, 42.44957], [1.58933, 42.46275], [1.65674, 42.47125], [1.66826, 42.50779], [1.70571, 42.48867], [1.72515, 42.50338]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AE",
+           iso1A3: "ARE",
+           iso1N3: "784",
+           wikidata: "Q878",
+           nameEn: "United Arab Emirates",
+           groups: ["145", "142", "UN"],
+           callingCodes: ["971"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[56.26534, 25.62825], [56.25341, 25.61443], [56.26636, 25.60643], [56.25365, 25.60211], [56.20473, 25.61119], [56.18363, 25.65508], [56.14826, 25.66351], [56.13579, 25.73524], [56.17416, 25.77239], [56.13963, 25.82765], [56.19334, 25.9795], [56.15498, 26.06828], [56.08666, 26.05038], [55.81777, 26.18798], [55.14145, 25.62624], [53.97892, 24.64436], [52.82259, 25.51697], [52.35509, 25.00368], [52.02277, 24.75635], [51.83108, 24.71675], [51.58834, 24.66608], [51.41644, 24.39615], [51.58871, 24.27256], [51.59617, 24.12041], [52.56622, 22.94341], [55.13599, 22.63334], [55.2137, 22.71065], [55.22634, 23.10378], [55.57358, 23.669], [55.48677, 23.94946], [55.73301, 24.05994], [55.8308, 24.01633], [56.01799, 24.07426], [55.95472, 24.2172], [55.83367, 24.20193], [55.77658, 24.23476], [55.76558, 24.23227], [55.75257, 24.23466], [55.75382, 24.2466], [55.75939, 24.26114], [55.76781, 24.26209], [55.79145, 24.27914], [55.80747, 24.31069], [55.83395, 24.32776], [55.83271, 24.41521], [55.76461, 24.5287], [55.83271, 24.68567], [55.83408, 24.77858], [55.81348, 24.80102], [55.81116, 24.9116], [55.85094, 24.96858], [55.90849, 24.96771], [55.96316, 25.00857], [56.05715, 24.95727], [56.05106, 24.87461], [55.97467, 24.89639], [55.97836, 24.87673], [56.03535, 24.81161], [56.06128, 24.74457], [56.13684, 24.73699], [56.20062, 24.78565], [56.20568, 24.85063], [56.30269, 24.88334], [56.34873, 24.93205], [56.3227, 24.97284], [56.86325, 25.03856], [56.82555, 25.7713], [56.26534, 25.62825]], [[56.26062, 25.33108], [56.3005, 25.31815], [56.3111, 25.30107], [56.35172, 25.30681], [56.34438, 25.26653], [56.27628, 25.23404], [56.24341, 25.22867], [56.20872, 25.24104], [56.20838, 25.25668], [56.24465, 25.27505], [56.25008, 25.28843], [56.23362, 25.31253], [56.26062, 25.33108]]], [[[56.28423, 25.26344], [56.29379, 25.2754], [56.28102, 25.28486], [56.2716, 25.27916], [56.27086, 25.26128], [56.28423, 25.26344]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AF",
+           iso1A3: "AFG",
+           iso1N3: "004",
+           wikidata: "Q889",
+           nameEn: "Afghanistan",
+           groups: ["034", "142", "UN"],
+           callingCodes: ["93"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[70.61526, 38.34774], [70.60407, 38.28046], [70.54673, 38.24541], [70.4898, 38.12546], [70.17206, 37.93276], [70.1863, 37.84296], [70.27694, 37.81258], [70.28243, 37.66706], [70.15015, 37.52519], [69.95971, 37.5659], [69.93362, 37.61378], [69.84435, 37.60616], [69.80041, 37.5746], [69.51888, 37.5844], [69.44954, 37.4869], [69.36645, 37.40462], [69.45022, 37.23315], [69.39529, 37.16752], [69.25152, 37.09426], [69.03274, 37.25174], [68.96407, 37.32603], [68.88168, 37.33368], [68.91189, 37.26704], [68.80889, 37.32494], [68.81438, 37.23862], [68.6798, 37.27906], [68.61851, 37.19815], [68.41888, 37.13906], [68.41201, 37.10402], [68.29253, 37.10621], [68.27605, 37.00977], [68.18542, 37.02074], [68.02194, 36.91923], [67.87917, 37.0591], [67.7803, 37.08978], [67.78329, 37.1834], [67.51868, 37.26102], [67.2581, 37.17216], [67.2224, 37.24545], [67.13039, 37.27168], [67.08232, 37.35469], [66.95598, 37.40162], [66.64699, 37.32958], [66.55743, 37.35409], [66.30993, 37.32409], [65.72274, 37.55438], [65.64137, 37.45061], [65.64263, 37.34388], [65.51778, 37.23881], [64.97945, 37.21913], [64.61141, 36.6351], [64.62514, 36.44311], [64.57295, 36.34362], [64.43288, 36.24401], [64.05385, 36.10433], [63.98519, 36.03773], [63.56496, 35.95106], [63.53475, 35.90881], [63.29579, 35.85985], [63.12276, 35.86208], [63.10318, 35.81782], [63.23262, 35.67487], [63.10079, 35.63024], [63.12276, 35.53196], [63.0898, 35.43131], [62.90853, 35.37086], [62.74098, 35.25432], [62.62288, 35.22067], [62.48006, 35.28796], [62.29878, 35.13312], [62.29191, 35.25964], [62.15871, 35.33278], [62.05709, 35.43803], [61.97743, 35.4604], [61.77693, 35.41341], [61.58742, 35.43803], [61.27371, 35.61482], [61.18187, 35.30249], [61.0991, 35.27845], [61.12831, 35.09938], [61.06926, 34.82139], [61.00197, 34.70631], [60.99922, 34.63064], [60.72316, 34.52857], [60.91321, 34.30411], [60.66502, 34.31539], [60.50209, 34.13992], [60.5838, 33.80793], [60.5485, 33.73422], [60.57762, 33.59772], [60.69573, 33.56054], [60.91133, 33.55596], [60.88908, 33.50219], [60.56485, 33.12944], [60.86191, 32.22565], [60.84541, 31.49561], [61.70929, 31.37391], [61.80569, 31.16167], [61.80957, 31.12576], [61.83257, 31.0452], [61.8335, 30.97669], [61.78268, 30.92724], [61.80829, 30.84224], [60.87231, 29.86514], [62.47751, 29.40782], [63.5876, 29.50456], [64.12966, 29.39157], [64.19796, 29.50407], [64.62116, 29.58903], [65.04005, 29.53957], [66.24175, 29.85181], [66.36042, 29.9583], [66.23609, 30.06321], [66.34869, 30.404], [66.28413, 30.57001], [66.39194, 30.9408], [66.42645, 30.95309], [66.58175, 30.97532], [66.68166, 31.07597], [66.72561, 31.20526], [66.83273, 31.26867], [67.04147, 31.31561], [67.03323, 31.24519], [67.29964, 31.19586], [67.78854, 31.33203], [67.7748, 31.4188], [67.62374, 31.40473], [67.58323, 31.52772], [67.72056, 31.52304], [67.86887, 31.63536], [68.00071, 31.6564], [68.1655, 31.82691], [68.25614, 31.80357], [68.27605, 31.75863], [68.44222, 31.76446], [68.57475, 31.83158], [68.6956, 31.75687], [68.79997, 31.61665], [68.91078, 31.59687], [68.95995, 31.64822], [69.00939, 31.62249], [69.11514, 31.70782], [69.20577, 31.85957], [69.3225, 31.93186], [69.27032, 32.14141], [69.27932, 32.29119], [69.23599, 32.45946], [69.2868, 32.53938], [69.38155, 32.56601], [69.44747, 32.6678], [69.43649, 32.7302], [69.38018, 32.76601], [69.47082, 32.85834], [69.5436, 32.8768], [69.49854, 32.88843], [69.49004, 33.01509], [69.57656, 33.09911], [69.71526, 33.09911], [69.79766, 33.13247], [69.85259, 33.09451], [70.02563, 33.14282], [70.07369, 33.22557], [70.13686, 33.21064], [70.32775, 33.34496], [70.17062, 33.53535], [70.20141, 33.64387], [70.14785, 33.6553], [70.14236, 33.71701], [70.00503, 33.73528], [69.85671, 33.93719], [69.87307, 33.9689], [69.90203, 34.04194], [70.54336, 33.9463], [70.88119, 33.97933], [71.07345, 34.06242], [71.06933, 34.10564], [71.09307, 34.11961], [71.09453, 34.13524], [71.13078, 34.16503], [71.12815, 34.26619], [71.17662, 34.36769], [71.02401, 34.44835], [71.0089, 34.54568], [71.11602, 34.63047], [71.08718, 34.69034], [71.28356, 34.80882], [71.29472, 34.87728], [71.50329, 34.97328], [71.49917, 35.00478], [71.55273, 35.02615], [71.52938, 35.09023], [71.67495, 35.21262], [71.5541, 35.28776], [71.54294, 35.31037], [71.65435, 35.4479], [71.49917, 35.6267], [71.55273, 35.71483], [71.37969, 35.95865], [71.19505, 36.04134], [71.60491, 36.39429], [71.80267, 36.49924], [72.18135, 36.71838], [72.6323, 36.84601], [73.82685, 36.91421], [74.04856, 36.82648], [74.43389, 37.00977], [74.53739, 36.96224], [74.56453, 37.03023], [74.49981, 37.24518], [74.80605, 37.21565], [74.88887, 37.23275], [74.8294, 37.3435], [74.68383, 37.3948], [74.56161, 37.37734], [74.41055, 37.3948], [74.23339, 37.41116], [74.20308, 37.34208], [73.8564, 37.26158], [73.82552, 37.22659], [73.64974, 37.23643], [73.61129, 37.27469], [73.76647, 37.33913], [73.77197, 37.4417], [73.29633, 37.46495], [73.06884, 37.31729], [72.79693, 37.22222], [72.66381, 37.02014], [72.54095, 37.00007], [72.31676, 36.98115], [71.83229, 36.68084], [71.67083, 36.67346], [71.57195, 36.74943], [71.51502, 36.89128], [71.48481, 36.93218], [71.46923, 36.99925], [71.45578, 37.03094], [71.43097, 37.05855], [71.44127, 37.11856], [71.4494, 37.18137], [71.4555, 37.21418], [71.47386, 37.2269], [71.48339, 37.23937], [71.4824, 37.24921], [71.48536, 37.26017], [71.50674, 37.31502], [71.49821, 37.31975], [71.4862, 37.33405], [71.47685, 37.40281], [71.49612, 37.4279], [71.5256, 37.47971], [71.50616, 37.50733], [71.49693, 37.53527], [71.5065, 37.60912], [71.51972, 37.61945], [71.54186, 37.69691], [71.55234, 37.73209], [71.53053, 37.76534], [71.54324, 37.77104], [71.55752, 37.78677], [71.59255, 37.79956], [71.58843, 37.92425], [71.51565, 37.95349], [71.32871, 37.88564], [71.296, 37.93403], [71.2809, 37.91995], [71.24969, 37.93031], [71.27278, 37.96496], [71.27622, 37.99946], [71.28922, 38.01272], [71.29878, 38.04429], [71.36444, 38.15358], [71.37803, 38.25641], [71.33869, 38.27335], [71.33114, 38.30339], [71.21291, 38.32797], [71.1451, 38.40106], [71.10957, 38.40671], [71.10592, 38.42077], [71.09542, 38.42517], [71.0556, 38.40176], [71.03545, 38.44779], [70.98693, 38.48862], [70.92728, 38.43021], [70.88719, 38.46826], [70.84376, 38.44688], [70.82538, 38.45394], [70.81697, 38.44507], [70.80521, 38.44447], [70.79766, 38.44944], [70.78702, 38.45031], [70.78581, 38.45502], [70.77132, 38.45548], [70.75455, 38.4252], [70.72485, 38.4131], [70.69807, 38.41861], [70.67438, 38.40597], [70.6761, 38.39144], [70.69189, 38.37031], [70.64966, 38.34999], [70.61526, 38.34774]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AG",
+           iso1A3: "ATG",
+           iso1N3: "028",
+           wikidata: "Q781",
+           nameEn: "Antigua and Barbuda",
+           groups: ["029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           callingCodes: ["1 268"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-61.66959, 18.6782], [-62.58307, 16.68909], [-62.1023, 16.97277], [-61.23098, 16.62484], [-61.66959, 18.6782]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AI",
+           iso1A3: "AIA",
+           iso1N3: "660",
+           wikidata: "Q25228",
+           nameEn: "Anguilla",
+           country: "GB",
+           groups: ["BOTS", "029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           callingCodes: ["1 264"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-63.79029, 19.11219], [-63.35989, 18.06012], [-62.62718, 18.26185], [-63.79029, 19.11219]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AL",
+           iso1A3: "ALB",
+           iso1N3: "008",
+           wikidata: "Q222",
+           nameEn: "Albania",
+           groups: ["039", "150", "UN"],
+           callingCodes: ["355"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[20.07761, 42.55582], [20.01834, 42.54622], [20.00842, 42.5109], [19.9324, 42.51699], [19.82333, 42.46581], [19.76549, 42.50237], [19.74731, 42.57422], [19.77375, 42.58517], [19.73244, 42.66299], [19.65972, 42.62774], [19.4836, 42.40831], [19.42352, 42.36546], [19.42, 42.33019], [19.28623, 42.17745], [19.40687, 42.10024], [19.37548, 42.06835], [19.36867, 42.02564], [19.37691, 41.96977], [19.34601, 41.95675], [19.33812, 41.90669], [19.37451, 41.8842], [19.37597, 41.84849], [19.26406, 41.74971], [19.0384, 40.35325], [19.95905, 39.82857], [19.97622, 39.78684], [19.92466, 39.69533], [19.98042, 39.6504], [20.00957, 39.69227], [20.05189, 39.69112], [20.12956, 39.65805], [20.15988, 39.652], [20.22376, 39.64532], [20.22707, 39.67459], [20.27412, 39.69884], [20.31961, 39.72799], [20.29152, 39.80421], [20.30804, 39.81563], [20.38572, 39.78516], [20.41475, 39.81437], [20.41546, 39.82832], [20.31135, 39.99438], [20.37911, 39.99058], [20.42373, 40.06777], [20.48487, 40.06271], [20.51297, 40.08168], [20.55593, 40.06524], [20.61081, 40.07866], [20.62566, 40.0897], [20.67162, 40.09433], [20.71789, 40.27739], [20.78234, 40.35803], [20.7906, 40.42726], [20.83688, 40.47882], [20.94925, 40.46625], [20.96908, 40.51526], [21.03932, 40.56299], [21.05833, 40.66586], [20.98134, 40.76046], [20.95752, 40.76982], [20.98396, 40.79109], [20.97887, 40.85475], [20.97693, 40.90103], [20.94305, 40.92399], [20.83671, 40.92752], [20.81567, 40.89662], [20.73504, 40.9081], [20.71634, 40.91781], [20.65558, 41.08009], [20.63454, 41.0889], [20.59832, 41.09066], [20.58546, 41.11179], [20.59715, 41.13644], [20.51068, 41.2323], [20.49432, 41.33679], [20.52119, 41.34381], [20.55976, 41.4087], [20.51301, 41.442], [20.49039, 41.49277], [20.45331, 41.51436], [20.45809, 41.5549], [20.52103, 41.56473], [20.55508, 41.58113], [20.51769, 41.65975], [20.52937, 41.69292], [20.51301, 41.72433], [20.53405, 41.78099], [20.57144, 41.7897], [20.55976, 41.87068], [20.59524, 41.8818], [20.57946, 41.91593], [20.63069, 41.94913], [20.59434, 42.03879], [20.55633, 42.08173], [20.56955, 42.12097], [20.48857, 42.25444], [20.3819, 42.3029], [20.34479, 42.32656], [20.24399, 42.32168], [20.21797, 42.41237], [20.17127, 42.50469], [20.07761, 42.55582]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AM",
+           iso1A3: "ARM",
+           iso1N3: "051",
+           wikidata: "Q399",
+           nameEn: "Armenia",
+           groups: ["145", "142", "UN"],
+           callingCodes: ["374"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[45.0133, 41.29747], [44.93493, 41.25685], [44.81437, 41.30371], [44.80053, 41.25949], [44.81749, 41.23488], [44.84358, 41.23088], [44.89911, 41.21366], [44.87887, 41.20195], [44.82084, 41.21513], [44.72814, 41.20338], [44.61462, 41.24018], [44.59322, 41.1933], [44.46791, 41.18204], [44.34417, 41.2382], [44.34337, 41.20312], [44.32139, 41.2079], [44.18148, 41.24644], [44.16591, 41.19141], [43.84835, 41.16329], [43.74717, 41.1117], [43.67712, 41.13398], [43.4717, 41.12611], [43.44984, 41.0988], [43.47319, 41.02251], [43.58683, 40.98961], [43.67712, 40.93084], [43.67712, 40.84846], [43.74872, 40.7365], [43.7425, 40.66805], [43.63664, 40.54159], [43.54791, 40.47413], [43.60862, 40.43267], [43.59928, 40.34019], [43.71136, 40.16673], [43.65221, 40.14889], [43.65688, 40.11199], [43.92307, 40.01787], [44.1057, 40.03555], [44.1778, 40.02845], [44.26973, 40.04866], [44.46635, 39.97733], [44.61845, 39.8281], [44.75779, 39.7148], [44.88354, 39.74432], [44.92869, 39.72157], [45.06604, 39.79277], [45.18554, 39.67846], [45.17464, 39.58614], [45.21784, 39.58074], [45.23535, 39.61373], [45.30385, 39.61373], [45.29606, 39.57654], [45.46992, 39.49888], [45.70547, 39.60174], [45.80804, 39.56716], [45.83, 39.46487], [45.79225, 39.3695], [45.99774, 39.28931], [46.02303, 39.09978], [46.06973, 39.0744], [46.14785, 38.84206], [46.20601, 38.85262], [46.34059, 38.92076], [46.53497, 38.86548], [46.51805, 38.94982], [46.54296, 39.07078], [46.44022, 39.19636], [46.52584, 39.18912], [46.54141, 39.15895], [46.58032, 39.21204], [46.63481, 39.23013], [46.56476, 39.24942], [46.50093, 39.33736], [46.43244, 39.35181], [46.37795, 39.42039], [46.4013, 39.45405], [46.53051, 39.47809], [46.51027, 39.52373], [46.57721, 39.54414], [46.57098, 39.56694], [46.52117, 39.58734], [46.42465, 39.57534], [46.40286, 39.63651], [46.18493, 39.60533], [45.96543, 39.78859], [45.82533, 39.82925], [45.7833, 39.9475], [45.60895, 39.97733], [45.59806, 40.0131], [45.78642, 40.03218], [45.83779, 39.98925], [45.97944, 40.181], [45.95609, 40.27846], [45.65098, 40.37696], [45.42994, 40.53804], [45.45484, 40.57707], [45.35366, 40.65979], [45.4206, 40.7424], [45.55914, 40.78366], [45.60584, 40.87436], [45.40814, 40.97904], [45.44083, 41.01663], [45.39725, 41.02603], [45.35677, 40.99784], [45.28859, 41.03757], [45.26162, 41.0228], [45.25897, 41.0027], [45.1994, 41.04518], [45.16493, 41.05068], [45.1634, 41.08082], [45.1313, 41.09369], [45.12923, 41.06059], [45.06784, 41.05379], [45.08028, 41.10917], [45.19942, 41.13299], [45.1969, 41.168], [45.11811, 41.19967], [45.05201, 41.19211], [45.02932, 41.2101], [45.05497, 41.2464], [45.0133, 41.29747]], [[45.21324, 40.9817], [45.21219, 40.99001], [45.20518, 40.99348], [45.19312, 40.98998], [45.18382, 41.0066], [45.20625, 41.01484], [45.23487, 41.00226], [45.23095, 40.97828], [45.21324, 40.9817]], [[45.00864, 41.03411], [44.9903, 41.05657], [44.96031, 41.06345], [44.95383, 41.07553], [44.97169, 41.09176], [45.00864, 41.09407], [45.03406, 41.07931], [45.04517, 41.06653], [45.03792, 41.03938], [45.00864, 41.03411]]], [[[45.50279, 40.58424], [45.56071, 40.64765], [45.51825, 40.67382], [45.47927, 40.65023], [45.50279, 40.58424]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AO",
+           iso1A3: "AGO",
+           iso1N3: "024",
+           wikidata: "Q916",
+           nameEn: "Angola",
+           groups: ["017", "202", "002", "UN"],
+           callingCodes: ["244"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[16.55507, -5.85631], [13.04371, -5.87078], [12.42245, -6.07585], [11.95767, -5.94705], [12.20376, -5.76338], [12.26557, -5.74031], [12.52318, -5.74353], [12.52301, -5.17481], [12.53599, -5.1618], [12.53586, -5.14658], [12.51589, -5.1332], [12.49815, -5.14058], [12.46297, -5.09408], [12.60251, -5.01715], [12.63465, -4.94632], [12.70868, -4.95505], [12.8733, -4.74346], [13.11195, -4.67745], [13.09648, -4.63739], [12.91489, -4.47907], [12.87096, -4.40315], [12.76844, -4.38709], [12.64835, -4.55937], [12.40964, -4.60609], [12.32324, -4.78415], [12.25587, -4.79437], [12.20901, -4.75642], [12.16068, -4.90089], [12.00924, -5.02627], [11.50888, -5.33417], [10.5065, -17.25284], [11.75063, -17.25013], [12.07076, -17.15165], [12.52111, -17.24495], [12.97145, -16.98567], [13.36212, -16.98048], [13.95896, -17.43141], [14.28743, -17.38814], [18.39229, -17.38927], [18.84226, -17.80375], [21.14283, -17.94318], [21.42741, -18.02787], [23.47474, -17.62877], [23.20038, -17.47563], [22.17217, -16.50269], [22.00323, -16.18028], [21.97988, -13.00148], [24.03339, -12.99091], [23.90937, -12.844], [24.06672, -12.29058], [23.98804, -12.13149], [24.02603, -11.15368], [24.00027, -10.89356], [23.86868, -11.02856], [23.45631, -10.946], [23.16602, -11.10577], [22.54205, -11.05784], [22.25951, -11.24911], [22.17954, -10.85884], [22.32604, -10.76291], [22.19039, -9.94628], [21.84856, -9.59871], [21.79824, -7.29628], [20.56263, -7.28566], [20.61689, -6.90876], [20.31846, -6.91953], [20.30218, -6.98955], [19.5469, -7.00195], [19.33698, -7.99743], [18.33635, -8.00126], [17.5828, -8.13784], [16.96282, -7.21787], [16.55507, -5.85631]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AQ",
+           iso1A3: "ATA",
+           iso1N3: "010",
+           wikidata: "Q51",
+           nameEn: "Antarctica",
+           level: "region",
+           callingCodes: ["672"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[180, -60], [-180, -60], [-180, -90], [180, -90], [180, -60]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AR",
+           iso1A3: "ARG",
+           iso1N3: "032",
+           wikidata: "Q414",
+           nameEn: "Argentina",
+           aliases: ["RA"],
+           groups: ["005", "419", "019", "UN"],
+           callingCodes: ["54"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-72.31343, -50.58411], [-72.33873, -51.59954], [-71.99889, -51.98018], [-69.97824, -52.00845], [-68.41683, -52.33516], [-68.60702, -52.65781], [-68.60733, -54.9125], [-68.01394, -54.8753], [-67.46182, -54.92205], [-67.11046, -54.94199], [-66.07313, -55.19618], [-63.67376, -55.11859], [-54.78916, -36.21945], [-57.83001, -34.69099], [-58.34425, -34.15035], [-58.44442, -33.84033], [-58.40475, -33.11777], [-58.1224, -32.98842], [-58.22362, -32.52416], [-58.10036, -32.25338], [-58.20252, -31.86966], [-58.00076, -31.65016], [-58.0023, -31.53084], [-58.07569, -31.44916], [-57.98127, -31.3872], [-57.9908, -31.34924], [-57.86729, -31.06352], [-57.89476, -30.95994], [-57.8024, -30.77193], [-57.89115, -30.49572], [-57.64859, -30.35095], [-57.61478, -30.25165], [-57.65132, -30.19229], [-57.09386, -29.74211], [-56.81251, -29.48154], [-56.62789, -29.18073], [-56.57295, -29.11357], [-56.54171, -29.11447], [-56.05265, -28.62651], [-56.00458, -28.60421], [-56.01729, -28.51223], [-55.65418, -28.18304], [-55.6262, -28.17124], [-55.33303, -27.94661], [-55.16872, -27.86224], [-55.1349, -27.89759], [-54.90805, -27.73149], [-54.90159, -27.63132], [-54.67657, -27.57214], [-54.50416, -27.48232], [-54.41888, -27.40882], [-54.19268, -27.30751], [-54.19062, -27.27639], [-54.15978, -27.2889], [-53.80144, -27.09844], [-53.73372, -26.6131], [-53.68269, -26.33359], [-53.64505, -26.28089], [-53.64186, -26.25976], [-53.64632, -26.24798], [-53.63881, -26.25075], [-53.63739, -26.2496], [-53.65237, -26.23289], [-53.65018, -26.19501], [-53.73968, -26.10012], [-53.73391, -26.07006], [-53.7264, -26.0664], [-53.73086, -26.05842], [-53.73511, -26.04211], [-53.83691, -25.94849], [-53.90831, -25.55513], [-54.52926, -25.62846], [-54.5502, -25.58915], [-54.59398, -25.59224], [-54.62063, -25.91213], [-54.60664, -25.9691], [-54.67359, -25.98607], [-54.69333, -26.37705], [-54.70732, -26.45099], [-54.80868, -26.55669], [-55.00584, -26.78754], [-55.06351, -26.80195], [-55.16948, -26.96068], [-55.25243, -26.93808], [-55.39611, -26.97679], [-55.62322, -27.1941], [-55.59094, -27.32444], [-55.74475, -27.44485], [-55.89195, -27.3467], [-56.18313, -27.29851], [-56.85337, -27.5165], [-58.04205, -27.2387], [-58.59549, -27.29973], [-58.65321, -27.14028], [-58.3198, -26.83443], [-58.1188, -26.16704], [-57.87176, -25.93604], [-57.57431, -25.47269], [-57.80821, -25.13863], [-58.25492, -24.92528], [-58.33055, -24.97099], [-59.33886, -24.49935], [-59.45482, -24.34787], [-60.03367, -24.00701], [-60.28163, -24.04436], [-60.99754, -23.80934], [-61.0782, -23.62932], [-61.9756, -23.0507], [-62.22768, -22.55807], [-62.51761, -22.37684], [-62.64455, -22.25091], [-62.8078, -22.12534], [-62.81124, -21.9987], [-63.66482, -21.99918], [-63.68113, -22.0544], [-63.70963, -21.99934], [-63.93287, -21.99934], [-64.22918, -22.55807], [-64.31489, -22.88824], [-64.35108, -22.73282], [-64.4176, -22.67692], [-64.58888, -22.25035], [-64.67174, -22.18957], [-64.90014, -22.12136], [-64.99524, -22.08255], [-65.47435, -22.08908], [-65.57743, -22.07675], [-65.58694, -22.09794], [-65.61166, -22.09504], [-65.7467, -22.10105], [-65.9261, -21.93335], [-66.04832, -21.9187], [-66.03836, -21.84829], [-66.24077, -21.77837], [-66.29714, -22.08741], [-66.7298, -22.23644], [-67.18382, -22.81525], [-66.99632, -22.99839], [-67.33563, -24.04237], [-68.24825, -24.42596], [-68.56909, -24.69831], [-68.38372, -25.08636], [-68.57622, -25.32505], [-68.38372, -26.15353], [-68.56909, -26.28146], [-68.59048, -26.49861], [-68.27677, -26.90626], [-68.43363, -27.08414], [-68.77586, -27.16029], [-69.22504, -27.95042], [-69.66709, -28.44055], [-69.80969, -29.07185], [-69.99507, -29.28351], [-69.8596, -30.26131], [-70.14479, -30.36595], [-70.55832, -31.51559], [-69.88099, -33.34489], [-69.87386, -34.13344], [-70.49416, -35.24145], [-70.38008, -36.02375], [-70.95047, -36.4321], [-71.24279, -37.20264], [-70.89532, -38.6923], [-71.37826, -38.91474], [-71.92726, -40.72714], [-71.74901, -42.11711], [-72.15541, -42.15941], [-72.14828, -42.85321], [-71.64206, -43.64774], [-71.81318, -44.38097], [-71.16436, -44.46244], [-71.26418, -44.75684], [-72.06985, -44.81756], [-71.35687, -45.22075], [-71.75614, -45.61611], [-71.68577, -46.55385], [-71.94152, -47.13595], [-72.50478, -47.80586], [-72.27662, -48.28727], [-72.54042, -48.52392], [-72.56894, -48.81116], [-73.09655, -49.14342], [-73.45156, -49.79461], [-73.55259, -49.92488], [-73.15765, -50.78337], [-72.31343, -50.58411]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AS",
+           iso1A3: "ASM",
+           iso1N3: "016",
+           wikidata: "Q16641",
+           nameEn: "American Samoa",
+           aliases: ["US-AS"],
+           country: "US",
+           groups: ["Q1352230", "061", "009", "UN"],
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["1 684"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-171.39864, -10.21587], [-170.99605, -15.1275], [-166.32598, -15.26169], [-171.39864, -10.21587]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AT",
+           iso1A3: "AUT",
+           iso1N3: "040",
+           wikidata: "Q40",
+           nameEn: "Austria",
+           groups: ["EU", "155", "150", "UN"],
+           callingCodes: ["43"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[15.34823, 48.98444], [15.28305, 48.98831], [15.26177, 48.95766], [15.16358, 48.94278], [15.15534, 48.99056], [14.99878, 49.01444], [14.97612, 48.96983], [14.98917, 48.90082], [14.95072, 48.79101], [14.98032, 48.77959], [14.9782, 48.7766], [14.98112, 48.77524], [14.9758, 48.76857], [14.95641, 48.75915], [14.94773, 48.76268], [14.81545, 48.7874], [14.80821, 48.77711], [14.80584, 48.73489], [14.72756, 48.69502], [14.71794, 48.59794], [14.66762, 48.58215], [14.60808, 48.62881], [14.56139, 48.60429], [14.4587, 48.64695], [14.43076, 48.58855], [14.33909, 48.55852], [14.20691, 48.5898], [14.09104, 48.5943], [14.01482, 48.63788], [14.06151, 48.66873], [13.84023, 48.76988], [13.82266, 48.75544], [13.81863, 48.73257], [13.79337, 48.71375], [13.81791, 48.69832], [13.81283, 48.68426], [13.81901, 48.6761], [13.82609, 48.62345], [13.80038, 48.59487], [13.80519, 48.58026], [13.76921, 48.55324], [13.7513, 48.5624], [13.74816, 48.53058], [13.72802, 48.51208], [13.66113, 48.53558], [13.65186, 48.55092], [13.62508, 48.55501], [13.59705, 48.57013], [13.57535, 48.55912], [13.51291, 48.59023], [13.50131, 48.58091], [13.50663, 48.57506], [13.46967, 48.55157], [13.45214, 48.56472], [13.43695, 48.55776], [13.45727, 48.51092], [13.42527, 48.45711], [13.43929, 48.43386], [13.40709, 48.37292], [13.30897, 48.31575], [13.26039, 48.29422], [13.18093, 48.29577], [13.126, 48.27867], [13.0851, 48.27711], [13.02083, 48.25689], [12.95306, 48.20629], [12.87126, 48.20318], [12.84475, 48.16556], [12.836, 48.1647], [12.8362, 48.15876], [12.82673, 48.15245], [12.80676, 48.14979], [12.78595, 48.12445], [12.7617, 48.12796], [12.74973, 48.10885], [12.76141, 48.07373], [12.8549, 48.01122], [12.87476, 47.96195], [12.91683, 47.95647], [12.9211, 47.95135], [12.91985, 47.94069], [12.92668, 47.93879], [12.93419, 47.94063], [12.93642, 47.94436], [12.93886, 47.94046], [12.94163, 47.92927], [13.00588, 47.84374], [12.98543, 47.82896], [12.96311, 47.79957], [12.93202, 47.77302], [12.94371, 47.76281], [12.9353, 47.74788], [12.91711, 47.74026], [12.90274, 47.72513], [12.91333, 47.7178], [12.92969, 47.71094], [12.98578, 47.7078], [13.01382, 47.72116], [13.07692, 47.68814], [13.09562, 47.63304], [13.06407, 47.60075], [13.06641, 47.58577], [13.04537, 47.58183], [13.05355, 47.56291], [13.03252, 47.53373], [13.04537, 47.49426], [12.9998, 47.46267], [12.98344, 47.48716], [12.9624, 47.47452], [12.85256, 47.52741], [12.84672, 47.54556], [12.80699, 47.54477], [12.77427, 47.58025], [12.82101, 47.61493], [12.76492, 47.64485], [12.77777, 47.66689], [12.7357, 47.6787], [12.6071, 47.6741], [12.57438, 47.63238], [12.53816, 47.63553], [12.50076, 47.62293], [12.44117, 47.6741], [12.43883, 47.6977], [12.37222, 47.68433], [12.336, 47.69534], [12.27991, 47.68827], [12.26004, 47.67725], [12.24017, 47.69534], [12.26238, 47.73544], [12.2542, 47.7433], [12.22571, 47.71776], [12.18303, 47.70065], [12.16217, 47.70105], [12.16769, 47.68167], [12.18347, 47.66663], [12.18507, 47.65984], [12.19895, 47.64085], [12.20801, 47.61082], [12.20398, 47.60667], [12.18568, 47.6049], [12.17737, 47.60121], [12.18145, 47.61019], [12.17824, 47.61506], [12.13734, 47.60639], [12.05788, 47.61742], [12.02282, 47.61033], [12.0088, 47.62451], [11.85572, 47.60166], [11.84052, 47.58354], [11.63934, 47.59202], [11.60681, 47.57881], [11.58811, 47.55515], [11.58578, 47.52281], [11.52618, 47.50939], [11.4362, 47.51413], [11.38128, 47.47465], [11.4175, 47.44621], [11.33804, 47.44937], [11.29597, 47.42566], [11.27844, 47.39956], [11.22002, 47.3964], [11.25157, 47.43277], [11.20482, 47.43198], [11.12536, 47.41222], [11.11835, 47.39719], [10.97111, 47.39561], [10.97111, 47.41617], [10.98513, 47.42882], [10.92437, 47.46991], [10.93839, 47.48018], [10.90918, 47.48571], [10.87061, 47.4786], [10.86945, 47.5015], [10.91268, 47.51334], [10.88814, 47.53701], [10.77596, 47.51729], [10.7596, 47.53228], [10.6965, 47.54253], [10.68832, 47.55752], [10.63456, 47.5591], [10.60337, 47.56755], [10.56912, 47.53584], [10.48849, 47.54057], [10.47329, 47.58552], [10.43473, 47.58394], [10.44992, 47.5524], [10.4324, 47.50111], [10.44291, 47.48453], [10.46278, 47.47901], [10.47446, 47.43318], [10.4359, 47.41183], [10.4324, 47.38494], [10.39851, 47.37623], [10.33424, 47.30813], [10.23257, 47.27088], [10.17531, 47.27167], [10.17648, 47.29149], [10.2147, 47.31014], [10.19998, 47.32832], [10.23757, 47.37609], [10.22774, 47.38904], [10.2127, 47.38019], [10.17648, 47.38889], [10.16362, 47.36674], [10.11805, 47.37228], [10.09819, 47.35724], [10.06897, 47.40709], [10.1052, 47.4316], [10.09001, 47.46005], [10.07131, 47.45531], [10.03859, 47.48927], [10.00003, 47.48216], [9.96029, 47.53899], [9.92407, 47.53111], [9.87733, 47.54688], [9.87499, 47.52953], [9.8189, 47.54688], [9.82591, 47.58158], [9.80254, 47.59419], [9.76748, 47.5934], [9.72736, 47.53457], [9.55125, 47.53629], [9.56312, 47.49495], [9.58208, 47.48344], [9.59482, 47.46305], [9.60205, 47.46165], [9.60484, 47.46358], [9.60841, 47.47178], [9.62158, 47.45858], [9.62475, 47.45685], [9.6423, 47.45599], [9.65728, 47.45383], [9.65863, 47.44847], [9.64483, 47.43842], [9.6446, 47.43233], [9.65043, 47.41937], [9.65136, 47.40504], [9.6629, 47.39591], [9.67334, 47.39191], [9.67445, 47.38429], [9.6711, 47.37824], [9.66243, 47.37136], [9.65427, 47.36824], [9.62476, 47.36639], [9.59978, 47.34671], [9.58513, 47.31334], [9.55857, 47.29919], [9.54773, 47.2809], [9.53116, 47.27029], [9.56766, 47.24281], [9.55176, 47.22585], [9.56981, 47.21926], [9.58264, 47.20673], [9.56539, 47.17124], [9.62623, 47.14685], [9.63395, 47.08443], [9.61216, 47.07732], [9.60717, 47.06091], [9.87935, 47.01337], [9.88266, 46.93343], [9.98058, 46.91434], [10.10715, 46.84296], [10.22675, 46.86942], [10.24128, 46.93147], [10.30031, 46.92093], [10.36933, 47.00212], [10.48376, 46.93891], [10.47197, 46.85698], [10.54783, 46.84505], [10.66405, 46.87614], [10.75753, 46.82258], [10.72974, 46.78972], [11.00764, 46.76896], [11.10618, 46.92966], [11.33355, 46.99862], [11.50739, 47.00644], [11.74789, 46.98484], [12.19254, 47.09331], [12.21781, 47.03996], [12.11675, 47.01241], [12.2006, 46.88854], [12.27591, 46.88651], [12.38708, 46.71529], [12.59992, 46.6595], [12.94445, 46.60401], [13.27627, 46.56059], [13.64088, 46.53438], [13.7148, 46.5222], [13.89837, 46.52331], [14.00422, 46.48474], [14.04002, 46.49117], [14.12097, 46.47724], [14.15989, 46.43327], [14.28326, 46.44315], [14.314, 46.43327], [14.42608, 46.44614], [14.45877, 46.41717], [14.52176, 46.42617], [14.56463, 46.37208], [14.5942, 46.43434], [14.66892, 46.44936], [14.72185, 46.49974], [14.81836, 46.51046], [14.83549, 46.56614], [14.86419, 46.59411], [14.87129, 46.61], [14.92283, 46.60848], [14.96002, 46.63459], [14.98024, 46.6009], [15.01451, 46.641], [15.14215, 46.66131], [15.23711, 46.63994], [15.41235, 46.65556], [15.45514, 46.63697], [15.46906, 46.61321], [15.54431, 46.6312], [15.55333, 46.64988], [15.54533, 46.66985], [15.59826, 46.68908], [15.62317, 46.67947], [15.63255, 46.68069], [15.6365, 46.6894], [15.6543, 46.69228], [15.6543, 46.70616], [15.67411, 46.70735], [15.69523, 46.69823], [15.72279, 46.69548], [15.73823, 46.70011], [15.76771, 46.69863], [15.78518, 46.70712], [15.8162, 46.71897], [15.87691, 46.7211], [15.94864, 46.68769], [15.98512, 46.68463], [15.99988, 46.67947], [16.04036, 46.6549], [16.04347, 46.68694], [16.02808, 46.71094], [15.99769, 46.7266], [15.98432, 46.74991], [15.99126, 46.78199], [15.99054, 46.82772], [16.05786, 46.83927], [16.10983, 46.867], [16.19904, 46.94134], [16.22403, 46.939], [16.27594, 46.9643], [16.28202, 47.00159], [16.51369, 47.00084], [16.43936, 47.03548], [16.52176, 47.05747], [16.46134, 47.09395], [16.52863, 47.13974], [16.44932, 47.14418], [16.46442, 47.16845], [16.4523, 47.18812], [16.42801, 47.18422], [16.41739, 47.20649], [16.43663, 47.21127], [16.44142, 47.25079], [16.47782, 47.25918], [16.45104, 47.41181], [16.49908, 47.39416], [16.52414, 47.41007], [16.57152, 47.40868], [16.6718, 47.46139], [16.64821, 47.50155], [16.71059, 47.52692], [16.64193, 47.63114], [16.58699, 47.61772], [16.4222, 47.66537], [16.55129, 47.72268], [16.53514, 47.73837], [16.54779, 47.75074], [16.61183, 47.76171], [16.65679, 47.74197], [16.72089, 47.73469], [16.7511, 47.67878], [16.82938, 47.68432], [16.86509, 47.72268], [16.87538, 47.68895], [17.08893, 47.70928], [17.05048, 47.79377], [17.07039, 47.81129], [17.00997, 47.86245], [17.08275, 47.87719], [17.11022, 47.92461], [17.09786, 47.97336], [17.16001, 48.00636], [17.07039, 48.0317], [17.09168, 48.09366], [17.05735, 48.14179], [17.02919, 48.13996], [16.97701, 48.17385], [16.89461, 48.31332], [16.90903, 48.32519], [16.84243, 48.35258], [16.83317, 48.38138], [16.83588, 48.3844], [16.8497, 48.38321], [16.85204, 48.44968], [16.94611, 48.53614], [16.93955, 48.60371], [16.90354, 48.71541], [16.79779, 48.70998], [16.71883, 48.73806], [16.68518, 48.7281], [16.67008, 48.77699], [16.46134, 48.80865], [16.40915, 48.74576], [16.37345, 48.729], [16.06034, 48.75436], [15.84404, 48.86921], [15.78087, 48.87644], [15.75341, 48.8516], [15.6921, 48.85973], [15.61622, 48.89541], [15.51357, 48.91549], [15.48027, 48.94481], [15.34823, 48.98444]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AU",
+           iso1A3: "AUS",
+           iso1N3: "036",
+           wikidata: "Q408",
+           nameEn: "Australia"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AW",
+           iso1A3: "ABW",
+           iso1N3: "533",
+           wikidata: "Q21203",
+           nameEn: "Aruba",
+           aliases: ["NL-AW"],
+           country: "NL",
+           groups: ["Q1451600", "029", "003", "419", "019", "UN"],
+           callingCodes: ["297"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-70.00823, 12.98375], [-70.35625, 12.58277], [-69.60231, 12.17], [-70.00823, 12.98375]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AX",
+           iso1A3: "ALA",
+           iso1N3: "248",
+           wikidata: "Q5689",
+           nameEn: "\xC5land Islands",
+           country: "FI",
+           groups: ["EU", "154", "150", "UN"],
+           callingCodes: ["358 18", "358 457"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[19.08191, 60.19152], [20.5104, 59.15546], [21.35468, 59.67511], [21.02509, 60.12142], [21.08159, 60.20167], [21.15143, 60.54555], [20.96741, 60.71528], [19.23413, 60.61414], [19.08191, 60.19152]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "AZ",
+           iso1A3: "AZE",
+           iso1N3: "031",
+           wikidata: "Q227",
+           nameEn: "Azerbaijan",
+           groups: ["145", "142", "UN"],
+           callingCodes: ["994"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[46.42738, 41.91323], [46.3984, 41.84399], [46.30863, 41.79133], [46.23962, 41.75811], [46.20538, 41.77205], [46.17891, 41.72094], [46.19759, 41.62327], [46.24429, 41.59883], [46.26531, 41.63339], [46.28182, 41.60089], [46.3253, 41.60912], [46.34039, 41.5947], [46.34126, 41.57454], [46.29794, 41.5724], [46.33925, 41.4963], [46.40307, 41.48464], [46.4669, 41.43331], [46.63658, 41.37727], [46.72375, 41.28609], [46.66148, 41.20533], [46.63969, 41.09515], [46.55096, 41.1104], [46.48558, 41.0576], [46.456, 41.09984], [46.37661, 41.10805], [46.27698, 41.19011], [46.13221, 41.19479], [45.95786, 41.17956], [45.80842, 41.2229], [45.69946, 41.29545], [45.75705, 41.35157], [45.71035, 41.36208], [45.68389, 41.3539], [45.45973, 41.45898], [45.4006, 41.42402], [45.31352, 41.47168], [45.26285, 41.46433], [45.1797, 41.42231], [45.09867, 41.34065], [45.0133, 41.29747], [45.05497, 41.2464], [45.02932, 41.2101], [45.05201, 41.19211], [45.11811, 41.19967], [45.1969, 41.168], [45.19942, 41.13299], [45.08028, 41.10917], [45.06784, 41.05379], [45.12923, 41.06059], [45.1313, 41.09369], [45.1634, 41.08082], [45.16493, 41.05068], [45.1994, 41.04518], [45.25897, 41.0027], [45.26162, 41.0228], [45.28859, 41.03757], [45.35677, 40.99784], [45.39725, 41.02603], [45.44083, 41.01663], [45.40814, 40.97904], [45.60584, 40.87436], [45.55914, 40.78366], [45.4206, 40.7424], [45.35366, 40.65979], [45.45484, 40.57707], [45.42994, 40.53804], [45.65098, 40.37696], [45.95609, 40.27846], [45.97944, 40.181], [45.83779, 39.98925], [45.78642, 40.03218], [45.59806, 40.0131], [45.60895, 39.97733], [45.7833, 39.9475], [45.82533, 39.82925], [45.96543, 39.78859], [46.18493, 39.60533], [46.40286, 39.63651], [46.42465, 39.57534], [46.52117, 39.58734], [46.57098, 39.56694], [46.57721, 39.54414], [46.51027, 39.52373], [46.53051, 39.47809], [46.4013, 39.45405], [46.37795, 39.42039], [46.43244, 39.35181], [46.50093, 39.33736], [46.56476, 39.24942], [46.63481, 39.23013], [46.58032, 39.21204], [46.54141, 39.15895], [46.52584, 39.18912], [46.44022, 39.19636], [46.54296, 39.07078], [46.51805, 38.94982], [46.53497, 38.86548], [46.75752, 39.03231], [46.83822, 39.13143], [46.92539, 39.16644], [46.95341, 39.13505], [47.05771, 39.20143], [47.05927, 39.24846], [47.31301, 39.37492], [47.38978, 39.45999], [47.50099, 39.49615], [47.84774, 39.66285], [47.98977, 39.70999], [48.34264, 39.42935], [48.37385, 39.37584], [48.15984, 39.30028], [48.12404, 39.25208], [48.15361, 39.19419], [48.31239, 39.09278], [48.33884, 39.03022], [48.28437, 38.97186], [48.08627, 38.94434], [48.07734, 38.91616], [48.01409, 38.90333], [48.02581, 38.82705], [48.24773, 38.71883], [48.3146, 38.59958], [48.45084, 38.61013], [48.58793, 38.45076], [48.62217, 38.40198], [48.70001, 38.40564], [48.78979, 38.45026], [48.81072, 38.44853], [48.84969, 38.45015], [48.88288, 38.43975], [52.39847, 39.43556], [48.80971, 41.95365], [48.5867, 41.84306], [48.55078, 41.77917], [48.42301, 41.65444], [48.40277, 41.60441], [48.2878, 41.56221], [48.22064, 41.51472], [48.07587, 41.49957], [47.87973, 41.21798], [47.75831, 41.19455], [47.62288, 41.22969], [47.54504, 41.20275], [47.49004, 41.26366], [47.34579, 41.27884], [47.10762, 41.59044], [47.03757, 41.55434], [46.99554, 41.59743], [47.00955, 41.63583], [46.8134, 41.76252], [46.75269, 41.8623], [46.58924, 41.80547], [46.5332, 41.87389], [46.42738, 41.91323]], [[45.50279, 40.58424], [45.47927, 40.65023], [45.51825, 40.67382], [45.56071, 40.64765], [45.50279, 40.58424]]], [[[45.00864, 41.03411], [45.03792, 41.03938], [45.04517, 41.06653], [45.03406, 41.07931], [45.00864, 41.09407], [44.97169, 41.09176], [44.95383, 41.07553], [44.96031, 41.06345], [44.9903, 41.05657], [45.00864, 41.03411]]], [[[45.21324, 40.9817], [45.23095, 40.97828], [45.23487, 41.00226], [45.20625, 41.01484], [45.18382, 41.0066], [45.19312, 40.98998], [45.20518, 40.99348], [45.21219, 40.99001], [45.21324, 40.9817]]], [[[45.46992, 39.49888], [45.29606, 39.57654], [45.30385, 39.61373], [45.23535, 39.61373], [45.21784, 39.58074], [45.17464, 39.58614], [45.18554, 39.67846], [45.06604, 39.79277], [44.92869, 39.72157], [44.88354, 39.74432], [44.75779, 39.7148], [44.80977, 39.65768], [44.81043, 39.62677], [44.88916, 39.59653], [44.96746, 39.42998], [45.05932, 39.36435], [45.08751, 39.35052], [45.16168, 39.21952], [45.30489, 39.18333], [45.40148, 39.09007], [45.40452, 39.07224], [45.44811, 39.04927], [45.44966, 38.99243], [45.6131, 38.964], [45.6155, 38.94304], [45.65172, 38.95199], [45.83883, 38.90768], [45.90266, 38.87739], [45.94624, 38.89072], [46.00228, 38.87376], [46.06766, 38.87861], [46.14785, 38.84206], [46.06973, 39.0744], [46.02303, 39.09978], [45.99774, 39.28931], [45.79225, 39.3695], [45.83, 39.46487], [45.80804, 39.56716], [45.70547, 39.60174], [45.46992, 39.49888]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BA",
+           iso1A3: "BIH",
+           iso1N3: "070",
+           wikidata: "Q225",
+           nameEn: "Bosnia and Herzegovina",
+           groups: ["039", "150", "UN"],
+           callingCodes: ["387"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[17.84826, 45.04489], [17.66571, 45.13408], [17.59104, 45.10816], [17.51469, 45.10791], [17.47589, 45.12656], [17.45615, 45.12523], [17.4498, 45.16119], [17.41229, 45.13335], [17.33573, 45.14521], [17.32092, 45.16246], [17.26815, 45.18444], [17.25131, 45.14957], [17.24325, 45.146], [17.18438, 45.14764], [17.0415, 45.20759], [16.9385, 45.22742], [16.92405, 45.27607], [16.83804, 45.18951], [16.81137, 45.18434], [16.78219, 45.19002], [16.74845, 45.20393], [16.64962, 45.20714], [16.60194, 45.23042], [16.56559, 45.22307], [16.5501, 45.2212], [16.52982, 45.22713], [16.49155, 45.21153], [16.4634, 45.14522], [16.40023, 45.1147], [16.38309, 45.05955], [16.38219, 45.05139], [16.3749, 45.05206], [16.35863, 45.03529], [16.35404, 45.00241], [16.29036, 44.99732], [16.12153, 45.09616], [15.98412, 45.23088], [15.83512, 45.22459], [15.76371, 45.16508], [15.78842, 45.11519], [15.74585, 45.0638], [15.78568, 44.97401], [15.74723, 44.96818], [15.76096, 44.87045], [15.79472, 44.8455], [15.72584, 44.82334], [15.8255, 44.71501], [15.89348, 44.74964], [16.05828, 44.61538], [16.00884, 44.58605], [16.03012, 44.55572], [16.10566, 44.52586], [16.16814, 44.40679], [16.12969, 44.38275], [16.21346, 44.35231], [16.18688, 44.27012], [16.36864, 44.08263], [16.43662, 44.07523], [16.43629, 44.02826], [16.50528, 44.0244], [16.55472, 43.95326], [16.70922, 43.84887], [16.75316, 43.77157], [16.80736, 43.76011], [17.00585, 43.58037], [17.15828, 43.49376], [17.24411, 43.49376], [17.29699, 43.44542], [17.25579, 43.40353], [17.286, 43.33065], [17.46986, 43.16559], [17.64268, 43.08595], [17.70879, 42.97223], [17.5392, 42.92787], [17.6444, 42.88641], [17.68151, 42.92725], [17.7948, 42.89556], [17.80854, 42.9182], [17.88201, 42.83668], [18.24318, 42.6112], [18.36197, 42.61423], [18.43735, 42.55921], [18.49778, 42.58409], [18.53751, 42.57376], [18.55504, 42.58409], [18.52232, 42.62279], [18.57373, 42.64429], [18.54841, 42.68328], [18.54603, 42.69171], [18.55221, 42.69045], [18.56789, 42.72074], [18.47324, 42.74992], [18.45921, 42.81682], [18.47633, 42.85829], [18.4935, 42.86433], [18.49661, 42.89306], [18.49076, 42.95553], [18.52232, 43.01451], [18.66254, 43.03928], [18.64735, 43.14766], [18.66605, 43.2056], [18.71747, 43.2286], [18.6976, 43.25243], [18.76538, 43.29838], [18.85342, 43.32426], [18.84794, 43.33735], [18.83912, 43.34795], [18.90911, 43.36383], [18.95819, 43.32899], [18.95001, 43.29327], [19.00844, 43.24988], [19.04233, 43.30008], [19.08206, 43.29668], [19.08673, 43.31453], [19.04071, 43.397], [19.01078, 43.43854], [18.96053, 43.45042], [18.95469, 43.49367], [18.91379, 43.50299], [19.01078, 43.55806], [19.04934, 43.50384], [19.13933, 43.5282], [19.15685, 43.53943], [19.22807, 43.5264], [19.24774, 43.53061], [19.2553, 43.5938], [19.33426, 43.58833], [19.36653, 43.60921], [19.41941, 43.54056], [19.42696, 43.57987], [19.50455, 43.58385], [19.5176, 43.71403], [19.3986, 43.79668], [19.23465, 43.98764], [19.24363, 44.01502], [19.38439, 43.96611], [19.52515, 43.95573], [19.56498, 43.99922], [19.61836, 44.01464], [19.61991, 44.05254], [19.57467, 44.04716], [19.55999, 44.06894], [19.51167, 44.08158], [19.47321, 44.1193], [19.48386, 44.14332], [19.47338, 44.15034], [19.43905, 44.13088], [19.40927, 44.16722], [19.3588, 44.18353], [19.34773, 44.23244], [19.32464, 44.27185], [19.26945, 44.26957], [19.23306, 44.26097], [19.20508, 44.2917], [19.18328, 44.28383], [19.16741, 44.28648], [19.13332, 44.31492], [19.13556, 44.338], [19.11547, 44.34218], [19.1083, 44.3558], [19.11865, 44.36712], [19.10298, 44.36924], [19.10365, 44.37795], [19.10704, 44.38249], [19.10749, 44.39421], [19.11785, 44.40313], [19.14681, 44.41463], [19.14837, 44.45253], [19.12278, 44.50132], [19.13369, 44.52521], [19.16699, 44.52197], [19.26388, 44.65412], [19.32543, 44.74058], [19.36722, 44.88164], [19.18183, 44.92055], [19.01994, 44.85493], [18.8704, 44.85097], [18.76347, 44.90669], [18.76369, 44.93707], [18.80661, 44.93561], [18.78357, 44.97741], [18.65723, 45.07544], [18.47939, 45.05871], [18.41896, 45.11083], [18.32077, 45.1021], [18.24387, 45.13699], [18.1624, 45.07654], [18.03121, 45.12632], [18.01594, 45.15163], [17.99479, 45.14958], [17.97834, 45.13831], [17.97336, 45.12245], [17.93706, 45.08016], [17.87148, 45.04645], [17.84826, 45.04489]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BB",
+           iso1A3: "BRB",
+           iso1N3: "052",
+           wikidata: "Q244",
+           nameEn: "Barbados",
+           groups: ["029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           callingCodes: ["1 246"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-58.56442, 13.24471], [-59.80731, 13.87556], [-59.82929, 12.70644], [-58.56442, 13.24471]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BD",
+           iso1A3: "BGD",
+           iso1N3: "050",
+           wikidata: "Q902",
+           nameEn: "Bangladesh",
+           groups: ["034", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["880"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[89.15869, 26.13708], [89.08899, 26.38845], [88.95612, 26.4564], [88.92357, 26.40711], [88.91321, 26.37984], [89.05328, 26.2469], [88.85004, 26.23211], [88.78961, 26.31093], [88.67837, 26.26291], [88.69485, 26.38353], [88.62144, 26.46783], [88.4298, 26.54489], [88.41196, 26.63837], [88.33093, 26.48929], [88.35153, 26.45241], [88.36938, 26.48683], [88.48749, 26.45855], [88.51649, 26.35923], [88.35153, 26.29123], [88.34757, 26.22216], [88.1844, 26.14417], [88.16581, 26.0238], [88.08804, 25.91334], [88.13138, 25.78773], [88.242, 25.80811], [88.45103, 25.66245], [88.4559, 25.59227], [88.677, 25.46959], [88.81296, 25.51546], [88.85278, 25.34679], [89.01105, 25.30303], [89.00463, 25.26583], [88.94067, 25.18534], [88.44766, 25.20149], [88.46277, 25.07468], [88.33917, 24.86803], [88.27325, 24.88796], [88.21832, 24.96642], [88.14004, 24.93529], [88.15515, 24.85806], [88.00683, 24.66477], [88.08786, 24.63232], [88.12296, 24.51301], [88.50934, 24.32474], [88.68801, 24.31464], [88.74841, 24.1959], [88.6976, 24.14703], [88.73743, 23.91751], [88.66189, 23.87607], [88.58087, 23.87105], [88.56507, 23.64044], [88.74841, 23.47361], [88.79351, 23.50535], [88.79254, 23.46028], [88.71133, 23.2492], [88.99148, 23.21134], [88.86377, 23.08759], [88.88327, 23.03885], [88.87063, 22.95235], [88.96713, 22.83346], [88.9151, 22.75228], [88.94614, 22.66941], [88.9367, 22.58527], [89.07114, 22.15335], [89.08044, 21.41871], [92.47409, 20.38654], [92.26071, 21.05697], [92.17752, 21.17445], [92.20087, 21.337], [92.37939, 21.47764], [92.43158, 21.37025], [92.55105, 21.3856], [92.60187, 21.24615], [92.68152, 21.28454], [92.59775, 21.6092], [92.62187, 21.87037], [92.60949, 21.97638], [92.56616, 22.13554], [92.60029, 22.1522], [92.5181, 22.71441], [92.37665, 22.9435], [92.38214, 23.28705], [92.26541, 23.70392], [92.15417, 23.73409], [92.04706, 23.64229], [91.95093, 23.73284], [91.95642, 23.47361], [91.84789, 23.42235], [91.76417, 23.26619], [91.81634, 23.08001], [91.7324, 23.00043], [91.61571, 22.93929], [91.54993, 23.01051], [91.46615, 23.2328], [91.4035, 23.27522], [91.40848, 23.07117], [91.36453, 23.06612], [91.28293, 23.37538], [91.15579, 23.6599], [91.25192, 23.83463], [91.22308, 23.89616], [91.29587, 24.0041], [91.35741, 23.99072], [91.37414, 24.10693], [91.55542, 24.08687], [91.63782, 24.1132], [91.65292, 24.22095], [91.73257, 24.14703], [91.76004, 24.23848], [91.82596, 24.22345], [91.89258, 24.14674], [91.96603, 24.3799], [92.11662, 24.38997], [92.15796, 24.54435], [92.25854, 24.9191], [92.38626, 24.86055], [92.49887, 24.88796], [92.39147, 25.01471], [92.33957, 25.07593], [92.0316, 25.1834], [91.63648, 25.12846], [91.25517, 25.20677], [90.87427, 25.15799], [90.65042, 25.17788], [90.40034, 25.1534], [90.1155, 25.22686], [89.90478, 25.31038], [89.87629, 25.28337], [89.83371, 25.29548], [89.84086, 25.31854], [89.81208, 25.37244], [89.86129, 25.61714], [89.84388, 25.70042], [89.80585, 25.82489], [89.86592, 25.93115], [89.77728, 26.04254], [89.77865, 26.08387], [89.73581, 26.15818], [89.70201, 26.15138], [89.63968, 26.22595], [89.57101, 25.9682], [89.53515, 26.00382], [89.35953, 26.0077], [89.15869, 26.13708]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BE",
+           iso1A3: "BEL",
+           iso1N3: "056",
+           wikidata: "Q31",
+           nameEn: "Belgium",
+           groups: ["EU", "155", "150", "UN"],
+           callingCodes: ["32"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[4.93295, 51.44945], [4.93909, 51.44632], [4.9524, 51.45014], [4.95244, 51.45207], [4.93295, 51.44945]]], [[[4.91493, 51.4353], [4.92652, 51.43329], [4.92952, 51.42984], [4.93986, 51.43064], [4.94265, 51.44003], [4.93471, 51.43861], [4.93416, 51.44185], [4.94025, 51.44193], [4.93544, 51.44634], [4.92879, 51.44161], [4.92815, 51.43856], [4.92566, 51.44273], [4.92811, 51.4437], [4.92287, 51.44741], [4.91811, 51.44621], [4.92227, 51.44252], [4.91935, 51.43634], [4.91493, 51.4353]]], [[[4.82946, 51.4213], [4.82409, 51.44736], [4.84139, 51.4799], [4.78803, 51.50284], [4.77321, 51.50529], [4.74578, 51.48937], [4.72935, 51.48424], [4.65442, 51.42352], [4.57489, 51.4324], [4.53521, 51.4243], [4.52846, 51.45002], [4.54675, 51.47265], [4.5388, 51.48184], [4.47736, 51.4778], [4.38122, 51.44905], [4.39747, 51.43316], [4.38064, 51.41965], [4.43777, 51.36989], [4.39292, 51.35547], [4.34086, 51.35738], [4.33265, 51.37687], [4.21923, 51.37443], [4.24024, 51.35371], [4.16721, 51.29348], [4.05165, 51.24171], [4.01957, 51.24504], [3.97889, 51.22537], [3.90125, 51.20371], [3.78783, 51.2151], [3.78999, 51.25766], [3.58939, 51.30064], [3.51502, 51.28697], [3.52698, 51.2458], [3.43488, 51.24135], [3.41704, 51.25933], [3.38289, 51.27331], [3.35847, 51.31572], [3.38696, 51.33436], [3.36263, 51.37112], [2.56575, 51.85301], [2.18458, 51.52087], [2.55904, 51.07014], [2.57551, 51.00326], [2.63074, 50.94746], [2.59093, 50.91751], [2.63331, 50.81457], [2.71165, 50.81295], [2.81056, 50.71773], [2.8483, 50.72276], [2.86985, 50.7033], [2.87937, 50.70298], [2.88504, 50.70656], [2.90069, 50.69263], [2.91036, 50.6939], [2.90873, 50.702], [2.95019, 50.75138], [2.96778, 50.75242], [3.00537, 50.76588], [3.04314, 50.77674], [3.09163, 50.77717], [3.10614, 50.78303], [3.11206, 50.79416], [3.11987, 50.79188], [3.1257, 50.78603], [3.15017, 50.79031], [3.16476, 50.76843], [3.18339, 50.74981], [3.18811, 50.74025], [3.20064, 50.73547], [3.19017, 50.72569], [3.20845, 50.71662], [3.22042, 50.71019], [3.24593, 50.71389], [3.26063, 50.70086], [3.26141, 50.69151], [3.2536, 50.68977], [3.264, 50.67668], [3.23951, 50.6585], [3.2729, 50.60718], [3.28575, 50.52724], [3.37693, 50.49538], [3.44629, 50.51009], [3.47385, 50.53397], [3.51564, 50.5256], [3.49509, 50.48885], [3.5683, 50.50192], [3.58361, 50.49049], [3.61014, 50.49568], [3.64426, 50.46275], [3.66153, 50.45165], [3.67494, 50.40239], [3.67262, 50.38663], [3.65709, 50.36873], [3.66976, 50.34563], [3.71009, 50.30305], [3.70987, 50.3191], [3.73911, 50.34809], [3.84314, 50.35219], [3.90781, 50.32814], [3.96771, 50.34989], [4.0268, 50.35793], [4.0689, 50.3254], [4.10237, 50.31247], [4.10957, 50.30234], [4.11954, 50.30425], [4.13665, 50.25609], [4.16808, 50.25786], [4.15524, 50.2833], [4.17347, 50.28838], [4.17861, 50.27443], [4.20651, 50.27333], [4.21945, 50.25539], [4.15524, 50.21103], [4.16014, 50.19239], [4.13561, 50.13078], [4.20147, 50.13535], [4.23101, 50.06945], [4.16294, 50.04719], [4.13508, 50.01976], [4.14239, 49.98034], [4.20532, 49.95803], [4.31963, 49.97043], [4.35051, 49.95315], [4.43488, 49.94122], [4.51098, 49.94659], [4.5414, 49.96911], [4.68695, 49.99685], [4.70064, 50.09384], [4.75237, 50.11314], [4.82438, 50.16878], [4.83279, 50.15331], [4.88602, 50.15182], [4.8382, 50.06738], [4.78827, 49.95609], [4.88529, 49.9236], [4.85134, 49.86457], [4.86965, 49.82271], [4.85464, 49.78995], [4.96714, 49.79872], [5.09249, 49.76193], [5.14545, 49.70287], [5.26232, 49.69456], [5.31465, 49.66846], [5.33039, 49.6555], [5.30214, 49.63055], [5.3137, 49.61225], [5.33851, 49.61599], [5.34837, 49.62889], [5.3974, 49.61596], [5.43713, 49.5707], [5.46734, 49.52648], [5.46541, 49.49825], [5.55001, 49.52729], [5.60909, 49.51228], [5.64505, 49.55146], [5.75649, 49.54321], [5.7577, 49.55915], [5.77435, 49.56298], [5.79195, 49.55228], [5.81838, 49.54777], [5.84143, 49.5533], [5.84692, 49.55663], [5.8424, 49.56082], [5.87256, 49.57539], [5.86986, 49.58756], [5.84971, 49.58674], [5.84826, 49.5969], [5.8762, 49.60898], [5.87609, 49.62047], [5.88393, 49.62802], [5.88552, 49.63507], [5.90599, 49.63853], [5.90164, 49.6511], [5.9069, 49.66377], [5.86175, 49.67862], [5.86527, 49.69291], [5.88677, 49.70951], [5.86503, 49.72739], [5.84193, 49.72161], [5.82562, 49.72395], [5.83149, 49.74729], [5.82245, 49.75048], [5.78871, 49.7962], [5.75409, 49.79239], [5.74953, 49.81428], [5.74364, 49.82058], [5.74844, 49.82435], [5.7404, 49.83452], [5.74076, 49.83823], [5.74975, 49.83933], [5.74953, 49.84709], [5.75884, 49.84811], [5.74567, 49.85368], [5.75861, 49.85631], [5.75269, 49.8711], [5.78415, 49.87922], [5.73621, 49.89796], [5.77314, 49.93646], [5.77291, 49.96056], [5.80833, 49.96451], [5.81163, 49.97142], [5.83467, 49.97823], [5.83968, 49.9892], [5.82331, 49.99662], [5.81866, 50.01286], [5.8551, 50.02683], [5.86904, 50.04614], [5.85474, 50.06342], [5.8857, 50.07824], [5.89488, 50.11476], [5.95929, 50.13295], [5.96453, 50.17259], [6.02488, 50.18283], [6.03093, 50.16362], [6.06406, 50.15344], [6.08577, 50.17246], [6.12028, 50.16374], [6.1137, 50.13668], [6.1379, 50.12964], [6.15298, 50.14126], [6.14132, 50.14971], [6.14588, 50.17106], [6.18739, 50.1822], [6.18364, 50.20815], [6.16853, 50.2234], [6.208, 50.25179], [6.28797, 50.27458], [6.29949, 50.30887], [6.32488, 50.32333], [6.35701, 50.31139], [6.40641, 50.32425], [6.40785, 50.33557], [6.3688, 50.35898], [6.34406, 50.37994], [6.36852, 50.40776], [6.37219, 50.45397], [6.34005, 50.46083], [6.3465, 50.48833], [6.30809, 50.50058], [6.26637, 50.50272], [6.22335, 50.49578], [6.20599, 50.52089], [6.19193, 50.5212], [6.18716, 50.52653], [6.19579, 50.5313], [6.19735, 50.53576], [6.17802, 50.54179], [6.17739, 50.55875], [6.20281, 50.56952], [6.22581, 50.5907], [6.24005, 50.58732], [6.24888, 50.59869], [6.2476, 50.60392], [6.26957, 50.62444], [6.17852, 50.6245], [6.11707, 50.72231], [6.04428, 50.72861], [6.0406, 50.71848], [6.0326, 50.72647], [6.03889, 50.74618], [6.01976, 50.75398], [5.97545, 50.75441], [5.95942, 50.7622], [5.89132, 50.75124], [5.89129, 50.75125], [5.88734, 50.77092], [5.84888, 50.75448], [5.84548, 50.76542], [5.80673, 50.7558], [5.77513, 50.78308], [5.76533, 50.78159], [5.74356, 50.7691], [5.73904, 50.75674], [5.72216, 50.76398], [5.69469, 50.75529], [5.68091, 50.75804], [5.70107, 50.7827], [5.68995, 50.79641], [5.70118, 50.80764], [5.65259, 50.82309], [5.64009, 50.84742], [5.64504, 50.87107], [5.67886, 50.88142], [5.69858, 50.91046], [5.71626, 50.90796], [5.72644, 50.91167], [5.72545, 50.92312], [5.74644, 50.94723], [5.75927, 50.95601], [5.74752, 50.96202], [5.72875, 50.95428], [5.71864, 50.96092], [5.76242, 50.99703], [5.77688, 51.02483], [5.75961, 51.03113], [5.77258, 51.06196], [5.79835, 51.05834], [5.79903, 51.09371], [5.82921, 51.09328], [5.83226, 51.10585], [5.8109, 51.10861], [5.80798, 51.11661], [5.85508, 51.14445], [5.82564, 51.16753], [5.77697, 51.1522], [5.77735, 51.17845], [5.74617, 51.18928], [5.70344, 51.1829], [5.65528, 51.18736], [5.65145, 51.19788], [5.5603, 51.22249], [5.5569, 51.26544], [5.515, 51.29462], [5.48476, 51.30053], [5.46519, 51.2849], [5.4407, 51.28169], [5.41672, 51.26248], [5.347, 51.27502], [5.33886, 51.26314], [5.29716, 51.26104], [5.26461, 51.26693], [5.23814, 51.26064], [5.22542, 51.26888], [5.24244, 51.30495], [5.2002, 51.32243], [5.16222, 51.31035], [5.13377, 51.31592], [5.13105, 51.34791], [5.07102, 51.39469], [5.10456, 51.43163], [5.07891, 51.4715], [5.04774, 51.47022], [5.03281, 51.48679], [5.0106, 51.47167], [5.00393, 51.44406], [4.92152, 51.39487], [4.90016, 51.41404], [4.84988, 51.41502], [4.78941, 51.41102], [4.77229, 51.41337], [4.76577, 51.43046], [4.78314, 51.43319], [4.82946, 51.4213]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BF",
+           iso1A3: "BFA",
+           iso1N3: "854",
+           wikidata: "Q965",
+           nameEn: "Burkina Faso",
+           groups: ["011", "202", "002", "UN"],
+           callingCodes: ["226"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[0.23859, 15.00135], [0.06588, 14.96961], [-0.24673, 15.07805], [-0.72004, 15.08655], [-1.05875, 14.7921], [-1.32166, 14.72774], [-1.68083, 14.50023], [-1.97945, 14.47709], [-1.9992, 14.19011], [-2.10223, 14.14878], [-2.47587, 14.29671], [-2.66175, 14.14713], [-2.84667, 14.05532], [-2.90831, 13.81174], [-2.88189, 13.64921], [-3.26407, 13.70699], [-3.28396, 13.5422], [-3.23599, 13.29035], [-3.43507, 13.27272], [-3.4313, 13.1588], [-3.54454, 13.1781], [-3.7911, 13.36665], [-3.96282, 13.38164], [-3.90558, 13.44375], [-3.96501, 13.49778], [-4.34477, 13.12927], [-4.21819, 12.95722], [-4.238, 12.71467], [-4.47356, 12.71252], [-4.41412, 12.31922], [-4.57703, 12.19875], [-4.54841, 12.1385], [-4.62546, 12.13204], [-4.62987, 12.06531], [-4.70692, 12.06746], [-4.72893, 12.01579], [-5.07897, 11.97918], [-5.26389, 11.84778], [-5.40258, 11.8327], [-5.26389, 11.75728], [-5.29251, 11.61715], [-5.22867, 11.60421], [-5.20665, 11.43811], [-5.25509, 11.36905], [-5.25949, 11.24816], [-5.32553, 11.21578], [-5.32994, 11.13371], [-5.49284, 11.07538], [-5.41579, 10.84628], [-5.47083, 10.75329], [-5.46643, 10.56074], [-5.51058, 10.43177], [-5.39602, 10.2929], [-5.12465, 10.29788], [-4.96453, 9.99923], [-4.96621, 9.89132], [-4.6426, 9.70696], [-4.31392, 9.60062], [-4.25999, 9.76012], [-3.69703, 9.94279], [-3.31779, 9.91125], [-3.27228, 9.84981], [-3.19306, 9.93781], [-3.16609, 9.85147], [-3.00765, 9.74019], [-2.93012, 9.57403], [-2.76494, 9.40778], [-2.68802, 9.49343], [-2.76534, 9.56589], [-2.74174, 9.83172], [-2.83108, 10.40252], [-2.94232, 10.64281], [-2.83373, 11.0067], [-0.67143, 10.99811], [-0.61937, 10.91305], [-0.44298, 11.04292], [-0.42391, 11.11661], [-0.38219, 11.12596], [-0.35955, 11.07801], [-0.28566, 11.12713], [-0.27374, 11.17157], [-0.13493, 11.14075], [0.50388, 11.01011], [0.48852, 10.98561], [0.50521, 10.98035], [0.4958, 10.93269], [0.66104, 10.99964], [0.91245, 10.99597], [0.9813, 11.08876], [1.03409, 11.04719], [1.42823, 11.46822], [2.00988, 11.42227], [2.29983, 11.68254], [2.39723, 11.89473], [2.05785, 12.35539], [2.26349, 12.41915], [0.99167, 13.10727], [0.99253, 13.37515], [1.18873, 13.31771], [1.21217, 13.37853], [1.24516, 13.33968], [1.28509, 13.35488], [1.24429, 13.39373], [1.20088, 13.38951], [1.02813, 13.46635], [0.99514, 13.5668], [0.77637, 13.64442], [0.77377, 13.6866], [0.61924, 13.68491], [0.38051, 14.05575], [0.16936, 14.51654], [0.23859, 15.00135]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BG",
+           iso1A3: "BGR",
+           iso1N3: "100",
+           wikidata: "Q219",
+           nameEn: "Bulgaria",
+           groups: ["EU", "151", "150", "UN"],
+           callingCodes: ["359"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[23.05288, 43.79494], [22.85314, 43.84452], [22.83753, 43.88055], [22.87873, 43.9844], [23.01674, 44.01946], [23.04988, 44.07694], [22.67173, 44.21564], [22.61711, 44.16938], [22.61688, 44.06534], [22.41449, 44.00514], [22.35558, 43.81281], [22.41043, 43.69566], [22.47582, 43.6558], [22.53397, 43.47225], [22.82036, 43.33665], [22.89727, 43.22417], [23.00806, 43.19279], [22.98104, 43.11199], [22.89521, 43.03625], [22.78397, 42.98253], [22.74826, 42.88701], [22.54302, 42.87774], [22.43309, 42.82057], [22.4997, 42.74144], [22.43983, 42.56851], [22.55669, 42.50144], [22.51961, 42.3991], [22.47498, 42.3915], [22.45919, 42.33822], [22.34773, 42.31725], [22.38136, 42.30339], [22.47251, 42.20393], [22.50289, 42.19527], [22.51224, 42.15457], [22.67701, 42.06614], [22.86749, 42.02275], [22.90254, 41.87587], [22.96682, 41.77137], [23.01239, 41.76527], [23.03342, 41.71034], [22.95513, 41.63265], [22.96331, 41.35782], [22.93334, 41.34104], [23.1833, 41.31755], [23.21953, 41.33773], [23.22771, 41.37106], [23.31301, 41.40525], [23.33639, 41.36317], [23.40416, 41.39999], [23.52453, 41.40262], [23.63203, 41.37632], [23.67644, 41.41139], [23.76525, 41.40175], [23.80148, 41.43943], [23.89613, 41.45257], [23.91483, 41.47971], [23.96975, 41.44118], [24.06908, 41.46132], [24.06323, 41.53222], [24.10063, 41.54796], [24.18126, 41.51735], [24.27124, 41.57682], [24.30513, 41.51297], [24.52599, 41.56808], [24.61129, 41.42278], [24.71529, 41.41928], [24.8041, 41.34913], [24.82514, 41.4035], [24.86136, 41.39298], [24.90928, 41.40876], [24.942, 41.38685], [25.11611, 41.34212], [25.28322, 41.23411], [25.48187, 41.28506], [25.52394, 41.2798], [25.55082, 41.31667], [25.61042, 41.30614], [25.66183, 41.31316], [25.70507, 41.29209], [25.8266, 41.34563], [25.87919, 41.30526], [26.12926, 41.35878], [26.16548, 41.42278], [26.20288, 41.43943], [26.14796, 41.47533], [26.176, 41.50072], [26.17951, 41.55409], [26.14328, 41.55496], [26.15146, 41.60828], [26.07083, 41.64584], [26.06148, 41.70345], [26.16841, 41.74858], [26.21325, 41.73223], [26.22888, 41.74139], [26.2654, 41.71544], [26.30255, 41.70925], [26.35957, 41.71149], [26.32952, 41.73637], [26.33589, 41.76802], [26.36952, 41.82265], [26.53968, 41.82653], [26.57961, 41.90024], [26.56051, 41.92995], [26.62996, 41.97644], [26.79143, 41.97386], [26.95638, 42.00741], [27.03277, 42.0809], [27.08486, 42.08735], [27.19251, 42.06028], [27.22376, 42.10152], [27.27411, 42.10409], [27.45478, 41.96591], [27.52379, 41.93756], [27.55191, 41.90928], [27.69949, 41.97515], [27.81235, 41.94803], [27.83492, 41.99709], [27.91479, 41.97902], [28.02971, 41.98066], [28.32297, 41.98371], [29.24336, 43.70874], [28.23293, 43.76], [27.99558, 43.84193], [27.92008, 44.00761], [27.73468, 43.95326], [27.64542, 44.04958], [27.60834, 44.01206], [27.39757, 44.0141], [27.26845, 44.12602], [26.95141, 44.13555], [26.62712, 44.05698], [26.38764, 44.04356], [26.10115, 43.96908], [26.05584, 43.90925], [25.94911, 43.85745], [25.72792, 43.69263], [25.39528, 43.61866], [25.17144, 43.70261], [25.10718, 43.6831], [24.96682, 43.72693], [24.73542, 43.68523], [24.62281, 43.74082], [24.50264, 43.76314], [24.35364, 43.70211], [24.18149, 43.68218], [23.73978, 43.80627], [23.61687, 43.79289], [23.4507, 43.84936], [23.26772, 43.84843], [23.05288, 43.79494]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BH",
+           iso1A3: "BHR",
+           iso1N3: "048",
+           wikidata: "Q398",
+           nameEn: "Bahrain",
+           groups: ["145", "142", "UN"],
+           callingCodes: ["973"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[50.93865, 26.30758], [50.71771, 26.73086], [50.38162, 26.53976], [50.26923, 26.08243], [50.302, 25.87592], [50.57069, 25.57887], [50.80824, 25.54641], [50.7801, 25.595], [50.86149, 25.6965], [50.81266, 25.88946], [50.93865, 26.30758]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BI",
+           iso1A3: "BDI",
+           iso1N3: "108",
+           wikidata: "Q967",
+           nameEn: "Burundi",
+           groups: ["014", "202", "002", "UN"],
+           callingCodes: ["257"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[30.54501, -2.41404], [30.42933, -2.31064], [30.14034, -2.43626], [29.95911, -2.33348], [29.88237, -2.75105], [29.36805, -2.82933], [29.32234, -2.6483], [29.0562, -2.58632], [29.04081, -2.7416], [29.00167, -2.78523], [29.00404, -2.81978], [29.0505, -2.81774], [29.09119, -2.87871], [29.09797, -2.91935], [29.16037, -2.95457], [29.17258, -2.99385], [29.25633, -3.05471], [29.21463, -3.3514], [29.23708, -3.75856], [29.43673, -4.44845], [29.63827, -4.44681], [29.75109, -4.45836], [29.77289, -4.41733], [29.82885, -4.36153], [29.88172, -4.35743], [30.03323, -4.26631], [30.22042, -4.01738], [30.45915, -3.56532], [30.84165, -3.25152], [30.83823, -2.97837], [30.6675, -2.98987], [30.57926, -2.89791], [30.4987, -2.9573], [30.40662, -2.86151], [30.52747, -2.65841], [30.41789, -2.66266], [30.54501, -2.41404]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BJ",
+           iso1A3: "BEN",
+           iso1N3: "204",
+           wikidata: "Q962",
+           nameEn: "Benin",
+           aliases: ["DY"],
+           groups: ["011", "202", "002", "UN"],
+           callingCodes: ["229"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[3.59375, 11.70269], [3.48187, 11.86092], [3.31613, 11.88495], [3.25352, 12.01467], [2.83978, 12.40585], [2.6593, 12.30631], [2.37783, 12.24804], [2.39657, 12.10952], [2.45824, 11.98672], [2.39723, 11.89473], [2.29983, 11.68254], [2.00988, 11.42227], [1.42823, 11.46822], [1.03409, 11.04719], [0.9813, 11.08876], [0.91245, 10.99597], [0.8804, 10.803], [0.80358, 10.71459], [0.77666, 10.37665], [1.35507, 9.99525], [1.36624, 9.5951], [1.33675, 9.54765], [1.41746, 9.3226], [1.5649, 9.16941], [1.61838, 9.0527], [1.64249, 6.99562], [1.55877, 6.99737], [1.61812, 6.74843], [1.58105, 6.68619], [1.76906, 6.43189], [1.79826, 6.28221], [1.62913, 6.24075], [1.67336, 6.02702], [2.74181, 6.13349], [2.70566, 6.38038], [2.70464, 6.50831], [2.74334, 6.57291], [2.7325, 6.64057], [2.78204, 6.70514], [2.78823, 6.76356], [2.73405, 6.78508], [2.74024, 6.92802], [2.71702, 6.95722], [2.76965, 7.13543], [2.74489, 7.42565], [2.79442, 7.43486], [2.78668, 7.5116], [2.73405, 7.5423], [2.73095, 7.7755], [2.67523, 7.87825], [2.77907, 9.06924], [3.08017, 9.10006], [3.14147, 9.28375], [3.13928, 9.47167], [3.25093, 9.61632], [3.34726, 9.70696], [3.32099, 9.78032], [3.35383, 9.83641], [3.54429, 9.87739], [3.66908, 10.18136], [3.57275, 10.27185], [3.6844, 10.46351], [3.78292, 10.40538], [3.84243, 10.59316], [3.71505, 11.13015], [3.49175, 11.29765], [3.59375, 11.70269]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BL",
+           iso1A3: "BLM",
+           iso1N3: "652",
+           wikidata: "Q25362",
+           nameEn: "Saint-Barth\xE9lemy",
+           country: "FR",
+           groups: ["Q1451600", "029", "003", "419", "019", "UN"],
+           callingCodes: ["590"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-62.62718, 18.26185], [-63.1055, 17.86651], [-62.34423, 17.49165], [-62.62718, 18.26185]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BM",
+           iso1A3: "BMU",
+           iso1N3: "060",
+           wikidata: "Q23635",
+           nameEn: "Bermuda",
+           country: "GB",
+           groups: ["BOTS", "021", "003", "019", "UN"],
+           driveSide: "left",
+           callingCodes: ["1 441"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-63.20987, 32.6953], [-65.31453, 32.68437], [-65.63955, 31.43417], [-63.20987, 32.6953]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BN",
+           iso1A3: "BRN",
+           iso1N3: "096",
+           wikidata: "Q921",
+           nameEn: "Brunei",
+           groups: ["Q36117", "035", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["673"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[115.16236, 5.01011], [115.02521, 5.35005], [114.10166, 4.76112], [114.07448, 4.58441], [114.15813, 4.57], [114.26876, 4.49878], [114.32176, 4.34942], [114.32176, 4.2552], [114.4416, 4.27588], [114.49922, 4.13108], [114.64211, 4.00694], [114.78539, 4.12205], [114.88039, 4.4257], [114.83189, 4.42387], [114.77303, 4.72871], [114.8266, 4.75062], [114.88841, 4.81905], [114.96982, 4.81146], [114.99417, 4.88201], [115.05038, 4.90275], [115.02955, 4.82087], [115.02278, 4.74137], [115.04064, 4.63706], [115.07737, 4.53418], [115.09978, 4.39123], [115.31275, 4.30806], [115.36346, 4.33563], [115.2851, 4.42295], [115.27819, 4.63661], [115.20737, 4.8256], [115.15092, 4.87604], [115.16236, 5.01011]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BO",
+           iso1A3: "BOL",
+           iso1N3: "068",
+           wikidata: "Q750",
+           nameEn: "Bolivia",
+           groups: ["005", "419", "019", "UN"],
+           callingCodes: ["591"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-63.90248, -12.52544], [-64.22539, -12.45267], [-64.30708, -12.46398], [-64.99778, -11.98604], [-65.30027, -11.48749], [-65.28141, -10.86289], [-65.35402, -10.78685], [-65.37923, -10.35141], [-65.29019, -9.86253], [-65.40615, -9.63894], [-65.56244, -9.84266], [-65.68343, -9.75323], [-67.17784, -10.34016], [-68.71533, -11.14749], [-68.7651, -11.0496], [-68.75179, -11.03688], [-68.75265, -11.02383], [-68.74802, -11.00891], [-69.42792, -10.93451], [-69.47839, -10.95254], [-69.57156, -10.94555], [-68.98115, -11.8979], [-68.65044, -12.50689], [-68.85615, -12.87769], [-68.8864, -13.40792], [-69.05265, -13.68546], [-68.88135, -14.18639], [-69.36254, -14.94634], [-69.14856, -15.23478], [-69.40336, -15.61358], [-69.20291, -16.16668], [-69.09986, -16.22693], [-68.96238, -16.194], [-68.79464, -16.33272], [-68.98358, -16.42165], [-69.04027, -16.57214], [-69.00853, -16.66769], [-69.16896, -16.72233], [-69.62883, -17.28142], [-69.46863, -17.37466], [-69.46897, -17.4988], [-69.46623, -17.60518], [-69.34126, -17.72753], [-69.28671, -17.94844], [-69.07496, -18.03715], [-69.14807, -18.16893], [-69.07432, -18.28259], [-68.94987, -18.93302], [-68.87082, -19.06003], [-68.80602, -19.08355], [-68.61989, -19.27584], [-68.41218, -19.40499], [-68.66761, -19.72118], [-68.54611, -19.84651], [-68.57132, -20.03134], [-68.74273, -20.08817], [-68.7276, -20.46178], [-68.44023, -20.62701], [-68.55383, -20.7355], [-68.53957, -20.91542], [-68.40403, -20.94562], [-68.18816, -21.28614], [-67.85114, -22.87076], [-67.54284, -22.89771], [-67.18382, -22.81525], [-66.7298, -22.23644], [-66.29714, -22.08741], [-66.24077, -21.77837], [-66.03836, -21.84829], [-66.04832, -21.9187], [-65.9261, -21.93335], [-65.7467, -22.10105], [-65.61166, -22.09504], [-65.58694, -22.09794], [-65.57743, -22.07675], [-65.47435, -22.08908], [-64.99524, -22.08255], [-64.90014, -22.12136], [-64.67174, -22.18957], [-64.58888, -22.25035], [-64.4176, -22.67692], [-64.35108, -22.73282], [-64.31489, -22.88824], [-64.22918, -22.55807], [-63.93287, -21.99934], [-63.70963, -21.99934], [-63.68113, -22.0544], [-63.66482, -21.99918], [-62.81124, -21.9987], [-62.8078, -22.12534], [-62.64455, -22.25091], [-62.2757, -21.06657], [-62.26883, -20.55311], [-61.93912, -20.10053], [-61.73723, -19.63958], [-60.00638, -19.2981], [-59.06965, -19.29148], [-58.23216, -19.80058], [-58.16225, -20.16193], [-57.8496, -19.98346], [-58.14215, -19.76276], [-57.78463, -19.03259], [-57.71113, -19.03161], [-57.69134, -19.00544], [-57.71995, -18.97546], [-57.71995, -18.89573], [-57.76764, -18.90087], [-57.56807, -18.25655], [-57.48237, -18.24219], [-57.69877, -17.8431], [-57.73949, -17.56095], [-57.90082, -17.44555], [-57.99661, -17.5273], [-58.32935, -17.28195], [-58.5058, -16.80958], [-58.30918, -16.3699], [-58.32431, -16.25861], [-58.41506, -16.32636], [-60.16069, -16.26479], [-60.23797, -15.50267], [-60.58224, -15.09887], [-60.23968, -15.09515], [-60.27887, -14.63021], [-60.46037, -14.22496], [-60.48053, -13.77981], [-61.05527, -13.50054], [-61.81151, -13.49564], [-63.76259, -12.42952], [-63.90248, -12.52544]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BQ",
+           iso1A3: "BES",
+           iso1N3: "535",
+           wikidata: "Q27561",
+           nameEn: "Caribbean Netherlands",
+           country: "NL"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BR",
+           iso1A3: "BRA",
+           iso1N3: "076",
+           wikidata: "Q155",
+           nameEn: "Brazil",
+           groups: ["005", "419", "019", "UN"],
+           callingCodes: ["55"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-59.69361, 4.34069], [-59.78878, 4.45637], [-60.15953, 4.53456], [-60.04189, 4.69801], [-59.98129, 5.07097], [-60.20944, 5.28754], [-60.32352, 5.21299], [-60.73204, 5.20931], [-60.5802, 4.94312], [-60.86539, 4.70512], [-60.98303, 4.54167], [-61.15703, 4.49839], [-61.31457, 4.54167], [-61.29675, 4.44216], [-61.48569, 4.43149], [-61.54629, 4.2822], [-62.13094, 4.08309], [-62.44822, 4.18621], [-62.57656, 4.04754], [-62.74411, 4.03331], [-62.7655, 3.73099], [-62.98296, 3.59935], [-63.21111, 3.96219], [-63.4464, 3.9693], [-63.42233, 3.89995], [-63.50611, 3.83592], [-63.67099, 4.01731], [-63.70218, 3.91417], [-63.86082, 3.94796], [-63.99183, 3.90172], [-64.14512, 4.12932], [-64.57648, 4.12576], [-64.72977, 4.28931], [-64.84028, 4.24665], [-64.48379, 3.7879], [-64.02908, 2.79797], [-64.0257, 2.48156], [-63.39114, 2.4317], [-63.39827, 2.16098], [-64.06135, 1.94722], [-64.08274, 1.64792], [-64.34654, 1.35569], [-64.38932, 1.5125], [-65.11657, 1.12046], [-65.57288, 0.62856], [-65.50158, 0.92086], [-65.6727, 1.01353], [-66.28507, 0.74585], [-66.85795, 1.22998], [-67.08222, 1.17441], [-67.15784, 1.80439], [-67.299, 1.87494], [-67.40488, 2.22258], [-67.9292, 1.82455], [-68.18632, 2.00091], [-68.26699, 1.83463], [-68.18128, 1.72881], [-69.38621, 1.70865], [-69.53746, 1.76408], [-69.83491, 1.69353], [-69.82987, 1.07864], [-69.26017, 1.06856], [-69.14422, 0.84172], [-69.20976, 0.57958], [-69.47696, 0.71065], [-70.04162, 0.55437], [-70.03658, -0.19681], [-69.603, -0.51947], [-69.59796, -0.75136], [-69.4215, -1.01853], [-69.43395, -1.42219], [-69.94708, -4.2431], [-70.00888, -4.37833], [-70.11305, -4.27281], [-70.19582, -4.3607], [-70.33236, -4.15214], [-70.77601, -4.15717], [-70.96814, -4.36915], [-71.87003, -4.51661], [-72.64391, -5.0391], [-72.83973, -5.14765], [-73.24579, -6.05764], [-73.12983, -6.43852], [-73.73986, -6.87919], [-73.77011, -7.28944], [-73.96938, -7.58465], [-73.65485, -7.77897], [-73.76576, -7.89884], [-72.92886, -9.04074], [-73.21498, -9.40904], [-72.72216, -9.41397], [-72.31883, -9.5184], [-72.14742, -9.98049], [-71.23394, -9.9668], [-70.53373, -9.42628], [-70.58453, -9.58303], [-70.55429, -9.76692], [-70.62487, -9.80666], [-70.64134, -11.0108], [-70.51395, -10.92249], [-70.38791, -11.07096], [-69.90896, -10.92744], [-69.57835, -10.94051], [-69.57156, -10.94555], [-69.47839, -10.95254], [-69.42792, -10.93451], [-68.74802, -11.00891], [-68.75265, -11.02383], [-68.75179, -11.03688], [-68.7651, -11.0496], [-68.71533, -11.14749], [-67.17784, -10.34016], [-65.68343, -9.75323], [-65.56244, -9.84266], [-65.40615, -9.63894], [-65.29019, -9.86253], [-65.37923, -10.35141], [-65.35402, -10.78685], [-65.28141, -10.86289], [-65.30027, -11.48749], [-64.99778, -11.98604], [-64.30708, -12.46398], [-64.22539, -12.45267], [-63.90248, -12.52544], [-63.76259, -12.42952], [-61.81151, -13.49564], [-61.05527, -13.50054], [-60.48053, -13.77981], [-60.46037, -14.22496], [-60.27887, -14.63021], [-60.23968, -15.09515], [-60.58224, -15.09887], [-60.23797, -15.50267], [-60.16069, -16.26479], [-58.41506, -16.32636], [-58.32431, -16.25861], [-58.30918, -16.3699], [-58.5058, -16.80958], [-58.32935, -17.28195], [-57.99661, -17.5273], [-57.90082, -17.44555], [-57.73949, -17.56095], [-57.69877, -17.8431], [-57.48237, -18.24219], [-57.56807, -18.25655], [-57.76764, -18.90087], [-57.71995, -18.89573], [-57.71995, -18.97546], [-57.69134, -19.00544], [-57.71113, -19.03161], [-57.78463, -19.03259], [-58.14215, -19.76276], [-57.8496, -19.98346], [-58.16225, -20.16193], [-57.84536, -20.93155], [-57.93492, -21.65505], [-57.88239, -21.6868], [-57.94642, -21.73799], [-57.98625, -22.09157], [-56.6508, -22.28387], [-56.5212, -22.11556], [-56.45893, -22.08072], [-56.23206, -22.25347], [-55.8331, -22.29008], [-55.74941, -22.46436], [-55.741, -22.52018], [-55.72366, -22.5519], [-55.6986, -22.56268], [-55.68742, -22.58407], [-55.62493, -22.62765], [-55.63849, -22.95122], [-55.5446, -23.22811], [-55.52288, -23.2595], [-55.5555, -23.28237], [-55.43585, -23.87157], [-55.44117, -23.9185], [-55.41784, -23.9657], [-55.12292, -23.99669], [-55.0518, -23.98666], [-55.02691, -23.97317], [-54.6238, -23.83078], [-54.32807, -24.01865], [-54.28207, -24.07305], [-54.4423, -25.13381], [-54.62033, -25.46026], [-54.60196, -25.48397], [-54.59509, -25.53696], [-54.59398, -25.59224], [-54.5502, -25.58915], [-54.52926, -25.62846], [-53.90831, -25.55513], [-53.83691, -25.94849], [-53.73511, -26.04211], [-53.73086, -26.05842], [-53.7264, -26.0664], [-53.73391, -26.07006], [-53.73968, -26.10012], [-53.65018, -26.19501], [-53.65237, -26.23289], [-53.63739, -26.2496], [-53.63881, -26.25075], [-53.64632, -26.24798], [-53.64186, -26.25976], [-53.64505, -26.28089], [-53.68269, -26.33359], [-53.73372, -26.6131], [-53.80144, -27.09844], [-54.15978, -27.2889], [-54.19062, -27.27639], [-54.19268, -27.30751], [-54.41888, -27.40882], [-54.50416, -27.48232], [-54.67657, -27.57214], [-54.90159, -27.63132], [-54.90805, -27.73149], [-55.1349, -27.89759], [-55.16872, -27.86224], [-55.33303, -27.94661], [-55.6262, -28.17124], [-55.65418, -28.18304], [-56.01729, -28.51223], [-56.00458, -28.60421], [-56.05265, -28.62651], [-56.54171, -29.11447], [-56.57295, -29.11357], [-56.62789, -29.18073], [-56.81251, -29.48154], [-57.09386, -29.74211], [-57.65132, -30.19229], [-57.22502, -30.26121], [-56.90236, -30.02578], [-56.49267, -30.39471], [-56.4795, -30.3899], [-56.4619, -30.38457], [-55.87388, -31.05053], [-55.58866, -30.84117], [-55.5634, -30.8686], [-55.55373, -30.8732], [-55.55218, -30.88193], [-55.54572, -30.89051], [-55.53431, -30.89714], [-55.53276, -30.90218], [-55.52712, -30.89997], [-55.51862, -30.89828], [-55.50841, -30.9027], [-55.50821, -30.91349], [-54.17384, -31.86168], [-53.76024, -32.0751], [-53.39572, -32.58596], [-53.37671, -32.57005], [-53.1111, -32.71147], [-53.53459, -33.16843], [-53.52794, -33.68908], [-53.44031, -33.69344], [-53.39593, -33.75169], [-53.37138, -33.74313], [-52.83257, -34.01481], [-28.34015, -20.99094], [-28.99601, 1.86593], [-51.35485, 4.8383], [-51.63798, 4.51394], [-51.61983, 4.14596], [-51.79599, 3.89336], [-51.82312, 3.85825], [-51.85573, 3.83427], [-52.31787, 3.17896], [-52.6906, 2.37298], [-52.96539, 2.1881], [-53.78743, 2.34412], [-54.16286, 2.10779], [-54.6084, 2.32856], [-55.01919, 2.564], [-55.71493, 2.40342], [-55.96292, 2.53188], [-56.13054, 2.27723], [-55.92159, 2.05236], [-55.89863, 1.89861], [-55.99278, 1.83137], [-56.47045, 1.95135], [-56.7659, 1.89509], [-57.07092, 1.95304], [-57.09109, 2.01854], [-57.23981, 1.95808], [-57.35073, 1.98327], [-57.55743, 1.69605], [-57.77281, 1.73344], [-57.97336, 1.64566], [-58.01873, 1.51966], [-58.33887, 1.58014], [-58.4858, 1.48399], [-58.53571, 1.29154], [-58.84229, 1.17749], [-58.92072, 1.31293], [-59.25583, 1.40559], [-59.74066, 1.87596], [-59.7264, 2.27497], [-59.91177, 2.36759], [-59.99733, 2.92312], [-59.79769, 3.37162], [-59.86899, 3.57089], [-59.51963, 3.91951], [-59.73353, 4.20399], [-59.69361, 4.34069]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BS",
+           iso1A3: "BHS",
+           iso1N3: "044",
+           wikidata: "Q778",
+           nameEn: "The Bahamas",
+           groups: ["029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           callingCodes: ["1 242"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-72.98446, 20.4801], [-71.70065, 25.7637], [-78.91214, 27.76553], [-80.65727, 23.71953], [-72.98446, 20.4801]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BT",
+           iso1A3: "BTN",
+           iso1N3: "064",
+           wikidata: "Q917",
+           nameEn: "Bhutan",
+           groups: ["034", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["975"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[91.6469, 27.76358], [91.5629, 27.84823], [91.48973, 27.93903], [91.46327, 28.0064], [91.25779, 28.07509], [91.20019, 27.98715], [90.69894, 28.07784], [90.58842, 28.02838], [90.13387, 28.19178], [89.79762, 28.23979], [89.59525, 28.16433], [89.12825, 27.62502], [89.0582, 27.60985], [88.97213, 27.51671], [88.95355, 27.4106], [89.00216, 27.32532], [88.96947, 27.30319], [88.93678, 27.33777], [88.91901, 27.32483], [88.74219, 27.144], [88.86984, 27.10937], [88.8714, 26.97488], [88.92301, 26.99286], [88.95807, 26.92668], [89.09554, 26.89089], [89.12825, 26.81661], [89.1926, 26.81329], [89.37913, 26.86224], [89.38319, 26.85963], [89.3901, 26.84225], [89.42349, 26.83727], [89.63369, 26.74402], [89.86124, 26.73307], [90.04535, 26.72422], [90.30402, 26.85098], [90.39271, 26.90704], [90.48504, 26.8594], [90.67715, 26.77215], [91.50067, 26.79223], [91.83181, 26.87318], [92.05523, 26.8692], [92.11863, 26.893], [92.03457, 27.07334], [92.04702, 27.26861], [92.12019, 27.27829], [92.01132, 27.47352], [91.65007, 27.48287], [91.55819, 27.6144], [91.6469, 27.76358]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BV",
+           iso1A3: "BVT",
+           iso1N3: "074",
+           wikidata: "Q23408",
+           nameEn: "Bouvet Island",
+           country: "NO",
+           groups: ["005", "419", "019", "UN"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[4.54042, -54.0949], [2.28941, -54.13089], [3.35353, -55.17558], [4.54042, -54.0949]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BW",
+           iso1A3: "BWA",
+           iso1N3: "072",
+           wikidata: "Q963",
+           nameEn: "Botswana",
+           groups: ["018", "202", "002", "UN"],
+           driveSide: "left",
+           callingCodes: ["267"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[25.26433, -17.79571], [25.16882, -17.78253], [25.05895, -17.84452], [24.95586, -17.79674], [24.73364, -17.89338], [24.71887, -17.9218], [24.6303, -17.9863], [24.57485, -18.07151], [24.40577, -17.95726], [24.19416, -18.01919], [23.61088, -18.4881], [23.29618, -17.99855], [23.0996, -18.00075], [21.45556, -18.31795], [20.99904, -18.31743], [20.99751, -22.00026], [19.99912, -21.99991], [19.99817, -24.76768], [20.02809, -24.78725], [20.03678, -24.81004], [20.29826, -24.94869], [20.64795, -25.47827], [20.86081, -26.14892], [20.61754, -26.4692], [20.63275, -26.78181], [20.68596, -26.9039], [20.87031, -26.80047], [21.13353, -26.86661], [21.37869, -26.82083], [21.69322, -26.86152], [21.7854, -26.79199], [21.77114, -26.69015], [21.83291, -26.65959], [21.90703, -26.66808], [22.06192, -26.61882], [22.21206, -26.3773], [22.41921, -26.23078], [22.56365, -26.19668], [22.70808, -25.99186], [22.86012, -25.50572], [23.03497, -25.29971], [23.47588, -25.29971], [23.9244, -25.64286], [24.18287, -25.62916], [24.36531, -25.773], [24.44703, -25.73021], [24.67319, -25.81749], [24.8946, -25.80723], [25.01718, -25.72507], [25.12266, -25.75931], [25.33076, -25.76616], [25.58543, -25.6343], [25.6643, -25.4491], [25.69661, -25.29284], [25.72702, -25.25503], [25.88571, -24.87802], [25.84295, -24.78661], [25.8515, -24.75727], [26.39409, -24.63468], [26.46346, -24.60358], [26.51667, -24.47219], [26.84165, -24.24885], [26.99749, -23.65486], [27.33768, -23.40917], [27.52393, -23.37952], [27.6066, -23.21894], [27.74154, -23.2137], [27.93539, -23.04941], [27.93729, -22.96194], [28.04752, -22.90243], [28.04562, -22.8394], [28.34874, -22.5694], [28.63287, -22.55887], [28.91889, -22.44299], [29.0151, -22.22907], [29.10881, -22.21202], [29.15268, -22.21399], [29.18974, -22.18599], [29.21955, -22.17771], [29.37703, -22.19581], [29.3533, -22.18363], [29.24648, -22.05967], [29.1974, -22.07472], [29.14501, -22.07275], [29.08495, -22.04867], [29.04108, -22.00563], [29.02191, -21.95665], [29.02191, -21.90647], [29.04023, -21.85864], [29.07763, -21.81877], [28.58114, -21.63455], [28.49942, -21.66634], [28.29416, -21.59037], [28.01669, -21.57624], [27.91407, -21.31621], [27.69171, -21.08409], [27.72972, -20.51735], [27.69361, -20.48531], [27.28865, -20.49873], [27.29831, -20.28935], [27.21278, -20.08244], [26.72246, -19.92707], [26.17227, -19.53709], [25.96226, -19.08152], [25.99837, -19.02943], [25.94326, -18.90362], [25.82353, -18.82808], [25.79217, -18.6355], [25.68859, -18.56165], [25.53465, -18.39041], [25.39972, -18.12691], [25.31799, -18.07091], [25.23909, -17.90832], [25.26433, -17.79571]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BY",
+           iso1A3: "BLR",
+           iso1N3: "112",
+           wikidata: "Q184",
+           nameEn: "Belarus",
+           groups: ["151", "150", "UN"],
+           callingCodes: ["375"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[28.15217, 56.16964], [27.97865, 56.11849], [27.63065, 55.89687], [27.61683, 55.78558], [27.3541, 55.8089], [27.27804, 55.78299], [27.1559, 55.85032], [26.97153, 55.8102], [26.87448, 55.7172], [26.76872, 55.67658], [26.71802, 55.70645], [26.64888, 55.70515], [26.63231, 55.67968], [26.63167, 55.57887], [26.55094, 55.5093], [26.5522, 55.40277], [26.44937, 55.34832], [26.5709, 55.32572], [26.6714, 55.33902], [26.80929, 55.31642], [26.83266, 55.30444], [26.835, 55.28182], [26.73017, 55.24226], [26.72983, 55.21788], [26.68075, 55.19787], [26.69243, 55.16718], [26.54753, 55.14181], [26.51481, 55.16051], [26.46249, 55.12814], [26.35121, 55.1525], [26.30628, 55.12536], [26.23202, 55.10439], [26.26941, 55.08032], [26.20397, 54.99729], [26.13386, 54.98924], [26.05907, 54.94631], [25.99129, 54.95705], [25.89462, 54.93438], [25.74122, 54.80108], [25.75977, 54.57252], [25.68045, 54.5321], [25.64813, 54.48704], [25.62203, 54.4656], [25.63371, 54.42075], [25.5376, 54.33158], [25.55425, 54.31591], [25.68513, 54.31727], [25.78553, 54.23327], [25.78563, 54.15747], [25.71084, 54.16704], [25.64875, 54.1259], [25.54724, 54.14925], [25.51452, 54.17799], [25.56823, 54.25212], [25.509, 54.30267], [25.35559, 54.26544], [25.22705, 54.26271], [25.19199, 54.219], [25.0728, 54.13419], [24.991, 54.14241], [24.96894, 54.17589], [24.77131, 54.11091], [24.85311, 54.02862], [24.74279, 53.96663], [24.69185, 53.96543], [24.69652, 54.01901], [24.62275, 54.00217], [24.44411, 53.90076], [24.34128, 53.90076], [24.19638, 53.96405], [23.98837, 53.92554], [23.95098, 53.9613], [23.81309, 53.94205], [23.80543, 53.89558], [23.71726, 53.93379], [23.61677, 53.92691], [23.51284, 53.95052], [23.62004, 53.60942], [23.81995, 53.24131], [23.85657, 53.22923], [23.91393, 53.16469], [23.87548, 53.0831], [23.92184, 53.02079], [23.94689, 52.95919], [23.91805, 52.94016], [23.93763, 52.71332], [23.73615, 52.6149], [23.58296, 52.59868], [23.45112, 52.53774], [23.34141, 52.44845], [23.18196, 52.28812], [23.20071, 52.22848], [23.47859, 52.18215], [23.54314, 52.12148], [23.61, 52.11264], [23.64066, 52.07626], [23.68733, 51.9906], [23.61523, 51.92066], [23.62691, 51.78208], [23.53198, 51.74298], [23.57053, 51.55938], [23.56236, 51.53673], [23.62751, 51.50512], [23.6736, 51.50255], [23.60906, 51.62122], [23.7766, 51.66809], [23.91118, 51.63316], [23.8741, 51.59734], [23.99907, 51.58369], [24.13075, 51.66979], [24.3163, 51.75063], [24.29021, 51.80841], [24.37123, 51.88222], [24.98784, 51.91273], [25.20228, 51.97143], [25.46163, 51.92205], [25.73673, 51.91973], [25.80574, 51.94556], [25.83217, 51.92587], [26.00408, 51.92967], [26.19084, 51.86781], [26.39367, 51.87315], [26.46962, 51.80501], [26.69759, 51.82284], [26.80043, 51.75777], [26.9489, 51.73788], [26.99422, 51.76933], [27.20602, 51.77291], [27.20948, 51.66713], [27.26613, 51.65957], [27.24828, 51.60161], [27.47212, 51.61184], [27.51058, 51.5854], [27.55727, 51.63486], [27.71932, 51.60672], [27.67125, 51.50854], [27.76052, 51.47604], [27.85253, 51.62293], [27.91844, 51.61952], [27.95827, 51.56065], [28.10658, 51.57857], [28.23452, 51.66988], [28.37592, 51.54505], [28.47051, 51.59734], [28.64429, 51.5664], [28.69161, 51.44695], [28.73143, 51.46236], [28.75615, 51.41442], [28.78224, 51.45294], [28.76027, 51.48802], [28.81795, 51.55552], [28.95528, 51.59222], [28.99098, 51.56833], [29.1187, 51.65872], [29.16402, 51.64679], [29.20659, 51.56918], [29.25603, 51.57089], [29.25191, 51.49828], [29.32881, 51.37843], [29.42357, 51.4187], [29.49773, 51.39814], [29.54372, 51.48372], [29.7408, 51.53417], [29.77376, 51.4461], [30.17888, 51.51025], [30.34642, 51.42555], [30.36153, 51.33984], [30.56203, 51.25655], [30.64992, 51.35014], [30.51946, 51.59649], [30.68804, 51.82806], [30.76443, 51.89739], [30.90897, 52.00699], [30.95589, 52.07775], [31.13332, 52.1004], [31.25142, 52.04131], [31.38326, 52.12991], [31.7822, 52.11406], [31.77877, 52.18636], [31.6895, 52.1973], [31.70735, 52.26711], [31.57971, 52.32146], [31.62084, 52.33849], [31.61397, 52.48843], [31.56316, 52.51518], [31.63869, 52.55361], [31.50406, 52.69707], [31.57277, 52.71613], [31.592, 52.79011], [31.35667, 52.97854], [31.24147, 53.031], [31.32283, 53.04101], [31.33519, 53.08805], [31.3915, 53.09712], [31.36403, 53.13504], [31.40523, 53.21406], [31.56316, 53.19432], [31.62496, 53.22886], [31.787, 53.18033], [31.82373, 53.10042], [32.15368, 53.07594], [32.40773, 53.18856], [32.51725, 53.28431], [32.73257, 53.33494], [32.74968, 53.45597], [32.47777, 53.5548], [32.40499, 53.6656], [32.50112, 53.68594], [32.45717, 53.74039], [32.36663, 53.7166], [32.12621, 53.81586], [31.89137, 53.78099], [31.77028, 53.80015], [31.85019, 53.91801], [31.88744, 54.03653], [31.89599, 54.0837], [31.57002, 54.14535], [31.30791, 54.25315], [31.3177, 54.34067], [31.22945, 54.46585], [31.08543, 54.50361], [31.21399, 54.63113], [31.19339, 54.66947], [30.99187, 54.67046], [30.98226, 54.68872], [31.0262, 54.70698], [30.97127, 54.71967], [30.95479, 54.74346], [30.75165, 54.80699], [30.8264, 54.90062], [30.81759, 54.94064], [30.93144, 54.9585], [30.95754, 54.98609], [30.9081, 55.02232], [30.94243, 55.03964], [31.00972, 55.02783], [31.02071, 55.06167], [30.97369, 55.17134], [30.87944, 55.28223], [30.81946, 55.27931], [30.8257, 55.3313], [30.93144, 55.3914], [30.90123, 55.46621], [30.95204, 55.50667], [30.93419, 55.6185], [30.86003, 55.63169], [30.7845, 55.58514], [30.72957, 55.66268], [30.67464, 55.64176], [30.63344, 55.73079], [30.51037, 55.76568], [30.51346, 55.78982], [30.48257, 55.81066], [30.30987, 55.83592], [30.27776, 55.86819], [30.12136, 55.8358], [29.97975, 55.87281], [29.80672, 55.79569], [29.61446, 55.77716], [29.51283, 55.70294], [29.3604, 55.75862], [29.44692, 55.95978], [29.21717, 55.98971], [29.08299, 56.03427], [28.73418, 55.97131], [28.63668, 56.07262], [28.68337, 56.10173], [28.5529, 56.11705], [28.43068, 56.09407], [28.37987, 56.11399], [28.36888, 56.05805], [28.30571, 56.06035], [28.15217, 56.16964]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "BZ",
+           iso1A3: "BLZ",
+           iso1N3: "084",
+           wikidata: "Q242",
+           nameEn: "Belize",
+           groups: ["013", "003", "419", "019", "UN"],
+           roadSpeedUnit: "mph",
+           callingCodes: ["501"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-88.3268, 18.49048], [-88.48242, 18.49164], [-88.71505, 18.0707], [-88.8716, 17.89535], [-89.03839, 18.0067], [-89.15105, 17.95104], [-89.14985, 17.81563], [-89.15025, 17.04813], [-89.22683, 15.88619], [-89.17418, 15.90898], [-89.02415, 15.9063], [-88.95358, 15.88698], [-88.40779, 16.09624], [-86.92368, 17.61462], [-87.84815, 18.18511], [-87.85693, 18.18266], [-87.86657, 18.19971], [-87.87604, 18.18313], [-87.90671, 18.15213], [-88.03165, 18.16657], [-88.03238, 18.41778], [-88.26593, 18.47617], [-88.29909, 18.47591], [-88.3268, 18.49048]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CA",
+           iso1A3: "CAN",
+           iso1N3: "124",
+           wikidata: "Q16",
+           nameEn: "Canada",
+           groups: ["021", "003", "019", "UN"],
+           callingCodes: ["1"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-67.20349, 45.1722], [-67.19603, 45.16771], [-67.15965, 45.16179], [-67.11316, 45.11176], [-67.0216, 44.95333], [-66.96824, 44.90965], [-66.98249, 44.87071], [-66.96824, 44.83078], [-66.93432, 44.82597], [-67.16117, 44.20069], [-61.98255, 37.34815], [-56.27503, 47.39728], [-53.12387, 41.40385], [-46.37635, 57.3249], [-77.52957, 77.23408], [-68.21821, 80.48551], [-49.33696, 84.57952], [-140.97446, 84.39275], [-141.00116, 60.30648], [-140.5227, 60.22077], [-140.45648, 60.30919], [-139.98024, 60.18027], [-139.68991, 60.33693], [-139.05831, 60.35205], [-139.20603, 60.08896], [-139.05365, 59.99655], [-138.71149, 59.90728], [-138.62145, 59.76431], [-137.60623, 59.24465], [-137.4925, 58.89415], [-136.82619, 59.16198], [-136.52365, 59.16752], [-136.47323, 59.46617], [-136.33727, 59.44466], [-136.22381, 59.55526], [-136.31566, 59.59083], [-135.48007, 59.79937], [-135.03069, 59.56208], [-135.00267, 59.28745], [-134.7047, 59.2458], [-134.55699, 59.1297], [-134.48059, 59.13231], [-134.27175, 58.8634], [-133.84645, 58.73543], [-133.38523, 58.42773], [-131.8271, 56.62247], [-130.77769, 56.36185], [-130.33965, 56.10849], [-130.10173, 56.12178], [-130.00093, 56.00325], [-130.00857, 55.91344], [-130.15373, 55.74895], [-129.97513, 55.28029], [-130.08035, 55.21556], [-130.18765, 55.07744], [-130.27203, 54.97174], [-130.44184, 54.85377], [-130.64499, 54.76912], [-130.61931, 54.70835], [-133.92876, 54.62289], [-133.36909, 48.51151], [-125.03842, 48.53282], [-123.50039, 48.21223], [-123.15614, 48.35395], [-123.26565, 48.6959], [-123.0093, 48.76586], [-123.0093, 48.83186], [-123.32163, 49.00419], [-95.15355, 48.9996], [-95.15357, 49.384], [-95.12903, 49.37056], [-95.05825, 49.35311], [-95.01419, 49.35647], [-94.99532, 49.36579], [-94.95681, 49.37035], [-94.85381, 49.32492], [-94.8159, 49.32299], [-94.82487, 49.29483], [-94.77355, 49.11998], [-94.75017, 49.09931], [-94.687, 48.84077], [-94.70087, 48.8339], [-94.70486, 48.82365], [-94.69669, 48.80918], [-94.69335, 48.77883], [-94.58903, 48.71803], [-94.54885, 48.71543], [-94.53826, 48.70216], [-94.44258, 48.69223], [-94.4174, 48.71049], [-94.27153, 48.70232], [-94.25172, 48.68404], [-94.25104, 48.65729], [-94.23215, 48.65202], [-93.85769, 48.63284], [-93.83288, 48.62745], [-93.80676, 48.58232], [-93.80939, 48.52439], [-93.79267, 48.51631], [-93.66382, 48.51845], [-93.47022, 48.54357], [-93.44472, 48.59147], [-93.40693, 48.60948], [-93.39758, 48.60364], [-93.3712, 48.60599], [-93.33946, 48.62787], [-93.25391, 48.64266], [-92.94973, 48.60866], [-92.7287, 48.54005], [-92.6342, 48.54133], [-92.62747, 48.50278], [-92.69927, 48.49573], [-92.71323, 48.46081], [-92.65606, 48.43471], [-92.50712, 48.44921], [-92.45588, 48.40624], [-92.48147, 48.36609], [-92.37185, 48.22259], [-92.27167, 48.25046], [-92.30939, 48.31251], [-92.26662, 48.35651], [-92.202, 48.35252], [-92.14732, 48.36578], [-92.05339, 48.35958], [-91.98929, 48.25409], [-91.86125, 48.21278], [-91.71231, 48.19875], [-91.70451, 48.11805], [-91.55649, 48.10611], [-91.58025, 48.04339], [-91.45829, 48.07454], [-91.43248, 48.04912], [-91.25025, 48.08522], [-91.08016, 48.18096], [-90.87588, 48.2484], [-90.75045, 48.09143], [-90.56444, 48.12184], [-90.56312, 48.09488], [-90.07418, 48.11043], [-89.89974, 47.98109], [-89.77248, 48.02607], [-89.57972, 48.00023], [-89.48837, 48.01412], [-88.37033, 48.30586], [-84.85871, 46.88881], [-84.55635, 46.45974], [-84.47607, 46.45225], [-84.4481, 46.48972], [-84.42101, 46.49853], [-84.34174, 46.50683], [-84.29893, 46.49127], [-84.26351, 46.49508], [-84.2264, 46.53337], [-84.1945, 46.54061], [-84.17723, 46.52753], [-84.12885, 46.53068], [-84.11196, 46.50248], [-84.13451, 46.39218], [-84.11254, 46.32329], [-84.11615, 46.2681], [-84.09756, 46.25512], [-84.1096, 46.23987], [-83.95399, 46.05634], [-83.90453, 46.05922], [-83.83329, 46.12169], [-83.57017, 46.105], [-83.43746, 45.99749], [-83.59589, 45.82131], [-82.48419, 45.30225], [-82.42469, 42.992], [-82.4146, 42.97626], [-82.4253, 42.95423], [-82.45331, 42.93139], [-82.4826, 42.8068], [-82.46613, 42.76615], [-82.51063, 42.66025], [-82.51858, 42.611], [-82.57583, 42.5718], [-82.58873, 42.54984], [-82.64242, 42.55594], [-82.82964, 42.37355], [-83.02253, 42.33045], [-83.07837, 42.30978], [-83.09837, 42.28877], [-83.12724, 42.2376], [-83.14962, 42.04089], [-83.11184, 41.95671], [-82.67862, 41.67615], [-78.93684, 42.82887], [-78.90712, 42.89733], [-78.90905, 42.93022], [-78.93224, 42.95229], [-78.96312, 42.95509], [-78.98126, 42.97], [-79.02074, 42.98444], [-79.02424, 43.01983], [-78.99941, 43.05612], [-79.01055, 43.06659], [-79.07486, 43.07845], [-79.05671, 43.10937], [-79.06881, 43.12029], [-79.0427, 43.13934], [-79.04652, 43.16396], [-79.05384, 43.17418], [-79.05002, 43.20133], [-79.05544, 43.21224], [-79.05512, 43.25375], [-79.06921, 43.26183], [-79.25796, 43.54052], [-76.79706, 43.63099], [-76.43859, 44.09393], [-76.35324, 44.13493], [-76.31222, 44.19894], [-76.244, 44.19643], [-76.1664, 44.23051], [-76.16285, 44.28262], [-76.00018, 44.34896], [-75.95947, 44.34463], [-75.8217, 44.43176], [-75.76813, 44.51537], [-75.41441, 44.76614], [-75.2193, 44.87821], [-75.01363, 44.95608], [-74.99101, 44.98051], [-74.8447, 45.00606], [-74.66689, 45.00646], [-74.32699, 44.99029], [-73.35025, 45.00942], [-71.50067, 45.01357], [-71.48735, 45.07784], [-71.42778, 45.12624], [-71.40364, 45.21382], [-71.44252, 45.2361], [-71.37133, 45.24624], [-71.29371, 45.29996], [-71.22338, 45.25184], [-71.19723, 45.25438], [-71.14568, 45.24128], [-71.08364, 45.30623], [-71.01866, 45.31573], [-71.0107, 45.34819], [-70.95193, 45.33895], [-70.91169, 45.29849], [-70.89864, 45.2398], [-70.84816, 45.22698], [-70.80236, 45.37444], [-70.82638, 45.39828], [-70.78372, 45.43269], [-70.65383, 45.37592], [-70.62518, 45.42286], [-70.72651, 45.49771], [-70.68516, 45.56964], [-70.54019, 45.67291], [-70.38934, 45.73215], [-70.41523, 45.79497], [-70.25976, 45.89675], [-70.24694, 45.95138], [-70.31025, 45.96424], [-70.23855, 46.1453], [-70.29078, 46.18832], [-70.18547, 46.35357], [-70.05812, 46.41768], [-69.99966, 46.69543], [-69.22119, 47.46461], [-69.05148, 47.42012], [-69.05073, 47.30076], [-69.05039, 47.2456], [-68.89222, 47.1807], [-68.70125, 47.24399], [-68.60575, 47.24659], [-68.57914, 47.28431], [-68.38332, 47.28723], [-68.37458, 47.35851], [-68.23244, 47.35712], [-67.94843, 47.1925], [-67.87993, 47.10377], [-67.78578, 47.06473], [-67.78111, 45.9392], [-67.75196, 45.91814], [-67.80961, 45.87531], [-67.75654, 45.82324], [-67.80653, 45.80022], [-67.80705, 45.69528], [-67.6049, 45.60725], [-67.43815, 45.59162], [-67.42144, 45.50584], [-67.50578, 45.48971], [-67.42394, 45.37969], [-67.48201, 45.27351], [-67.34927, 45.122], [-67.29754, 45.14865], [-67.29748, 45.18173], [-67.27039, 45.1934], [-67.22751, 45.16344], [-67.20349, 45.1722]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CC",
+           iso1A3: "CCK",
+           iso1N3: "166",
+           wikidata: "Q36004",
+           nameEn: "Cocos (Keeling) Islands",
+           country: "AU",
+           groups: ["053", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["61"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[96.61846, -10.82438], [96.02343, -12.68334], [97.93979, -12.33309], [96.61846, -10.82438]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CD",
+           iso1A3: "COD",
+           iso1N3: "180",
+           wikidata: "Q974",
+           nameEn: "Democratic Republic of the Congo",
+           aliases: ["ZR"],
+           groups: ["017", "202", "002", "UN"],
+           callingCodes: ["243"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[27.44012, 5.07349], [27.09575, 5.22305], [26.93064, 5.13535], [26.85579, 5.03887], [26.74572, 5.10685], [26.48595, 5.04984], [26.13371, 5.25594], [25.86073, 5.19455], [25.53271, 5.37431], [25.34558, 5.29101], [25.31256, 5.03668], [24.71816, 4.90509], [24.46719, 5.0915], [23.38847, 4.60013], [22.94817, 4.82392], [22.89094, 4.79321], [22.84691, 4.69887], [22.78526, 4.71423], [22.6928, 4.47285], [22.60915, 4.48821], [22.5431, 4.22041], [22.45504, 4.13039], [22.27682, 4.11347], [22.10721, 4.20723], [21.6405, 4.317], [21.55904, 4.25553], [21.25744, 4.33676], [21.21341, 4.29285], [21.11214, 4.33895], [21.08793, 4.39603], [20.90383, 4.44877], [20.60184, 4.42394], [18.62755, 3.47564], [18.63857, 3.19342], [18.10683, 2.26876], [18.08034, 1.58553], [17.85887, 1.04327], [17.86989, 0.58873], [17.95255, 0.48128], [17.93877, 0.32424], [17.81204, 0.23884], [17.66051, -0.26535], [17.72112, -0.52707], [17.32438, -0.99265], [16.97999, -1.12762], [16.70724, -1.45815], [16.50336, -1.8795], [16.16173, -2.16586], [16.22785, -2.59528], [16.1755, -3.25014], [16.21407, -3.2969], [15.89448, -3.9513], [15.53081, -4.042], [15.48121, -4.22062], [15.41785, -4.28381], [15.32693, -4.27282], [15.25411, -4.31121], [15.1978, -4.32388], [14.83101, -4.80838], [14.67948, -4.92093], [14.5059, -4.84956], [14.41499, -4.8825], [14.37366, -4.56125], [14.47284, -4.42941], [14.3957, -4.36623], [14.40672, -4.28381], [13.9108, -4.50906], [13.81162, -4.41842], [13.71794, -4.44864], [13.70417, -4.72601], [13.50305, -4.77818], [13.41764, -4.89897], [13.11182, -4.5942], [13.09648, -4.63739], [13.11195, -4.67745], [12.8733, -4.74346], [12.70868, -4.95505], [12.63465, -4.94632], [12.60251, -5.01715], [12.46297, -5.09408], [12.49815, -5.14058], [12.51589, -5.1332], [12.53586, -5.14658], [12.53599, -5.1618], [12.52301, -5.17481], [12.52318, -5.74353], [12.26557, -5.74031], [12.20376, -5.76338], [11.95767, -5.94705], [12.42245, -6.07585], [13.04371, -5.87078], [16.55507, -5.85631], [16.96282, -7.21787], [17.5828, -8.13784], [18.33635, -8.00126], [19.33698, -7.99743], [19.5469, -7.00195], [20.30218, -6.98955], [20.31846, -6.91953], [20.61689, -6.90876], [20.56263, -7.28566], [21.79824, -7.29628], [21.84856, -9.59871], [22.19039, -9.94628], [22.32604, -10.76291], [22.17954, -10.85884], [22.25951, -11.24911], [22.54205, -11.05784], [23.16602, -11.10577], [23.45631, -10.946], [23.86868, -11.02856], [24.00027, -10.89356], [24.34528, -11.06816], [24.42612, -11.44975], [25.34069, -11.19707], [25.33058, -11.65767], [26.01777, -11.91488], [26.88687, -12.01868], [27.04351, -11.61312], [27.22541, -11.60323], [27.21025, -11.76157], [27.59932, -12.22123], [28.33199, -12.41375], [29.01918, -13.41353], [29.60531, -13.21685], [29.65078, -13.41844], [29.81551, -13.44683], [29.8139, -12.14898], [29.48404, -12.23604], [29.4992, -12.43843], [29.18592, -12.37921], [28.48357, -11.87532], [28.37241, -11.57848], [28.65032, -10.65133], [28.62795, -9.92942], [28.68532, -9.78], [28.56208, -9.49122], [28.51627, -9.44726], [28.52636, -9.35379], [28.36562, -9.30091], [28.38526, -9.23393], [28.9711, -8.66935], [28.88917, -8.4831], [30.79243, -8.27382], [30.2567, -7.14121], [29.52552, -6.2731], [29.43673, -4.44845], [29.23708, -3.75856], [29.21463, -3.3514], [29.25633, -3.05471], [29.17258, -2.99385], [29.16037, -2.95457], [29.09797, -2.91935], [29.09119, -2.87871], [29.0505, -2.81774], [29.00404, -2.81978], [29.00167, -2.78523], [29.04081, -2.7416], [29.00357, -2.70596], [28.94346, -2.69124], [28.89793, -2.66111], [28.90226, -2.62385], [28.89288, -2.55848], [28.87943, -2.55165], [28.86193, -2.53185], [28.86209, -2.5231], [28.87497, -2.50887], [28.88846, -2.50493], [28.89342, -2.49017], [28.89132, -2.47557], [28.86846, -2.44866], [28.86826, -2.41888], [28.89601, -2.37321], [28.95642, -2.37321], [29.00051, -2.29001], [29.105, -2.27043], [29.17562, -2.12278], [29.11847, -1.90576], [29.24458, -1.69663], [29.24323, -1.66826], [29.36322, -1.50887], [29.45038, -1.5054], [29.53062, -1.40499], [29.59061, -1.39016], [29.58388, -0.89821], [29.63006, -0.8997], [29.62708, -0.71055], [29.67176, -0.55714], [29.67474, -0.47969], [29.65091, -0.46777], [29.72687, -0.08051], [29.7224, 0.07291], [29.77454, 0.16675], [29.81922, 0.16824], [29.87284, 0.39166], [29.97413, 0.52124], [29.95477, 0.64486], [29.98307, 0.84295], [30.1484, 0.89805], [30.22139, 0.99635], [30.24671, 1.14974], [30.48503, 1.21675], [31.30127, 2.11006], [31.28042, 2.17853], [31.20148, 2.2217], [31.1985, 2.29462], [31.12104, 2.27676], [31.07934, 2.30207], [31.06593, 2.35862], [30.96911, 2.41071], [30.91102, 2.33332], [30.83059, 2.42559], [30.74271, 2.43601], [30.75612, 2.5863], [30.8857, 2.83923], [30.8574, 2.9508], [30.77101, 3.04897], [30.84251, 3.26908], [30.93486, 3.40737], [30.94081, 3.50847], [30.85153, 3.48867], [30.85997, 3.5743], [30.80713, 3.60506], [30.78512, 3.67097], [30.56277, 3.62703], [30.57378, 3.74567], [30.55396, 3.84451], [30.47691, 3.83353], [30.27658, 3.95653], [30.22374, 3.93896], [30.1621, 4.10586], [30.06964, 4.13221], [29.79666, 4.37809], [29.82087, 4.56246], [29.49726, 4.7007], [29.43341, 4.50101], [29.22207, 4.34297], [29.03054, 4.48784], [28.8126, 4.48784], [28.6651, 4.42638], [28.20719, 4.35614], [27.79551, 4.59976], [27.76469, 4.79284], [27.65462, 4.89375], [27.56656, 4.89375], [27.44012, 5.07349]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CF",
+           iso1A3: "CAF",
+           iso1N3: "140",
+           wikidata: "Q929",
+           nameEn: "Central African Republic",
+           groups: ["017", "202", "002", "UN"],
+           callingCodes: ["236"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[22.87758, 10.91915], [22.45889, 11.00246], [21.72139, 10.64136], [21.71479, 10.29932], [21.63553, 10.217], [21.52766, 10.2105], [21.34934, 9.95907], [21.26348, 9.97642], [20.82979, 9.44696], [20.36748, 9.11019], [19.06421, 9.00367], [18.86388, 8.87971], [19.11044, 8.68172], [18.79783, 8.25929], [18.67455, 8.22226], [18.62612, 8.14163], [18.64153, 8.08714], [18.6085, 8.05009], [18.02731, 8.01085], [17.93926, 7.95853], [17.67288, 7.98905], [16.8143, 7.53971], [16.6668, 7.67281], [16.658, 7.75353], [16.59415, 7.76444], [16.58315, 7.88657], [16.41583, 7.77971], [16.40703, 7.68809], [15.79942, 7.44149], [15.73118, 7.52006], [15.49743, 7.52179], [15.23397, 7.25135], [15.04717, 6.77085], [14.96311, 6.75693], [14.79966, 6.39043], [14.80122, 6.34866], [14.74206, 6.26356], [14.56149, 6.18928], [14.43073, 6.08867], [14.42917, 6.00508], [14.49455, 5.91683], [14.60974, 5.91838], [14.62375, 5.70466], [14.58951, 5.59777], [14.62531, 5.51411], [14.52724, 5.28319], [14.57083, 5.23979], [14.65489, 5.21343], [14.73383, 4.6135], [15.00825, 4.41458], [15.08609, 4.30282], [15.10644, 4.1362], [15.17482, 4.05131], [15.07686, 4.01805], [15.73522, 3.24348], [15.77725, 3.26835], [16.05449, 3.02306], [16.08252, 2.45708], [16.19357, 2.21537], [16.50126, 2.84739], [16.46701, 2.92512], [16.57598, 3.47999], [16.68283, 3.54257], [17.01746, 3.55136], [17.35649, 3.63045], [17.46876, 3.70515], [17.60966, 3.63705], [17.83421, 3.61068], [17.85842, 3.53378], [18.05656, 3.56893], [18.14902, 3.54476], [18.17323, 3.47665], [18.24148, 3.50302], [18.2723, 3.57992], [18.39558, 3.58212], [18.49245, 3.63924], [18.58711, 3.49423], [18.62755, 3.47564], [20.60184, 4.42394], [20.90383, 4.44877], [21.08793, 4.39603], [21.11214, 4.33895], [21.21341, 4.29285], [21.25744, 4.33676], [21.55904, 4.25553], [21.6405, 4.317], [22.10721, 4.20723], [22.27682, 4.11347], [22.45504, 4.13039], [22.5431, 4.22041], [22.60915, 4.48821], [22.6928, 4.47285], [22.78526, 4.71423], [22.84691, 4.69887], [22.89094, 4.79321], [22.94817, 4.82392], [23.38847, 4.60013], [24.46719, 5.0915], [24.71816, 4.90509], [25.31256, 5.03668], [25.34558, 5.29101], [25.53271, 5.37431], [25.86073, 5.19455], [26.13371, 5.25594], [26.48595, 5.04984], [26.74572, 5.10685], [26.85579, 5.03887], [26.93064, 5.13535], [27.09575, 5.22305], [27.44012, 5.07349], [27.26886, 5.25876], [27.23017, 5.37167], [27.28621, 5.56382], [27.22705, 5.62889], [27.22705, 5.71254], [26.51721, 6.09655], [26.58259, 6.1987], [26.32729, 6.36272], [26.38022, 6.63493], [25.90076, 7.09549], [25.37461, 7.33024], [25.35281, 7.42595], [25.20337, 7.50312], [25.20649, 7.61115], [25.29214, 7.66675], [25.25319, 7.8487], [24.98855, 7.96588], [24.85156, 8.16933], [24.35965, 8.26177], [24.13238, 8.36959], [24.25691, 8.69288], [23.51905, 8.71749], [23.59065, 8.99743], [23.44744, 8.99128], [23.4848, 9.16959], [23.56263, 9.19418], [23.64358, 9.28637], [23.64981, 9.44303], [23.62179, 9.53823], [23.69155, 9.67566], [23.67164, 9.86923], [23.3128, 10.45214], [23.02221, 10.69235], [22.87758, 10.91915]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CG",
+           iso1A3: "COG",
+           iso1N3: "178",
+           wikidata: "Q971",
+           nameEn: "Republic of the Congo",
+           groups: ["017", "202", "002", "UN"],
+           callingCodes: ["242"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[18.62755, 3.47564], [18.58711, 3.49423], [18.49245, 3.63924], [18.39558, 3.58212], [18.2723, 3.57992], [18.24148, 3.50302], [18.17323, 3.47665], [18.14902, 3.54476], [18.05656, 3.56893], [17.85842, 3.53378], [17.83421, 3.61068], [17.60966, 3.63705], [17.46876, 3.70515], [17.35649, 3.63045], [17.01746, 3.55136], [16.68283, 3.54257], [16.57598, 3.47999], [16.46701, 2.92512], [16.50126, 2.84739], [16.19357, 2.21537], [16.15568, 2.18955], [16.08563, 2.19733], [16.05294, 1.9811], [16.14634, 1.70259], [16.02647, 1.65591], [16.02959, 1.76483], [15.48942, 1.98265], [15.34776, 1.91264], [15.22634, 2.03243], [15.00996, 1.98887], [14.61145, 2.17866], [13.29457, 2.16106], [13.13461, 1.57238], [13.25447, 1.32339], [13.15519, 1.23368], [13.89582, 1.4261], [14.25186, 1.39842], [14.48179, 0.9152], [14.26066, 0.57255], [14.10909, 0.58563], [13.88648, 0.26652], [13.90632, -0.2287], [14.06862, -0.20826], [14.2165, -0.38261], [14.41887, -0.44799], [14.52569, -0.57818], [14.41838, -1.89412], [14.25932, -1.97624], [14.23518, -2.15671], [14.16202, -2.23916], [14.23829, -2.33715], [14.10442, -2.49268], [13.85846, -2.46935], [13.92073, -2.35581], [13.75884, -2.09293], [13.47977, -2.43224], [13.02759, -2.33098], [12.82172, -1.91091], [12.61312, -1.8129], [12.44656, -1.92025], [12.47925, -2.32626], [12.04895, -2.41704], [11.96866, -2.33559], [11.74605, -2.39936], [11.57637, -2.33379], [11.64487, -2.61865], [11.5359, -2.85654], [11.64798, -2.81146], [11.80365, -3.00424], [11.70558, -3.0773], [11.70227, -3.17465], [11.96554, -3.30267], [11.8318, -3.5812], [11.92719, -3.62768], [11.87083, -3.71571], [11.68608, -3.68942], [11.57949, -3.52798], [11.48764, -3.51089], [11.22301, -3.69888], [11.12647, -3.94169], [10.75913, -4.39519], [11.50888, -5.33417], [12.00924, -5.02627], [12.16068, -4.90089], [12.20901, -4.75642], [12.25587, -4.79437], [12.32324, -4.78415], [12.40964, -4.60609], [12.64835, -4.55937], [12.76844, -4.38709], [12.87096, -4.40315], [12.91489, -4.47907], [13.09648, -4.63739], [13.11182, -4.5942], [13.41764, -4.89897], [13.50305, -4.77818], [13.70417, -4.72601], [13.71794, -4.44864], [13.81162, -4.41842], [13.9108, -4.50906], [14.40672, -4.28381], [14.3957, -4.36623], [14.47284, -4.42941], [14.37366, -4.56125], [14.41499, -4.8825], [14.5059, -4.84956], [14.67948, -4.92093], [14.83101, -4.80838], [15.1978, -4.32388], [15.25411, -4.31121], [15.32693, -4.27282], [15.41785, -4.28381], [15.48121, -4.22062], [15.53081, -4.042], [15.89448, -3.9513], [16.21407, -3.2969], [16.1755, -3.25014], [16.22785, -2.59528], [16.16173, -2.16586], [16.50336, -1.8795], [16.70724, -1.45815], [16.97999, -1.12762], [17.32438, -0.99265], [17.72112, -0.52707], [17.66051, -0.26535], [17.81204, 0.23884], [17.93877, 0.32424], [17.95255, 0.48128], [17.86989, 0.58873], [17.85887, 1.04327], [18.08034, 1.58553], [18.10683, 2.26876], [18.63857, 3.19342], [18.62755, 3.47564]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CH",
+           iso1A3: "CHE",
+           iso1N3: "756",
+           wikidata: "Q39",
+           nameEn: "Switzerland",
+           groups: ["155", "150", "UN"],
+           callingCodes: ["41"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[8.72809, 47.69282], [8.72617, 47.69651], [8.73671, 47.7169], [8.70543, 47.73121], [8.74251, 47.75168], [8.71778, 47.76571], [8.68985, 47.75686], [8.68022, 47.78599], [8.65292, 47.80066], [8.64425, 47.76398], [8.62408, 47.7626], [8.61657, 47.79998], [8.56415, 47.80633], [8.56814, 47.78001], [8.48868, 47.77215], [8.45771, 47.7493], [8.44807, 47.72426], [8.40569, 47.69855], [8.4211, 47.68407], [8.40473, 47.67499], [8.41346, 47.66676], [8.42264, 47.66667], [8.44711, 47.65379], [8.4667, 47.65747], [8.46605, 47.64103], [8.49656, 47.64709], [8.5322, 47.64687], [8.52801, 47.66059], [8.56141, 47.67088], [8.57683, 47.66158], [8.6052, 47.67258], [8.61113, 47.66332], [8.62884, 47.65098], [8.62049, 47.63757], [8.60412, 47.63735], [8.61471, 47.64514], [8.60701, 47.65271], [8.59545, 47.64298], [8.60348, 47.61204], [8.57586, 47.59537], [8.55756, 47.62394], [8.51686, 47.63476], [8.50747, 47.61897], [8.45578, 47.60121], [8.46637, 47.58389], [8.48949, 47.588], [8.49431, 47.58107], [8.43235, 47.56617], [8.39477, 47.57826], [8.38273, 47.56608], [8.32735, 47.57133], [8.30277, 47.58607], [8.29524, 47.5919], [8.29722, 47.60603], [8.2824, 47.61225], [8.26313, 47.6103], [8.25863, 47.61571], [8.23809, 47.61204], [8.22577, 47.60385], [8.22011, 47.6181], [8.20617, 47.62141], [8.19378, 47.61636], [8.1652, 47.5945], [8.14947, 47.59558], [8.13823, 47.59147], [8.13662, 47.58432], [8.11543, 47.5841], [8.10395, 47.57918], [8.10002, 47.56504], [8.08557, 47.55768], [8.06663, 47.56374], [8.04383, 47.55443], [8.02136, 47.55096], [8.00113, 47.55616], [7.97581, 47.55493], [7.95682, 47.55789], [7.94494, 47.54511], [7.91251, 47.55031], [7.90673, 47.57674], [7.88664, 47.58854], [7.84412, 47.5841], [7.81901, 47.58798], [7.79486, 47.55691], [7.75261, 47.54599], [7.71961, 47.54219], [7.69642, 47.53297], [7.68101, 47.53232], [7.6656, 47.53752], [7.66174, 47.54554], [7.65083, 47.54662], [7.63338, 47.56256], [7.67655, 47.56435], [7.68904, 47.57133], [7.67115, 47.5871], [7.68486, 47.59601], [7.69385, 47.60099], [7.68229, 47.59905], [7.67395, 47.59212], [7.64599, 47.59695], [7.64213, 47.5944], [7.64309, 47.59151], [7.61929, 47.57683], [7.60459, 47.57869], [7.60523, 47.58519], [7.58945, 47.59017], [7.58386, 47.57536], [7.56684, 47.57785], [7.56548, 47.57617], [7.55689, 47.57232], [7.55652, 47.56779], [7.53634, 47.55553], [7.52831, 47.55347], [7.51723, 47.54578], [7.50873, 47.54546], [7.49691, 47.53821], [7.50588, 47.52856], [7.51904, 47.53515], [7.53199, 47.5284], [7.5229, 47.51644], [7.49804, 47.51798], [7.51076, 47.49651], [7.47534, 47.47932], [7.43356, 47.49712], [7.42923, 47.48628], [7.4583, 47.47216], [7.4462, 47.46264], [7.43088, 47.45846], [7.40308, 47.43638], [7.35603, 47.43432], [7.33526, 47.44186], [7.24669, 47.4205], [7.17026, 47.44312], [7.19583, 47.49455], [7.16249, 47.49025], [7.12781, 47.50371], [7.07425, 47.48863], [7.0231, 47.50522], [6.98425, 47.49432], [7.0024, 47.45264], [6.93953, 47.43388], [6.93744, 47.40714], [6.88542, 47.37262], [6.87959, 47.35335], [7.03125, 47.36996], [7.0564, 47.35134], [7.05305, 47.33304], [6.94316, 47.28747], [6.95108, 47.26428], [6.9508, 47.24338], [6.8489, 47.15933], [6.76788, 47.1208], [6.68823, 47.06616], [6.71531, 47.0494], [6.43341, 46.92703], [6.46456, 46.88865], [6.43216, 46.80336], [6.45209, 46.77502], [6.38351, 46.73171], [6.27135, 46.68251], [6.11084, 46.57649], [6.1567, 46.54402], [6.07269, 46.46244], [6.08427, 46.44305], [6.06407, 46.41676], [6.09926, 46.40768], [6.15016, 46.3778], [6.15985, 46.37721], [6.16987, 46.36759], [6.15738, 46.3491], [6.13876, 46.33844], [6.1198, 46.31157], [6.11697, 46.29547], [6.1013, 46.28512], [6.11926, 46.2634], [6.12446, 46.25059], [6.10071, 46.23772], [6.08563, 46.24651], [6.07072, 46.24085], [6.0633, 46.24583], [6.05029, 46.23518], [6.04602, 46.23127], [6.03342, 46.2383], [6.02461, 46.23313], [5.97542, 46.21525], [5.96515, 46.19638], [5.99573, 46.18587], [5.98846, 46.17046], [5.98188, 46.17392], [5.97508, 46.15863], [5.9641, 46.14412], [5.95781, 46.12925], [5.97893, 46.13303], [5.9871, 46.14499], [6.01791, 46.14228], [6.03614, 46.13712], [6.04564, 46.14031], [6.05203, 46.15191], [6.07491, 46.14879], [6.09199, 46.15191], [6.09926, 46.14373], [6.13397, 46.1406], [6.15305, 46.15194], [6.18116, 46.16187], [6.18871, 46.16644], [6.18707, 46.17999], [6.19552, 46.18401], [6.19807, 46.18369], [6.20539, 46.19163], [6.21114, 46.1927], [6.21273, 46.19409], [6.21603, 46.19507], [6.21844, 46.19837], [6.22222, 46.19888], [6.22175, 46.20045], [6.23544, 46.20714], [6.23913, 46.20511], [6.24821, 46.20531], [6.26007, 46.21165], [6.27694, 46.21566], [6.29663, 46.22688], [6.31041, 46.24417], [6.29474, 46.26221], [6.26749, 46.24745], [6.24952, 46.26255], [6.23775, 46.27822], [6.25137, 46.29014], [6.24826, 46.30175], [6.21981, 46.31304], [6.25432, 46.3632], [6.53358, 46.45431], [6.82312, 46.42661], [6.8024, 46.39171], [6.77152, 46.34784], [6.86052, 46.28512], [6.78968, 46.14058], [6.89321, 46.12548], [6.87868, 46.03855], [6.93862, 46.06502], [7.00946, 45.9944], [7.04151, 45.92435], [7.10685, 45.85653], [7.56343, 45.97421], [7.85949, 45.91485], [7.9049, 45.99945], [7.98881, 45.99867], [8.02906, 46.10331], [8.11383, 46.11577], [8.16866, 46.17817], [8.08814, 46.26692], [8.31162, 46.38044], [8.30648, 46.41587], [8.42464, 46.46367], [8.46317, 46.43712], [8.45032, 46.26869], [8.62242, 46.12112], [8.75697, 46.10395], [8.80778, 46.10085], [8.85617, 46.0748], [8.79414, 46.00913], [8.78585, 45.98973], [8.79362, 45.99207], [8.8319, 45.9879], [8.85121, 45.97239], [8.86688, 45.96135], [8.88904, 45.95465], [8.93649, 45.86775], [8.94372, 45.86587], [8.93504, 45.86245], [8.91129, 45.8388], [8.94737, 45.84285], [8.9621, 45.83707], [8.99663, 45.83466], [9.00324, 45.82055], [9.0298, 45.82127], [9.03279, 45.82865], [9.03793, 45.83548], [9.03505, 45.83976], [9.04059, 45.8464], [9.04546, 45.84968], [9.06642, 45.8761], [9.09065, 45.89906], [8.99257, 45.9698], [9.01618, 46.04928], [9.24503, 46.23616], [9.29226, 46.32717], [9.25502, 46.43743], [9.28136, 46.49685], [9.36128, 46.5081], [9.40487, 46.46621], [9.45936, 46.50873], [9.46117, 46.37481], [9.57015, 46.2958], [9.71273, 46.29266], [9.73086, 46.35071], [9.95249, 46.38045], [10.07055, 46.21668], [10.14439, 46.22992], [10.17862, 46.25626], [10.10506, 46.3372], [10.165, 46.41051], [10.03715, 46.44479], [10.10307, 46.61003], [10.23674, 46.63484], [10.25309, 46.57432], [10.46136, 46.53164], [10.49375, 46.62049], [10.44686, 46.64162], [10.40475, 46.63671], [10.38659, 46.67847], [10.47197, 46.85698], [10.48376, 46.93891], [10.36933, 47.00212], [10.30031, 46.92093], [10.24128, 46.93147], [10.22675, 46.86942], [10.10715, 46.84296], [9.98058, 46.91434], [9.88266, 46.93343], [9.87935, 47.01337], [9.60717, 47.06091], [9.55721, 47.04762], [9.54041, 47.06495], [9.47548, 47.05257], [9.47139, 47.06402], [9.51362, 47.08505], [9.52089, 47.10019], [9.51044, 47.13727], [9.48774, 47.17402], [9.4891, 47.19346], [9.50318, 47.22153], [9.52406, 47.24959], [9.53116, 47.27029], [9.54773, 47.2809], [9.55857, 47.29919], [9.58513, 47.31334], [9.59978, 47.34671], [9.62476, 47.36639], [9.65427, 47.36824], [9.66243, 47.37136], [9.6711, 47.37824], [9.67445, 47.38429], [9.67334, 47.39191], [9.6629, 47.39591], [9.65136, 47.40504], [9.65043, 47.41937], [9.6446, 47.43233], [9.64483, 47.43842], [9.65863, 47.44847], [9.65728, 47.45383], [9.6423, 47.45599], [9.62475, 47.45685], [9.62158, 47.45858], [9.60841, 47.47178], [9.60484, 47.46358], [9.60205, 47.46165], [9.59482, 47.46305], [9.58208, 47.48344], [9.56312, 47.49495], [9.55125, 47.53629], [9.25619, 47.65939], [9.18203, 47.65598], [9.17593, 47.65399], [9.1755, 47.65584], [9.1705, 47.65513], [9.15181, 47.66904], [9.13845, 47.66389], [9.09891, 47.67801], [9.02093, 47.6868], [8.94093, 47.65596], [8.89946, 47.64769], [8.87625, 47.65441], [8.87383, 47.67045], [8.85065, 47.68209], [8.86989, 47.70504], [8.82002, 47.71458], [8.80663, 47.73821], [8.77309, 47.72059], [8.76965, 47.7075], [8.79966, 47.70222], [8.79511, 47.67462], [8.75856, 47.68969], [8.72809, 47.69282]], [[8.95861, 45.96485], [8.96668, 45.98436], [8.97741, 45.98317], [8.97604, 45.96151], [8.95861, 45.96485]], [[8.70847, 47.68904], [8.68985, 47.69552], [8.66837, 47.68437], [8.65769, 47.68928], [8.67508, 47.6979], [8.66416, 47.71367], [8.70237, 47.71453], [8.71773, 47.69088], [8.70847, 47.68904]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CI",
+           iso1A3: "CIV",
+           iso1N3: "384",
+           wikidata: "Q1008",
+           nameEn: "C\xF4te d'Ivoire",
+           groups: ["011", "202", "002", "UN"],
+           callingCodes: ["225"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-7.52774, 3.7105], [-3.34019, 4.17519], [-3.10675, 5.08515], [-3.11073, 5.12675], [-3.063, 5.13665], [-2.96554, 5.10397], [-2.95261, 5.12477], [-2.75502, 5.10657], [-2.73074, 5.1364], [-2.77625, 5.34621], [-2.72737, 5.34789], [-2.76614, 5.60963], [-2.85378, 5.65156], [-2.93132, 5.62137], [-2.96671, 5.6415], [-2.95323, 5.71865], [-3.01896, 5.71697], [-3.25999, 6.62521], [-3.21954, 6.74407], [-3.23327, 6.81744], [-2.95438, 7.23737], [-2.97822, 7.27165], [-2.92339, 7.60847], [-2.79467, 7.86002], [-2.78395, 7.94974], [-2.74819, 7.92613], [-2.67787, 8.02055], [-2.61232, 8.02645], [-2.62901, 8.11495], [-2.49037, 8.20872], [-2.58243, 8.7789], [-2.66357, 9.01771], [-2.77799, 9.04949], [-2.69814, 9.22717], [-2.68802, 9.49343], [-2.76494, 9.40778], [-2.93012, 9.57403], [-3.00765, 9.74019], [-3.16609, 9.85147], [-3.19306, 9.93781], [-3.27228, 9.84981], [-3.31779, 9.91125], [-3.69703, 9.94279], [-4.25999, 9.76012], [-4.31392, 9.60062], [-4.6426, 9.70696], [-4.96621, 9.89132], [-4.96453, 9.99923], [-5.12465, 10.29788], [-5.39602, 10.2929], [-5.51058, 10.43177], [-5.65135, 10.46767], [-5.78124, 10.43952], [-5.99478, 10.19694], [-6.18851, 10.24244], [-6.1731, 10.46983], [-6.24795, 10.74248], [-6.325, 10.68624], [-6.40646, 10.69922], [-6.42847, 10.5694], [-6.52974, 10.59104], [-6.63541, 10.66893], [-6.68164, 10.35074], [-6.93921, 10.35291], [-7.01186, 10.25111], [-6.97444, 10.21644], [-7.00966, 10.15794], [-7.0603, 10.14711], [-7.13331, 10.24877], [-7.3707, 10.24677], [-7.44555, 10.44602], [-7.52261, 10.4655], [-7.54462, 10.40921], [-7.63048, 10.46334], [-7.92107, 10.15577], [-7.97971, 10.17117], [-8.01225, 10.1021], [-8.11921, 10.04577], [-8.15652, 9.94288], [-8.09434, 9.86936], [-8.14657, 9.55062], [-8.03463, 9.39604], [-7.85056, 9.41812], [-7.90777, 9.20456], [-7.73862, 9.08422], [-7.92518, 8.99332], [-7.95503, 8.81146], [-7.69882, 8.66148], [-7.65653, 8.36873], [-7.92518, 8.50652], [-8.22991, 8.48438], [-8.2411, 8.24196], [-8.062, 8.16071], [-7.98675, 8.20134], [-7.99919, 8.11023], [-7.94695, 8.00925], [-8.06449, 8.04989], [-8.13414, 7.87991], [-8.09931, 7.78626], [-8.21374, 7.54466], [-8.4003, 7.6285], [-8.47114, 7.55676], [-8.41935, 7.51203], [-8.37458, 7.25794], [-8.29249, 7.1691], [-8.31736, 6.82837], [-8.59456, 6.50612], [-8.48652, 6.43797], [-8.45666, 6.49977], [-8.38453, 6.35887], [-8.3298, 6.36381], [-8.17557, 6.28222], [-8.00642, 6.31684], [-7.90692, 6.27728], [-7.83478, 6.20309], [-7.8497, 6.08932], [-7.79747, 6.07696], [-7.78254, 5.99037], [-7.70294, 5.90625], [-7.67309, 5.94337], [-7.48155, 5.80974], [-7.46165, 5.84934], [-7.43677, 5.84687], [-7.43926, 5.74787], [-7.37209, 5.61173], [-7.43428, 5.42355], [-7.36463, 5.32944], [-7.46165, 5.26256], [-7.48901, 5.14118], [-7.55369, 5.08667], [-7.53876, 4.94294], [-7.59349, 4.8909], [-7.53259, 4.35145], [-7.52774, 3.7105]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CK",
+           iso1A3: "COK",
+           iso1N3: "184",
+           wikidata: "Q26988",
+           nameEn: "Cook Islands",
+           country: "NZ",
+           groups: ["061", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["682"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-168.15106, -10.26955], [-156.45576, -31.75456], [-156.48634, -15.52824], [-156.50903, -7.4975], [-168.15106, -10.26955]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CL",
+           iso1A3: "CHL",
+           iso1N3: "152",
+           wikidata: "Q298",
+           nameEn: "Chile",
+           groups: ["005", "419", "019", "UN"],
+           callingCodes: ["56"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-68.60702, -52.65781], [-68.41683, -52.33516], [-69.97824, -52.00845], [-71.99889, -51.98018], [-72.33873, -51.59954], [-72.31343, -50.58411], [-73.15765, -50.78337], [-73.55259, -49.92488], [-73.45156, -49.79461], [-73.09655, -49.14342], [-72.56894, -48.81116], [-72.54042, -48.52392], [-72.27662, -48.28727], [-72.50478, -47.80586], [-71.94152, -47.13595], [-71.68577, -46.55385], [-71.75614, -45.61611], [-71.35687, -45.22075], [-72.06985, -44.81756], [-71.26418, -44.75684], [-71.16436, -44.46244], [-71.81318, -44.38097], [-71.64206, -43.64774], [-72.14828, -42.85321], [-72.15541, -42.15941], [-71.74901, -42.11711], [-71.92726, -40.72714], [-71.37826, -38.91474], [-70.89532, -38.6923], [-71.24279, -37.20264], [-70.95047, -36.4321], [-70.38008, -36.02375], [-70.49416, -35.24145], [-69.87386, -34.13344], [-69.88099, -33.34489], [-70.55832, -31.51559], [-70.14479, -30.36595], [-69.8596, -30.26131], [-69.99507, -29.28351], [-69.80969, -29.07185], [-69.66709, -28.44055], [-69.22504, -27.95042], [-68.77586, -27.16029], [-68.43363, -27.08414], [-68.27677, -26.90626], [-68.59048, -26.49861], [-68.56909, -26.28146], [-68.38372, -26.15353], [-68.57622, -25.32505], [-68.38372, -25.08636], [-68.56909, -24.69831], [-68.24825, -24.42596], [-67.33563, -24.04237], [-66.99632, -22.99839], [-67.18382, -22.81525], [-67.54284, -22.89771], [-67.85114, -22.87076], [-68.18816, -21.28614], [-68.40403, -20.94562], [-68.53957, -20.91542], [-68.55383, -20.7355], [-68.44023, -20.62701], [-68.7276, -20.46178], [-68.74273, -20.08817], [-68.57132, -20.03134], [-68.54611, -19.84651], [-68.66761, -19.72118], [-68.41218, -19.40499], [-68.61989, -19.27584], [-68.80602, -19.08355], [-68.87082, -19.06003], [-68.94987, -18.93302], [-69.07432, -18.28259], [-69.14807, -18.16893], [-69.07496, -18.03715], [-69.28671, -17.94844], [-69.34126, -17.72753], [-69.46623, -17.60518], [-69.46897, -17.4988], [-69.66483, -17.65083], [-69.79087, -17.65563], [-69.82868, -17.72048], [-69.75305, -17.94605], [-69.81607, -18.12582], [-69.96732, -18.25992], [-70.16394, -18.31737], [-70.31267, -18.31258], [-70.378, -18.3495], [-70.59118, -18.35072], [-113.52687, -26.52828], [-68.11646, -58.14883], [-66.07313, -55.19618], [-67.11046, -54.94199], [-67.46182, -54.92205], [-68.01394, -54.8753], [-68.60733, -54.9125], [-68.60702, -52.65781]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CM",
+           iso1A3: "CMR",
+           iso1N3: "120",
+           wikidata: "Q1009",
+           nameEn: "Cameroon",
+           groups: ["017", "202", "002", "UN"],
+           callingCodes: ["237"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[14.83314, 12.62963], [14.55058, 12.78256], [14.56101, 12.91036], [14.46881, 13.08259], [14.08251, 13.0797], [14.20204, 12.53405], [14.17523, 12.41916], [14.22215, 12.36533], [14.4843, 12.35223], [14.6474, 12.17466], [14.61612, 11.7798], [14.55207, 11.72001], [14.64591, 11.66166], [14.6124, 11.51283], [14.17821, 11.23831], [13.97489, 11.30258], [13.78945, 11.00154], [13.7403, 11.00593], [13.70753, 10.94451], [13.73434, 10.9255], [13.54964, 10.61236], [13.5705, 10.53183], [13.43644, 10.13326], [13.34111, 10.12299], [13.25025, 10.03647], [13.25323, 10.00127], [13.286, 9.9822], [13.27409, 9.93232], [13.24132, 9.91031], [13.25025, 9.86042], [13.29941, 9.8296], [13.25472, 9.76795], [13.22642, 9.57266], [13.02385, 9.49334], [12.85628, 9.36698], [12.91958, 9.33905], [12.90022, 9.11411], [12.81085, 8.91992], [12.79, 8.75361], [12.71701, 8.7595], [12.68722, 8.65938], [12.44146, 8.6152], [12.4489, 8.52536], [12.26123, 8.43696], [12.24782, 8.17904], [12.19271, 8.10826], [12.20909, 7.97553], [11.99908, 7.67302], [12.01844, 7.52981], [11.93205, 7.47812], [11.84864, 7.26098], [11.87396, 7.09398], [11.63117, 6.9905], [11.55818, 6.86186], [11.57755, 6.74059], [11.51499, 6.60892], [11.42264, 6.5882], [11.42041, 6.53789], [11.09495, 6.51717], [11.09644, 6.68437], [10.94302, 6.69325], [10.8179, 6.83377], [10.83727, 6.9358], [10.60789, 7.06885], [10.59746, 7.14719], [10.57214, 7.16345], [10.53639, 6.93432], [10.21466, 6.88996], [10.15135, 7.03781], [9.86314, 6.77756], [9.77824, 6.79088], [9.70674, 6.51717], [9.51757, 6.43874], [8.84209, 5.82562], [8.88156, 5.78857], [8.83687, 5.68483], [8.92029, 5.58403], [8.78027, 5.1243], [8.60302, 4.87353], [8.34397, 4.30689], [9.22018, 3.72052], [9.81162, 2.33797], [9.82123, 2.35097], [9.83754, 2.32428], [9.83238, 2.29079], [9.84716, 2.24676], [9.89012, 2.20457], [9.90749, 2.20049], [9.991, 2.16561], [11.3561, 2.17217], [11.37116, 2.29975], [13.28534, 2.25716], [13.29457, 2.16106], [14.61145, 2.17866], [15.00996, 1.98887], [15.22634, 2.03243], [15.34776, 1.91264], [15.48942, 1.98265], [16.02959, 1.76483], [16.02647, 1.65591], [16.14634, 1.70259], [16.05294, 1.9811], [16.08563, 2.19733], [16.15568, 2.18955], [16.19357, 2.21537], [16.08252, 2.45708], [16.05449, 3.02306], [15.77725, 3.26835], [15.73522, 3.24348], [15.07686, 4.01805], [15.17482, 4.05131], [15.10644, 4.1362], [15.08609, 4.30282], [15.00825, 4.41458], [14.73383, 4.6135], [14.65489, 5.21343], [14.57083, 5.23979], [14.52724, 5.28319], [14.62531, 5.51411], [14.58951, 5.59777], [14.62375, 5.70466], [14.60974, 5.91838], [14.49455, 5.91683], [14.42917, 6.00508], [14.43073, 6.08867], [14.56149, 6.18928], [14.74206, 6.26356], [14.80122, 6.34866], [14.79966, 6.39043], [14.96311, 6.75693], [15.04717, 6.77085], [15.23397, 7.25135], [15.49743, 7.52179], [15.56964, 7.58936], [15.59272, 7.7696], [15.50743, 7.79302], [15.20426, 8.50892], [15.09484, 8.65982], [14.83566, 8.80557], [14.35707, 9.19611], [14.37094, 9.2954], [13.97544, 9.6365], [14.01793, 9.73169], [14.1317, 9.82413], [14.20411, 10.00055], [14.4673, 10.00264], [14.80082, 9.93818], [14.95722, 9.97926], [15.05999, 9.94845], [15.14043, 9.99246], [15.24618, 9.99246], [15.41408, 9.92876], [15.68761, 9.99344], [15.50535, 10.1098], [15.30874, 10.31063], [15.23724, 10.47764], [15.14936, 10.53915], [15.15532, 10.62846], [15.06737, 10.80921], [15.09127, 10.87431], [15.04957, 11.02347], [15.10021, 11.04101], [15.0585, 11.40481], [15.13149, 11.5537], [15.06595, 11.71126], [15.11579, 11.79313], [15.04808, 11.8731], [15.05786, 12.0608], [15.0349, 12.10698], [15.00146, 12.1223], [14.96952, 12.0925], [14.89019, 12.16593], [14.90827, 12.3269], [14.83314, 12.62963]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CN",
+           iso1A3: "CHN",
+           iso1N3: "156",
+           wikidata: "Q148",
+           nameEn: "People's Republic of China"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CO",
+           iso1A3: "COL",
+           iso1N3: "170",
+           wikidata: "Q739",
+           nameEn: "Colombia",
+           groups: ["005", "419", "019", "UN"],
+           callingCodes: ["57"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-71.19849, 12.65801], [-81.58685, 18.0025], [-82.06974, 14.49418], [-82.56142, 11.91792], [-78.79327, 9.93766], [-77.58292, 9.22278], [-77.32389, 8.81247], [-77.45064, 8.49991], [-77.17257, 7.97422], [-77.57185, 7.51147], [-77.72514, 7.72348], [-77.72157, 7.47612], [-77.81426, 7.48319], [-77.89178, 7.22681], [-78.06168, 7.07793], [-82.12561, 4.00341], [-78.87137, 1.47457], [-78.42749, 1.15389], [-77.85677, 0.80197], [-77.7148, 0.85003], [-77.68613, 0.83029], [-77.66416, 0.81604], [-77.67815, 0.73863], [-77.49984, 0.64476], [-77.52001, 0.40782], [-76.89177, 0.24736], [-76.4094, 0.24015], [-76.41215, 0.38228], [-76.23441, 0.42294], [-75.82927, 0.09578], [-75.25764, -0.11943], [-75.18513, -0.0308], [-74.42701, -0.50218], [-74.26675, -0.97229], [-73.65312, -1.26222], [-72.92587, -2.44514], [-71.75223, -2.15058], [-70.94377, -2.23142], [-70.04609, -2.73906], [-70.71396, -3.7921], [-70.52393, -3.87553], [-70.3374, -3.79505], [-69.94708, -4.2431], [-69.43395, -1.42219], [-69.4215, -1.01853], [-69.59796, -0.75136], [-69.603, -0.51947], [-70.03658, -0.19681], [-70.04162, 0.55437], [-69.47696, 0.71065], [-69.20976, 0.57958], [-69.14422, 0.84172], [-69.26017, 1.06856], [-69.82987, 1.07864], [-69.83491, 1.69353], [-69.53746, 1.76408], [-69.38621, 1.70865], [-68.18128, 1.72881], [-68.26699, 1.83463], [-68.18632, 2.00091], [-67.9292, 1.82455], [-67.40488, 2.22258], [-67.299, 1.87494], [-67.15784, 1.80439], [-67.08222, 1.17441], [-66.85795, 1.22998], [-67.21967, 2.35778], [-67.65696, 2.81691], [-67.85862, 2.79173], [-67.85862, 2.86727], [-67.30945, 3.38393], [-67.50067, 3.75812], [-67.62671, 3.74303], [-67.85358, 4.53249], [-67.83341, 5.31104], [-67.59141, 5.5369], [-67.63914, 5.64963], [-67.58558, 5.84537], [-67.43513, 5.98835], [-67.4625, 6.20625], [-67.60654, 6.2891], [-69.41843, 6.1072], [-70.10716, 6.96516], [-70.7596, 7.09799], [-71.03941, 6.98163], [-71.37234, 7.01588], [-71.42212, 7.03854], [-71.44118, 7.02116], [-71.82441, 7.04314], [-72.04895, 7.03837], [-72.19437, 7.37034], [-72.43132, 7.40034], [-72.47415, 7.48928], [-72.45321, 7.57232], [-72.47827, 7.65604], [-72.46763, 7.79518], [-72.44454, 7.86031], [-72.46183, 7.90682], [-72.45806, 7.91141], [-72.47042, 7.92306], [-72.48183, 7.92909], [-72.48801, 7.94329], [-72.47213, 7.96106], [-72.39137, 8.03534], [-72.35163, 8.01163], [-72.36987, 8.19976], [-72.4042, 8.36513], [-72.65474, 8.61428], [-72.77415, 9.10165], [-72.94052, 9.10663], [-73.02119, 9.27584], [-73.36905, 9.16636], [-72.98085, 9.85253], [-72.88002, 10.44309], [-72.4767, 11.1117], [-72.24983, 11.14138], [-71.9675, 11.65536], [-71.3275, 11.85], [-70.92579, 11.96275], [-71.19849, 12.65801]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CP",
+           iso1A3: "CPT",
+           wikidata: "Q161258",
+           nameEn: "Clipperton Island",
+           country: "FR",
+           groups: ["013", "003", "019", "UN"],
+           isoStatus: "excRes"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-110.36279, 9.79626], [-108.755, 9.84085], [-109.04145, 11.13245], [-110.36279, 9.79626]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CR",
+           iso1A3: "CRI",
+           iso1N3: "188",
+           wikidata: "Q800",
+           nameEn: "Costa Rica",
+           groups: ["013", "003", "419", "019", "UN"],
+           callingCodes: ["506"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-83.68276, 11.01562], [-83.66597, 10.79916], [-83.90838, 10.71161], [-84.68197, 11.07568], [-84.92439, 10.9497], [-85.60529, 11.22607], [-85.71223, 11.06868], [-86.14524, 11.09059], [-87.41779, 5.02401], [-82.94503, 7.93865], [-82.89978, 8.04083], [-82.89137, 8.05755], [-82.88641, 8.10219], [-82.9388, 8.26634], [-83.05209, 8.33394], [-82.93056, 8.43465], [-82.8679, 8.44042], [-82.8382, 8.48117], [-82.83322, 8.52464], [-82.83975, 8.54755], [-82.82739, 8.60153], [-82.8794, 8.6981], [-82.92068, 8.74832], [-82.91377, 8.774], [-82.88253, 8.83331], [-82.72126, 8.97125], [-82.93516, 9.07687], [-82.93516, 9.46741], [-82.84871, 9.4973], [-82.87919, 9.62645], [-82.77206, 9.59573], [-82.66667, 9.49746], [-82.61345, 9.49881], [-82.56507, 9.57279], [-82.51044, 9.65379], [-83.54024, 10.96805], [-83.68276, 11.01562]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CU",
+           iso1A3: "CUB",
+           iso1N3: "192",
+           wikidata: "Q241",
+           nameEn: "Cuba",
+           groups: ["029", "003", "419", "019", "UN"],
+           callingCodes: ["53"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-73.62304, 20.6935], [-82.02215, 24.23074], [-85.77883, 21.92705], [-74.81171, 18.82201], [-73.62304, 20.6935]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CV",
+           iso1A3: "CPV",
+           iso1N3: "132",
+           wikidata: "Q1011",
+           nameEn: "Cape Verde",
+           groups: ["Q105472", "011", "202", "002", "UN"],
+           callingCodes: ["238"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-28.81604, 14.57305], [-20.39702, 14.12816], [-23.37101, 19.134], [-28.81604, 14.57305]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CW",
+           iso1A3: "CUW",
+           iso1N3: "531",
+           wikidata: "Q25279",
+           nameEn: "Cura\xE7ao",
+           aliases: ["NL-CW"],
+           country: "NL",
+           groups: ["Q1451600", "029", "003", "419", "019", "UN"],
+           callingCodes: ["599"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-68.90012, 12.62309], [-69.59009, 12.46019], [-68.99639, 11.79035], [-68.33524, 11.78151], [-68.90012, 12.62309]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CX",
+           iso1A3: "CXR",
+           iso1N3: "162",
+           wikidata: "Q31063",
+           nameEn: "Christmas Island",
+           country: "AU",
+           groups: ["053", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["61"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[105.66835, -9.31927], [104.67494, -11.2566], [106.66176, -11.14349], [105.66835, -9.31927]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CY",
+           iso1A3: "CYP",
+           iso1N3: "196",
+           wikidata: "Q229",
+           nameEn: "Republic of Cyprus",
+           groups: ["Q644636", "EU", "145", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["357"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[32.46489, 35.48584], [30.15137, 34.08517], [32.74412, 34.43926], [32.75515, 34.64985], [32.76136, 34.68318], [32.79433, 34.67883], [32.82717, 34.70622], [32.86014, 34.70585], [32.86167, 34.68734], [32.9068, 34.66102], [32.91398, 34.67343], [32.93043, 34.67091], [32.92807, 34.66736], [32.93449, 34.66241], [32.93693, 34.67027], [32.94379, 34.67111], [32.94683, 34.67907], [32.95539, 34.68471], [32.99135, 34.68061], [32.98668, 34.67268], [32.99014, 34.65518], [32.97736, 34.65277], [32.97079, 34.66112], [32.95325, 34.66462], [32.94796, 34.6587], [32.94976, 34.65204], [32.95471, 34.64528], [32.95323, 34.64075], [32.95891, 34.62919], [32.96718, 34.63446], [32.96968, 34.64046], [33.0138, 34.64424], [33.26744, 34.49942], [33.83531, 34.73974], [33.70575, 34.97947], [33.70639, 34.99303], [33.71514, 35.00294], [33.69731, 35.01754], [33.69938, 35.03123], [33.67678, 35.03866], [33.63765, 35.03869], [33.61215, 35.0527], [33.59658, 35.03635], [33.567, 35.04803], [33.57478, 35.06049], [33.53975, 35.08151], [33.48915, 35.06594], [33.47666, 35.00701], [33.45256, 35.00288], [33.45178, 35.02078], [33.47825, 35.04103], [33.48136, 35.0636], [33.46813, 35.10564], [33.41675, 35.16325], [33.4076, 35.20062], [33.38575, 35.2018], [33.37248, 35.18698], [33.3717, 35.1788], [33.36569, 35.17479], [33.35612, 35.17402], [33.35596, 35.17942], [33.34964, 35.17803], [33.35056, 35.18328], [33.31955, 35.18096], [33.3072, 35.16816], [33.27068, 35.16815], [33.15138, 35.19504], [33.11105, 35.15639], [33.08249, 35.17319], [33.01192, 35.15639], [32.94471, 35.09422], [32.86406, 35.1043], [32.85733, 35.07742], [32.70779, 35.14127], [32.70947, 35.18328], [32.64864, 35.19967], [32.60361, 35.16647], [32.46489, 35.48584]]], [[[33.74144, 35.01053], [33.7492, 35.01319], [33.74983, 35.02274], [33.74265, 35.02329], [33.73781, 35.02181], [33.7343, 35.01178], [33.74144, 35.01053]]], [[[33.77312, 34.9976], [33.75994, 35.00113], [33.75682, 34.99916], [33.76605, 34.99543], [33.76738, 34.99188], [33.7778, 34.98981], [33.77843, 34.988], [33.78149, 34.98854], [33.78318, 34.98699], [33.78571, 34.98951], [33.78917, 34.98854], [33.79191, 34.98914], [33.78516, 34.99582], [33.77553, 34.99518], [33.77312, 34.9976]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "CZ",
+           iso1A3: "CZE",
+           iso1N3: "203",
+           wikidata: "Q213",
+           nameEn: "Czechia",
+           groups: ["EU", "151", "150", "UN"],
+           callingCodes: ["420"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[14.82803, 50.86966], [14.79139, 50.81438], [14.70661, 50.84096], [14.61993, 50.86049], [14.63434, 50.8883], [14.65259, 50.90513], [14.64802, 50.93241], [14.58024, 50.91443], [14.56374, 50.922], [14.59702, 50.96148], [14.59908, 50.98685], [14.58215, 50.99306], [14.56432, 51.01008], [14.53438, 51.00374], [14.53321, 51.01679], [14.49873, 51.02242], [14.50809, 51.0427], [14.49991, 51.04692], [14.49154, 51.04382], [14.49202, 51.02286], [14.45827, 51.03712], [14.41335, 51.02086], [14.30098, 51.05515], [14.25665, 50.98935], [14.28776, 50.97718], [14.32353, 50.98556], [14.32793, 50.97379], [14.30251, 50.96606], [14.31422, 50.95243], [14.39848, 50.93866], [14.38691, 50.89907], [14.30098, 50.88448], [14.27123, 50.89386], [14.24314, 50.88761], [14.22331, 50.86049], [14.02982, 50.80662], [13.98864, 50.8177], [13.89113, 50.78533], [13.89444, 50.74142], [13.82942, 50.7251], [13.76316, 50.73487], [13.70204, 50.71771], [13.65977, 50.73096], [13.52474, 50.70394], [13.53748, 50.67654], [13.5226, 50.64721], [13.49742, 50.63133], [13.46413, 50.60102], [13.42189, 50.61243], [13.37485, 50.64931], [13.37805, 50.627], [13.32264, 50.60317], [13.32594, 50.58009], [13.29454, 50.57904], [13.25158, 50.59268], [13.19043, 50.50237], [13.13424, 50.51709], [13.08301, 50.50132], [13.0312, 50.50944], [13.02038, 50.4734], [13.02147, 50.44763], [12.98433, 50.42016], [12.94058, 50.40944], [12.82465, 50.45738], [12.73476, 50.43237], [12.73044, 50.42268], [12.70731, 50.39948], [12.67261, 50.41949], [12.51356, 50.39694], [12.48747, 50.37278], [12.49214, 50.35228], [12.48256, 50.34784], [12.46643, 50.35527], [12.43722, 50.33774], [12.43371, 50.32506], [12.39924, 50.32302], [12.40158, 50.29521], [12.36594, 50.28289], [12.35425, 50.23993], [12.33263, 50.24367], [12.32445, 50.20442], [12.33847, 50.19432], [12.32596, 50.17146], [12.29232, 50.17524], [12.28063, 50.19544], [12.28755, 50.22429], [12.23943, 50.24594], [12.24791, 50.25525], [12.26953, 50.25189], [12.25119, 50.27079], [12.20823, 50.2729], [12.18013, 50.32146], [12.10907, 50.32041], [12.13716, 50.27396], [12.09287, 50.25032], [12.19335, 50.19997], [12.21484, 50.16399], [12.1917, 50.13434], [12.2073, 50.10315], [12.23709, 50.10213], [12.27433, 50.0771], [12.26111, 50.06331], [12.30798, 50.05719], [12.49908, 49.97305], [12.47264, 49.94222], [12.55197, 49.92094], [12.48256, 49.83575], [12.46603, 49.78882], [12.40489, 49.76321], [12.4462, 49.70233], [12.52553, 49.68415], [12.53544, 49.61888], [12.56188, 49.6146], [12.60155, 49.52887], [12.64782, 49.52565], [12.64121, 49.47628], [12.669, 49.42935], [12.71227, 49.42363], [12.75854, 49.3989], [12.78168, 49.34618], [12.88414, 49.33541], [12.88249, 49.35479], [12.94859, 49.34079], [13.03618, 49.30417], [13.02957, 49.27399], [13.05883, 49.26259], [13.17665, 49.16713], [13.17019, 49.14339], [13.20405, 49.12303], [13.23689, 49.11412], [13.28242, 49.1228], [13.39479, 49.04812], [13.40802, 48.98851], [13.50221, 48.93752], [13.50552, 48.97441], [13.58319, 48.96899], [13.61624, 48.9462], [13.67739, 48.87886], [13.73854, 48.88538], [13.76994, 48.83537], [13.78977, 48.83319], [13.8096, 48.77877], [13.84023, 48.76988], [14.06151, 48.66873], [14.01482, 48.63788], [14.09104, 48.5943], [14.20691, 48.5898], [14.33909, 48.55852], [14.43076, 48.58855], [14.4587, 48.64695], [14.56139, 48.60429], [14.60808, 48.62881], [14.66762, 48.58215], [14.71794, 48.59794], [14.72756, 48.69502], [14.80584, 48.73489], [14.80821, 48.77711], [14.81545, 48.7874], [14.94773, 48.76268], [14.95641, 48.75915], [14.9758, 48.76857], [14.98112, 48.77524], [14.9782, 48.7766], [14.98032, 48.77959], [14.95072, 48.79101], [14.98917, 48.90082], [14.97612, 48.96983], [14.99878, 49.01444], [15.15534, 48.99056], [15.16358, 48.94278], [15.26177, 48.95766], [15.28305, 48.98831], [15.34823, 48.98444], [15.48027, 48.94481], [15.51357, 48.91549], [15.61622, 48.89541], [15.6921, 48.85973], [15.75341, 48.8516], [15.78087, 48.87644], [15.84404, 48.86921], [16.06034, 48.75436], [16.37345, 48.729], [16.40915, 48.74576], [16.46134, 48.80865], [16.67008, 48.77699], [16.68518, 48.7281], [16.71883, 48.73806], [16.79779, 48.70998], [16.90354, 48.71541], [16.93955, 48.60371], [17.00215, 48.70887], [17.11202, 48.82925], [17.19355, 48.87602], [17.29054, 48.85546], [17.3853, 48.80936], [17.45671, 48.85004], [17.5295, 48.81117], [17.7094, 48.86721], [17.73126, 48.87885], [17.77944, 48.92318], [17.87831, 48.92679], [17.91814, 49.01784], [18.06885, 49.03157], [18.1104, 49.08624], [18.15022, 49.24518], [18.18456, 49.28909], [18.36446, 49.3267], [18.4139, 49.36517], [18.4084, 49.40003], [18.44686, 49.39467], [18.54848, 49.47059], [18.53063, 49.49022], [18.57183, 49.51162], [18.6144, 49.49824], [18.67757, 49.50895], [18.74761, 49.492], [18.84521, 49.51672], [18.84786, 49.5446], [18.80479, 49.6815], [18.72838, 49.68163], [18.69817, 49.70473], [18.62676, 49.71983], [18.62943, 49.74603], [18.62645, 49.75002], [18.61368, 49.75426], [18.61278, 49.7618], [18.57183, 49.83334], [18.60341, 49.86256], [18.57045, 49.87849], [18.57697, 49.91565], [18.54299, 49.92537], [18.54495, 49.9079], [18.53423, 49.89906], [18.41604, 49.93498], [18.33562, 49.94747], [18.33278, 49.92415], [18.31914, 49.91565], [18.27794, 49.93863], [18.27107, 49.96779], [18.21752, 49.97309], [18.20241, 49.99958], [18.10628, 50.00223], [18.07898, 50.04535], [18.03212, 50.06574], [18.00396, 50.04954], [18.04585, 50.03311], [18.04585, 50.01194], [18.00191, 50.01723], [17.86886, 49.97452], [17.77669, 50.02253], [17.7506, 50.07896], [17.6888, 50.12037], [17.66683, 50.10275], [17.59404, 50.16437], [17.70528, 50.18812], [17.76296, 50.23382], [17.72176, 50.25665], [17.74648, 50.29966], [17.69292, 50.32859], [17.67764, 50.28977], [17.58889, 50.27837], [17.3702, 50.28123], [17.34548, 50.2628], [17.34273, 50.32947], [17.27681, 50.32246], [17.19991, 50.3654], [17.19579, 50.38817], [17.14498, 50.38117], [17.1224, 50.39494], [16.89229, 50.45117], [16.85933, 50.41093], [16.90877, 50.38642], [16.94448, 50.31281], [16.99803, 50.30316], [17.02138, 50.27772], [16.99803, 50.25753], [17.02825, 50.23118], [17.00353, 50.21449], [16.98018, 50.24172], [16.8456, 50.20834], [16.7014, 50.09659], [16.63137, 50.1142], [16.55446, 50.16613], [16.56407, 50.21009], [16.42674, 50.32509], [16.39379, 50.3207], [16.3622, 50.34875], [16.36495, 50.37679], [16.30289, 50.38292], [16.28118, 50.36891], [16.22821, 50.41054], [16.21585, 50.40627], [16.19526, 50.43291], [16.31413, 50.50274], [16.34572, 50.49575], [16.44597, 50.58041], [16.33611, 50.66579], [16.23174, 50.67101], [16.20839, 50.63096], [16.10265, 50.66405], [16.02437, 50.60046], [15.98317, 50.61528], [16.0175, 50.63009], [15.97219, 50.69799], [15.87331, 50.67188], [15.81683, 50.75666], [15.73186, 50.73885], [15.43798, 50.80833], [15.3803, 50.77187], [15.36656, 50.83956], [15.2773, 50.8907], [15.27043, 50.97724], [15.2361, 50.99886], [15.1743, 50.9833], [15.16744, 51.01959], [15.11937, 50.99021], [15.10152, 51.01095], [15.06218, 51.02269], [15.03895, 51.0123], [15.02433, 51.0242], [14.96419, 50.99108], [15.01088, 50.97984], [14.99852, 50.86817], [14.82803, 50.86966]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "DE",
+           iso1A3: "DEU",
+           iso1N3: "276",
+           wikidata: "Q183",
+           nameEn: "Germany",
+           groups: ["EU", "155", "150", "UN"],
+           callingCodes: ["49"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[8.70847, 47.68904], [8.71773, 47.69088], [8.70237, 47.71453], [8.66416, 47.71367], [8.67508, 47.6979], [8.65769, 47.68928], [8.66837, 47.68437], [8.68985, 47.69552], [8.70847, 47.68904]]], [[[8.72617, 47.69651], [8.72809, 47.69282], [8.75856, 47.68969], [8.79511, 47.67462], [8.79966, 47.70222], [8.76965, 47.7075], [8.77309, 47.72059], [8.80663, 47.73821], [8.82002, 47.71458], [8.86989, 47.70504], [8.85065, 47.68209], [8.87383, 47.67045], [8.87625, 47.65441], [8.89946, 47.64769], [8.94093, 47.65596], [9.02093, 47.6868], [9.09891, 47.67801], [9.13845, 47.66389], [9.15181, 47.66904], [9.1705, 47.65513], [9.1755, 47.65584], [9.17593, 47.65399], [9.18203, 47.65598], [9.25619, 47.65939], [9.55125, 47.53629], [9.72736, 47.53457], [9.76748, 47.5934], [9.80254, 47.59419], [9.82591, 47.58158], [9.8189, 47.54688], [9.87499, 47.52953], [9.87733, 47.54688], [9.92407, 47.53111], [9.96029, 47.53899], [10.00003, 47.48216], [10.03859, 47.48927], [10.07131, 47.45531], [10.09001, 47.46005], [10.1052, 47.4316], [10.06897, 47.40709], [10.09819, 47.35724], [10.11805, 47.37228], [10.16362, 47.36674], [10.17648, 47.38889], [10.2127, 47.38019], [10.22774, 47.38904], [10.23757, 47.37609], [10.19998, 47.32832], [10.2147, 47.31014], [10.17648, 47.29149], [10.17531, 47.27167], [10.23257, 47.27088], [10.33424, 47.30813], [10.39851, 47.37623], [10.4324, 47.38494], [10.4359, 47.41183], [10.47446, 47.43318], [10.46278, 47.47901], [10.44291, 47.48453], [10.4324, 47.50111], [10.44992, 47.5524], [10.43473, 47.58394], [10.47329, 47.58552], [10.48849, 47.54057], [10.56912, 47.53584], [10.60337, 47.56755], [10.63456, 47.5591], [10.68832, 47.55752], [10.6965, 47.54253], [10.7596, 47.53228], [10.77596, 47.51729], [10.88814, 47.53701], [10.91268, 47.51334], [10.86945, 47.5015], [10.87061, 47.4786], [10.90918, 47.48571], [10.93839, 47.48018], [10.92437, 47.46991], [10.98513, 47.42882], [10.97111, 47.41617], [10.97111, 47.39561], [11.11835, 47.39719], [11.12536, 47.41222], [11.20482, 47.43198], [11.25157, 47.43277], [11.22002, 47.3964], [11.27844, 47.39956], [11.29597, 47.42566], [11.33804, 47.44937], [11.4175, 47.44621], [11.38128, 47.47465], [11.4362, 47.51413], [11.52618, 47.50939], [11.58578, 47.52281], [11.58811, 47.55515], [11.60681, 47.57881], [11.63934, 47.59202], [11.84052, 47.58354], [11.85572, 47.60166], [12.0088, 47.62451], [12.02282, 47.61033], [12.05788, 47.61742], [12.13734, 47.60639], [12.17824, 47.61506], [12.18145, 47.61019], [12.17737, 47.60121], [12.18568, 47.6049], [12.20398, 47.60667], [12.20801, 47.61082], [12.19895, 47.64085], [12.18507, 47.65984], [12.18347, 47.66663], [12.16769, 47.68167], [12.16217, 47.70105], [12.18303, 47.70065], [12.22571, 47.71776], [12.2542, 47.7433], [12.26238, 47.73544], [12.24017, 47.69534], [12.26004, 47.67725], [12.27991, 47.68827], [12.336, 47.69534], [12.37222, 47.68433], [12.43883, 47.6977], [12.44117, 47.6741], [12.50076, 47.62293], [12.53816, 47.63553], [12.57438, 47.63238], [12.6071, 47.6741], [12.7357, 47.6787], [12.77777, 47.66689], [12.76492, 47.64485], [12.82101, 47.61493], [12.77427, 47.58025], [12.80699, 47.54477], [12.84672, 47.54556], [12.85256, 47.52741], [12.9624, 47.47452], [12.98344, 47.48716], [12.9998, 47.46267], [13.04537, 47.49426], [13.03252, 47.53373], [13.05355, 47.56291], [13.04537, 47.58183], [13.06641, 47.58577], [13.06407, 47.60075], [13.09562, 47.63304], [13.07692, 47.68814], [13.01382, 47.72116], [12.98578, 47.7078], [12.92969, 47.71094], [12.91333, 47.7178], [12.90274, 47.72513], [12.91711, 47.74026], [12.9353, 47.74788], [12.94371, 47.76281], [12.93202, 47.77302], [12.96311, 47.79957], [12.98543, 47.82896], [13.00588, 47.84374], [12.94163, 47.92927], [12.93886, 47.94046], [12.93642, 47.94436], [12.93419, 47.94063], [12.92668, 47.93879], [12.91985, 47.94069], [12.9211, 47.95135], [12.91683, 47.95647], [12.87476, 47.96195], [12.8549, 48.01122], [12.76141, 48.07373], [12.74973, 48.10885], [12.7617, 48.12796], [12.78595, 48.12445], [12.80676, 48.14979], [12.82673, 48.15245], [12.8362, 48.15876], [12.836, 48.1647], [12.84475, 48.16556], [12.87126, 48.20318], [12.95306, 48.20629], [13.02083, 48.25689], [13.0851, 48.27711], [13.126, 48.27867], [13.18093, 48.29577], [13.26039, 48.29422], [13.30897, 48.31575], [13.40709, 48.37292], [13.43929, 48.43386], [13.42527, 48.45711], [13.45727, 48.51092], [13.43695, 48.55776], [13.45214, 48.56472], [13.46967, 48.55157], [13.50663, 48.57506], [13.50131, 48.58091], [13.51291, 48.59023], [13.57535, 48.55912], [13.59705, 48.57013], [13.62508, 48.55501], [13.65186, 48.55092], [13.66113, 48.53558], [13.72802, 48.51208], [13.74816, 48.53058], [13.7513, 48.5624], [13.76921, 48.55324], [13.80519, 48.58026], [13.80038, 48.59487], [13.82609, 48.62345], [13.81901, 48.6761], [13.81283, 48.68426], [13.81791, 48.69832], [13.79337, 48.71375], [13.81863, 48.73257], [13.82266, 48.75544], [13.84023, 48.76988], [13.8096, 48.77877], [13.78977, 48.83319], [13.76994, 48.83537], [13.73854, 48.88538], [13.67739, 48.87886], [13.61624, 48.9462], [13.58319, 48.96899], [13.50552, 48.97441], [13.50221, 48.93752], [13.40802, 48.98851], [13.39479, 49.04812], [13.28242, 49.1228], [13.23689, 49.11412], [13.20405, 49.12303], [13.17019, 49.14339], [13.17665, 49.16713], [13.05883, 49.26259], [13.02957, 49.27399], [13.03618, 49.30417], [12.94859, 49.34079], [12.88249, 49.35479], [12.88414, 49.33541], [12.78168, 49.34618], [12.75854, 49.3989], [12.71227, 49.42363], [12.669, 49.42935], [12.64121, 49.47628], [12.64782, 49.52565], [12.60155, 49.52887], [12.56188, 49.6146], [12.53544, 49.61888], [12.52553, 49.68415], [12.4462, 49.70233], [12.40489, 49.76321], [12.46603, 49.78882], [12.48256, 49.83575], [12.55197, 49.92094], [12.47264, 49.94222], [12.49908, 49.97305], [12.30798, 50.05719], [12.26111, 50.06331], [12.27433, 50.0771], [12.23709, 50.10213], [12.2073, 50.10315], [12.1917, 50.13434], [12.21484, 50.16399], [12.19335, 50.19997], [12.09287, 50.25032], [12.13716, 50.27396], [12.10907, 50.32041], [12.18013, 50.32146], [12.20823, 50.2729], [12.25119, 50.27079], [12.26953, 50.25189], [12.24791, 50.25525], [12.23943, 50.24594], [12.28755, 50.22429], [12.28063, 50.19544], [12.29232, 50.17524], [12.32596, 50.17146], [12.33847, 50.19432], [12.32445, 50.20442], [12.33263, 50.24367], [12.35425, 50.23993], [12.36594, 50.28289], [12.40158, 50.29521], [12.39924, 50.32302], [12.43371, 50.32506], [12.43722, 50.33774], [12.46643, 50.35527], [12.48256, 50.34784], [12.49214, 50.35228], [12.48747, 50.37278], [12.51356, 50.39694], [12.67261, 50.41949], [12.70731, 50.39948], [12.73044, 50.42268], [12.73476, 50.43237], [12.82465, 50.45738], [12.94058, 50.40944], [12.98433, 50.42016], [13.02147, 50.44763], [13.02038, 50.4734], [13.0312, 50.50944], [13.08301, 50.50132], [13.13424, 50.51709], [13.19043, 50.50237], [13.25158, 50.59268], [13.29454, 50.57904], [13.32594, 50.58009], [13.32264, 50.60317], [13.37805, 50.627], [13.37485, 50.64931], [13.42189, 50.61243], [13.46413, 50.60102], [13.49742, 50.63133], [13.5226, 50.64721], [13.53748, 50.67654], [13.52474, 50.70394], [13.65977, 50.73096], [13.70204, 50.71771], [13.76316, 50.73487], [13.82942, 50.7251], [13.89444, 50.74142], [13.89113, 50.78533], [13.98864, 50.8177], [14.02982, 50.80662], [14.22331, 50.86049], [14.24314, 50.88761], [14.27123, 50.89386], [14.30098, 50.88448], [14.38691, 50.89907], [14.39848, 50.93866], [14.31422, 50.95243], [14.30251, 50.96606], [14.32793, 50.97379], [14.32353, 50.98556], [14.28776, 50.97718], [14.25665, 50.98935], [14.30098, 51.05515], [14.41335, 51.02086], [14.45827, 51.03712], [14.49202, 51.02286], [14.49154, 51.04382], [14.49991, 51.04692], [14.50809, 51.0427], [14.49873, 51.02242], [14.53321, 51.01679], [14.53438, 51.00374], [14.56432, 51.01008], [14.58215, 50.99306], [14.59908, 50.98685], [14.59702, 50.96148], [14.56374, 50.922], [14.58024, 50.91443], [14.64802, 50.93241], [14.65259, 50.90513], [14.63434, 50.8883], [14.61993, 50.86049], [14.70661, 50.84096], [14.79139, 50.81438], [14.82803, 50.86966], [14.81664, 50.88148], [14.89681, 50.9422], [14.89252, 50.94999], [14.92942, 50.99744], [14.95529, 51.04552], [14.97938, 51.07742], [14.98229, 51.11354], [14.99689, 51.12205], [14.99079, 51.14284], [14.99646, 51.14365], [15.00083, 51.14974], [14.99414, 51.15813], [14.99311, 51.16249], [15.0047, 51.16874], [15.01242, 51.21285], [15.04288, 51.28387], [14.98008, 51.33449], [14.96899, 51.38367], [14.9652, 51.44793], [14.94749, 51.47155], [14.73219, 51.52922], [14.72652, 51.53902], [14.73047, 51.54606], [14.71125, 51.56209], [14.7727, 51.61263], [14.75759, 51.62318], [14.75392, 51.67445], [14.69065, 51.70842], [14.66386, 51.73282], [14.64625, 51.79472], [14.60493, 51.80473], [14.59089, 51.83302], [14.6588, 51.88359], [14.6933, 51.9044], [14.70601, 51.92944], [14.7177, 51.94048], [14.72163, 51.95188], [14.71836, 51.95606], [14.7139, 51.95643], [14.70488, 51.97679], [14.71339, 52.00337], [14.76026, 52.06624], [14.72971, 52.09167], [14.6917, 52.10283], [14.67683, 52.13936], [14.70616, 52.16927], [14.68344, 52.19612], [14.71319, 52.22144], [14.70139, 52.25038], [14.58149, 52.28007], [14.56378, 52.33838], [14.55228, 52.35264], [14.54423, 52.42568], [14.63056, 52.48993], [14.60081, 52.53116], [14.6289, 52.57136], [14.61073, 52.59847], [14.22071, 52.81175], [14.13806, 52.82392], [14.12256, 52.84311], [14.15873, 52.87715], [14.14056, 52.95786], [14.25954, 53.00264], [14.35044, 53.05829], [14.38679, 53.13669], [14.36696, 53.16444], [14.37853, 53.20405], [14.40662, 53.21098], [14.45125, 53.26241], [14.44133, 53.27427], [14.4215, 53.27724], [14.35209, 53.49506], [14.3273, 53.50587], [14.30416, 53.55499], [14.31904, 53.61581], [14.2853, 53.63392], [14.28477, 53.65955], [14.27133, 53.66613], [14.2836, 53.67721], [14.26782, 53.69866], [14.27249, 53.74464], [14.21323, 53.8664], [14.20823, 53.90776], [14.18544, 53.91258], [14.20647, 53.91671], [14.22634, 53.9291], [14.20811, 54.12784], [13.93395, 54.84044], [12.85844, 54.82438], [11.90309, 54.38543], [11.00303, 54.63689], [10.31111, 54.65968], [10.16755, 54.73883], [9.89314, 54.84171], [9.73563, 54.8247], [9.61187, 54.85548], [9.62734, 54.88057], [9.58937, 54.88785], [9.4659, 54.83131], [9.43155, 54.82586], [9.41213, 54.84254], [9.38532, 54.83968], [9.36496, 54.81749], [9.33849, 54.80233], [9.32771, 54.80602], [9.2474, 54.8112], [9.23445, 54.83432], [9.24631, 54.84726], [9.20571, 54.85841], [9.14275, 54.87421], [9.04629, 54.87249], [8.92795, 54.90452], [8.81178, 54.90518], [8.76387, 54.8948], [8.63979, 54.91069], [8.55769, 54.91837], [8.45719, 55.06747], [8.02459, 55.09613], [5.45168, 54.20039], [6.91025, 53.44221], [7.00198, 53.32672], [7.19052, 53.31866], [7.21679, 53.20058], [7.22681, 53.18165], [7.17898, 53.13817], [7.21694, 53.00742], [7.07253, 52.81083], [7.04557, 52.63318], [6.77307, 52.65375], [6.71641, 52.62905], [6.69507, 52.488], [6.94293, 52.43597], [6.99041, 52.47235], [7.03417, 52.40237], [7.07044, 52.37805], [7.02703, 52.27941], [7.06365, 52.23789], [7.03729, 52.22695], [6.9897, 52.2271], [6.97189, 52.20329], [6.83984, 52.11728], [6.76117, 52.11895], [6.68128, 52.05052], [6.83035, 51.9905], [6.82357, 51.96711], [6.72319, 51.89518], [6.68386, 51.91861], [6.58556, 51.89386], [6.50231, 51.86313], [6.47179, 51.85395], [6.38815, 51.87257], [6.40704, 51.82771], [6.30593, 51.84998], [6.29872, 51.86801], [6.21443, 51.86801], [6.15349, 51.90439], [6.11551, 51.89769], [6.16902, 51.84094], [6.10337, 51.84829], [6.06705, 51.86136], [5.99848, 51.83195], [5.94568, 51.82786], [5.98665, 51.76944], [5.95003, 51.7493], [6.04091, 51.71821], [6.02767, 51.6742], [6.11759, 51.65609], [6.09055, 51.60564], [6.18017, 51.54096], [6.21724, 51.48568], [6.20654, 51.40049], [6.22641, 51.39948], [6.22674, 51.36135], [6.16977, 51.33169], [6.07889, 51.24432], [6.07889, 51.17038], [6.17384, 51.19589], [6.16706, 51.15677], [5.98292, 51.07469], [5.9541, 51.03496], [5.9134, 51.06736], [5.86735, 51.05182], [5.87849, 51.01969], [5.90493, 51.00198], [5.90296, 50.97356], [5.95282, 50.98728], [6.02697, 50.98303], [6.01615, 50.93367], [6.09297, 50.92066], [6.07486, 50.89307], [6.08805, 50.87223], [6.07693, 50.86025], [6.07431, 50.84674], [6.05702, 50.85179], [6.05623, 50.8572], [6.01921, 50.84435], [6.02328, 50.81694], [6.00462, 50.80065], [5.98404, 50.80988], [5.97497, 50.79992], [6.02624, 50.77453], [6.01976, 50.75398], [6.03889, 50.74618], [6.0326, 50.72647], [6.0406, 50.71848], [6.04428, 50.72861], [6.11707, 50.72231], [6.17852, 50.6245], [6.26957, 50.62444], [6.2476, 50.60392], [6.24888, 50.59869], [6.24005, 50.58732], [6.22581, 50.5907], [6.20281, 50.56952], [6.17739, 50.55875], [6.17802, 50.54179], [6.19735, 50.53576], [6.19579, 50.5313], [6.18716, 50.52653], [6.19193, 50.5212], [6.20599, 50.52089], [6.22335, 50.49578], [6.26637, 50.50272], [6.30809, 50.50058], [6.3465, 50.48833], [6.34005, 50.46083], [6.37219, 50.45397], [6.36852, 50.40776], [6.34406, 50.37994], [6.3688, 50.35898], [6.40785, 50.33557], [6.40641, 50.32425], [6.35701, 50.31139], [6.32488, 50.32333], [6.29949, 50.30887], [6.28797, 50.27458], [6.208, 50.25179], [6.16853, 50.2234], [6.18364, 50.20815], [6.18739, 50.1822], [6.14588, 50.17106], [6.14132, 50.14971], [6.15298, 50.14126], [6.1379, 50.12964], [6.12055, 50.09171], [6.11274, 50.05916], [6.13458, 50.04141], [6.13044, 50.02929], [6.14666, 50.02207], [6.13794, 50.01466], [6.13273, 50.02019], [6.1295, 50.01849], [6.13806, 50.01056], [6.14948, 50.00908], [6.14147, 49.99563], [6.1701, 49.98518], [6.16466, 49.97086], [6.17872, 49.9537], [6.18554, 49.95622], [6.18045, 49.96611], [6.19089, 49.96991], [6.19856, 49.95053], [6.22094, 49.94955], [6.22608, 49.929], [6.21882, 49.92403], [6.22926, 49.92096], [6.23496, 49.89972], [6.26146, 49.88203], [6.28874, 49.87592], [6.29692, 49.86685], [6.30963, 49.87021], [6.32303, 49.85133], [6.32098, 49.83728], [6.33585, 49.83785], [6.34267, 49.84974], [6.36576, 49.85032], [6.40022, 49.82029], [6.42521, 49.81591], [6.42905, 49.81091], [6.44131, 49.81443], [6.45425, 49.81164], [6.47111, 49.82263], [6.48718, 49.81267], [6.50647, 49.80916], [6.51215, 49.80124], [6.52121, 49.81338], [6.53122, 49.80666], [6.52169, 49.79787], [6.50534, 49.78952], [6.51669, 49.78336], [6.51056, 49.77515], [6.51828, 49.76855], [6.51646, 49.75961], [6.50174, 49.75292], [6.50193, 49.73291], [6.51805, 49.72425], [6.51397, 49.72058], [6.50261, 49.72718], [6.49535, 49.72645], [6.49694, 49.72205], [6.5042, 49.71808], [6.50647, 49.71353], [6.49785, 49.71118], [6.48014, 49.69767], [6.46048, 49.69092], [6.44654, 49.67799], [6.42937, 49.66857], [6.42726, 49.66078], [6.43768, 49.66021], [6.4413, 49.65722], [6.41861, 49.61723], [6.39822, 49.60081], [6.385, 49.59946], [6.37464, 49.58886], [6.38342, 49.5799], [6.38024, 49.57593], [6.36676, 49.57813], [6.35825, 49.57053], [6.38228, 49.55855], [6.38072, 49.55171], [6.35666, 49.52931], [6.36788, 49.50377], [6.36907, 49.48931], [6.36778, 49.46937], [6.38352, 49.46463], [6.39168, 49.4667], [6.40274, 49.46546], [6.42432, 49.47683], [6.55404, 49.42464], [6.533, 49.40748], [6.60091, 49.36864], [6.58807, 49.35358], [6.572, 49.35027], [6.60186, 49.31055], [6.66583, 49.28065], [6.69274, 49.21661], [6.71843, 49.2208], [6.73256, 49.20486], [6.71137, 49.18808], [6.73765, 49.16375], [6.78265, 49.16793], [6.83385, 49.15162], [6.84703, 49.15734], [6.86225, 49.18185], [6.85016, 49.19354], [6.85119, 49.20038], [6.83555, 49.21249], [6.85939, 49.22376], [6.89298, 49.20863], [6.91875, 49.22261], [6.93831, 49.2223], [6.94028, 49.21641], [6.95963, 49.203], [6.97273, 49.2099], [7.01318, 49.19018], [7.03459, 49.19096], [7.0274, 49.17042], [7.03178, 49.15734], [7.04662, 49.13724], [7.04409, 49.12123], [7.04843, 49.11422], [7.05548, 49.11185], [7.06642, 49.11415], [7.07162, 49.1255], [7.09007, 49.13094], [7.07859, 49.15031], [7.10715, 49.15631], [7.10384, 49.13787], [7.12504, 49.14253], [7.1358, 49.1282], [7.1593, 49.1204], [7.23473, 49.12971], [7.29514, 49.11426], [7.3195, 49.14231], [7.35995, 49.14399], [7.3662, 49.17308], [7.44052, 49.18354], [7.44455, 49.16765], [7.49473, 49.17], [7.49172, 49.13915], [7.53012, 49.09818], [7.56416, 49.08136], [7.62575, 49.07654], [7.63618, 49.05428], [7.75948, 49.04562], [7.79557, 49.06583], [7.86386, 49.03499], [7.93641, 49.05544], [7.97783, 49.03161], [8.14189, 48.97833], [8.22604, 48.97352], [8.20031, 48.95856], [8.19989, 48.95825], [8.12813, 48.87985], [8.10253, 48.81829], [8.06802, 48.78957], [8.0326, 48.79017], [8.01534, 48.76085], [7.96994, 48.75606], [7.96812, 48.72491], [7.89002, 48.66317], [7.84098, 48.64217], [7.80057, 48.5857], [7.80167, 48.54758], [7.80647, 48.51239], [7.76833, 48.48945], [7.73109, 48.39192], [7.74562, 48.32736], [7.69022, 48.30018], [7.6648, 48.22219], [7.57137, 48.12292], [7.56966, 48.03265], [7.62302, 47.97898], [7.55673, 47.87371], [7.52921, 47.77747], [7.54761, 47.72912], [7.53722, 47.71635], [7.51266, 47.70197], [7.51915, 47.68335], [7.52067, 47.66437], [7.53384, 47.65115], [7.5591, 47.63849], [7.57423, 47.61628], [7.58851, 47.60794], [7.59301, 47.60058], [7.58945, 47.59017], [7.60523, 47.58519], [7.60459, 47.57869], [7.61929, 47.57683], [7.64309, 47.59151], [7.64213, 47.5944], [7.64599, 47.59695], [7.67395, 47.59212], [7.68229, 47.59905], [7.69385, 47.60099], [7.68486, 47.59601], [7.67115, 47.5871], [7.68904, 47.57133], [7.67655, 47.56435], [7.63338, 47.56256], [7.65083, 47.54662], [7.66174, 47.54554], [7.6656, 47.53752], [7.68101, 47.53232], [7.69642, 47.53297], [7.71961, 47.54219], [7.75261, 47.54599], [7.79486, 47.55691], [7.81901, 47.58798], [7.84412, 47.5841], [7.88664, 47.58854], [7.90673, 47.57674], [7.91251, 47.55031], [7.94494, 47.54511], [7.95682, 47.55789], [7.97581, 47.55493], [8.00113, 47.55616], [8.02136, 47.55096], [8.04383, 47.55443], [8.06663, 47.56374], [8.08557, 47.55768], [8.10002, 47.56504], [8.10395, 47.57918], [8.11543, 47.5841], [8.13662, 47.58432], [8.13823, 47.59147], [8.14947, 47.59558], [8.1652, 47.5945], [8.19378, 47.61636], [8.20617, 47.62141], [8.22011, 47.6181], [8.22577, 47.60385], [8.23809, 47.61204], [8.25863, 47.61571], [8.26313, 47.6103], [8.2824, 47.61225], [8.29722, 47.60603], [8.29524, 47.5919], [8.30277, 47.58607], [8.32735, 47.57133], [8.35512, 47.57014], [8.38273, 47.56608], [8.39477, 47.57826], [8.43235, 47.56617], [8.49431, 47.58107], [8.48949, 47.588], [8.46637, 47.58389], [8.45578, 47.60121], [8.50747, 47.61897], [8.51686, 47.63476], [8.55756, 47.62394], [8.57586, 47.59537], [8.60348, 47.61204], [8.59545, 47.64298], [8.60701, 47.65271], [8.61471, 47.64514], [8.60412, 47.63735], [8.62049, 47.63757], [8.62884, 47.65098], [8.61113, 47.66332], [8.6052, 47.67258], [8.57683, 47.66158], [8.56141, 47.67088], [8.52801, 47.66059], [8.5322, 47.64687], [8.49656, 47.64709], [8.46605, 47.64103], [8.4667, 47.65747], [8.44711, 47.65379], [8.42264, 47.66667], [8.41346, 47.66676], [8.40473, 47.67499], [8.4211, 47.68407], [8.40569, 47.69855], [8.44807, 47.72426], [8.45771, 47.7493], [8.48868, 47.77215], [8.56814, 47.78001], [8.56415, 47.80633], [8.61657, 47.79998], [8.62408, 47.7626], [8.64425, 47.76398], [8.65292, 47.80066], [8.68022, 47.78599], [8.68985, 47.75686], [8.71778, 47.76571], [8.74251, 47.75168], [8.70543, 47.73121], [8.73671, 47.7169], [8.72617, 47.69651]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "DG",
+           iso1A3: "DGA",
+           wikidata: "Q184851",
+           nameEn: "Diego Garcia",
+           country: "GB",
+           groups: ["IO", "BOTS", "014", "202", "002", "UN"],
+           isoStatus: "excRes",
+           callingCodes: ["246"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[73.14823, -7.76302], [73.09982, -6.07324], [71.43792, -7.73904], [73.14823, -7.76302]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "DJ",
+           iso1A3: "DJI",
+           iso1N3: "262",
+           wikidata: "Q977",
+           nameEn: "Djibouti",
+           groups: ["014", "202", "002", "UN"],
+           callingCodes: ["253"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[43.90659, 12.3823], [43.90659, 12.3823], [43.32909, 12.59711], [43.29075, 12.79154], [42.86195, 12.58747], [42.7996, 12.42629], [42.6957, 12.36201], [42.46941, 12.52661], [42.4037, 12.46478], [41.95461, 11.81157], [41.82878, 11.72361], [41.77727, 11.49902], [41.8096, 11.33606], [41.80056, 10.97127], [42.06302, 10.92599], [42.13691, 10.97586], [42.42669, 10.98493], [42.62989, 11.09711], [42.75111, 11.06992], [42.79037, 10.98493], [42.95776, 10.98533], [43.90659, 12.3823]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "DK",
+           iso1A3: "DNK",
+           iso1N3: "208",
+           wikidata: "Q756617",
+           nameEn: "Kingdom of Denmark"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "DM",
+           iso1A3: "DMA",
+           iso1N3: "212",
+           wikidata: "Q784",
+           nameEn: "Dominica",
+           groups: ["029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           callingCodes: ["1 767"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-61.32485, 14.91445], [-60.86656, 15.82603], [-61.95646, 15.5094], [-61.32485, 14.91445]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "DO",
+           iso1A3: "DOM",
+           iso1N3: "214",
+           wikidata: "Q786",
+           nameEn: "Dominican Republic",
+           groups: ["029", "003", "419", "019", "UN"],
+           callingCodes: ["1 809", "1 829", "1 849"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-67.87844, 21.7938], [-72.38946, 20.27111], [-71.77419, 19.73128], [-71.75865, 19.70231], [-71.7429, 19.58445], [-71.71449, 19.55364], [-71.71268, 19.53374], [-71.6802, 19.45008], [-71.69448, 19.37866], [-71.77766, 19.33823], [-71.73229, 19.26686], [-71.62642, 19.21212], [-71.65337, 19.11759], [-71.69938, 19.10916], [-71.71088, 19.08353], [-71.74088, 19.0437], [-71.88102, 18.95007], [-71.77766, 18.95007], [-71.72624, 18.87802], [-71.71885, 18.78423], [-71.82556, 18.62551], [-71.95412, 18.64939], [-72.00201, 18.62312], [-71.88102, 18.50125], [-71.90875, 18.45821], [-71.69952, 18.34101], [-71.78271, 18.18302], [-71.75465, 18.14405], [-71.74994, 18.11115], [-71.73783, 18.07177], [-71.75671, 18.03456], [-72.29523, 17.48026], [-68.39466, 16.14167], [-67.87844, 21.7938]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "DZ",
+           iso1A3: "DZA",
+           iso1N3: "012",
+           wikidata: "Q262",
+           nameEn: "Algeria",
+           groups: ["015", "002", "UN"],
+           callingCodes: ["213"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[8.59123, 37.14286], [5.10072, 39.89531], [-2.27707, 35.35051], [-2.21248, 35.08532], [-2.21445, 35.04378], [-2.04734, 34.93218], [-1.97833, 34.93218], [-1.97469, 34.886], [-1.73707, 34.74226], [-1.84569, 34.61907], [-1.69788, 34.48056], [-1.78042, 34.39018], [-1.64666, 34.10405], [-1.73494, 33.71721], [-1.59508, 33.59929], [-1.67067, 33.27084], [-1.46249, 33.0499], [-1.54244, 32.95499], [-1.37794, 32.73628], [-0.9912, 32.52467], [-1.24998, 32.32993], [-1.24453, 32.1917], [-1.15735, 32.12096], [-1.22829, 32.07832], [-2.46166, 32.16603], [-2.93873, 32.06557], [-2.82784, 31.79459], [-3.66314, 31.6339], [-3.66386, 31.39202], [-3.77647, 31.31912], [-3.77103, 31.14984], [-3.54944, 31.0503], [-3.65418, 30.85566], [-3.64735, 30.67539], [-4.31774, 30.53229], [-4.6058, 30.28343], [-5.21671, 29.95253], [-5.58831, 29.48103], [-5.72121, 29.52322], [-5.75616, 29.61407], [-6.69965, 29.51623], [-6.78351, 29.44634], [-6.95824, 29.50924], [-7.61585, 29.36252], [-8.6715, 28.71194], [-8.66879, 27.6666], [-8.66674, 27.31569], [-4.83423, 24.99935], [1.15698, 21.12843], [1.20992, 20.73533], [3.24648, 19.81703], [3.12501, 19.1366], [3.36082, 18.9745], [4.26651, 19.14224], [5.8153, 19.45101], [7.38361, 20.79165], [7.48273, 20.87258], [11.96886, 23.51735], [11.62498, 24.26669], [11.41061, 24.21456], [10.85323, 24.5595], [10.33159, 24.5465], [10.02432, 24.98124], [10.03146, 25.35635], [9.38834, 26.19288], [9.51696, 26.39148], [9.89569, 26.57696], [9.78136, 29.40961], [9.3876, 30.16738], [9.55544, 30.23971], [9.07483, 32.07865], [8.35999, 32.50101], [8.31895, 32.83483], [8.1179, 33.05086], [8.11433, 33.10175], [7.83028, 33.18851], [7.73687, 33.42114], [7.54088, 33.7726], [7.52851, 34.06493], [7.66174, 34.20167], [7.74207, 34.16492], [7.81242, 34.21841], [7.86264, 34.3987], [8.20482, 34.57575], [8.29655, 34.72798], [8.25189, 34.92009], [8.30727, 34.95378], [8.3555, 35.10007], [8.47318, 35.23376], [8.30329, 35.29884], [8.36086, 35.47774], [8.35371, 35.66373], [8.26472, 35.73669], [8.2626, 35.91733], [8.40731, 36.42208], [8.18936, 36.44939], [8.16167, 36.48817], [8.47609, 36.66607], [8.46537, 36.7706], [8.57613, 36.78062], [8.67706, 36.8364], [8.62972, 36.86499], [8.64044, 36.9401], [8.59123, 37.14286]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "EA",
+           wikidata: "Q28868874",
+           nameEn: "Ceuta, Melilla",
+           country: "ES",
+           level: "territory",
+           isoStatus: "excRes"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "EC",
+           iso1A3: "ECU",
+           iso1N3: "218",
+           wikidata: "Q736",
+           nameEn: "Ecuador"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "EE",
+           iso1A3: "EST",
+           iso1N3: "233",
+           wikidata: "Q191",
+           nameEn: "Estonia",
+           aliases: ["EW"],
+           groups: ["EU", "154", "150", "UN"],
+           callingCodes: ["372"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[26.32936, 60.00121], [20.5104, 59.15546], [19.84909, 57.57876], [22.80496, 57.87798], [23.20055, 57.56697], [24.26221, 57.91787], [24.3579, 57.87471], [25.19484, 58.0831], [25.28237, 57.98539], [25.29581, 58.08288], [25.73499, 57.90193], [26.05949, 57.84744], [26.0324, 57.79037], [26.02456, 57.78342], [26.027, 57.78158], [26.0266, 57.77441], [26.02069, 57.77169], [26.02415, 57.76865], [26.03332, 57.7718], [26.0543, 57.76105], [26.08098, 57.76619], [26.2029, 57.7206], [26.1866, 57.6849], [26.29253, 57.59244], [26.46527, 57.56885], [26.54675, 57.51813], [26.90364, 57.62823], [27.34698, 57.52242], [27.31919, 57.57672], [27.40393, 57.62125], [27.3746, 57.66834], [27.52615, 57.72843], [27.50171, 57.78842], [27.56689, 57.83356], [27.78526, 57.83963], [27.81841, 57.89244], [27.67282, 57.92627], [27.62393, 58.09462], [27.48541, 58.22615], [27.55489, 58.39525], [27.36366, 58.78381], [27.74429, 58.98351], [27.80482, 59.1116], [27.87978, 59.18097], [27.90911, 59.24353], [28.00689, 59.28351], [28.14215, 59.28934], [28.19284, 59.35791], [28.20537, 59.36491], [28.21137, 59.38058], [28.19061, 59.39962], [28.04187, 59.47017], [27.85643, 59.58538], [26.90044, 59.63819], [26.32936, 60.00121]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "EG",
+           iso1A3: "EGY",
+           iso1N3: "818",
+           wikidata: "Q79",
+           nameEn: "Egypt",
+           groups: ["015", "002", "UN"],
+           callingCodes: ["20"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[33.62659, 31.82938], [26.92891, 33.39516], [24.8458, 31.39877], [25.01077, 30.73861], [24.71117, 30.17441], [24.99968, 29.24574], [24.99885, 21.99535], [33.17563, 22.00405], [34.0765, 22.00501], [37.8565, 22.00903], [34.4454, 27.91479], [34.8812, 29.36878], [34.92298, 29.45305], [34.26742, 31.21998], [34.24012, 31.29591], [34.23572, 31.2966], [34.21853, 31.32363], [34.052, 31.46619], [33.62659, 31.82938]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "EH",
+           iso1A3: "ESH",
+           iso1N3: "732",
+           wikidata: "Q6250",
+           nameEn: "Western Sahara",
+           groups: ["015", "002"],
+           callingCodes: ["212"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-8.66879, 27.6666], [-8.77527, 27.66663], [-8.71787, 26.9898], [-9.08698, 26.98639], [-9.56957, 26.90042], [-9.81998, 26.71379], [-10.68417, 26.90984], [-11.35695, 26.8505], [-11.23622, 26.72023], [-11.38635, 26.611], [-11.62052, 26.05229], [-12.06001, 26.04442], [-12.12281, 25.13682], [-12.92147, 24.39502], [-13.00628, 24.01923], [-13.75627, 23.77231], [-14.10361, 22.75501], [-14.1291, 22.41636], [-14.48112, 22.00886], [-14.47329, 21.63839], [-14.78487, 21.36587], [-16.44269, 21.39745], [-16.9978, 21.36239], [-17.02707, 21.34022], [-17.21511, 21.34226], [-17.35589, 20.80492], [-17.0471, 20.76408], [-17.0695, 20.85742], [-17.06781, 20.92697], [-17.0396, 20.9961], [-17.0357, 21.05368], [-16.99806, 21.12142], [-16.95474, 21.33997], [-13.01525, 21.33343], [-13.08438, 22.53866], [-13.15313, 22.75649], [-13.10753, 22.89493], [-13.00412, 23.02297], [-12.5741, 23.28975], [-12.36213, 23.3187], [-12.14969, 23.41935], [-12.00251, 23.4538], [-12.0002, 25.9986], [-8.66721, 25.99918], [-8.66674, 27.31569], [-8.66879, 27.6666]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "ER",
+           iso1A3: "ERI",
+           iso1N3: "232",
+           wikidata: "Q986",
+           nameEn: "Eritrea",
+           groups: ["014", "202", "002", "UN"],
+           callingCodes: ["291"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[40.99158, 15.81743], [39.63762, 18.37348], [38.57727, 17.98125], [38.45916, 17.87167], [38.37133, 17.66269], [38.13362, 17.53906], [37.50967, 17.32199], [37.42694, 17.04041], [36.99777, 17.07172], [36.92193, 16.23451], [36.76371, 15.80831], [36.69761, 15.75323], [36.54276, 15.23478], [36.44337, 15.14963], [36.54376, 14.25597], [36.56536, 14.26177], [36.55659, 14.28237], [36.63364, 14.31172], [36.85787, 14.32201], [37.01622, 14.2561], [37.09486, 14.27155], [37.13206, 14.40746], [37.3106, 14.44657], [37.47319, 14.2149], [37.528, 14.18413], [37.91287, 14.89447], [38.0364, 14.72745], [38.25562, 14.67287], [38.3533, 14.51323], [38.45748, 14.41445], [38.78306, 14.4754], [38.98058, 14.54895], [39.02834, 14.63717], [39.16074, 14.65187], [39.14772, 14.61827], [39.19547, 14.56996], [39.23888, 14.56365], [39.26927, 14.48801], [39.2302, 14.44598], [39.2519, 14.40393], [39.37685, 14.54402], [39.52756, 14.49011], [39.50585, 14.55735], [39.58182, 14.60987], [39.76632, 14.54264], [39.9443, 14.41024], [40.07236, 14.54264], [40.14649, 14.53969], [40.21128, 14.39342], [40.25686, 14.41445], [40.9167, 14.11152], [41.25097, 13.60787], [41.62864, 13.38626], [42.05841, 12.80912], [42.21469, 12.75832], [42.2798, 12.6355], [42.4037, 12.46478], [42.46941, 12.52661], [42.6957, 12.36201], [42.7996, 12.42629], [42.86195, 12.58747], [43.29075, 12.79154], [40.99158, 15.81743]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "ES",
+           iso1A3: "ESP",
+           iso1N3: "724",
+           wikidata: "Q29",
+           nameEn: "Spain"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "ET",
+           iso1A3: "ETH",
+           iso1N3: "231",
+           wikidata: "Q115",
+           nameEn: "Ethiopia",
+           groups: ["014", "202", "002", "UN"],
+           callingCodes: ["251"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[42.4037, 12.46478], [42.2798, 12.6355], [42.21469, 12.75832], [42.05841, 12.80912], [41.62864, 13.38626], [41.25097, 13.60787], [40.9167, 14.11152], [40.25686, 14.41445], [40.21128, 14.39342], [40.14649, 14.53969], [40.07236, 14.54264], [39.9443, 14.41024], [39.76632, 14.54264], [39.58182, 14.60987], [39.50585, 14.55735], [39.52756, 14.49011], [39.37685, 14.54402], [39.2519, 14.40393], [39.2302, 14.44598], [39.26927, 14.48801], [39.23888, 14.56365], [39.19547, 14.56996], [39.14772, 14.61827], [39.16074, 14.65187], [39.02834, 14.63717], [38.98058, 14.54895], [38.78306, 14.4754], [38.45748, 14.41445], [38.3533, 14.51323], [38.25562, 14.67287], [38.0364, 14.72745], [37.91287, 14.89447], [37.528, 14.18413], [37.47319, 14.2149], [37.3106, 14.44657], [37.13206, 14.40746], [37.09486, 14.27155], [37.01622, 14.2561], [36.85787, 14.32201], [36.63364, 14.31172], [36.55659, 14.28237], [36.56536, 14.26177], [36.54376, 14.25597], [36.44653, 13.95666], [36.48824, 13.83954], [36.38993, 13.56459], [36.24545, 13.36759], [36.13374, 12.92665], [36.16651, 12.88019], [36.14268, 12.70879], [36.01458, 12.72478], [35.70476, 12.67101], [35.24302, 11.91132], [35.11492, 11.85156], [35.05832, 11.71158], [35.09556, 11.56278], [34.95704, 11.24448], [35.01215, 11.19626], [34.93172, 10.95946], [34.97789, 10.91559], [34.97491, 10.86147], [34.86916, 10.78832], [34.86618, 10.74588], [34.77532, 10.69027], [34.77383, 10.74588], [34.59062, 10.89072], [34.4372, 10.781], [34.2823, 10.53508], [34.34783, 10.23914], [34.32102, 10.11599], [34.22718, 10.02506], [34.20484, 9.9033], [34.13186, 9.7492], [34.08717, 9.55243], [34.10229, 9.50238], [34.14304, 9.04654], [34.14453, 8.60204], [34.01346, 8.50041], [33.89579, 8.4842], [33.87195, 8.41938], [33.71407, 8.3678], [33.66938, 8.44442], [33.54575, 8.47094], [33.3119, 8.45474], [33.19721, 8.40317], [33.1853, 8.29264], [33.18083, 8.13047], [33.08401, 8.05822], [33.0006, 7.90333], [33.04944, 7.78989], [33.24637, 7.77939], [33.32531, 7.71297], [33.44745, 7.7543], [33.71407, 7.65983], [33.87642, 7.5491], [34.02984, 7.36449], [34.03878, 7.27437], [34.01495, 7.25664], [34.19369, 7.12807], [34.19369, 7.04382], [34.35753, 6.91963], [34.47669, 6.91076], [34.53925, 6.82794], [34.53776, 6.74808], [34.65096, 6.72589], [34.77459, 6.5957], [34.87736, 6.60161], [35.01738, 6.46991], [34.96227, 6.26415], [35.00546, 5.89387], [35.12611, 5.68937], [35.13058, 5.62118], [35.31188, 5.50106], [35.29938, 5.34042], [35.50792, 5.42431], [35.8576, 5.33413], [35.81968, 5.10757], [35.82118, 4.77382], [35.9419, 4.61933], [35.95449, 4.53244], [36.03924, 4.44406], [36.84474, 4.44518], [37.07724, 4.33503], [38.14168, 3.62487], [38.45812, 3.60445], [38.52336, 3.62551], [38.91938, 3.51198], [39.07736, 3.5267], [39.19954, 3.47834], [39.49444, 3.45521], [39.51551, 3.40895], [39.55132, 3.39634], [39.58339, 3.47434], [39.76808, 3.67058], [39.86043, 3.86974], [40.77498, 4.27683], [41.1754, 3.94079], [41.89488, 3.97375], [42.07619, 4.17667], [42.55853, 4.20518], [42.84526, 4.28357], [42.97746, 4.44032], [43.04177, 4.57923], [43.40263, 4.79289], [44.02436, 4.9451], [44.98104, 4.91821], [47.97917, 8.00124], [47.92477, 8.00111], [46.99339, 7.9989], [44.19222, 8.93028], [43.32613, 9.59205], [43.23518, 9.84605], [43.0937, 9.90579], [42.87643, 10.18441], [42.69452, 10.62672], [42.95776, 10.98533], [42.79037, 10.98493], [42.75111, 11.06992], [42.62989, 11.09711], [42.42669, 10.98493], [42.13691, 10.97586], [42.06302, 10.92599], [41.80056, 10.97127], [41.8096, 11.33606], [41.77727, 11.49902], [41.82878, 11.72361], [41.95461, 11.81157], [42.4037, 12.46478]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "EU",
+           iso1A3: "EUE",
+           wikidata: "Q458",
+           nameEn: "European Union",
+           level: "union",
+           isoStatus: "excRes"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "FI",
+           iso1A3: "FIN",
+           iso1N3: "246",
+           wikidata: "Q33",
+           nameEn: "Finland",
+           aliases: ["SF"]
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "FJ",
+           iso1A3: "FJI",
+           iso1N3: "242",
+           wikidata: "Q712",
+           nameEn: "Fiji",
+           groups: ["054", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["679"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[174.245, -23.1974], [179.99999, -22.5], [179.99999, -11.5], [174, -11.5], [174.245, -23.1974]]], [[[-176.76826, -14.95183], [-180, -14.96041], [-180, -22.90585], [-176.74538, -22.89767], [-176.76826, -14.95183]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "FK",
+           iso1A3: "FLK",
+           iso1N3: "238",
+           wikidata: "Q9648",
+           nameEn: "Falkland Islands",
+           country: "GB",
+           groups: ["BOTS", "005", "419", "019", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["500"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-63.67376, -55.11859], [-54.56126, -51.26248], [-61.26735, -50.63919], [-63.67376, -55.11859]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "FM",
+           iso1A3: "FSM",
+           iso1N3: "583",
+           wikidata: "Q702",
+           nameEn: "Federated States of Micronesia",
+           groups: ["057", "009", "UN"],
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["691"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[138.20583, 13.3783], [136.27107, 6.73747], [156.88247, -1.39237], [165.19726, 6.22546], [138.20583, 13.3783]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "FO",
+           iso1A3: "FRO",
+           iso1N3: "234",
+           wikidata: "Q4628",
+           nameEn: "Faroe Islands",
+           country: "DK",
+           groups: ["154", "150", "UN"],
+           callingCodes: ["298"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-8.51774, 62.35338], [-6.51083, 60.95272], [-5.70102, 62.77194], [-8.51774, 62.35338]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "FR",
+           iso1A3: "FRA",
+           iso1N3: "250",
+           wikidata: "Q142",
+           nameEn: "France"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "FX",
+           iso1A3: "FXX",
+           iso1N3: "249",
+           wikidata: "Q212429",
+           nameEn: "Metropolitan France",
+           country: "FR",
+           groups: ["EU", "155", "150", "UN"],
+           isoStatus: "excRes",
+           callingCodes: ["33"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[2.55904, 51.07014], [2.18458, 51.52087], [1.17405, 50.74239], [-2.02963, 49.91866], [-2.09454, 49.46288], [-1.83944, 49.23037], [-2.00491, 48.86706], [-2.65349, 49.15373], [-6.28985, 48.93406], [-1.81005, 43.59738], [-1.77289, 43.38957], [-1.79319, 43.37497], [-1.78332, 43.36399], [-1.78714, 43.35476], [-1.77068, 43.34396], [-1.75334, 43.34107], [-1.75079, 43.3317], [-1.7397, 43.32979], [-1.73074, 43.29481], [-1.69407, 43.31378], [-1.62481, 43.30726], [-1.63052, 43.28591], [-1.61341, 43.25269], [-1.57674, 43.25269], [-1.55963, 43.28828], [-1.50992, 43.29481], [-1.45289, 43.27049], [-1.40942, 43.27272], [-1.3758, 43.24511], [-1.41562, 43.12815], [-1.47555, 43.08372], [-1.44067, 43.047], [-1.35272, 43.02658], [-1.34419, 43.09665], [-1.32209, 43.1127], [-1.27118, 43.11961], [-1.30052, 43.09581], [-1.30531, 43.06859], [-1.25244, 43.04164], [-1.22881, 43.05534], [-1.10333, 43.0059], [-1.00963, 42.99279], [-0.97133, 42.96239], [-0.81652, 42.95166], [-0.75478, 42.96916], [-0.72037, 42.92541], [-0.73422, 42.91228], [-0.72608, 42.89318], [-0.69837, 42.87945], [-0.67637, 42.88303], [-0.55497, 42.77846], [-0.50863, 42.82713], [-0.44334, 42.79939], [-0.41319, 42.80776], [-0.38833, 42.80132], [-0.3122, 42.84788], [-0.17939, 42.78974], [-0.16141, 42.79535], [-0.10519, 42.72761], [-0.02468, 42.68513], [0.17569, 42.73424], [0.25336, 42.7174], [0.29407, 42.67431], [0.36251, 42.72282], [0.40214, 42.69779], [0.67873, 42.69458], [0.65421, 42.75872], [0.66121, 42.84021], [0.711, 42.86372], [0.93089, 42.79154], [0.96166, 42.80629], [0.98292, 42.78754], [1.0804, 42.78569], [1.15928, 42.71407], [1.35562, 42.71944], [1.44197, 42.60217], [1.47986, 42.61346], [1.46718, 42.63296], [1.48043, 42.65203], [1.50867, 42.64483], [1.55418, 42.65669], [1.60085, 42.62703], [1.63485, 42.62957], [1.6625, 42.61982], [1.68267, 42.62533], [1.73452, 42.61515], [1.72588, 42.59098], [1.7858, 42.57698], [1.73683, 42.55492], [1.72515, 42.50338], [1.76335, 42.48863], [1.83037, 42.48395], [1.88853, 42.4501], [1.93663, 42.45439], [1.94292, 42.44316], [1.94061, 42.43333], [1.94084, 42.43039], [1.9574, 42.42401], [1.96482, 42.37787], [2.00488, 42.35399], [2.06241, 42.35906], [2.11621, 42.38393], [2.12789, 42.41291], [2.16599, 42.42314], [2.20578, 42.41633], [2.25551, 42.43757], [2.38504, 42.39977], [2.43299, 42.39423], [2.43508, 42.37568], [2.48457, 42.33933], [2.54382, 42.33406], [2.55516, 42.35351], [2.57934, 42.35808], [2.6747, 42.33974], [2.65311, 42.38771], [2.72056, 42.42298], [2.75497, 42.42578], [2.77464, 42.41046], [2.84335, 42.45724], [2.85675, 42.45444], [2.86983, 42.46843], [2.88413, 42.45938], [2.92107, 42.4573], [2.94283, 42.48174], [2.96518, 42.46692], [3.03734, 42.47363], [3.08167, 42.42748], [3.10027, 42.42621], [3.11379, 42.43646], [3.17156, 42.43545], [3.75438, 42.33445], [7.60802, 41.05927], [10.09675, 41.44089], [9.56115, 43.20816], [7.50102, 43.51859], [7.42422, 43.72209], [7.40903, 43.7296], [7.41113, 43.73156], [7.41291, 43.73168], [7.41298, 43.73311], [7.41233, 43.73439], [7.42062, 43.73977], [7.42299, 43.74176], [7.42443, 43.74087], [7.42809, 43.74396], [7.43013, 43.74895], [7.43624, 43.75014], [7.43708, 43.75197], [7.4389, 43.75151], [7.4379, 43.74963], [7.47823, 43.73341], [7.53006, 43.78405], [7.50423, 43.84345], [7.49355, 43.86551], [7.51162, 43.88301], [7.56075, 43.89932], [7.56858, 43.94506], [7.60771, 43.95772], [7.65266, 43.9763], [7.66848, 43.99943], [7.6597, 44.03009], [7.72508, 44.07578], [7.66878, 44.12795], [7.68694, 44.17487], [7.63245, 44.17877], [7.62155, 44.14881], [7.36364, 44.11882], [7.34547, 44.14359], [7.27827, 44.1462], [7.16929, 44.20352], [7.00764, 44.23736], [6.98221, 44.28289], [6.89171, 44.36637], [6.88784, 44.42043], [6.94504, 44.43112], [6.86233, 44.49834], [6.85507, 44.53072], [6.96042, 44.62129], [6.95133, 44.66264], [7.00582, 44.69364], [7.07484, 44.68073], [7.00401, 44.78782], [7.02217, 44.82519], [6.93499, 44.8664], [6.90774, 44.84322], [6.75518, 44.89915], [6.74519, 44.93661], [6.74791, 45.01939], [6.66981, 45.02324], [6.62803, 45.11175], [6.7697, 45.16044], [6.85144, 45.13226], [6.96706, 45.20841], [7.07074, 45.21228], [7.13115, 45.25386], [7.10572, 45.32924], [7.18019, 45.40071], [7.00037, 45.509], [6.98948, 45.63869], [6.80785, 45.71864], [6.80785, 45.83265], [6.95315, 45.85163], [7.04151, 45.92435], [7.00946, 45.9944], [6.93862, 46.06502], [6.87868, 46.03855], [6.89321, 46.12548], [6.78968, 46.14058], [6.86052, 46.28512], [6.77152, 46.34784], [6.8024, 46.39171], [6.82312, 46.42661], [6.53358, 46.45431], [6.25432, 46.3632], [6.21981, 46.31304], [6.24826, 46.30175], [6.25137, 46.29014], [6.23775, 46.27822], [6.24952, 46.26255], [6.26749, 46.24745], [6.29474, 46.26221], [6.31041, 46.24417], [6.29663, 46.22688], [6.27694, 46.21566], [6.26007, 46.21165], [6.24821, 46.20531], [6.23913, 46.20511], [6.23544, 46.20714], [6.22175, 46.20045], [6.22222, 46.19888], [6.21844, 46.19837], [6.21603, 46.19507], [6.21273, 46.19409], [6.21114, 46.1927], [6.20539, 46.19163], [6.19807, 46.18369], [6.19552, 46.18401], [6.18707, 46.17999], [6.18871, 46.16644], [6.18116, 46.16187], [6.15305, 46.15194], [6.13397, 46.1406], [6.09926, 46.14373], [6.09199, 46.15191], [6.07491, 46.14879], [6.05203, 46.15191], [6.04564, 46.14031], [6.03614, 46.13712], [6.01791, 46.14228], [5.9871, 46.14499], [5.97893, 46.13303], [5.95781, 46.12925], [5.9641, 46.14412], [5.97508, 46.15863], [5.98188, 46.17392], [5.98846, 46.17046], [5.99573, 46.18587], [5.96515, 46.19638], [5.97542, 46.21525], [6.02461, 46.23313], [6.03342, 46.2383], [6.04602, 46.23127], [6.05029, 46.23518], [6.0633, 46.24583], [6.07072, 46.24085], [6.08563, 46.24651], [6.10071, 46.23772], [6.12446, 46.25059], [6.11926, 46.2634], [6.1013, 46.28512], [6.11697, 46.29547], [6.1198, 46.31157], [6.13876, 46.33844], [6.15738, 46.3491], [6.16987, 46.36759], [6.15985, 46.37721], [6.15016, 46.3778], [6.09926, 46.40768], [6.06407, 46.41676], [6.08427, 46.44305], [6.07269, 46.46244], [6.1567, 46.54402], [6.11084, 46.57649], [6.27135, 46.68251], [6.38351, 46.73171], [6.45209, 46.77502], [6.43216, 46.80336], [6.46456, 46.88865], [6.43341, 46.92703], [6.71531, 47.0494], [6.68823, 47.06616], [6.76788, 47.1208], [6.8489, 47.15933], [6.9508, 47.24338], [6.95108, 47.26428], [6.94316, 47.28747], [7.05305, 47.33304], [7.0564, 47.35134], [7.03125, 47.36996], [6.87959, 47.35335], [6.88542, 47.37262], [6.93744, 47.40714], [6.93953, 47.43388], [7.0024, 47.45264], [6.98425, 47.49432], [7.0231, 47.50522], [7.07425, 47.48863], [7.12781, 47.50371], [7.16249, 47.49025], [7.19583, 47.49455], [7.17026, 47.44312], [7.24669, 47.4205], [7.33526, 47.44186], [7.35603, 47.43432], [7.40308, 47.43638], [7.43088, 47.45846], [7.4462, 47.46264], [7.4583, 47.47216], [7.42923, 47.48628], [7.43356, 47.49712], [7.47534, 47.47932], [7.51076, 47.49651], [7.49804, 47.51798], [7.5229, 47.51644], [7.53199, 47.5284], [7.51904, 47.53515], [7.50588, 47.52856], [7.49691, 47.53821], [7.50873, 47.54546], [7.51723, 47.54578], [7.52831, 47.55347], [7.53634, 47.55553], [7.55652, 47.56779], [7.55689, 47.57232], [7.56548, 47.57617], [7.56684, 47.57785], [7.58386, 47.57536], [7.58945, 47.59017], [7.59301, 47.60058], [7.58851, 47.60794], [7.57423, 47.61628], [7.5591, 47.63849], [7.53384, 47.65115], [7.52067, 47.66437], [7.51915, 47.68335], [7.51266, 47.70197], [7.53722, 47.71635], [7.54761, 47.72912], [7.52921, 47.77747], [7.55673, 47.87371], [7.62302, 47.97898], [7.56966, 48.03265], [7.57137, 48.12292], [7.6648, 48.22219], [7.69022, 48.30018], [7.74562, 48.32736], [7.73109, 48.39192], [7.76833, 48.48945], [7.80647, 48.51239], [7.80167, 48.54758], [7.80057, 48.5857], [7.84098, 48.64217], [7.89002, 48.66317], [7.96812, 48.72491], [7.96994, 48.75606], [8.01534, 48.76085], [8.0326, 48.79017], [8.06802, 48.78957], [8.10253, 48.81829], [8.12813, 48.87985], [8.19989, 48.95825], [8.20031, 48.95856], [8.22604, 48.97352], [8.14189, 48.97833], [7.97783, 49.03161], [7.93641, 49.05544], [7.86386, 49.03499], [7.79557, 49.06583], [7.75948, 49.04562], [7.63618, 49.05428], [7.62575, 49.07654], [7.56416, 49.08136], [7.53012, 49.09818], [7.49172, 49.13915], [7.49473, 49.17], [7.44455, 49.16765], [7.44052, 49.18354], [7.3662, 49.17308], [7.35995, 49.14399], [7.3195, 49.14231], [7.29514, 49.11426], [7.23473, 49.12971], [7.1593, 49.1204], [7.1358, 49.1282], [7.12504, 49.14253], [7.10384, 49.13787], [7.10715, 49.15631], [7.07859, 49.15031], [7.09007, 49.13094], [7.07162, 49.1255], [7.06642, 49.11415], [7.05548, 49.11185], [7.04843, 49.11422], [7.04409, 49.12123], [7.04662, 49.13724], [7.03178, 49.15734], [7.0274, 49.17042], [7.03459, 49.19096], [7.01318, 49.19018], [6.97273, 49.2099], [6.95963, 49.203], [6.94028, 49.21641], [6.93831, 49.2223], [6.91875, 49.22261], [6.89298, 49.20863], [6.85939, 49.22376], [6.83555, 49.21249], [6.85119, 49.20038], [6.85016, 49.19354], [6.86225, 49.18185], [6.84703, 49.15734], [6.83385, 49.15162], [6.78265, 49.16793], [6.73765, 49.16375], [6.71137, 49.18808], [6.73256, 49.20486], [6.71843, 49.2208], [6.69274, 49.21661], [6.66583, 49.28065], [6.60186, 49.31055], [6.572, 49.35027], [6.58807, 49.35358], [6.60091, 49.36864], [6.533, 49.40748], [6.55404, 49.42464], [6.42432, 49.47683], [6.40274, 49.46546], [6.39168, 49.4667], [6.38352, 49.46463], [6.36778, 49.46937], [6.3687, 49.4593], [6.28818, 49.48465], [6.27875, 49.503], [6.25029, 49.50609], [6.2409, 49.51408], [6.19543, 49.50536], [6.17386, 49.50934], [6.15366, 49.50226], [6.16115, 49.49297], [6.14321, 49.48796], [6.12814, 49.49365], [6.12346, 49.4735], [6.10325, 49.4707], [6.09845, 49.46351], [6.10072, 49.45268], [6.08373, 49.45594], [6.07887, 49.46399], [6.05553, 49.46663], [6.04176, 49.44801], [6.02743, 49.44845], [6.02648, 49.45451], [5.97693, 49.45513], [5.96876, 49.49053], [5.94224, 49.49608], [5.94128, 49.50034], [5.86571, 49.50015], [5.83389, 49.52152], [5.83467, 49.52717], [5.84466, 49.53027], [5.83648, 49.5425], [5.81664, 49.53775], [5.80871, 49.5425], [5.81838, 49.54777], [5.79195, 49.55228], [5.77435, 49.56298], [5.7577, 49.55915], [5.75649, 49.54321], [5.64505, 49.55146], [5.60909, 49.51228], [5.55001, 49.52729], [5.46541, 49.49825], [5.46734, 49.52648], [5.43713, 49.5707], [5.3974, 49.61596], [5.34837, 49.62889], [5.33851, 49.61599], [5.3137, 49.61225], [5.30214, 49.63055], [5.33039, 49.6555], [5.31465, 49.66846], [5.26232, 49.69456], [5.14545, 49.70287], [5.09249, 49.76193], [4.96714, 49.79872], [4.85464, 49.78995], [4.86965, 49.82271], [4.85134, 49.86457], [4.88529, 49.9236], [4.78827, 49.95609], [4.8382, 50.06738], [4.88602, 50.15182], [4.83279, 50.15331], [4.82438, 50.16878], [4.75237, 50.11314], [4.70064, 50.09384], [4.68695, 49.99685], [4.5414, 49.96911], [4.51098, 49.94659], [4.43488, 49.94122], [4.35051, 49.95315], [4.31963, 49.97043], [4.20532, 49.95803], [4.14239, 49.98034], [4.13508, 50.01976], [4.16294, 50.04719], [4.23101, 50.06945], [4.20147, 50.13535], [4.13561, 50.13078], [4.16014, 50.19239], [4.15524, 50.21103], [4.21945, 50.25539], [4.20651, 50.27333], [4.17861, 50.27443], [4.17347, 50.28838], [4.15524, 50.2833], [4.16808, 50.25786], [4.13665, 50.25609], [4.11954, 50.30425], [4.10957, 50.30234], [4.10237, 50.31247], [4.0689, 50.3254], [4.0268, 50.35793], [3.96771, 50.34989], [3.90781, 50.32814], [3.84314, 50.35219], [3.73911, 50.34809], [3.70987, 50.3191], [3.71009, 50.30305], [3.66976, 50.34563], [3.65709, 50.36873], [3.67262, 50.38663], [3.67494, 50.40239], [3.66153, 50.45165], [3.64426, 50.46275], [3.61014, 50.49568], [3.58361, 50.49049], [3.5683, 50.50192], [3.49509, 50.48885], [3.51564, 50.5256], [3.47385, 50.53397], [3.44629, 50.51009], [3.37693, 50.49538], [3.28575, 50.52724], [3.2729, 50.60718], [3.23951, 50.6585], [3.264, 50.67668], [3.2536, 50.68977], [3.26141, 50.69151], [3.26063, 50.70086], [3.24593, 50.71389], [3.22042, 50.71019], [3.20845, 50.71662], [3.19017, 50.72569], [3.20064, 50.73547], [3.18811, 50.74025], [3.18339, 50.74981], [3.16476, 50.76843], [3.15017, 50.79031], [3.1257, 50.78603], [3.11987, 50.79188], [3.11206, 50.79416], [3.10614, 50.78303], [3.09163, 50.77717], [3.04314, 50.77674], [3.00537, 50.76588], [2.96778, 50.75242], [2.95019, 50.75138], [2.90873, 50.702], [2.91036, 50.6939], [2.90069, 50.69263], [2.88504, 50.70656], [2.87937, 50.70298], [2.86985, 50.7033], [2.8483, 50.72276], [2.81056, 50.71773], [2.71165, 50.81295], [2.63331, 50.81457], [2.59093, 50.91751], [2.63074, 50.94746], [2.57551, 51.00326], [2.55904, 51.07014]], [[1.99838, 42.44682], [1.98378, 42.44697], [1.96125, 42.45364], [1.95606, 42.45785], [1.96215, 42.47854], [1.97003, 42.48081], [1.97227, 42.48487], [1.97697, 42.48568], [1.98022, 42.49569], [1.98916, 42.49351], [1.99766, 42.4858], [1.98579, 42.47486], [1.99216, 42.46208], [2.01564, 42.45171], [1.99838, 42.44682]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GA",
+           iso1A3: "GAB",
+           iso1N3: "266",
+           wikidata: "Q1000",
+           nameEn: "Gabon",
+           groups: ["017", "202", "002", "UN"],
+           callingCodes: ["241"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[13.29457, 2.16106], [13.28534, 2.25716], [11.37116, 2.29975], [11.3561, 2.17217], [11.35307, 1.00251], [9.79648, 1.0019], [9.75065, 1.06753], [9.66433, 1.06723], [7.24416, -0.64092], [10.75913, -4.39519], [11.12647, -3.94169], [11.22301, -3.69888], [11.48764, -3.51089], [11.57949, -3.52798], [11.68608, -3.68942], [11.87083, -3.71571], [11.92719, -3.62768], [11.8318, -3.5812], [11.96554, -3.30267], [11.70227, -3.17465], [11.70558, -3.0773], [11.80365, -3.00424], [11.64798, -2.81146], [11.5359, -2.85654], [11.64487, -2.61865], [11.57637, -2.33379], [11.74605, -2.39936], [11.96866, -2.33559], [12.04895, -2.41704], [12.47925, -2.32626], [12.44656, -1.92025], [12.61312, -1.8129], [12.82172, -1.91091], [13.02759, -2.33098], [13.47977, -2.43224], [13.75884, -2.09293], [13.92073, -2.35581], [13.85846, -2.46935], [14.10442, -2.49268], [14.23829, -2.33715], [14.16202, -2.23916], [14.23518, -2.15671], [14.25932, -1.97624], [14.41838, -1.89412], [14.52569, -0.57818], [14.41887, -0.44799], [14.2165, -0.38261], [14.06862, -0.20826], [13.90632, -0.2287], [13.88648, 0.26652], [14.10909, 0.58563], [14.26066, 0.57255], [14.48179, 0.9152], [14.25186, 1.39842], [13.89582, 1.4261], [13.15519, 1.23368], [13.25447, 1.32339], [13.13461, 1.57238], [13.29457, 2.16106]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GB",
+           iso1A3: "GBR",
+           iso1N3: "826",
+           wikidata: "Q145",
+           ccTLD: ".uk",
+           nameEn: "United Kingdom",
+           aliases: ["UK"]
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GD",
+           iso1A3: "GRD",
+           iso1N3: "308",
+           wikidata: "Q769",
+           nameEn: "Grenada",
+           aliases: ["WG"],
+           groups: ["029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           callingCodes: ["1 473"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-62.64026, 12.69984], [-61.77886, 11.36496], [-59.94058, 12.34011], [-62.64026, 12.69984]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GE",
+           iso1A3: "GEO",
+           iso1N3: "268",
+           wikidata: "Q230",
+           nameEn: "Georgia",
+           groups: ["145", "142", "UN"],
+           callingCodes: ["995"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[46.42738, 41.91323], [45.61676, 42.20768], [45.78692, 42.48358], [45.36501, 42.55268], [45.15318, 42.70598], [44.88754, 42.74934], [44.80941, 42.61277], [44.70002, 42.74679], [44.54202, 42.75699], [43.95517, 42.55396], [43.73119, 42.62043], [43.81453, 42.74297], [43.0419, 43.02413], [43.03322, 43.08883], [42.75889, 43.19651], [42.66667, 43.13917], [42.40563, 43.23226], [41.64935, 43.22331], [40.65957, 43.56212], [40.10657, 43.57344], [40.04445, 43.47776], [40.03312, 43.44262], [40.01007, 43.42411], [40.01552, 43.42025], [40.00853, 43.40578], [40.0078, 43.38551], [39.81147, 43.06294], [40.89217, 41.72528], [41.54366, 41.52185], [41.7148, 41.4932], [41.7124, 41.47417], [41.81939, 41.43621], [41.95134, 41.52466], [42.26387, 41.49346], [42.51772, 41.43606], [42.59202, 41.58183], [42.72794, 41.59714], [42.84471, 41.58912], [42.78995, 41.50126], [42.84899, 41.47265], [42.8785, 41.50516], [43.02956, 41.37891], [43.21707, 41.30331], [43.13373, 41.25503], [43.1945, 41.25242], [43.23096, 41.17536], [43.36118, 41.2028], [43.44973, 41.17666], [43.4717, 41.12611], [43.67712, 41.13398], [43.74717, 41.1117], [43.84835, 41.16329], [44.16591, 41.19141], [44.18148, 41.24644], [44.32139, 41.2079], [44.34337, 41.20312], [44.34417, 41.2382], [44.46791, 41.18204], [44.59322, 41.1933], [44.61462, 41.24018], [44.72814, 41.20338], [44.82084, 41.21513], [44.87887, 41.20195], [44.89911, 41.21366], [44.84358, 41.23088], [44.81749, 41.23488], [44.80053, 41.25949], [44.81437, 41.30371], [44.93493, 41.25685], [45.0133, 41.29747], [45.09867, 41.34065], [45.1797, 41.42231], [45.26285, 41.46433], [45.31352, 41.47168], [45.4006, 41.42402], [45.45973, 41.45898], [45.68389, 41.3539], [45.71035, 41.36208], [45.75705, 41.35157], [45.69946, 41.29545], [45.80842, 41.2229], [45.95786, 41.17956], [46.13221, 41.19479], [46.27698, 41.19011], [46.37661, 41.10805], [46.456, 41.09984], [46.48558, 41.0576], [46.55096, 41.1104], [46.63969, 41.09515], [46.66148, 41.20533], [46.72375, 41.28609], [46.63658, 41.37727], [46.4669, 41.43331], [46.40307, 41.48464], [46.33925, 41.4963], [46.29794, 41.5724], [46.34126, 41.57454], [46.34039, 41.5947], [46.3253, 41.60912], [46.28182, 41.60089], [46.26531, 41.63339], [46.24429, 41.59883], [46.19759, 41.62327], [46.17891, 41.72094], [46.20538, 41.77205], [46.23962, 41.75811], [46.30863, 41.79133], [46.3984, 41.84399], [46.42738, 41.91323]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GF",
+           iso1A3: "GUF",
+           iso1N3: "254",
+           wikidata: "Q3769",
+           nameEn: "French Guiana",
+           country: "FR",
+           groups: ["Q3320166", "EU", "005", "419", "019", "UN"],
+           callingCodes: ["594"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-51.35485, 4.8383], [-53.7094, 6.2264], [-54.01074, 5.68785], [-54.01877, 5.52789], [-54.26916, 5.26909], [-54.4717, 4.91964], [-54.38444, 4.13222], [-54.19367, 3.84387], [-54.05128, 3.63557], [-53.98914, 3.627], [-53.9849, 3.58697], [-54.28534, 2.67798], [-54.42864, 2.42442], [-54.6084, 2.32856], [-54.16286, 2.10779], [-53.78743, 2.34412], [-52.96539, 2.1881], [-52.6906, 2.37298], [-52.31787, 3.17896], [-51.85573, 3.83427], [-51.82312, 3.85825], [-51.79599, 3.89336], [-51.61983, 4.14596], [-51.63798, 4.51394], [-51.35485, 4.8383]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GG",
+           iso1A3: "GGY",
+           iso1N3: "831",
+           wikidata: "Q25230",
+           nameEn: "Bailiwick of Guernsey",
+           country: "GB"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GH",
+           iso1A3: "GHA",
+           iso1N3: "288",
+           wikidata: "Q117",
+           nameEn: "Ghana",
+           groups: ["011", "202", "002", "UN"],
+           callingCodes: ["233"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-0.13493, 11.14075], [-0.27374, 11.17157], [-0.28566, 11.12713], [-0.35955, 11.07801], [-0.38219, 11.12596], [-0.42391, 11.11661], [-0.44298, 11.04292], [-0.61937, 10.91305], [-0.67143, 10.99811], [-2.83373, 11.0067], [-2.94232, 10.64281], [-2.83108, 10.40252], [-2.74174, 9.83172], [-2.76534, 9.56589], [-2.68802, 9.49343], [-2.69814, 9.22717], [-2.77799, 9.04949], [-2.66357, 9.01771], [-2.58243, 8.7789], [-2.49037, 8.20872], [-2.62901, 8.11495], [-2.61232, 8.02645], [-2.67787, 8.02055], [-2.74819, 7.92613], [-2.78395, 7.94974], [-2.79467, 7.86002], [-2.92339, 7.60847], [-2.97822, 7.27165], [-2.95438, 7.23737], [-3.23327, 6.81744], [-3.21954, 6.74407], [-3.25999, 6.62521], [-3.01896, 5.71697], [-2.95323, 5.71865], [-2.96671, 5.6415], [-2.93132, 5.62137], [-2.85378, 5.65156], [-2.76614, 5.60963], [-2.72737, 5.34789], [-2.77625, 5.34621], [-2.73074, 5.1364], [-2.75502, 5.10657], [-2.95261, 5.12477], [-2.96554, 5.10397], [-3.063, 5.13665], [-3.11073, 5.12675], [-3.10675, 5.08515], [-3.34019, 4.17519], [1.07031, 5.15655], [1.27574, 5.93551], [1.19771, 6.11522], [1.19966, 6.17069], [1.09187, 6.17074], [1.05969, 6.22998], [1.03108, 6.24064], [0.99652, 6.33779], [0.89283, 6.33779], [0.71048, 6.53083], [0.74862, 6.56517], [0.63659, 6.63857], [0.6497, 6.73682], [0.58176, 6.76049], [0.57406, 6.80348], [0.52853, 6.82921], [0.56508, 6.92971], [0.52098, 6.94391], [0.52217, 6.9723], [0.59606, 7.01252], [0.65327, 7.31643], [0.62943, 7.41099], [0.57223, 7.39326], [0.52455, 7.45354], [0.51979, 7.58706], [0.58295, 7.62368], [0.62943, 7.85751], [0.58891, 8.12779], [0.6056, 8.13959], [0.61156, 8.18324], [0.5913, 8.19622], [0.63897, 8.25873], [0.73432, 8.29529], [0.64731, 8.48866], [0.47211, 8.59945], [0.37319, 8.75262], [0.52455, 8.87746], [0.45424, 9.04581], [0.56388, 9.40697], [0.49118, 9.48339], [0.36485, 9.49749], [0.33148, 9.44812], [0.25758, 9.42696], [0.2254, 9.47869], [0.31241, 9.50337], [0.30406, 9.521], [0.2409, 9.52335], [0.23851, 9.57389], [0.38153, 9.58682], [0.36008, 9.6256], [0.29334, 9.59387], [0.26712, 9.66437], [0.28261, 9.69022], [0.32313, 9.6491], [0.34816, 9.66907], [0.34816, 9.71607], [0.32075, 9.72781], [0.36366, 10.03309], [0.41252, 10.02018], [0.41371, 10.06361], [0.35293, 10.09412], [0.39584, 10.31112], [0.33028, 10.30408], [0.29453, 10.41546], [0.18846, 10.4096], [0.12886, 10.53149], [-0.05945, 10.63458], [-0.09141, 10.7147], [-0.07327, 10.71845], [-0.07183, 10.76794], [-0.0228, 10.81916], [-0.02685, 10.8783], [-908e-5, 10.91644], [-63e-4, 10.96417], [0.03355, 10.9807], [0.02395, 11.06229], [342e-5, 11.08317], [-514e-5, 11.10763], [-0.0275, 11.11202], [-0.05733, 11.08628], [-0.14462, 11.10811], [-0.13493, 11.14075]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GI",
+           iso1A3: "GIB",
+           iso1N3: "292",
+           wikidata: "Q1410",
+           nameEn: "Gibraltar",
+           country: "GB",
+           groups: ["Q12837", "BOTS", "039", "150", "UN"],
+           callingCodes: ["350"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-5.34064, 36.03744], [-5.27801, 36.14942], [-5.33822, 36.15272], [-5.34536, 36.15501], [-5.40526, 36.15488], [-5.34064, 36.03744]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GL",
+           iso1A3: "GRL",
+           iso1N3: "304",
+           wikidata: "Q223",
+           nameEn: "Greenland",
+           country: "DK",
+           groups: ["Q1451600", "021", "003", "019", "UN"],
+           callingCodes: ["299"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-49.33696, 84.57952], [-68.21821, 80.48551], [-77.52957, 77.23408], [-46.37635, 57.3249], [-9.68082, 72.73731], [-5.7106, 84.28058], [-49.33696, 84.57952]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GM",
+           iso1A3: "GMB",
+           iso1N3: "270",
+           wikidata: "Q1005",
+           nameEn: "The Gambia",
+           groups: ["011", "202", "002", "UN"],
+           callingCodes: ["220"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-15.14917, 13.57989], [-14.36795, 13.23033], [-13.79409, 13.34472], [-13.8955, 13.59126], [-14.34721, 13.46578], [-14.93719, 13.80173], [-15.36504, 13.79313], [-15.47902, 13.58758], [-17.43598, 13.59273], [-17.43966, 13.04579], [-16.74676, 13.06025], [-16.69343, 13.16791], [-15.80355, 13.16729], [-15.80478, 13.34832], [-15.26908, 13.37768], [-15.14917, 13.57989]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GN",
+           iso1A3: "GIN",
+           iso1N3: "324",
+           wikidata: "Q1006",
+           nameEn: "Guinea",
+           groups: ["011", "202", "002", "UN"],
+           callingCodes: ["224"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-11.37536, 12.40788], [-11.46267, 12.44559], [-11.91331, 12.42008], [-12.35415, 12.32758], [-12.87336, 12.51892], [-13.06603, 12.49342], [-13.05296, 12.64003], [-13.70523, 12.68013], [-13.7039, 12.60313], [-13.65089, 12.49515], [-13.64168, 12.42764], [-13.70851, 12.24978], [-13.92745, 12.24077], [-13.94589, 12.16869], [-13.7039, 12.00869], [-13.7039, 11.70195], [-14.09799, 11.63649], [-14.26623, 11.67486], [-14.31513, 11.60713], [-14.51173, 11.49708], [-14.66677, 11.51188], [-14.77786, 11.36323], [-14.95993, 10.99244], [-15.07174, 10.89557], [-15.96748, 10.162], [-14.36218, 8.64107], [-13.29911, 9.04245], [-13.18586, 9.0925], [-13.08953, 9.0409], [-12.94095, 9.26335], [-12.76788, 9.3133], [-12.47254, 9.86834], [-12.24262, 9.92386], [-12.12634, 9.87203], [-11.91023, 9.93927], [-11.89624, 9.99763], [-11.2118, 10.00098], [-10.6534, 9.29919], [-10.74484, 9.07998], [-10.5783, 9.06386], [-10.56197, 8.81225], [-10.47707, 8.67669], [-10.61422, 8.5314], [-10.70565, 8.29235], [-10.63934, 8.35326], [-10.54891, 8.31174], [-10.37257, 8.48941], [-10.27575, 8.48711], [-10.203, 8.47991], [-10.14579, 8.52665], [-10.05375, 8.50697], [-10.05873, 8.42578], [-9.77763, 8.54633], [-9.47415, 8.35195], [-9.50898, 8.18455], [-9.41445, 8.02448], [-9.44928, 7.9284], [-9.35724, 7.74111], [-9.37465, 7.62032], [-9.48161, 7.37122], [-9.41943, 7.41809], [-9.305, 7.42056], [-9.20798, 7.38109], [-9.18311, 7.30461], [-9.09107, 7.1985], [-8.93435, 7.2824], [-8.85724, 7.26019], [-8.8448, 7.35149], [-8.72789, 7.51429], [-8.67814, 7.69428], [-8.55874, 7.70167], [-8.55874, 7.62525], [-8.47114, 7.55676], [-8.4003, 7.6285], [-8.21374, 7.54466], [-8.09931, 7.78626], [-8.13414, 7.87991], [-8.06449, 8.04989], [-7.94695, 8.00925], [-7.99919, 8.11023], [-7.98675, 8.20134], [-8.062, 8.16071], [-8.2411, 8.24196], [-8.22991, 8.48438], [-7.92518, 8.50652], [-7.65653, 8.36873], [-7.69882, 8.66148], [-7.95503, 8.81146], [-7.92518, 8.99332], [-7.73862, 9.08422], [-7.90777, 9.20456], [-7.85056, 9.41812], [-8.03463, 9.39604], [-8.14657, 9.55062], [-8.09434, 9.86936], [-8.15652, 9.94288], [-8.11921, 10.04577], [-8.01225, 10.1021], [-7.97971, 10.17117], [-7.9578, 10.2703], [-8.10207, 10.44649], [-8.22711, 10.41722], [-8.32614, 10.69273], [-8.2667, 10.91762], [-8.35083, 11.06234], [-8.66923, 10.99397], [-8.40058, 11.37466], [-8.80854, 11.66715], [-8.94784, 12.34842], [-9.13689, 12.50875], [-9.38067, 12.48446], [-9.32097, 12.29009], [-9.63938, 12.18312], [-9.714, 12.0226], [-10.30604, 12.24634], [-10.71897, 11.91552], [-10.80355, 12.1053], [-10.99758, 12.24634], [-11.24136, 12.01286], [-11.50006, 12.17826], [-11.37536, 12.40788]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GP",
+           iso1A3: "GLP",
+           iso1N3: "312",
+           wikidata: "Q17012",
+           nameEn: "Guadeloupe",
+           country: "FR",
+           groups: ["Q3320166", "EU", "029", "003", "419", "019", "UN"],
+           callingCodes: ["590"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-60.03183, 16.1129], [-61.60296, 16.73066], [-63.00549, 15.26166], [-60.03183, 16.1129]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GQ",
+           iso1A3: "GNQ",
+           iso1N3: "226",
+           wikidata: "Q983",
+           nameEn: "Equatorial Guinea",
+           groups: ["017", "202", "002", "UN"],
+           callingCodes: ["240"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[9.22018, 3.72052], [8.34397, 4.30689], [7.71762, 0.6674], [3.35016, -3.29031], [9.66433, 1.06723], [9.75065, 1.06753], [9.79648, 1.0019], [11.35307, 1.00251], [11.3561, 2.17217], [9.991, 2.16561], [9.90749, 2.20049], [9.89012, 2.20457], [9.84716, 2.24676], [9.83238, 2.29079], [9.83754, 2.32428], [9.82123, 2.35097], [9.81162, 2.33797], [9.22018, 3.72052]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GR",
+           iso1A3: "GRC",
+           iso1N3: "300",
+           wikidata: "Q41",
+           nameEn: "Greece",
+           aliases: ["EL"],
+           groups: ["EU", "039", "150", "UN"],
+           callingCodes: ["30"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[26.03489, 40.73051], [26.0754, 40.72772], [26.08638, 40.73214], [26.12495, 40.74283], [26.12854, 40.77339], [26.15685, 40.80709], [26.21351, 40.83298], [26.20856, 40.86048], [26.26169, 40.9168], [26.29441, 40.89119], [26.28623, 40.93005], [26.32259, 40.94042], [26.35894, 40.94292], [26.33297, 40.98388], [26.3606, 41.02027], [26.31928, 41.07386], [26.32259, 41.24929], [26.39861, 41.25053], [26.5209, 41.33993], [26.5837, 41.32131], [26.62997, 41.34613], [26.61767, 41.42281], [26.59742, 41.48058], [26.59196, 41.60491], [26.5209, 41.62592], [26.47958, 41.67037], [26.35957, 41.71149], [26.30255, 41.70925], [26.2654, 41.71544], [26.22888, 41.74139], [26.21325, 41.73223], [26.16841, 41.74858], [26.06148, 41.70345], [26.07083, 41.64584], [26.15146, 41.60828], [26.14328, 41.55496], [26.17951, 41.55409], [26.176, 41.50072], [26.14796, 41.47533], [26.20288, 41.43943], [26.16548, 41.42278], [26.12926, 41.35878], [25.87919, 41.30526], [25.8266, 41.34563], [25.70507, 41.29209], [25.66183, 41.31316], [25.61042, 41.30614], [25.55082, 41.31667], [25.52394, 41.2798], [25.48187, 41.28506], [25.28322, 41.23411], [25.11611, 41.34212], [24.942, 41.38685], [24.90928, 41.40876], [24.86136, 41.39298], [24.82514, 41.4035], [24.8041, 41.34913], [24.71529, 41.41928], [24.61129, 41.42278], [24.52599, 41.56808], [24.30513, 41.51297], [24.27124, 41.57682], [24.18126, 41.51735], [24.10063, 41.54796], [24.06323, 41.53222], [24.06908, 41.46132], [23.96975, 41.44118], [23.91483, 41.47971], [23.89613, 41.45257], [23.80148, 41.43943], [23.76525, 41.40175], [23.67644, 41.41139], [23.63203, 41.37632], [23.52453, 41.40262], [23.40416, 41.39999], [23.33639, 41.36317], [23.31301, 41.40525], [23.22771, 41.37106], [23.21953, 41.33773], [23.1833, 41.31755], [22.93334, 41.34104], [22.81199, 41.3398], [22.76408, 41.32225], [22.74538, 41.16321], [22.71266, 41.13945], [22.65306, 41.18168], [22.62852, 41.14385], [22.58295, 41.11568], [22.5549, 41.13065], [22.42285, 41.11921], [22.26744, 41.16409], [22.17629, 41.15969], [22.1424, 41.12449], [22.06527, 41.15617], [21.90869, 41.09191], [21.91102, 41.04786], [21.7556, 40.92525], [21.69601, 40.9429], [21.57448, 40.86076], [21.53007, 40.90759], [21.41555, 40.9173], [21.35595, 40.87578], [21.25779, 40.86165], [21.21105, 40.8855], [21.15262, 40.85546], [20.97887, 40.85475], [20.98396, 40.79109], [20.95752, 40.76982], [20.98134, 40.76046], [21.05833, 40.66586], [21.03932, 40.56299], [20.96908, 40.51526], [20.94925, 40.46625], [20.83688, 40.47882], [20.7906, 40.42726], [20.78234, 40.35803], [20.71789, 40.27739], [20.67162, 40.09433], [20.62566, 40.0897], [20.61081, 40.07866], [20.55593, 40.06524], [20.51297, 40.08168], [20.48487, 40.06271], [20.42373, 40.06777], [20.37911, 39.99058], [20.31135, 39.99438], [20.41546, 39.82832], [20.41475, 39.81437], [20.38572, 39.78516], [20.30804, 39.81563], [20.29152, 39.80421], [20.31961, 39.72799], [20.27412, 39.69884], [20.22707, 39.67459], [20.22376, 39.64532], [20.15988, 39.652], [20.12956, 39.65805], [20.05189, 39.69112], [20.00957, 39.69227], [19.98042, 39.6504], [19.92466, 39.69533], [19.97622, 39.78684], [19.95905, 39.82857], [19.0384, 40.35325], [19.20409, 39.7532], [22.5213, 33.45682], [29.73302, 35.92555], [29.69611, 36.10365], [29.61805, 36.14179], [29.61002, 36.1731], [29.48192, 36.18377], [29.30783, 36.01033], [28.23708, 36.56812], [27.95037, 36.46155], [27.89482, 36.69898], [27.46117, 36.53789], [27.24613, 36.71622], [27.45627, 36.9008], [27.20312, 36.94571], [27.14757, 37.32], [26.95583, 37.64989], [26.99377, 37.69034], [27.16428, 37.72343], [27.05537, 37.9131], [26.21136, 38.17558], [26.24183, 38.44695], [26.32173, 38.48731], [26.21136, 38.65436], [26.61814, 38.81372], [26.70773, 39.0312], [26.43357, 39.43096], [25.94257, 39.39358], [25.61285, 40.17161], [26.04292, 40.3958], [25.94795, 40.72797], [26.03489, 40.73051]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GS",
+           iso1A3: "SGS",
+           iso1N3: "239",
+           wikidata: "Q35086",
+           nameEn: "South Georgia and South Sandwich Islands",
+           country: "GB",
+           groups: ["BOTS", "005", "419", "019", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["500"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-35.26394, -43.68272], [-53.39656, -59.87088], [-22.31757, -59.85974], [-35.26394, -43.68272]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GT",
+           iso1A3: "GTM",
+           iso1N3: "320",
+           wikidata: "Q774",
+           nameEn: "Guatemala",
+           groups: ["013", "003", "419", "019", "UN"],
+           callingCodes: ["502"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-89.14985, 17.81563], [-90.98678, 17.81655], [-90.99199, 17.25192], [-91.43809, 17.25373], [-91.04436, 16.92175], [-90.69064, 16.70697], [-90.61212, 16.49832], [-90.40499, 16.40524], [-90.44567, 16.07573], [-91.73182, 16.07371], [-92.20983, 15.26077], [-92.0621, 15.07406], [-92.1454, 14.98143], [-92.1423, 14.88647], [-92.18161, 14.84147], [-92.1454, 14.6804], [-92.2261, 14.53423], [-92.37213, 14.39277], [-90.55276, 12.8866], [-90.11344, 13.73679], [-90.10505, 13.85104], [-89.88937, 14.0396], [-89.81807, 14.07073], [-89.76103, 14.02923], [-89.73251, 14.04133], [-89.75569, 14.07073], [-89.70756, 14.1537], [-89.61844, 14.21937], [-89.52397, 14.22628], [-89.50614, 14.26084], [-89.58814, 14.33165], [-89.57441, 14.41637], [-89.39028, 14.44561], [-89.34776, 14.43013], [-89.35189, 14.47553], [-89.23719, 14.58046], [-89.15653, 14.57802], [-89.13132, 14.71582], [-89.23467, 14.85596], [-89.15149, 14.97775], [-89.18048, 14.99967], [-89.15149, 15.07392], [-88.97343, 15.14039], [-88.32467, 15.63665], [-88.31459, 15.66942], [-88.24022, 15.69247], [-88.22552, 15.72294], [-88.20359, 16.03858], [-88.40779, 16.09624], [-88.95358, 15.88698], [-89.02415, 15.9063], [-89.17418, 15.90898], [-89.22683, 15.88619], [-89.15025, 17.04813], [-89.14985, 17.81563]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GU",
+           iso1A3: "GUM",
+           iso1N3: "316",
+           wikidata: "Q16635",
+           nameEn: "Guam",
+           aliases: ["US-GU"],
+           country: "US",
+           groups: ["Q1352230", "Q153732", "057", "009", "UN"],
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["1 671"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[146.25931, 13.85876], [143.82485, 13.92273], [144.61642, 12.82462], [146.25931, 13.85876]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GW",
+           iso1A3: "GNB",
+           iso1N3: "624",
+           wikidata: "Q1007",
+           nameEn: "Guinea-Bissau",
+           groups: ["011", "202", "002", "UN"],
+           callingCodes: ["245"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-14.31513, 11.60713], [-14.26623, 11.67486], [-14.09799, 11.63649], [-13.7039, 11.70195], [-13.7039, 12.00869], [-13.94589, 12.16869], [-13.92745, 12.24077], [-13.70851, 12.24978], [-13.64168, 12.42764], [-13.65089, 12.49515], [-13.7039, 12.60313], [-13.70523, 12.68013], [-15.17582, 12.6847], [-15.67302, 12.42974], [-16.20591, 12.46157], [-16.38191, 12.36449], [-16.70562, 12.34803], [-17.4623, 11.92379], [-15.96748, 10.162], [-15.07174, 10.89557], [-14.95993, 10.99244], [-14.77786, 11.36323], [-14.66677, 11.51188], [-14.51173, 11.49708], [-14.31513, 11.60713]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "GY",
+           iso1A3: "GUY",
+           iso1N3: "328",
+           wikidata: "Q734",
+           nameEn: "Guyana",
+           groups: ["005", "419", "019", "UN"],
+           driveSide: "left",
+           callingCodes: ["592"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-56.84822, 6.73257], [-59.54058, 8.6862], [-59.98508, 8.53046], [-59.85562, 8.35213], [-59.80661, 8.28906], [-59.83156, 8.23261], [-59.97059, 8.20791], [-60.02407, 8.04557], [-60.38056, 7.8302], [-60.51959, 7.83373], [-60.64793, 7.56877], [-60.71923, 7.55817], [-60.59802, 7.33194], [-60.63367, 7.25061], [-60.54098, 7.14804], [-60.44116, 7.20817], [-60.28074, 7.1162], [-60.39419, 6.94847], [-60.54873, 6.8631], [-61.13632, 6.70922], [-61.20762, 6.58174], [-61.15058, 6.19558], [-61.4041, 5.95304], [-60.73204, 5.20931], [-60.32352, 5.21299], [-60.20944, 5.28754], [-59.98129, 5.07097], [-60.04189, 4.69801], [-60.15953, 4.53456], [-59.78878, 4.45637], [-59.69361, 4.34069], [-59.73353, 4.20399], [-59.51963, 3.91951], [-59.86899, 3.57089], [-59.79769, 3.37162], [-59.99733, 2.92312], [-59.91177, 2.36759], [-59.7264, 2.27497], [-59.74066, 1.87596], [-59.25583, 1.40559], [-58.92072, 1.31293], [-58.84229, 1.17749], [-58.53571, 1.29154], [-58.4858, 1.48399], [-58.33887, 1.58014], [-58.01873, 1.51966], [-57.97336, 1.64566], [-57.77281, 1.73344], [-57.55743, 1.69605], [-57.35073, 1.98327], [-57.23981, 1.95808], [-57.09109, 2.01854], [-57.07092, 1.95304], [-56.7659, 1.89509], [-56.47045, 1.95135], [-56.55439, 2.02003], [-56.70519, 2.02964], [-57.35891, 3.32121], [-58.0307, 3.95513], [-57.8699, 4.89394], [-57.37442, 5.0208], [-57.22536, 5.15605], [-57.31629, 5.33714], [-56.84822, 6.73257]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "HK",
+           iso1A3: "HKG",
+           iso1N3: "344",
+           wikidata: "Q8646",
+           nameEn: "Hong Kong",
+           country: "CN",
+           groups: ["030", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["852"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[113.92195, 22.13873], [114.50148, 22.15017], [114.44998, 22.55977], [114.25154, 22.55977], [114.22888, 22.5436], [114.22185, 22.55343], [114.20655, 22.55706], [114.18338, 22.55444], [114.17247, 22.55944], [114.1597, 22.56041], [114.15123, 22.55163], [114.1482, 22.54091], [114.13823, 22.54319], [114.12665, 22.54003], [114.11656, 22.53415], [114.11181, 22.52878], [114.1034, 22.5352], [114.09692, 22.53435], [114.09048, 22.53716], [114.08606, 22.53276], [114.07817, 22.52997], [114.07267, 22.51855], [114.06272, 22.51617], [114.05729, 22.51104], [114.05438, 22.5026], [114.03113, 22.5065], [113.86771, 22.42972], [113.81621, 22.2163], [113.83338, 22.1826], [113.92195, 22.13873]]]]
          }
-         /**
-          * getTiles() returns an array of tiles that cover the map view
-          */
-
-
-         tiler.getTiles = function (projection) {
-           var origin = [projection.scale() * Math.PI - projection.translate()[0], projection.scale() * Math.PI - projection.translate()[1]];
-           this.size(projection.clipExtent()[1]).scale(projection.scale() * 2 * Math.PI).translate(projection.translate());
-           var tiles = tiler();
-           var ts = tiles.scale;
-           return tiles.map(function (tile) {
-             if (_skipNullIsland && nearNullIsland(tile)) {
-               return false;
-             }
-
-             var x = tile[0] * ts - origin[0];
-             var y = tile[1] * ts - origin[1];
-             return {
-               id: tile.toString(),
-               xyz: tile,
-               extent: geoExtent(projection.invert([x, y + ts]), projection.invert([x + ts, y]))
-             };
-           }).filter(Boolean);
-         };
-         /**
-          * getGeoJSON() returns a FeatureCollection for debugging tiles
-          */
-
-
-         tiler.getGeoJSON = function (projection) {
-           var features = tiler.getTiles(projection).map(function (tile) {
-             return {
-               type: 'Feature',
-               properties: {
-                 id: tile.id,
-                 name: tile.id
-               },
-               geometry: {
-                 type: 'Polygon',
-                 coordinates: [tile.extent.polygon()]
-               }
-             };
-           });
-           return {
-             type: 'FeatureCollection',
-             features: features
-           };
-         };
-
-         tiler.tileSize = function (val) {
-           if (!arguments.length) return _tileSize;
-           _tileSize = val;
-           return tiler;
-         };
-
-         tiler.zoomExtent = function (val) {
-           if (!arguments.length) return _zoomExtent;
-           _zoomExtent = val;
-           return tiler;
-         };
-
-         tiler.size = function (val) {
-           if (!arguments.length) return _size;
-           _size = val;
-           return tiler;
-         };
-
-         tiler.scale = function (val) {
-           if (!arguments.length) return _scale;
-           _scale = val;
-           return tiler;
-         };
-
-         tiler.translate = function (val) {
-           if (!arguments.length) return _translate;
-           _translate = val;
-           return tiler;
-         }; // number to extend the rows/columns beyond those covering the viewport
-
-
-         tiler.margin = function (val) {
-           if (!arguments.length) return _margin;
-           _margin = +val;
-           return tiler;
-         };
-
-         tiler.skipNullIsland = function (val) {
-           if (!arguments.length) return _skipNullIsland;
-           _skipNullIsland = val;
-           return tiler;
-         };
-
-         return tiler;
-       }
-
-       function utilTriggerEvent(target, type) {
-         target.each(function () {
-           var evt = document.createEvent('HTMLEvents');
-           evt.initEvent(type, true, true);
-           this.dispatchEvent(evt);
-         });
-       }
-
-       var _mainLocalizer = coreLocalizer(); // singleton
-
-
-       var _t = _mainLocalizer.t;
-       // coreLocalizer manages language and locale parameters including translated strings
-       //
-
-       function coreLocalizer() {
-         var localizer = {};
-         var _dataLanguages = {}; // `_dataLocales` is an object containing all _supported_ locale codes -> language info.
-         // * `rtl` - right-to-left or left-to-right text direction
-         // * `pct` - the percent of strings translated; 1 = 100%, full coverage
-         //
-         // {
-         // en: { rtl: false, pct: {…} },
-         // de: { rtl: false, pct: {…} },
-         // …
-         // }
-
-         var _dataLocales = {}; // `localeStrings` is an object containing all _loaded_ locale codes -> string data.
-         // {
-         // en: { icons: {…}, toolbar: {…}, modes: {…}, operations: {…}, … },
-         // de: { icons: {…}, toolbar: {…}, modes: {…}, operations: {…}, … },
-         // …
-         // }
-
-         var _localeStrings = {}; // the current locale
-
-         var _localeCode = 'en-US'; // `_localeCodes` must contain `_localeCode` first, optionally followed by fallbacks
-
-         var _localeCodes = ['en-US', 'en'];
-         var _languageCode = 'en';
-         var _textDirection = 'ltr';
-         var _usesMetric = false;
-         var _languageNames = {};
-         var _scriptNames = {}; // getters for the current locale parameters
-
-         localizer.localeCode = function () {
-           return _localeCode;
-         };
-
-         localizer.localeCodes = function () {
-           return _localeCodes;
-         };
-
-         localizer.languageCode = function () {
-           return _languageCode;
-         };
-
-         localizer.textDirection = function () {
-           return _textDirection;
-         };
-
-         localizer.usesMetric = function () {
-           return _usesMetric;
-         };
-
-         localizer.languageNames = function () {
-           return _languageNames;
-         };
-
-         localizer.scriptNames = function () {
-           return _scriptNames;
-         }; // The client app may want to manually set the locale, regardless of the
-         // settings provided by the browser
-
-
-         var _preferredLocaleCodes = [];
-
-         localizer.preferredLocaleCodes = function (codes) {
-           if (!arguments.length) return _preferredLocaleCodes;
-
-           if (typeof codes === 'string') {
-             // be generous and accept delimited strings as input
-             _preferredLocaleCodes = codes.split(/,|;| /gi).filter(Boolean);
-           } else {
-             _preferredLocaleCodes = codes;
-           }
-
-           return localizer;
-         };
-
-         var _loadPromise;
-
-         localizer.ensureLoaded = function () {
-           if (_loadPromise) return _loadPromise;
-           return _loadPromise = Promise.all([// load the list of languages
-           _mainFileFetcher.get('languages'), // load the list of supported locales
-           _mainFileFetcher.get('locales')]).then(function (results) {
-             _dataLanguages = results[0];
-             _dataLocales = results[1];
-           }).then(function () {
-             var requestedLocales = (_preferredLocaleCodes || []). // List of locales preferred by the browser in priority order.
-             concat(utilDetect().browserLocales) // fallback to English since it's the only guaranteed complete language
-             .concat(['en']);
-
-             _localeCodes = localesToUseFrom(requestedLocales); // Run iD in the highest-priority locale; the rest are fallbacks
-
-             _localeCode = _localeCodes[0]; // Will always return the index for `en` if nothing else
-
-             var fullCoverageIndex = _localeCodes.findIndex(function (locale) {
-               return _dataLocales[locale].pct === 1;
-             }); // We only need to load locales up until we find one with full coverage
-
-
-             var loadStringsPromises = _localeCodes.slice(0, fullCoverageIndex + 1).map(function (code) {
-               return localizer.loadLocale(code);
-             });
-
-             return Promise.all(loadStringsPromises);
-           }).then(function () {
-             updateForCurrentLocale();
-           })["catch"](function (err) {
-             return console.error(err);
-           }); // eslint-disable-line
-         }; // Returns the locales from `requestedLocales` supported by iD that we should use
-
-
-         function localesToUseFrom(requestedLocales) {
-           var supportedLocales = _dataLocales;
-           var toUse = [];
-
-           for (var i in requestedLocales) {
-             var locale = requestedLocales[i];
-             if (supportedLocales[locale]) toUse.push(locale);
-
-             if (locale.includes('-')) {
-               // Full locale ('es-ES'), add fallback to the base ('es')
-               var langPart = locale.split('-')[0];
-               if (supportedLocales[langPart]) toUse.push(langPart);
-             }
-           } // remove duplicates
-
-
-           return utilArrayUniq(toUse);
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "HM",
+           iso1A3: "HMD",
+           iso1N3: "334",
+           wikidata: "Q131198",
+           nameEn: "Heard Island and McDonald Islands",
+           country: "AU",
+           groups: ["053", "009", "UN"],
+           driveSide: "left"
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[71.08716, -53.87687], [75.44182, -53.99822], [72.87012, -51.48322], [71.08716, -53.87687]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "HN",
+           iso1A3: "HND",
+           iso1N3: "340",
+           wikidata: "Q783",
+           nameEn: "Honduras",
+           groups: ["013", "003", "419", "019", "UN"],
+           callingCodes: ["504"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-83.86109, 17.73736], [-88.20359, 16.03858], [-88.22552, 15.72294], [-88.24022, 15.69247], [-88.31459, 15.66942], [-88.32467, 15.63665], [-88.97343, 15.14039], [-89.15149, 15.07392], [-89.18048, 14.99967], [-89.15149, 14.97775], [-89.23467, 14.85596], [-89.13132, 14.71582], [-89.15653, 14.57802], [-89.23719, 14.58046], [-89.35189, 14.47553], [-89.34776, 14.43013], [-89.04187, 14.33644], [-88.94608, 14.20207], [-88.85785, 14.17763], [-88.815, 14.11652], [-88.73182, 14.10919], [-88.70661, 14.04317], [-88.49738, 13.97224], [-88.48982, 13.86458], [-88.25791, 13.91108], [-88.23018, 13.99915], [-88.07641, 13.98447], [-88.00331, 13.86948], [-87.7966, 13.91353], [-87.68821, 13.80829], [-87.73106, 13.75443], [-87.78148, 13.52906], [-87.71657, 13.50577], [-87.72115, 13.46083], [-87.73841, 13.44169], [-87.77354, 13.45767], [-87.83467, 13.44655], [-87.84675, 13.41078], [-87.80177, 13.35689], [-87.73714, 13.32715], [-87.69751, 13.25228], [-87.55124, 13.12523], [-87.37107, 12.98646], [-87.06306, 13.00892], [-87.03785, 12.98682], [-86.93197, 13.05313], [-86.93383, 13.18677], [-86.87066, 13.30641], [-86.71267, 13.30348], [-86.76812, 13.79605], [-86.35219, 13.77157], [-86.14801, 14.04317], [-86.00685, 14.08474], [-86.03458, 13.99181], [-85.75477, 13.8499], [-85.73964, 13.9698], [-85.45762, 14.11304], [-85.32149, 14.2562], [-85.18602, 14.24929], [-85.1575, 14.53934], [-84.90082, 14.80489], [-84.82596, 14.82212], [-84.70119, 14.68078], [-84.48373, 14.63249], [-84.10584, 14.76353], [-83.89551, 14.76697], [-83.62101, 14.89448], [-83.49268, 15.01158], [-83.13724, 15.00002], [-83.04763, 15.03256], [-82.06974, 14.49418], [-81.58685, 18.0025], [-83.86109, 17.73736]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "HR",
+           iso1A3: "HRV",
+           iso1N3: "191",
+           wikidata: "Q224",
+           nameEn: "Croatia",
+           groups: ["EU", "039", "150", "UN"],
+           callingCodes: ["385"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[17.6444, 42.88641], [17.5392, 42.92787], [17.70879, 42.97223], [17.64268, 43.08595], [17.46986, 43.16559], [17.286, 43.33065], [17.25579, 43.40353], [17.29699, 43.44542], [17.24411, 43.49376], [17.15828, 43.49376], [17.00585, 43.58037], [16.80736, 43.76011], [16.75316, 43.77157], [16.70922, 43.84887], [16.55472, 43.95326], [16.50528, 44.0244], [16.43629, 44.02826], [16.43662, 44.07523], [16.36864, 44.08263], [16.18688, 44.27012], [16.21346, 44.35231], [16.12969, 44.38275], [16.16814, 44.40679], [16.10566, 44.52586], [16.03012, 44.55572], [16.00884, 44.58605], [16.05828, 44.61538], [15.89348, 44.74964], [15.8255, 44.71501], [15.72584, 44.82334], [15.79472, 44.8455], [15.76096, 44.87045], [15.74723, 44.96818], [15.78568, 44.97401], [15.74585, 45.0638], [15.78842, 45.11519], [15.76371, 45.16508], [15.83512, 45.22459], [15.98412, 45.23088], [16.12153, 45.09616], [16.29036, 44.99732], [16.35404, 45.00241], [16.35863, 45.03529], [16.3749, 45.05206], [16.38219, 45.05139], [16.38309, 45.05955], [16.40023, 45.1147], [16.4634, 45.14522], [16.49155, 45.21153], [16.52982, 45.22713], [16.5501, 45.2212], [16.56559, 45.22307], [16.60194, 45.23042], [16.64962, 45.20714], [16.74845, 45.20393], [16.78219, 45.19002], [16.81137, 45.18434], [16.83804, 45.18951], [16.92405, 45.27607], [16.9385, 45.22742], [17.0415, 45.20759], [17.18438, 45.14764], [17.24325, 45.146], [17.25131, 45.14957], [17.26815, 45.18444], [17.32092, 45.16246], [17.33573, 45.14521], [17.41229, 45.13335], [17.4498, 45.16119], [17.45615, 45.12523], [17.47589, 45.12656], [17.51469, 45.10791], [17.59104, 45.10816], [17.66571, 45.13408], [17.84826, 45.04489], [17.87148, 45.04645], [17.93706, 45.08016], [17.97336, 45.12245], [17.97834, 45.13831], [17.99479, 45.14958], [18.01594, 45.15163], [18.03121, 45.12632], [18.1624, 45.07654], [18.24387, 45.13699], [18.32077, 45.1021], [18.41896, 45.11083], [18.47939, 45.05871], [18.65723, 45.07544], [18.78357, 44.97741], [18.80661, 44.93561], [18.76369, 44.93707], [18.76347, 44.90669], [18.8704, 44.85097], [19.01994, 44.85493], [18.98957, 44.90645], [19.02871, 44.92541], [19.06853, 44.89915], [19.15573, 44.95409], [19.05205, 44.97692], [19.1011, 45.01191], [19.07952, 45.14668], [19.14063, 45.12972], [19.19144, 45.17863], [19.43589, 45.17137], [19.41941, 45.23475], [19.28208, 45.23813], [19.10774, 45.29547], [18.97446, 45.37528], [18.99918, 45.49333], [19.08364, 45.48804], [19.07471, 45.53086], [18.94562, 45.53712], [18.88776, 45.57253], [18.96691, 45.66731], [18.90305, 45.71863], [18.85783, 45.85493], [18.81394, 45.91329], [18.80211, 45.87995], [18.6792, 45.92057], [18.57483, 45.80772], [18.44368, 45.73972], [18.12439, 45.78905], [18.08869, 45.76511], [17.99805, 45.79671], [17.87377, 45.78522], [17.66545, 45.84207], [17.56821, 45.93728], [17.35672, 45.95209], [17.14592, 46.16697], [16.8903, 46.28122], [16.8541, 46.36255], [16.7154, 46.39523], [16.6639, 46.45203], [16.59527, 46.47524], [16.52604, 46.47831], [16.5007, 46.49644], [16.44036, 46.5171], [16.38771, 46.53608], [16.37193, 46.55008], [16.29793, 46.5121], [16.26733, 46.51505], [16.26759, 46.50566], [16.23961, 46.49653], [16.25124, 46.48067], [16.27398, 46.42875], [16.27329, 46.41467], [16.30162, 46.40437], [16.30233, 46.37837], [16.18824, 46.38282], [16.14859, 46.40547], [16.05281, 46.39141], [16.05065, 46.3833], [16.07314, 46.36458], [16.07616, 46.3463], [15.97965, 46.30652], [15.79284, 46.25811], [15.78817, 46.21719], [15.75479, 46.20336], [15.75436, 46.21969], [15.67395, 46.22478], [15.6434, 46.21396], [15.64904, 46.19229], [15.59909, 46.14761], [15.6083, 46.11992], [15.62317, 46.09103], [15.72977, 46.04682], [15.71246, 46.01196], [15.70327, 46.00015], [15.70636, 45.92116], [15.67967, 45.90455], [15.68383, 45.88867], [15.68232, 45.86819], [15.70411, 45.8465], [15.66662, 45.84085], [15.64185, 45.82915], [15.57952, 45.84953], [15.52234, 45.82195], [15.47325, 45.8253], [15.47531, 45.79802], [15.40836, 45.79491], [15.25423, 45.72275], [15.30872, 45.69014], [15.34919, 45.71623], [15.4057, 45.64727], [15.38952, 45.63682], [15.34214, 45.64702], [15.34695, 45.63382], [15.31027, 45.6303], [15.27747, 45.60504], [15.29837, 45.5841], [15.30249, 45.53224], [15.38188, 45.48752], [15.33051, 45.45258], [15.27758, 45.46678], [15.16862, 45.42309], [15.05187, 45.49079], [15.02385, 45.48533], [14.92266, 45.52788], [14.90554, 45.47769], [14.81992, 45.45913], [14.80124, 45.49515], [14.71718, 45.53442], [14.68605, 45.53006], [14.69694, 45.57366], [14.59576, 45.62812], [14.60977, 45.66403], [14.57397, 45.67165], [14.53816, 45.6205], [14.5008, 45.60852], [14.49769, 45.54424], [14.36693, 45.48642], [14.32487, 45.47142], [14.27681, 45.4902], [14.26611, 45.48239], [14.24239, 45.50607], [14.22371, 45.50388], [14.20348, 45.46896], [14.07116, 45.48752], [14.00578, 45.52352], [13.96063, 45.50825], [13.99488, 45.47551], [13.97309, 45.45258], [13.90771, 45.45149], [13.88124, 45.42637], [13.81742, 45.43729], [13.7785, 45.46787], [13.67398, 45.4436], [13.62902, 45.45898], [13.56979, 45.4895], [13.45644, 45.59464], [13.05142, 45.33128], [13.12821, 44.48877], [16.15283, 42.18525], [18.45131, 42.21682], [18.54128, 42.39171], [18.52152, 42.42302], [18.43588, 42.48556], [18.44307, 42.51077], [18.43735, 42.55921], [18.36197, 42.61423], [18.24318, 42.6112], [17.88201, 42.83668], [17.80854, 42.9182], [17.7948, 42.89556], [17.68151, 42.92725], [17.6444, 42.88641]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "HT",
+           iso1A3: "HTI",
+           iso1N3: "332",
+           wikidata: "Q790",
+           nameEn: "Haiti",
+           aliases: ["RH"],
+           groups: ["029", "003", "419", "019", "UN"],
+           callingCodes: ["509"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-71.71885, 18.78423], [-71.72624, 18.87802], [-71.77766, 18.95007], [-71.88102, 18.95007], [-71.74088, 19.0437], [-71.71088, 19.08353], [-71.69938, 19.10916], [-71.65337, 19.11759], [-71.62642, 19.21212], [-71.73229, 19.26686], [-71.77766, 19.33823], [-71.69448, 19.37866], [-71.6802, 19.45008], [-71.71268, 19.53374], [-71.71449, 19.55364], [-71.7429, 19.58445], [-71.75865, 19.70231], [-71.77419, 19.73128], [-72.38946, 20.27111], [-73.37289, 20.43199], [-74.7289, 18.71009], [-74.76465, 18.06252], [-72.29523, 17.48026], [-71.75671, 18.03456], [-71.73783, 18.07177], [-71.74994, 18.11115], [-71.75465, 18.14405], [-71.78271, 18.18302], [-71.69952, 18.34101], [-71.90875, 18.45821], [-71.88102, 18.50125], [-72.00201, 18.62312], [-71.95412, 18.64939], [-71.82556, 18.62551], [-71.71885, 18.78423]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "HU",
+           iso1A3: "HUN",
+           iso1N3: "348",
+           wikidata: "Q28",
+           nameEn: "Hungary",
+           groups: ["EU", "151", "150", "UN"],
+           callingCodes: ["36"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[21.72525, 48.34628], [21.67134, 48.3989], [21.6068, 48.50365], [21.44063, 48.58456], [21.11516, 48.49546], [20.83248, 48.5824], [20.5215, 48.53336], [20.29943, 48.26104], [20.24312, 48.2784], [19.92452, 48.1283], [19.63338, 48.25006], [19.52489, 48.19791], [19.47957, 48.09437], [19.28182, 48.08336], [19.23924, 48.0595], [19.01952, 48.07052], [18.82176, 48.04206], [18.76134, 47.97499], [18.76821, 47.87469], [18.8506, 47.82308], [18.74074, 47.8157], [18.66521, 47.76772], [18.56496, 47.76588], [18.29305, 47.73541], [18.02938, 47.75665], [17.71215, 47.7548], [17.23699, 48.02094], [17.16001, 48.00636], [17.09786, 47.97336], [17.11022, 47.92461], [17.08275, 47.87719], [17.00997, 47.86245], [17.07039, 47.81129], [17.05048, 47.79377], [17.08893, 47.70928], [16.87538, 47.68895], [16.86509, 47.72268], [16.82938, 47.68432], [16.7511, 47.67878], [16.72089, 47.73469], [16.65679, 47.74197], [16.61183, 47.76171], [16.54779, 47.75074], [16.53514, 47.73837], [16.55129, 47.72268], [16.4222, 47.66537], [16.58699, 47.61772], [16.64193, 47.63114], [16.71059, 47.52692], [16.64821, 47.50155], [16.6718, 47.46139], [16.57152, 47.40868], [16.52414, 47.41007], [16.49908, 47.39416], [16.45104, 47.41181], [16.47782, 47.25918], [16.44142, 47.25079], [16.43663, 47.21127], [16.41739, 47.20649], [16.42801, 47.18422], [16.4523, 47.18812], [16.46442, 47.16845], [16.44932, 47.14418], [16.52863, 47.13974], [16.46134, 47.09395], [16.52176, 47.05747], [16.43936, 47.03548], [16.51369, 47.00084], [16.28202, 47.00159], [16.27594, 46.9643], [16.22403, 46.939], [16.19904, 46.94134], [16.10983, 46.867], [16.14365, 46.8547], [16.15711, 46.85434], [16.21892, 46.86961], [16.2365, 46.87775], [16.2941, 46.87137], [16.34547, 46.83836], [16.3408, 46.80641], [16.31303, 46.79838], [16.30966, 46.7787], [16.37816, 46.69975], [16.42641, 46.69228], [16.41863, 46.66238], [16.38594, 46.6549], [16.39217, 46.63673], [16.50139, 46.56684], [16.52885, 46.53303], [16.52604, 46.5051], [16.59527, 46.47524], [16.6639, 46.45203], [16.7154, 46.39523], [16.8541, 46.36255], [16.8903, 46.28122], [17.14592, 46.16697], [17.35672, 45.95209], [17.56821, 45.93728], [17.66545, 45.84207], [17.87377, 45.78522], [17.99805, 45.79671], [18.08869, 45.76511], [18.12439, 45.78905], [18.44368, 45.73972], [18.57483, 45.80772], [18.6792, 45.92057], [18.80211, 45.87995], [18.81394, 45.91329], [18.99712, 45.93537], [19.01284, 45.96529], [19.0791, 45.96458], [19.10388, 46.04015], [19.14543, 45.9998], [19.28826, 45.99694], [19.52473, 46.1171], [19.56113, 46.16824], [19.66007, 46.19005], [19.81491, 46.1313], [19.93508, 46.17553], [20.01816, 46.17696], [20.03533, 46.14509], [20.09713, 46.17315], [20.26068, 46.12332], [20.28324, 46.1438], [20.35573, 46.16629], [20.45377, 46.14405], [20.49718, 46.18721], [20.63863, 46.12728], [20.76085, 46.21002], [20.74574, 46.25467], [20.86797, 46.28884], [21.06572, 46.24897], [21.16872, 46.30118], [21.28061, 46.44941], [21.26929, 46.4993], [21.33214, 46.63035], [21.43926, 46.65109], [21.5151, 46.72147], [21.48935, 46.7577], [21.52028, 46.84118], [21.59307, 46.86935], [21.59581, 46.91628], [21.68645, 46.99595], [21.648, 47.03902], [21.78395, 47.11104], [21.94463, 47.38046], [22.01055, 47.37767], [22.03389, 47.42508], [22.00917, 47.50492], [22.31816, 47.76126], [22.41979, 47.7391], [22.46559, 47.76583], [22.67247, 47.7871], [22.76617, 47.8417], [22.77991, 47.87211], [22.89849, 47.95851], [22.84276, 47.98602], [22.87847, 48.04665], [22.81804, 48.11363], [22.73427, 48.12005], [22.66835, 48.09162], [22.58733, 48.10813], [22.59007, 48.15121], [22.49806, 48.25189], [22.38133, 48.23726], [22.2083, 48.42534], [22.14689, 48.4005], [21.83339, 48.36242], [21.8279, 48.33321], [21.72525, 48.34628]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "IC",
+           wikidata: "Q5813",
+           nameEn: "Canary Islands",
+           country: "ES",
+           groups: ["Q3320166", "Q105472", "EU", "039", "150", "UN"],
+           isoStatus: "excRes",
+           callingCodes: ["34"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-12.00985, 30.24121], [-25.3475, 27.87574], [-14.43883, 27.02969], [-12.00985, 30.24121]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "ID",
+           iso1A3: "IDN",
+           iso1N3: "360",
+           wikidata: "Q252",
+           nameEn: "Indonesia",
+           aliases: ["RI"]
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "IE",
+           iso1A3: "IRL",
+           iso1N3: "372",
+           wikidata: "Q27",
+           nameEn: "Republic of Ireland",
+           groups: ["EU", "Q22890", "154", "150", "UN"],
+           driveSide: "left",
+           callingCodes: ["353"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-6.26218, 54.09785], [-6.29003, 54.11278], [-6.32694, 54.09337], [-6.36279, 54.11248], [-6.36605, 54.07234], [-6.47849, 54.06947], [-6.62842, 54.03503], [-6.66264, 54.0666], [-6.6382, 54.17071], [-6.70175, 54.20218], [-6.74575, 54.18788], [-6.81583, 54.22791], [-6.85179, 54.29176], [-6.87775, 54.34682], [-7.02034, 54.4212], [-7.19145, 54.31296], [-7.14908, 54.22732], [-7.25012, 54.20063], [-7.26316, 54.13863], [-7.29493, 54.12013], [-7.29687, 54.1354], [-7.28017, 54.16714], [-7.29157, 54.17191], [-7.34005, 54.14698], [-7.30553, 54.11869], [-7.32834, 54.11475], [-7.44567, 54.1539], [-7.4799, 54.12239], [-7.55812, 54.12239], [-7.69501, 54.20731], [-7.81397, 54.20159], [-7.8596, 54.21779], [-7.87101, 54.29299], [-8.04555, 54.36292], [-8.179, 54.46763], [-8.04538, 54.48941], [-7.99812, 54.54427], [-7.8596, 54.53671], [-7.70315, 54.62077], [-7.93293, 54.66603], [-7.83352, 54.73854], [-7.75041, 54.7103], [-7.64449, 54.75265], [-7.54671, 54.74606], [-7.54508, 54.79401], [-7.47626, 54.83084], [-7.4473, 54.87003], [-7.44404, 54.9403], [-7.40004, 54.94498], [-7.4033, 55.00391], [-7.34464, 55.04688], [-7.2471, 55.06933], [-6.34755, 55.49206], [-7.75229, 55.93854], [-22.01468, 48.19557], [-6.03913, 51.13217], [-5.37267, 53.63269], [-6.26218, 54.09785]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "IL",
+           iso1A3: "ISR",
+           iso1N3: "376",
+           wikidata: "Q801",
+           nameEn: "Israel",
+           groups: ["145", "142", "UN"],
+           callingCodes: ["972"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[34.052, 31.46619], [34.29262, 31.70393], [34.48681, 31.59711], [34.56797, 31.54197], [34.48892, 31.48365], [34.40077, 31.40926], [34.36505, 31.36404], [34.37381, 31.30598], [34.36523, 31.28963], [34.29417, 31.24194], [34.26742, 31.21998], [34.92298, 29.45305], [34.97718, 29.54294], [34.98207, 29.58147], [35.02147, 29.66343], [35.14108, 30.07374], [35.19183, 30.34636], [35.16218, 30.43535], [35.19595, 30.50297], [35.21379, 30.60401], [35.29311, 30.71365], [35.33456, 30.81224], [35.33984, 30.8802], [35.41371, 30.95565], [35.43658, 31.12444], [35.40316, 31.25535], [35.47672, 31.49578], [35.39675, 31.49572], [35.22921, 31.37445], [35.13033, 31.3551], [35.02459, 31.35979], [34.92571, 31.34337], [34.88932, 31.37093], [34.87833, 31.39321], [34.89756, 31.43891], [34.93258, 31.47816], [34.94356, 31.50743], [34.9415, 31.55601], [34.95249, 31.59813], [35.00879, 31.65426], [35.08226, 31.69107], [35.10782, 31.71594], [35.11895, 31.71454], [35.12933, 31.7325], [35.13931, 31.73012], [35.15119, 31.73634], [35.15474, 31.73352], [35.16478, 31.73242], [35.18023, 31.72067], [35.20538, 31.72388], [35.21937, 31.71578], [35.22392, 31.71899], [35.23972, 31.70896], [35.24315, 31.71244], [35.2438, 31.7201], [35.24981, 31.72543], [35.25182, 31.73945], [35.26319, 31.74846], [35.25225, 31.7678], [35.26058, 31.79064], [35.25573, 31.81362], [35.26404, 31.82567], [35.251, 31.83085], [35.25753, 31.8387], [35.24816, 31.8458], [35.2304, 31.84222], [35.2249, 31.85433], [35.22817, 31.8638], [35.22567, 31.86745], [35.22294, 31.87889], [35.22014, 31.88264], [35.2136, 31.88241], [35.21276, 31.88153], [35.21016, 31.88237], [35.20945, 31.8815], [35.20791, 31.8821], [35.20673, 31.88151], [35.20381, 31.86716], [35.21128, 31.863], [35.216, 31.83894], [35.21469, 31.81835], [35.19461, 31.82687], [35.18169, 31.82542], [35.18603, 31.80901], [35.14174, 31.81325], [35.07677, 31.85627], [35.05617, 31.85685], [35.01978, 31.82944], [34.9724, 31.83352], [34.99712, 31.85569], [35.03489, 31.85919], [35.03978, 31.89276], [35.03489, 31.92448], [35.00124, 31.93264], [34.98682, 31.96935], [35.00261, 32.027], [34.9863, 32.09551], [34.99437, 32.10962], [34.98507, 32.12606], [34.99039, 32.14626], [34.96009, 32.17503], [34.95703, 32.19522], [34.98885, 32.20758], [35.01841, 32.23981], [35.02939, 32.2671], [35.01119, 32.28684], [35.01772, 32.33863], [35.04243, 32.35008], [35.05142, 32.3667], [35.0421, 32.38242], [35.05311, 32.4024], [35.05423, 32.41754], [35.07059, 32.4585], [35.08564, 32.46948], [35.09236, 32.47614], [35.10024, 32.47856], [35.10882, 32.4757], [35.15937, 32.50466], [35.2244, 32.55289], [35.25049, 32.52453], [35.29306, 32.50947], [35.30685, 32.51024], [35.35212, 32.52047], [35.40224, 32.50136], [35.42034, 32.46009], [35.41598, 32.45593], [35.41048, 32.43706], [35.42078, 32.41562], [35.55807, 32.38674], [35.55494, 32.42687], [35.57485, 32.48669], [35.56614, 32.64393], [35.59813, 32.65159], [35.61669, 32.67999], [35.66527, 32.681], [35.68467, 32.70715], [35.75983, 32.74803], [35.78745, 32.77938], [35.83758, 32.82817], [35.84021, 32.8725], [35.87012, 32.91976], [35.89298, 32.9456], [35.87188, 32.98028], [35.84802, 33.1031], [35.81911, 33.11077], [35.81911, 33.1336], [35.84285, 33.16673], [35.83846, 33.19397], [35.81647, 33.2028], [35.81295, 33.24841], [35.77513, 33.27342], [35.813, 33.3172], [35.77477, 33.33609], [35.62019, 33.27278], [35.62283, 33.24226], [35.58502, 33.26653], [35.58326, 33.28381], [35.56523, 33.28969], [35.55555, 33.25844], [35.54544, 33.25513], [35.54808, 33.236], [35.5362, 33.23196], [35.54228, 33.19865], [35.52573, 33.11921], [35.50335, 33.114], [35.50272, 33.09056], [35.448, 33.09264], [35.43059, 33.06659], [35.35223, 33.05617], [35.31429, 33.10515], [35.1924, 33.08743], [35.10645, 33.09318], [34.78515, 33.20368], [33.62659, 31.82938], [34.052, 31.46619]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "IM",
+           iso1A3: "IMN",
+           iso1N3: "833",
+           wikidata: "Q9676",
+           nameEn: "Isle of Man",
+           country: "GB",
+           groups: ["Q185086", "154", "150", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["44 01624", "44 07624", "44 07524", "44 07924"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-3.98763, 54.07351], [-4.1819, 54.57861], [-5.6384, 53.81157], [-3.98763, 54.07351]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "IN",
+           iso1A3: "IND",
+           iso1N3: "356",
+           wikidata: "Q668",
+           nameEn: "India"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "IO",
+           iso1A3: "IOT",
+           iso1N3: "086",
+           wikidata: "Q43448",
+           nameEn: "British Indian Ocean Territory",
+           country: "GB"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "IQ",
+           iso1A3: "IRQ",
+           iso1N3: "368",
+           wikidata: "Q796",
+           nameEn: "Iraq",
+           groups: ["145", "142", "UN"],
+           callingCodes: ["964"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[42.78887, 37.38615], [42.56725, 37.14878], [42.35724, 37.10998], [42.36697, 37.0627], [41.81736, 36.58782], [41.40058, 36.52502], [41.28864, 36.35368], [41.2564, 36.06012], [41.37027, 35.84095], [41.38184, 35.62502], [41.26569, 35.42708], [41.21654, 35.1508], [41.2345, 34.80049], [41.12388, 34.65742], [40.97676, 34.39788], [40.64314, 34.31604], [38.79171, 33.37328], [39.08202, 32.50304], [38.98762, 32.47694], [39.04251, 32.30203], [39.26157, 32.35555], [39.29903, 32.23259], [40.01521, 32.05667], [42.97601, 30.72204], [42.97796, 30.48295], [44.72255, 29.19736], [46.42415, 29.05947], [46.5527, 29.10283], [46.89695, 29.50584], [47.15166, 30.01044], [47.37192, 30.10421], [47.7095, 30.10453], [48.01114, 29.98906], [48.06782, 30.02906], [48.17332, 30.02448], [48.40479, 29.85763], [48.59531, 29.66815], [48.83867, 29.78572], [48.61441, 29.93675], [48.51011, 29.96238], [48.44785, 30.00148], [48.4494, 30.04456], [48.43384, 30.08233], [48.38869, 30.11062], [48.38714, 30.13485], [48.41671, 30.17254], [48.41117, 30.19846], [48.26393, 30.3408], [48.24385, 30.33846], [48.21279, 30.31644], [48.19425, 30.32796], [48.18321, 30.39703], [48.14585, 30.44133], [48.02443, 30.4789], [48.03221, 30.9967], [47.68219, 31.00004], [47.6804, 31.39086], [47.86337, 31.78422], [47.64771, 32.07666], [47.52474, 32.15972], [47.57144, 32.20583], [47.37529, 32.47808], [47.17218, 32.45393], [46.46788, 32.91992], [46.32298, 32.9731], [46.17198, 32.95612], [46.09103, 32.98354], [46.15175, 33.07229], [46.03966, 33.09577], [46.05367, 33.13097], [46.11905, 33.11924], [46.20623, 33.20395], [45.99919, 33.5082], [45.86687, 33.49263], [45.96183, 33.55751], [45.89801, 33.63661], [45.77814, 33.60938], [45.50261, 33.94968], [45.42789, 33.9458], [45.41077, 33.97421], [45.47264, 34.03099], [45.56176, 34.15088], [45.58667, 34.30147], [45.53552, 34.35148], [45.49171, 34.3439], [45.46697, 34.38221], [45.43879, 34.45949], [45.51883, 34.47692], [45.53219, 34.60441], [45.59074, 34.55558], [45.60224, 34.55057], [45.73923, 34.54416], [45.70031, 34.69277], [45.65672, 34.7222], [45.68284, 34.76624], [45.70031, 34.82322], [45.73641, 34.83975], [45.79682, 34.85133], [45.78904, 34.91135], [45.86532, 34.89858], [45.89477, 34.95805], [45.87864, 35.03441], [45.92173, 35.0465], [45.92203, 35.09538], [45.93108, 35.08148], [45.94756, 35.09188], [46.06508, 35.03699], [46.07747, 35.0838], [46.11763, 35.07551], [46.19116, 35.11097], [46.15642, 35.1268], [46.16229, 35.16984], [46.19738, 35.18536], [46.18457, 35.22561], [46.11367, 35.23729], [46.15474, 35.2883], [46.13152, 35.32548], [46.05358, 35.38568], [45.98453, 35.49848], [46.01518, 35.52012], [45.97584, 35.58132], [46.03028, 35.57416], [46.01307, 35.59756], [46.0165, 35.61501], [45.99452, 35.63574], [46.0117, 35.65059], [46.01631, 35.69139], [46.23736, 35.71414], [46.34166, 35.78363], [46.32921, 35.82655], [46.17198, 35.8013], [46.08325, 35.8581], [45.94711, 35.82218], [45.89784, 35.83708], [45.81442, 35.82107], [45.76145, 35.79898], [45.6645, 35.92872], [45.60018, 35.96069], [45.55245, 35.99943], [45.46594, 36.00042], [45.38275, 35.97156], [45.33916, 35.99424], [45.37652, 36.06222], [45.37312, 36.09917], [45.32235, 36.17383], [45.30038, 36.27769], [45.26261, 36.3001], [45.27394, 36.35846], [45.23953, 36.43257], [45.11811, 36.40751], [45.00759, 36.5402], [45.06985, 36.62645], [45.06985, 36.6814], [45.01537, 36.75128], [44.84725, 36.77622], [44.83479, 36.81362], [44.90173, 36.86096], [44.91199, 36.91468], [44.89862, 37.01897], [44.81611, 37.04383], [44.75229, 37.11958], [44.78319, 37.1431], [44.76698, 37.16162], [44.63179, 37.19229], [44.42631, 37.05825], [44.38117, 37.05825], [44.35315, 37.04955], [44.35937, 37.02843], [44.30645, 36.97373], [44.25975, 36.98119], [44.18503, 37.09551], [44.22239, 37.15756], [44.27998, 37.16501], [44.2613, 37.25055], [44.13521, 37.32486], [44.02002, 37.33229], [43.90949, 37.22453], [43.84878, 37.22205], [43.82699, 37.19477], [43.8052, 37.22825], [43.7009, 37.23692], [43.63085, 37.21957], [43.56702, 37.25675], [43.50787, 37.24436], [43.33508, 37.33105], [43.30083, 37.30629], [43.11403, 37.37436], [42.93705, 37.32015], [42.78887, 37.38615]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "IR",
+           iso1A3: "IRN",
+           iso1N3: "364",
+           wikidata: "Q794",
+           nameEn: "Iran",
+           groups: ["034", "142", "UN"],
+           callingCodes: ["98"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[44.96746, 39.42998], [44.88916, 39.59653], [44.81043, 39.62677], [44.71806, 39.71124], [44.65422, 39.72163], [44.6137, 39.78393], [44.47298, 39.68788], [44.48111, 39.61579], [44.41849, 39.56659], [44.42832, 39.4131], [44.37921, 39.4131], [44.29818, 39.378], [44.22452, 39.4169], [44.03667, 39.39223], [44.1043, 39.19842], [44.20946, 39.13975], [44.18863, 38.93881], [44.30322, 38.81581], [44.26155, 38.71427], [44.28065, 38.6465], [44.32058, 38.62752], [44.3207, 38.49799], [44.3119, 38.37887], [44.38309, 38.36117], [44.44386, 38.38295], [44.50115, 38.33939], [44.42476, 38.25763], [44.22509, 37.88859], [44.3883, 37.85433], [44.45948, 37.77065], [44.55498, 37.783], [44.62096, 37.71985], [44.56887, 37.6429], [44.61401, 37.60165], [44.58449, 37.45018], [44.81021, 37.2915], [44.75986, 37.21549], [44.7868, 37.16644], [44.78319, 37.1431], [44.75229, 37.11958], [44.81611, 37.04383], [44.89862, 37.01897], [44.91199, 36.91468], [44.90173, 36.86096], [44.83479, 36.81362], [44.84725, 36.77622], [45.01537, 36.75128], [45.06985, 36.6814], [45.06985, 36.62645], [45.00759, 36.5402], [45.11811, 36.40751], [45.23953, 36.43257], [45.27394, 36.35846], [45.26261, 36.3001], [45.30038, 36.27769], [45.32235, 36.17383], [45.37312, 36.09917], [45.37652, 36.06222], [45.33916, 35.99424], [45.38275, 35.97156], [45.46594, 36.00042], [45.55245, 35.99943], [45.60018, 35.96069], [45.6645, 35.92872], [45.76145, 35.79898], [45.81442, 35.82107], [45.89784, 35.83708], [45.94711, 35.82218], [46.08325, 35.8581], [46.17198, 35.8013], [46.32921, 35.82655], [46.34166, 35.78363], [46.23736, 35.71414], [46.01631, 35.69139], [46.0117, 35.65059], [45.99452, 35.63574], [46.0165, 35.61501], [46.01307, 35.59756], [46.03028, 35.57416], [45.97584, 35.58132], [46.01518, 35.52012], [45.98453, 35.49848], [46.05358, 35.38568], [46.13152, 35.32548], [46.15474, 35.2883], [46.11367, 35.23729], [46.18457, 35.22561], [46.19738, 35.18536], [46.16229, 35.16984], [46.15642, 35.1268], [46.19116, 35.11097], [46.11763, 35.07551], [46.07747, 35.0838], [46.06508, 35.03699], [45.94756, 35.09188], [45.93108, 35.08148], [45.92203, 35.09538], [45.92173, 35.0465], [45.87864, 35.03441], [45.89477, 34.95805], [45.86532, 34.89858], [45.78904, 34.91135], [45.79682, 34.85133], [45.73641, 34.83975], [45.70031, 34.82322], [45.68284, 34.76624], [45.65672, 34.7222], [45.70031, 34.69277], [45.73923, 34.54416], [45.60224, 34.55057], [45.59074, 34.55558], [45.53219, 34.60441], [45.51883, 34.47692], [45.43879, 34.45949], [45.46697, 34.38221], [45.49171, 34.3439], [45.53552, 34.35148], [45.58667, 34.30147], [45.56176, 34.15088], [45.47264, 34.03099], [45.41077, 33.97421], [45.42789, 33.9458], [45.50261, 33.94968], [45.77814, 33.60938], [45.89801, 33.63661], [45.96183, 33.55751], [45.86687, 33.49263], [45.99919, 33.5082], [46.20623, 33.20395], [46.11905, 33.11924], [46.05367, 33.13097], [46.03966, 33.09577], [46.15175, 33.07229], [46.09103, 32.98354], [46.17198, 32.95612], [46.32298, 32.9731], [46.46788, 32.91992], [47.17218, 32.45393], [47.37529, 32.47808], [47.57144, 32.20583], [47.52474, 32.15972], [47.64771, 32.07666], [47.86337, 31.78422], [47.6804, 31.39086], [47.68219, 31.00004], [48.03221, 30.9967], [48.02443, 30.4789], [48.14585, 30.44133], [48.18321, 30.39703], [48.19425, 30.32796], [48.21279, 30.31644], [48.24385, 30.33846], [48.26393, 30.3408], [48.41117, 30.19846], [48.41671, 30.17254], [48.38714, 30.13485], [48.38869, 30.11062], [48.43384, 30.08233], [48.4494, 30.04456], [48.44785, 30.00148], [48.51011, 29.96238], [48.61441, 29.93675], [48.83867, 29.78572], [49.98877, 27.87827], [50.37726, 27.89227], [54.39838, 25.68383], [55.14145, 25.62624], [55.81777, 26.18798], [56.2644, 26.58649], [56.68954, 26.76645], [56.79239, 26.41236], [56.82555, 25.7713], [56.86325, 25.03856], [61.46682, 24.57869], [61.6433, 25.27541], [61.683, 25.66638], [61.83968, 25.7538], [61.83831, 26.07249], [61.89391, 26.26251], [62.05117, 26.31647], [62.21304, 26.26601], [62.31484, 26.528], [62.77352, 26.64099], [63.1889, 26.65072], [63.18688, 26.83844], [63.25005, 26.84212], [63.25005, 27.08692], [63.32283, 27.14437], [63.19649, 27.25674], [62.80604, 27.22412], [62.79684, 27.34381], [62.84905, 27.47627], [62.7638, 28.02992], [62.79412, 28.28108], [62.59499, 28.24842], [62.40259, 28.42703], [61.93581, 28.55284], [61.65978, 28.77937], [61.53765, 29.00507], [61.31508, 29.38903], [60.87231, 29.86514], [61.80829, 30.84224], [61.78268, 30.92724], [61.8335, 30.97669], [61.83257, 31.0452], [61.80957, 31.12576], [61.80569, 31.16167], [61.70929, 31.37391], [60.84541, 31.49561], [60.86191, 32.22565], [60.56485, 33.12944], [60.88908, 33.50219], [60.91133, 33.55596], [60.69573, 33.56054], [60.57762, 33.59772], [60.5485, 33.73422], [60.5838, 33.80793], [60.50209, 34.13992], [60.66502, 34.31539], [60.91321, 34.30411], [60.72316, 34.52857], [60.99922, 34.63064], [61.00197, 34.70631], [61.06926, 34.82139], [61.12831, 35.09938], [61.0991, 35.27845], [61.18187, 35.30249], [61.27371, 35.61482], [61.22719, 35.67038], [61.26152, 35.80749], [61.22444, 35.92879], [61.12007, 35.95992], [61.22719, 36.12759], [61.1393, 36.38782], [61.18187, 36.55348], [61.14516, 36.64644], [60.34767, 36.63214], [60.00768, 37.04102], [59.74678, 37.12499], [59.55178, 37.13594], [59.39385, 37.34257], [59.39797, 37.47892], [59.33507, 37.53146], [59.22905, 37.51161], [58.9338, 37.67374], [58.6921, 37.64548], [58.5479, 37.70526], [58.47786, 37.6433], [58.39959, 37.63134], [58.22999, 37.6856], [58.21399, 37.77281], [57.79534, 37.89299], [57.35042, 37.98546], [57.37236, 38.09321], [57.21169, 38.28965], [57.03453, 38.18717], [56.73928, 38.27887], [56.62255, 38.24005], [56.43303, 38.26054], [56.32454, 38.18502], [56.33278, 38.08132], [55.97847, 38.08024], [55.76561, 38.12238], [55.44152, 38.08564], [55.13412, 37.94705], [54.851, 37.75739], [54.77684, 37.62264], [54.81804, 37.61285], [54.77822, 37.51597], [54.67247, 37.43532], [54.58664, 37.45809], [54.36211, 37.34912], [54.24565, 37.32047], [53.89734, 37.3464], [48.88288, 38.43975], [48.84969, 38.45015], [48.81072, 38.44853], [48.78979, 38.45026], [48.70001, 38.40564], [48.62217, 38.40198], [48.58793, 38.45076], [48.45084, 38.61013], [48.3146, 38.59958], [48.24773, 38.71883], [48.02581, 38.82705], [48.01409, 38.90333], [48.07734, 38.91616], [48.08627, 38.94434], [48.28437, 38.97186], [48.33884, 39.03022], [48.31239, 39.09278], [48.15361, 39.19419], [48.12404, 39.25208], [48.15984, 39.30028], [48.37385, 39.37584], [48.34264, 39.42935], [47.98977, 39.70999], [47.84774, 39.66285], [47.50099, 39.49615], [47.38978, 39.45999], [47.31301, 39.37492], [47.05927, 39.24846], [47.05771, 39.20143], [46.95341, 39.13505], [46.92539, 39.16644], [46.83822, 39.13143], [46.75752, 39.03231], [46.53497, 38.86548], [46.34059, 38.92076], [46.20601, 38.85262], [46.14785, 38.84206], [46.06766, 38.87861], [46.00228, 38.87376], [45.94624, 38.89072], [45.90266, 38.87739], [45.83883, 38.90768], [45.65172, 38.95199], [45.6155, 38.94304], [45.6131, 38.964], [45.44966, 38.99243], [45.44811, 39.04927], [45.40452, 39.07224], [45.40148, 39.09007], [45.30489, 39.18333], [45.16168, 39.21952], [45.08751, 39.35052], [45.05932, 39.36435], [44.96746, 39.42998]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "IS",
+           iso1A3: "ISL",
+           iso1N3: "352",
+           wikidata: "Q189",
+           nameEn: "Iceland",
+           groups: ["154", "150", "UN"],
+           callingCodes: ["354"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-33.15676, 62.62995], [-8.25539, 63.0423], [-15.70914, 69.67442], [-33.15676, 62.62995]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "IT",
+           iso1A3: "ITA",
+           iso1N3: "380",
+           wikidata: "Q38",
+           nameEn: "Italy",
+           groups: ["EU", "039", "150", "UN"],
+           callingCodes: ["39"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[8.95861, 45.96485], [8.97604, 45.96151], [8.97741, 45.98317], [8.96668, 45.98436], [8.95861, 45.96485]]], [[[7.63035, 43.57419], [9.56115, 43.20816], [10.09675, 41.44089], [7.60802, 41.05927], [7.89009, 38.19924], [11.2718, 37.6713], [12.13667, 34.20326], [14.02721, 36.53141], [17.67657, 35.68918], [18.83516, 40.36999], [16.15283, 42.18525], [13.12821, 44.48877], [13.05142, 45.33128], [13.45644, 45.59464], [13.6076, 45.64761], [13.7198, 45.59352], [13.74587, 45.59811], [13.78445, 45.5825], [13.84106, 45.58185], [13.86771, 45.59898], [13.8695, 45.60835], [13.9191, 45.6322], [13.87933, 45.65207], [13.83422, 45.68703], [13.83332, 45.70855], [13.8235, 45.7176], [13.66986, 45.79955], [13.59784, 45.8072], [13.58858, 45.83503], [13.57563, 45.8425], [13.58644, 45.88173], [13.59565, 45.89446], [13.60857, 45.89907], [13.61931, 45.91782], [13.63815, 45.93607], [13.6329, 45.94894], [13.64307, 45.98326], [13.63458, 45.98947], [13.62074, 45.98388], [13.58903, 45.99009], [13.56759, 45.96991], [13.52963, 45.96588], [13.50104, 45.98078], [13.47474, 46.00546], [13.49702, 46.01832], [13.50998, 46.04498], [13.49568, 46.04839], [13.50104, 46.05986], [13.57072, 46.09022], [13.64053, 46.13587], [13.66472, 46.17392], [13.64451, 46.18966], [13.56682, 46.18703], [13.56114, 46.2054], [13.47587, 46.22725], [13.42218, 46.20758], [13.37671, 46.29668], [13.44808, 46.33507], [13.43418, 46.35992], [13.47019, 46.3621], [13.5763, 46.40915], [13.5763, 46.42613], [13.59777, 46.44137], [13.68684, 46.43881], [13.7148, 46.5222], [13.64088, 46.53438], [13.27627, 46.56059], [12.94445, 46.60401], [12.59992, 46.6595], [12.38708, 46.71529], [12.27591, 46.88651], [12.2006, 46.88854], [12.11675, 47.01241], [12.21781, 47.03996], [12.19254, 47.09331], [11.74789, 46.98484], [11.50739, 47.00644], [11.33355, 46.99862], [11.10618, 46.92966], [11.00764, 46.76896], [10.72974, 46.78972], [10.75753, 46.82258], [10.66405, 46.87614], [10.54783, 46.84505], [10.47197, 46.85698], [10.38659, 46.67847], [10.40475, 46.63671], [10.44686, 46.64162], [10.49375, 46.62049], [10.46136, 46.53164], [10.25309, 46.57432], [10.23674, 46.63484], [10.10307, 46.61003], [10.03715, 46.44479], [10.165, 46.41051], [10.10506, 46.3372], [10.17862, 46.25626], [10.14439, 46.22992], [10.07055, 46.21668], [9.95249, 46.38045], [9.73086, 46.35071], [9.71273, 46.29266], [9.57015, 46.2958], [9.46117, 46.37481], [9.45936, 46.50873], [9.40487, 46.46621], [9.36128, 46.5081], [9.28136, 46.49685], [9.25502, 46.43743], [9.29226, 46.32717], [9.24503, 46.23616], [9.01618, 46.04928], [8.99257, 45.9698], [9.09065, 45.89906], [9.06642, 45.8761], [9.04546, 45.84968], [9.04059, 45.8464], [9.03505, 45.83976], [9.03793, 45.83548], [9.03279, 45.82865], [9.0298, 45.82127], [9.00324, 45.82055], [8.99663, 45.83466], [8.9621, 45.83707], [8.94737, 45.84285], [8.91129, 45.8388], [8.93504, 45.86245], [8.94372, 45.86587], [8.93649, 45.86775], [8.88904, 45.95465], [8.86688, 45.96135], [8.85121, 45.97239], [8.8319, 45.9879], [8.79362, 45.99207], [8.78585, 45.98973], [8.79414, 46.00913], [8.85617, 46.0748], [8.80778, 46.10085], [8.75697, 46.10395], [8.62242, 46.12112], [8.45032, 46.26869], [8.46317, 46.43712], [8.42464, 46.46367], [8.30648, 46.41587], [8.31162, 46.38044], [8.08814, 46.26692], [8.16866, 46.17817], [8.11383, 46.11577], [8.02906, 46.10331], [7.98881, 45.99867], [7.9049, 45.99945], [7.85949, 45.91485], [7.56343, 45.97421], [7.10685, 45.85653], [7.04151, 45.92435], [6.95315, 45.85163], [6.80785, 45.83265], [6.80785, 45.71864], [6.98948, 45.63869], [7.00037, 45.509], [7.18019, 45.40071], [7.10572, 45.32924], [7.13115, 45.25386], [7.07074, 45.21228], [6.96706, 45.20841], [6.85144, 45.13226], [6.7697, 45.16044], [6.62803, 45.11175], [6.66981, 45.02324], [6.74791, 45.01939], [6.74519, 44.93661], [6.75518, 44.89915], [6.90774, 44.84322], [6.93499, 44.8664], [7.02217, 44.82519], [7.00401, 44.78782], [7.07484, 44.68073], [7.00582, 44.69364], [6.95133, 44.66264], [6.96042, 44.62129], [6.85507, 44.53072], [6.86233, 44.49834], [6.94504, 44.43112], [6.88784, 44.42043], [6.89171, 44.36637], [6.98221, 44.28289], [7.00764, 44.23736], [7.16929, 44.20352], [7.27827, 44.1462], [7.34547, 44.14359], [7.36364, 44.11882], [7.62155, 44.14881], [7.63245, 44.17877], [7.68694, 44.17487], [7.66878, 44.12795], [7.72508, 44.07578], [7.6597, 44.03009], [7.66848, 43.99943], [7.65266, 43.9763], [7.60771, 43.95772], [7.56858, 43.94506], [7.56075, 43.89932], [7.51162, 43.88301], [7.49355, 43.86551], [7.50423, 43.84345], [7.53006, 43.78405], [7.63035, 43.57419]], [[12.45181, 41.90056], [12.44834, 41.90095], [12.44582, 41.90194], [12.44815, 41.90326], [12.44984, 41.90545], [12.45091, 41.90625], [12.45543, 41.90738], [12.45561, 41.90629], [12.45762, 41.9058], [12.45755, 41.9033], [12.45826, 41.90281], [12.45834, 41.90174], [12.4577, 41.90115], [12.45691, 41.90125], [12.45626, 41.90172], [12.45435, 41.90143], [12.45446, 41.90028], [12.45181, 41.90056]], [[12.45648, 43.89369], [12.44184, 43.90498], [12.41641, 43.89991], [12.40935, 43.9024], [12.41233, 43.90956], [12.40733, 43.92379], [12.41551, 43.92984], [12.41165, 43.93769], [12.40506, 43.94325], [12.40415, 43.95485], [12.41414, 43.95273], [12.42005, 43.9578], [12.43662, 43.95698], [12.44684, 43.96597], [12.46205, 43.97463], [12.47853, 43.98052], [12.49406, 43.98492], [12.50678, 43.99113], [12.51463, 43.99122], [12.5154, 43.98508], [12.51064, 43.98165], [12.51109, 43.97201], [12.50622, 43.97131], [12.50875, 43.96198], [12.50655, 43.95796], [12.51427, 43.94897], [12.51553, 43.94096], [12.50496, 43.93017], [12.50269, 43.92363], [12.49724, 43.92248], [12.49247, 43.91774], [12.49429, 43.90973], [12.48771, 43.89706], [12.45648, 43.89369]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "JE",
+           iso1A3: "JEY",
+           iso1N3: "832",
+           wikidata: "Q785",
+           nameEn: "Bailiwick of Jersey",
+           country: "GB",
+           groups: ["830", "Q185086", "154", "150", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["44 01534"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-2.00491, 48.86706], [-1.83944, 49.23037], [-2.09454, 49.46288], [-2.65349, 49.15373], [-2.00491, 48.86706]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "JM",
+           iso1A3: "JAM",
+           iso1N3: "388",
+           wikidata: "Q766",
+           nameEn: "Jamaica",
+           aliases: ["JA"],
+           groups: ["029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           callingCodes: ["1 876", "1 658"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-74.09729, 17.36817], [-78.9741, 19.59515], [-78.34606, 16.57862], [-74.09729, 17.36817]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "JO",
+           iso1A3: "JOR",
+           iso1N3: "400",
+           wikidata: "Q810",
+           nameEn: "Jordan",
+           groups: ["145", "142", "UN"],
+           callingCodes: ["962"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[39.04251, 32.30203], [38.98762, 32.47694], [39.08202, 32.50304], [38.79171, 33.37328], [36.83946, 32.31293], [36.40959, 32.37908], [36.23948, 32.50108], [36.20875, 32.49529], [36.20379, 32.52751], [36.08074, 32.51463], [36.02239, 32.65911], [35.96633, 32.66237], [35.93307, 32.71966], [35.88405, 32.71321], [35.75983, 32.74803], [35.68467, 32.70715], [35.66527, 32.681], [35.61669, 32.67999], [35.59813, 32.65159], [35.56614, 32.64393], [35.57485, 32.48669], [35.55494, 32.42687], [35.55807, 32.38674], [35.57111, 32.21877], [35.52012, 32.04076], [35.54375, 31.96587], [35.52758, 31.9131], [35.55941, 31.76535], [35.47672, 31.49578], [35.40316, 31.25535], [35.43658, 31.12444], [35.41371, 30.95565], [35.33984, 30.8802], [35.33456, 30.81224], [35.29311, 30.71365], [35.21379, 30.60401], [35.19595, 30.50297], [35.16218, 30.43535], [35.19183, 30.34636], [35.14108, 30.07374], [35.02147, 29.66343], [34.98207, 29.58147], [34.97718, 29.54294], [34.92298, 29.45305], [34.8812, 29.36878], [36.07081, 29.18469], [36.50005, 29.49696], [36.75083, 29.86903], [37.4971, 29.99949], [37.66395, 30.33245], [37.99354, 30.49998], [36.99791, 31.50081], [38.99233, 31.99721], [39.29903, 32.23259], [39.26157, 32.35555], [39.04251, 32.30203]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "JP",
+           iso1A3: "JPN",
+           iso1N3: "392",
+           wikidata: "Q17",
+           nameEn: "Japan",
+           groups: ["030", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["81"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[145.82361, 43.38904], [145.23667, 43.76813], [145.82343, 44.571], [140.9182, 45.92937], [133.61399, 37.41], [129.2669, 34.87122], [122.26612, 25.98197], [123.92912, 17.8782], [155.16731, 23.60141], [145.82361, 43.38904]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "KE",
+           iso1A3: "KEN",
+           iso1N3: "404",
+           wikidata: "Q114",
+           nameEn: "Kenya",
+           groups: ["014", "202", "002", "UN"],
+           driveSide: "left",
+           callingCodes: ["254"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[35.9419, 4.61933], [35.51424, 4.61643], [35.42366, 4.76969], [35.47843, 4.91872], [35.30992, 4.90402], [35.34151, 5.02364], [34.47601, 4.72162], [33.9873, 4.23316], [34.06046, 4.15235], [34.15429, 3.80464], [34.45815, 3.67385], [34.44922, 3.51627], [34.39112, 3.48802], [34.41794, 3.44342], [34.40006, 3.37949], [34.45815, 3.18319], [34.56242, 3.11478], [34.60114, 2.93034], [34.65774, 2.8753], [34.73967, 2.85447], [34.78137, 2.76223], [34.77244, 2.70272], [34.95267, 2.47209], [34.90947, 2.42447], [34.98692, 1.97348], [34.9899, 1.6668], [34.92734, 1.56109], [34.87819, 1.5596], [34.7918, 1.36752], [34.82606, 1.30944], [34.82606, 1.26626], [34.80223, 1.22754], [34.67562, 1.21265], [34.58029, 1.14712], [34.57427, 1.09868], [34.52369, 1.10692], [34.43349, 0.85254], [34.40041, 0.80266], [34.31516, 0.75693], [34.27345, 0.63182], [34.20196, 0.62289], [34.13493, 0.58118], [34.11408, 0.48884], [34.08727, 0.44713], [34.10067, 0.36372], [33.90936, 0.10581], [33.98449, -0.13079], [33.9264, -0.54188], [33.93107, -0.99298], [34.02286, -1.00779], [34.03084, -1.05101], [34.0824, -1.02264], [37.67199, -3.06222], [37.71745, -3.304], [37.5903, -3.42735], [37.63099, -3.50723], [37.75036, -3.54243], [37.81321, -3.69179], [39.21631, -4.67835], [39.44306, -4.93877], [39.62121, -4.68136], [41.75542, -1.85308], [41.56362, -1.66375], [41.56, -1.59812], [41.00099, -0.83068], [40.98767, 2.82959], [41.31368, 3.14314], [41.89488, 3.97375], [41.1754, 3.94079], [40.77498, 4.27683], [39.86043, 3.86974], [39.76808, 3.67058], [39.58339, 3.47434], [39.55132, 3.39634], [39.51551, 3.40895], [39.49444, 3.45521], [39.19954, 3.47834], [39.07736, 3.5267], [38.91938, 3.51198], [38.52336, 3.62551], [38.45812, 3.60445], [38.14168, 3.62487], [37.07724, 4.33503], [36.84474, 4.44518], [36.03924, 4.44406], [35.95449, 4.53244], [35.9419, 4.61933]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "KG",
+           iso1A3: "KGZ",
+           iso1N3: "417",
+           wikidata: "Q813",
+           nameEn: "Kyrgyzstan",
+           groups: ["143", "142", "UN"],
+           callingCodes: ["996"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[74.88756, 42.98612], [74.75, 42.99029], [74.70331, 43.02519], [74.64615, 43.05881], [74.57491, 43.13702], [74.22489, 43.24657], [73.55634, 43.03071], [73.50992, 42.82356], [73.44393, 42.43098], [71.88792, 42.83578], [71.62405, 42.76613], [71.53272, 42.8014], [71.2724, 42.77853], [71.22785, 42.69248], [71.17807, 42.67381], [71.15232, 42.60486], [70.97717, 42.50147], [70.85973, 42.30188], [70.94483, 42.26238], [71.13263, 42.28356], [71.28719, 42.18033], [70.69777, 41.92554], [70.17682, 41.5455], [70.48909, 41.40335], [70.67586, 41.47953], [70.78572, 41.36419], [70.77885, 41.24813], [70.86263, 41.23833], [70.9615, 41.16393], [71.02193, 41.19494], [71.11806, 41.15359], [71.25813, 41.18796], [71.27187, 41.11015], [71.34877, 41.16807], [71.40198, 41.09436], [71.46148, 41.13958], [71.43814, 41.19644], [71.46688, 41.31883], [71.57227, 41.29175], [71.6787, 41.42111], [71.65914, 41.49599], [71.73054, 41.54713], [71.71132, 41.43012], [71.76625, 41.4466], [71.83914, 41.3546], [71.91457, 41.2982], [71.85964, 41.19081], [72.07249, 41.11739], [72.10745, 41.15483], [72.16433, 41.16483], [72.17594, 41.15522], [72.14864, 41.13363], [72.1792, 41.10621], [72.21061, 41.05607], [72.17594, 41.02377], [72.18339, 40.99571], [72.324, 41.03381], [72.34026, 41.04539], [72.34757, 41.06104], [72.36138, 41.04384], [72.38511, 41.02785], [72.45206, 41.03018], [72.48742, 40.97136], [72.55109, 40.96046], [72.59136, 40.86947], [72.68157, 40.84942], [72.84291, 40.85512], [72.94454, 40.8094], [73.01869, 40.84681], [73.13267, 40.83512], [73.13412, 40.79122], [73.0612, 40.76678], [72.99133, 40.76457], [72.93296, 40.73089], [72.8722, 40.71111], [72.85372, 40.7116], [72.84754, 40.67229], [72.80137, 40.67856], [72.74866, 40.60873], [72.74894, 40.59592], [72.75982, 40.57273], [72.74862, 40.57131], [72.74768, 40.58051], [72.73995, 40.58409], [72.69579, 40.59778], [72.66713, 40.59076], [72.66713, 40.5219], [72.47795, 40.5532], [72.40517, 40.61917], [72.34406, 40.60144], [72.41714, 40.55736], [72.38384, 40.51535], [72.41513, 40.50856], [72.44191, 40.48222], [72.40346, 40.4007], [72.24368, 40.46091], [72.18648, 40.49893], [71.96401, 40.31907], [72.05464, 40.27586], [71.85002, 40.25647], [71.82646, 40.21872], [71.73054, 40.14818], [71.71719, 40.17886], [71.69621, 40.18492], [71.70569, 40.20391], [71.68386, 40.26984], [71.61931, 40.26775], [71.61725, 40.20615], [71.51549, 40.22986], [71.51215, 40.26943], [71.4246, 40.28619], [71.36663, 40.31593], [71.13042, 40.34106], [71.05901, 40.28765], [70.95789, 40.28761], [70.9818, 40.22392], [70.80495, 40.16813], [70.7928, 40.12797], [70.65827, 40.0981], [70.65946, 39.9878], [70.58912, 39.95211], [70.55033, 39.96619], [70.47557, 39.93216], [70.57384, 39.99394], [70.58297, 40.00891], [70.01283, 40.23288], [69.67001, 40.10639], [69.64704, 40.12165], [69.57615, 40.10524], [69.55555, 40.12296], [69.53794, 40.11833], [69.53855, 40.0887], [69.5057, 40.03277], [69.53615, 39.93991], [69.43557, 39.92877], [69.43134, 39.98431], [69.35649, 40.01994], [69.26938, 39.8127], [69.3594, 39.52516], [69.68677, 39.59281], [69.87491, 39.53882], [70.11111, 39.58223], [70.2869, 39.53141], [70.44757, 39.60128], [70.64087, 39.58792], [70.7854, 39.38933], [71.06418, 39.41586], [71.08752, 39.50704], [71.49814, 39.61397], [71.55856, 39.57588], [71.5517, 39.45722], [71.62688, 39.44056], [71.76816, 39.45456], [71.80164, 39.40631], [71.7522, 39.32031], [71.79202, 39.27355], [71.90601, 39.27674], [72.04059, 39.36704], [72.09689, 39.26823], [72.17242, 39.2661], [72.23834, 39.17248], [72.33173, 39.33093], [72.62027, 39.39696], [72.85934, 39.35116], [73.18454, 39.35536], [73.31912, 39.38615], [73.45096, 39.46677], [73.59831, 39.46425], [73.87018, 39.47879], [73.94683, 39.60733], [73.92354, 39.69565], [73.9051, 39.75073], [73.83006, 39.76136], [73.97049, 40.04378], [74.25533, 40.13191], [74.35063, 40.09742], [74.69875, 40.34668], [74.85996, 40.32857], [74.78168, 40.44886], [74.82013, 40.52197], [75.08243, 40.43945], [75.22834, 40.45382], [75.5854, 40.66874], [75.69663, 40.28642], [75.91361, 40.2948], [75.96168, 40.38064], [76.33659, 40.3482], [76.5261, 40.46114], [76.75681, 40.95354], [76.99302, 41.0696], [77.28004, 41.0033], [77.3693, 41.0375], [77.52723, 41.00227], [77.76206, 41.01574], [77.81287, 41.14307], [78.12873, 41.23091], [78.15757, 41.38565], [78.3732, 41.39603], [79.92977, 42.04113], [80.17842, 42.03211], [80.17807, 42.21166], [79.97364, 42.42816], [79.52921, 42.44778], [79.19763, 42.804], [78.91502, 42.76839], [78.48469, 42.89649], [75.82823, 42.94848], [75.72174, 42.79672], [75.29966, 42.86183], [75.22619, 42.85528], [74.88756, 42.98612]], [[70.74189, 39.86319], [70.63105, 39.77923], [70.59667, 39.83542], [70.54998, 39.85137], [70.52631, 39.86989], [70.53651, 39.89155], [70.74189, 39.86319]], [[71.86463, 39.98598], [71.84316, 39.95582], [71.7504, 39.93701], [71.71511, 39.96348], [71.78838, 40.01404], [71.86463, 39.98598]], [[71.21139, 40.03369], [71.1427, 39.95026], [71.23067, 39.93581], [71.16101, 39.88423], [71.10531, 39.91354], [71.04979, 39.89808], [71.10501, 39.95568], [71.09063, 39.99], [71.11668, 39.99291], [71.11037, 40.01984], [71.01035, 40.05481], [71.00236, 40.18154], [71.06305, 40.1771], [71.12218, 40.03052], [71.21139, 40.03369]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "KH",
+           iso1A3: "KHM",
+           iso1N3: "116",
+           wikidata: "Q424",
+           nameEn: "Cambodia",
+           groups: ["035", "142", "UN"],
+           callingCodes: ["855"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[105.87328, 11.55953], [105.81645, 11.56876], [105.80867, 11.60536], [105.8507, 11.66635], [105.88962, 11.67854], [105.95188, 11.63738], [106.00792, 11.7197], [106.02038, 11.77457], [106.06708, 11.77761], [106.13158, 11.73283], [106.18539, 11.75171], [106.26478, 11.72122], [106.30525, 11.67549], [106.37219, 11.69836], [106.44691, 11.66787], [106.45158, 11.68616], [106.41577, 11.76999], [106.44535, 11.8279], [106.44068, 11.86294], [106.4687, 11.86751], [106.4111, 11.97413], [106.70687, 11.96956], [106.79405, 12.0807], [106.92325, 12.06548], [106.99953, 12.08983], [107.15831, 12.27547], [107.34511, 12.33327], [107.42917, 12.24657], [107.4463, 12.29373], [107.55059, 12.36824], [107.5755, 12.52177], [107.55993, 12.7982], [107.49611, 12.88926], [107.49144, 13.01215], [107.62843, 13.3668], [107.61909, 13.52577], [107.53503, 13.73908], [107.45252, 13.78897], [107.46498, 13.91593], [107.44318, 13.99751], [107.38247, 13.99147], [107.35757, 14.02319], [107.37158, 14.07906], [107.33577, 14.11832], [107.40427, 14.24509], [107.39493, 14.32655], [107.44941, 14.41552], [107.48521, 14.40346], [107.52569, 14.54665], [107.52102, 14.59034], [107.55371, 14.628], [107.54361, 14.69092], [107.47238, 14.61523], [107.44435, 14.52785], [107.37897, 14.54443], [107.3276, 14.58812], [107.29803, 14.58963], [107.26534, 14.54292], [107.256, 14.48716], [107.21241, 14.48716], [107.17038, 14.41782], [107.09722, 14.3937], [107.03962, 14.45099], [107.04585, 14.41782], [106.98825, 14.36806], [106.9649, 14.3198], [106.90574, 14.33639], [106.8497, 14.29416], [106.80767, 14.31226], [106.73762, 14.42687], [106.63333, 14.44194], [106.59908, 14.50977], [106.57106, 14.50525], [106.54148, 14.59565], [106.50723, 14.58963], [106.45898, 14.55045], [106.47766, 14.50977], [106.43874, 14.52032], [106.40916, 14.45249], [106.32355, 14.44043], [106.25194, 14.48415], [106.21302, 14.36203], [106.00131, 14.36957], [105.99509, 14.32734], [106.02311, 14.30623], [106.04801, 14.20363], [106.10872, 14.18401], [106.11962, 14.11307], [106.18656, 14.06324], [106.16632, 14.01794], [106.10094, 13.98471], [106.10405, 13.9137], [105.90791, 13.92881], [105.78182, 14.02247], [105.78338, 14.08438], [105.5561, 14.15684], [105.44869, 14.10703], [105.36775, 14.09948], [105.2759, 14.17496], [105.20894, 14.34967], [105.17748, 14.34432], [105.14012, 14.23873], [105.08408, 14.20402], [105.02804, 14.23722], [104.97667, 14.38806], [104.69335, 14.42726], [104.55014, 14.36091], [104.27616, 14.39861], [103.93836, 14.3398], [103.70175, 14.38052], [103.71109, 14.4348], [103.53518, 14.42575], [103.39353, 14.35639], [103.16469, 14.33075], [102.93275, 14.19044], [102.91251, 14.01531], [102.77864, 13.93374], [102.72727, 13.77806], [102.56848, 13.69366], [102.5481, 13.6589], [102.58635, 13.6286], [102.62483, 13.60883], [102.57573, 13.60461], [102.5358, 13.56933], [102.44601, 13.5637], [102.36859, 13.57488], [102.33828, 13.55613], [102.361, 13.50551], [102.35563, 13.47307], [102.35692, 13.38274], [102.34611, 13.35618], [102.36001, 13.31142], [102.36146, 13.26006], [102.43422, 13.09061], [102.46011, 13.08057], [102.52275, 12.99813], [102.48694, 12.97537], [102.49335, 12.92711], [102.53053, 12.77506], [102.4994, 12.71736], [102.51963, 12.66117], [102.57567, 12.65358], [102.7796, 12.43781], [102.78116, 12.40284], [102.73134, 12.37091], [102.70176, 12.1686], [102.77026, 12.06815], [102.78427, 11.98746], [102.83957, 11.8519], [102.90973, 11.75613], [102.91449, 11.65512], [102.52395, 11.25257], [102.47649, 9.66162], [103.99198, 10.48391], [104.43778, 10.42386], [104.47963, 10.43046], [104.49869, 10.4057], [104.59018, 10.53073], [104.87933, 10.52833], [104.95094, 10.64003], [105.09571, 10.72722], [105.02722, 10.89236], [105.08326, 10.95656], [105.11449, 10.96332], [105.34011, 10.86179], [105.42884, 10.96878], [105.50045, 10.94586], [105.77751, 11.03671], [105.86376, 10.89839], [105.84603, 10.85873], [105.93403, 10.83853], [105.94535, 10.9168], [106.06708, 10.8098], [106.18539, 10.79451], [106.14301, 10.98176], [106.20095, 10.97795], [106.1757, 11.07301], [106.1527, 11.10476], [106.10444, 11.07879], [105.86782, 11.28343], [105.88962, 11.43605], [105.87328, 11.55953]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "KI",
+           iso1A3: "KIR",
+           iso1N3: "296",
+           wikidata: "Q710",
+           nameEn: "Kiribati",
+           groups: ["057", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["686"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[169, 3.9], [169, -3.5], [178, -3.5], [178, 3.9], [169, 3.9]]], [[[-161.06795, 5.2462], [-158.12991, -1.86122], [-175.33482, -1.40631], [-175.31804, -7.54825], [-156.50903, -7.4975], [-156.48634, -15.52824], [-135.59706, -4.70473], [-161.06795, 5.2462]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "KM",
+           iso1A3: "COM",
+           iso1N3: "174",
+           wikidata: "Q970",
+           nameEn: "Comoros",
+           groups: ["014", "202", "002", "UN"],
+           callingCodes: ["269"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[42.63904, -10.02522], [43.28731, -13.97126], [45.4971, -11.75965], [42.63904, -10.02522]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "KN",
+           iso1A3: "KNA",
+           iso1N3: "659",
+           wikidata: "Q763",
+           nameEn: "St. Kitts and Nevis",
+           groups: ["029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           callingCodes: ["1 869"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-62.29333, 17.43155], [-62.76692, 17.64353], [-63.09677, 17.21372], [-62.63813, 16.65446], [-62.29333, 17.43155]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "KP",
+           iso1A3: "PRK",
+           iso1N3: "408",
+           wikidata: "Q423",
+           nameEn: "North Korea",
+           groups: ["030", "142", "UN"],
+           callingCodes: ["850"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[130.26095, 42.9027], [130.09764, 42.91425], [130.12957, 42.98361], [129.96409, 42.97306], [129.95082, 43.01051], [129.8865, 43.00395], [129.85261, 42.96494], [129.83277, 42.86746], [129.80719, 42.79218], [129.7835, 42.76521], [129.77183, 42.69435], [129.75294, 42.59409], [129.72541, 42.43739], [129.60482, 42.44461], [129.54701, 42.37254], [129.42882, 42.44702], [129.28541, 42.41574], [129.22423, 42.3553], [129.22285, 42.26491], [129.15178, 42.17224], [128.96068, 42.06657], [128.94007, 42.03537], [128.04487, 42.01769], [128.15119, 41.74568], [128.30716, 41.60322], [128.20061, 41.40895], [128.18546, 41.41279], [128.12967, 41.37931], [128.03311, 41.39232], [128.02633, 41.42103], [127.92943, 41.44291], [127.29712, 41.49473], [127.17841, 41.59714], [126.90729, 41.79955], [126.60631, 41.65565], [126.53189, 41.35206], [126.242, 41.15454], [126.00335, 40.92835], [125.76869, 40.87908], [125.71172, 40.85223], [124.86913, 40.45387], [124.40719, 40.13655], [124.38556, 40.11047], [124.3322, 40.05573], [124.37089, 40.03004], [124.35029, 39.95639], [124.23201, 39.9248], [124.17532, 39.8232], [123.90497, 38.79949], [123.85601, 37.49093], [124.67666, 38.05679], [124.84224, 37.977], [124.87921, 37.80827], [125.06408, 37.66334], [125.37112, 37.62643], [125.81159, 37.72949], [126.13074, 37.70512], [126.18776, 37.74728], [126.19097, 37.81462], [126.24402, 37.83113], [126.43239, 37.84095], [126.46818, 37.80873], [126.56709, 37.76857], [126.59918, 37.76364], [126.66067, 37.7897], [126.68793, 37.83728], [126.68793, 37.9175], [126.67023, 37.95852], [126.84961, 38.0344], [126.88106, 38.10246], [126.95887, 38.1347], [126.95338, 38.17735], [127.04479, 38.25518], [127.15749, 38.30722], [127.38727, 38.33227], [127.49672, 38.30647], [127.55013, 38.32257], [128.02917, 38.31861], [128.27652, 38.41657], [128.31105, 38.58462], [128.37487, 38.62345], [128.65655, 38.61914], [131.95041, 41.5445], [130.65022, 42.32281], [130.66367, 42.38024], [130.64181, 42.41422], [130.60805, 42.4317], [130.56835, 42.43281], [130.55143, 42.52158], [130.50123, 42.61636], [130.44361, 42.54849], [130.41826, 42.6011], [130.2385, 42.71127], [130.23068, 42.80125], [130.26095, 42.9027]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "KR",
+           iso1A3: "KOR",
+           iso1N3: "410",
+           wikidata: "Q884",
+           nameEn: "South Korea",
+           groups: ["030", "142", "UN"],
+           callingCodes: ["82"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[133.11729, 37.53115], [128.65655, 38.61914], [128.37487, 38.62345], [128.31105, 38.58462], [128.27652, 38.41657], [128.02917, 38.31861], [127.55013, 38.32257], [127.49672, 38.30647], [127.38727, 38.33227], [127.15749, 38.30722], [127.04479, 38.25518], [126.95338, 38.17735], [126.95887, 38.1347], [126.88106, 38.10246], [126.84961, 38.0344], [126.67023, 37.95852], [126.68793, 37.9175], [126.68793, 37.83728], [126.66067, 37.7897], [126.59918, 37.76364], [126.56709, 37.76857], [126.46818, 37.80873], [126.43239, 37.84095], [126.24402, 37.83113], [126.19097, 37.81462], [126.18776, 37.74728], [126.13074, 37.70512], [125.81159, 37.72949], [125.37112, 37.62643], [125.06408, 37.66334], [124.87921, 37.80827], [124.84224, 37.977], [124.67666, 38.05679], [123.85601, 37.49093], [122.80525, 33.30571], [125.99728, 32.63328], [129.2669, 34.87122], [133.11729, 37.53115]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "KW",
+           iso1A3: "KWT",
+           iso1N3: "414",
+           wikidata: "Q817",
+           nameEn: "Kuwait",
+           groups: ["145", "142", "UN"],
+           callingCodes: ["965"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[49.00421, 28.81495], [48.59531, 29.66815], [48.40479, 29.85763], [48.17332, 30.02448], [48.06782, 30.02906], [48.01114, 29.98906], [47.7095, 30.10453], [47.37192, 30.10421], [47.15166, 30.01044], [46.89695, 29.50584], [46.5527, 29.10283], [47.46202, 29.0014], [47.58376, 28.83382], [47.59863, 28.66798], [47.70561, 28.5221], [48.42991, 28.53628], [49.00421, 28.81495]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "KY",
+           iso1A3: "CYM",
+           iso1N3: "136",
+           wikidata: "Q5785",
+           nameEn: "Cayman Islands",
+           country: "GB",
+           groups: ["BOTS", "029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["1 345"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-82.11509, 19.60401], [-80.36068, 18.11751], [-79.32727, 20.06742], [-82.11509, 19.60401]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "KZ",
+           iso1A3: "KAZ",
+           iso1N3: "398",
+           wikidata: "Q232",
+           nameEn: "Kazakhstan",
+           groups: ["143", "142", "UN"],
+           callingCodes: ["7"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[68.90865, 55.38148], [68.19206, 55.18823], [68.26661, 55.09226], [68.21308, 54.98645], [65.20174, 54.55216], [65.24663, 54.35721], [65.11033, 54.33028], [64.97216, 54.4212], [63.97686, 54.29763], [64.02715, 54.22679], [63.91224, 54.20013], [63.80604, 54.27079], [62.58651, 54.05871], [62.56876, 53.94047], [62.45931, 53.90737], [62.38535, 54.03961], [62.00966, 54.04134], [62.03913, 53.94768], [61.65318, 54.02445], [61.56941, 53.95703], [61.47603, 54.08048], [61.3706, 54.08464], [61.26863, 53.92797], [60.99796, 53.93699], [61.14283, 53.90063], [61.22574, 53.80268], [60.90626, 53.62937], [61.55706, 53.57144], [61.57185, 53.50112], [61.37957, 53.45887], [61.29082, 53.50992], [61.14291, 53.41481], [61.19024, 53.30536], [62.14574, 53.09626], [62.12799, 52.99133], [62.0422, 52.96105], [61.23462, 53.03227], [61.05842, 52.92217], [60.71989, 52.75923], [60.71693, 52.66245], [60.84118, 52.63912], [60.84709, 52.52228], [60.98021, 52.50068], [61.05417, 52.35096], [60.78201, 52.22067], [60.72581, 52.15538], [60.48915, 52.15175], [60.19925, 51.99173], [59.99809, 51.98263], [60.09867, 51.87135], [60.50986, 51.7964], [60.36787, 51.66815], [60.5424, 51.61675], [60.92401, 51.61124], [60.95655, 51.48615], [61.50677, 51.40687], [61.55114, 51.32746], [61.6813, 51.25716], [61.56889, 51.23679], [61.4431, 50.80679], [60.81833, 50.6629], [60.31914, 50.67705], [60.17262, 50.83312], [60.01288, 50.8163], [59.81172, 50.54451], [59.51886, 50.49937], [59.48928, 50.64216], [58.87974, 50.70852], [58.3208, 51.15151], [57.75578, 51.13852], [57.74986, 50.93017], [57.44221, 50.88354], [57.17302, 51.11253], [56.17906, 50.93204], [56.11398, 50.7471], [55.67774, 50.54508], [54.72067, 51.03261], [54.56685, 51.01958], [54.71476, 50.61214], [54.55797, 50.52006], [54.41894, 50.61214], [54.46331, 50.85554], [54.12248, 51.11542], [53.69299, 51.23466], [53.46165, 51.49445], [52.54329, 51.48444], [52.36119, 51.74161], [51.8246, 51.67916], [51.77431, 51.49536], [51.301, 51.48799], [51.26254, 51.68466], [50.59695, 51.61859], [50.26859, 51.28677], [49.97277, 51.2405], [49.76866, 51.11067], [49.39001, 51.09396], [49.41959, 50.85927], [49.12673, 50.78639], [48.86936, 50.61589], [48.57946, 50.63278], [48.90782, 50.02281], [48.68352, 49.89546], [48.42564, 49.82283], [48.24519, 49.86099], [48.10044, 50.09242], [47.58551, 50.47867], [47.30448, 50.30894], [47.34589, 50.09308], [47.18319, 49.93721], [46.9078, 49.86707], [46.78398, 49.34026], [47.04658, 49.19834], [47.00857, 49.04921], [46.78392, 48.95352], [46.49011, 48.43019], [47.11516, 48.27188], [47.12107, 47.83687], [47.38731, 47.68176], [47.41689, 47.83687], [47.64973, 47.76559], [48.15348, 47.74545], [48.45173, 47.40818], [48.52326, 47.4102], [49.01136, 46.72716], [48.51142, 46.69268], [48.54988, 46.56267], [49.16518, 46.38542], [49.32259, 46.26944], [49.88945, 46.04554], [49.2134, 44.84989], [52.26048, 41.69249], [52.47884, 41.78034], [52.97575, 42.1308], [54.20635, 42.38477], [54.95182, 41.92424], [55.45471, 41.25609], [56.00314, 41.32584], [55.97584, 44.99322], [55.97584, 44.99328], [55.97584, 44.99338], [55.97584, 44.99343], [55.97584, 44.99348], [55.97584, 44.99353], [55.97584, 44.99359], [55.97584, 44.99369], [55.97584, 44.99374], [55.97584, 44.99384], [55.97584, 44.9939], [55.97584, 44.994], [55.97584, 44.99405], [55.97584, 44.99415], [55.97584, 44.99421], [55.97584, 44.99426], [55.97584, 44.99431], [55.97584, 44.99436], [55.97584, 44.99441], [55.97594, 44.99446], [55.97605, 44.99452], [55.97605, 44.99457], [55.97605, 44.99462], [55.97605, 44.99467], [55.97605, 44.99477], [55.97615, 44.99477], [55.97615, 44.99483], [55.97615, 44.99493], [55.97615, 44.99498], [55.97615, 44.99503], [55.97615, 44.99508], [55.97625, 44.99514], [55.97636, 44.99519], [55.97636, 44.99524], [55.97646, 44.99529], [55.97646, 44.99534], [55.97656, 44.99539], [55.97667, 44.99545], [55.97677, 44.9955], [55.97677, 44.99555], [55.97677, 44.9956], [55.97687, 44.9956], [55.97698, 44.99565], [55.97698, 44.9957], [55.97708, 44.99576], [55.97718, 44.99581], [55.97729, 44.99586], [55.97739, 44.99586], [55.97739, 44.99591], [55.97749, 44.99591], [55.9776, 44.99591], [55.9777, 44.99596], [55.9777, 44.99601], [55.9778, 44.99607], [55.97791, 44.99607], [55.97801, 44.99607], [55.97801, 44.99612], [55.97811, 44.99617], [55.97822, 44.99617], [55.97832, 44.99622], [55.97842, 44.99622], [58.59711, 45.58671], [61.01475, 44.41383], [62.01711, 43.51008], [63.34656, 43.64003], [64.53885, 43.56941], [64.96464, 43.74748], [65.18666, 43.48835], [65.53277, 43.31856], [65.85194, 42.85481], [66.09482, 42.93426], [66.00546, 41.94455], [66.53302, 41.87388], [66.69129, 41.1311], [67.9644, 41.14611], [67.98511, 41.02794], [68.08273, 41.08148], [68.1271, 41.0324], [67.96736, 40.83798], [68.49983, 40.56437], [68.63, 40.59358], [68.58444, 40.91447], [68.49983, 40.99669], [68.62221, 41.03019], [68.65662, 40.93861], [68.73945, 40.96989], [68.7217, 41.05025], [69.01308, 41.22804], [69.05006, 41.36183], [69.15137, 41.43078], [69.17701, 41.43769], [69.18528, 41.45175], [69.20439, 41.45391], [69.22671, 41.46298], [69.23332, 41.45847], [69.25059, 41.46693], [69.29778, 41.43673], [69.35554, 41.47211], [69.37468, 41.46555], [69.45081, 41.46246], [69.39485, 41.51518], [69.45751, 41.56863], [69.49545, 41.545], [70.94483, 42.26238], [70.85973, 42.30188], [70.97717, 42.50147], [71.15232, 42.60486], [71.17807, 42.67381], [71.22785, 42.69248], [71.2724, 42.77853], [71.53272, 42.8014], [71.62405, 42.76613], [71.88792, 42.83578], [73.44393, 42.43098], [73.50992, 42.82356], [73.55634, 43.03071], [74.22489, 43.24657], [74.57491, 43.13702], [74.64615, 43.05881], [74.70331, 43.02519], [74.75, 42.99029], [74.88756, 42.98612], [75.22619, 42.85528], [75.29966, 42.86183], [75.72174, 42.79672], [75.82823, 42.94848], [78.48469, 42.89649], [78.91502, 42.76839], [79.19763, 42.804], [79.52921, 42.44778], [79.97364, 42.42816], [80.17807, 42.21166], [80.26841, 42.23797], [80.16892, 42.61137], [80.26886, 42.8366], [80.38169, 42.83142], [80.58999, 42.9011], [80.3735, 43.01557], [80.62913, 43.141], [80.78817, 43.14235], [80.77771, 43.30065], [80.69718, 43.32589], [80.75156, 43.44948], [80.40031, 44.10986], [80.40229, 44.23319], [80.38384, 44.63073], [79.8987, 44.89957], [80.11169, 45.03352], [81.73278, 45.3504], [82.51374, 45.1755], [82.58474, 45.40027], [82.21792, 45.56619], [83.04622, 47.19053], [83.92184, 46.98912], [84.73077, 47.01394], [84.93995, 46.87399], [85.22443, 47.04816], [85.54294, 47.06171], [85.69696, 47.2898], [85.61067, 47.49753], [85.5169, 48.05493], [85.73581, 48.3939], [86.38069, 48.46064], [86.75343, 48.70331], [86.73568, 48.99918], [86.87238, 49.12432], [87.28386, 49.11626], [87.31465, 49.23603], [87.03071, 49.25142], [86.82606, 49.51796], [86.61307, 49.60239], [86.79056, 49.74787], [86.63674, 49.80136], [86.18709, 49.50259], [85.24047, 49.60239], [84.99198, 50.06793], [84.29385, 50.27257], [83.8442, 50.87375], [83.14607, 51.00796], [82.55443, 50.75412], [81.94999, 50.79307], [81.46581, 50.77658], [81.41248, 50.97524], [81.06091, 50.94833], [81.16999, 51.15662], [80.80318, 51.28262], [80.44819, 51.20855], [80.4127, 50.95581], [80.08138, 50.77658], [79.11255, 52.01171], [77.90383, 53.29807], [76.54243, 53.99329], [76.44076, 54.16017], [76.82266, 54.1798], [76.91052, 54.4677], [75.3668, 54.07439], [75.43398, 53.98652], [75.07405, 53.80831], [73.39218, 53.44623], [73.25412, 53.61532], [73.68921, 53.86522], [73.74778, 54.07194], [73.37963, 53.96132], [72.71026, 54.1161], [72.43415, 53.92685], [72.17477, 54.36303], [71.96141, 54.17736], [71.10379, 54.13326], [71.08706, 54.33376], [71.24185, 54.64965], [71.08288, 54.71253], [70.96009, 55.10558], [70.76493, 55.3027], [70.19179, 55.1476], [69.74917, 55.35545], [69.34224, 55.36344], [68.90865, 55.38148]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "LA",
+           iso1A3: "LAO",
+           iso1N3: "418",
+           wikidata: "Q819",
+           nameEn: "Laos",
+           groups: ["035", "142", "UN"],
+           callingCodes: ["856"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[102.1245, 22.43372], [102.03633, 22.46164], [101.98487, 22.42766], [101.91344, 22.44417], [101.90714, 22.38688], [101.86828, 22.38397], [101.7685, 22.50337], [101.68973, 22.46843], [101.61306, 22.27515], [101.56789, 22.28876], [101.53638, 22.24794], [101.60675, 22.13513], [101.57525, 22.13026], [101.62566, 21.96574], [101.7791, 21.83019], [101.74555, 21.72852], [101.83257, 21.61562], [101.80001, 21.57461], [101.7475, 21.5873], [101.7727, 21.51794], [101.74224, 21.48276], [101.74014, 21.30967], [101.84412, 21.25291], [101.83887, 21.20983], [101.76745, 21.21571], [101.79266, 21.19025], [101.7622, 21.14813], [101.70548, 21.14911], [101.66977, 21.20004], [101.60886, 21.17947], [101.59491, 21.18621], [101.6068, 21.23329], [101.54563, 21.25668], [101.29326, 21.17254], [101.2229, 21.23271], [101.26912, 21.36482], [101.19349, 21.41959], [101.2124, 21.56422], [101.15156, 21.56129], [101.16198, 21.52808], [101.00234, 21.39612], [100.80173, 21.2934], [100.72716, 21.31786], [100.63578, 21.05639], [100.55281, 21.02796], [100.50974, 20.88574], [100.64628, 20.88279], [100.60112, 20.8347], [100.51079, 20.82194], [100.36375, 20.82783], [100.1957, 20.68247], [100.08404, 20.36626], [100.09999, 20.31614], [100.09337, 20.26293], [100.11785, 20.24787], [100.1712, 20.24324], [100.16668, 20.2986], [100.22076, 20.31598], [100.25769, 20.3992], [100.33383, 20.4028], [100.37439, 20.35156], [100.41473, 20.25625], [100.44992, 20.23644], [100.4537, 20.19971], [100.47567, 20.19133], [100.51052, 20.14928], [100.55218, 20.17741], [100.58808, 20.15791], [100.5094, 19.87904], [100.398, 19.75047], [100.49604, 19.53504], [100.58219, 19.49164], [100.64606, 19.55884], [100.77231, 19.48324], [100.90302, 19.61901], [101.08928, 19.59748], [101.26545, 19.59242], [101.26991, 19.48324], [101.21347, 19.46223], [101.20604, 19.35296], [101.24911, 19.33334], [101.261, 19.12717], [101.35606, 19.04716], [101.25803, 18.89545], [101.22832, 18.73377], [101.27585, 18.68875], [101.06047, 18.43247], [101.18227, 18.34367], [101.15108, 18.25624], [101.19118, 18.2125], [101.1793, 18.0544], [101.02185, 17.87637], [100.96541, 17.57926], [101.15108, 17.47586], [101.44667, 17.7392], [101.72294, 17.92867], [101.78087, 18.07559], [101.88485, 18.02474], [102.11359, 18.21532], [102.45523, 17.97106], [102.59234, 17.96127], [102.60971, 17.95411], [102.61432, 17.92273], [102.5896, 17.84889], [102.59485, 17.83537], [102.68194, 17.80151], [102.69946, 17.81686], [102.67543, 17.84529], [102.68538, 17.86653], [102.75954, 17.89561], [102.79044, 17.93612], [102.81988, 17.94233], [102.86323, 17.97531], [102.95812, 18.0054], [102.9912, 17.9949], [103.01998, 17.97095], [103.0566, 18.00144], [103.07823, 18.03833], [103.07343, 18.12351], [103.1493, 18.17799], [103.14994, 18.23172], [103.17093, 18.2618], [103.29757, 18.30475], [103.23818, 18.34875], [103.24779, 18.37807], [103.30977, 18.4341], [103.41044, 18.4486], [103.47773, 18.42841], [103.60957, 18.40528], [103.699, 18.34125], [103.82449, 18.33979], [103.85642, 18.28666], [103.93916, 18.33914], [103.97725, 18.33631], [104.06533, 18.21656], [104.10927, 18.10826], [104.21776, 17.99335], [104.2757, 17.86139], [104.35432, 17.82871], [104.45404, 17.66788], [104.69867, 17.53038], [104.80061, 17.39367], [104.80716, 17.19025], [104.73712, 17.01404], [104.7373, 16.91125], [104.76442, 16.84752], [104.7397, 16.81005], [104.76099, 16.69302], [104.73349, 16.565], [104.88057, 16.37311], [105.00262, 16.25627], [105.06204, 16.09792], [105.42001, 16.00657], [105.38508, 15.987], [105.34115, 15.92737], [105.37959, 15.84074], [105.42285, 15.76971], [105.46573, 15.74742], [105.61756, 15.68792], [105.60446, 15.53301], [105.58191, 15.41031], [105.47635, 15.3796], [105.4692, 15.33709], [105.50662, 15.32054], [105.58043, 15.32724], [105.46661, 15.13132], [105.61162, 15.00037], [105.5121, 14.80802], [105.53864, 14.55731], [105.43783, 14.43865], [105.20894, 14.34967], [105.2759, 14.17496], [105.36775, 14.09948], [105.44869, 14.10703], [105.5561, 14.15684], [105.78338, 14.08438], [105.78182, 14.02247], [105.90791, 13.92881], [106.10405, 13.9137], [106.10094, 13.98471], [106.16632, 14.01794], [106.18656, 14.06324], [106.11962, 14.11307], [106.10872, 14.18401], [106.04801, 14.20363], [106.02311, 14.30623], [105.99509, 14.32734], [106.00131, 14.36957], [106.21302, 14.36203], [106.25194, 14.48415], [106.32355, 14.44043], [106.40916, 14.45249], [106.43874, 14.52032], [106.47766, 14.50977], [106.45898, 14.55045], [106.50723, 14.58963], [106.54148, 14.59565], [106.57106, 14.50525], [106.59908, 14.50977], [106.63333, 14.44194], [106.73762, 14.42687], [106.80767, 14.31226], [106.8497, 14.29416], [106.90574, 14.33639], [106.9649, 14.3198], [106.98825, 14.36806], [107.04585, 14.41782], [107.03962, 14.45099], [107.09722, 14.3937], [107.17038, 14.41782], [107.21241, 14.48716], [107.256, 14.48716], [107.26534, 14.54292], [107.29803, 14.58963], [107.3276, 14.58812], [107.37897, 14.54443], [107.44435, 14.52785], [107.47238, 14.61523], [107.54361, 14.69092], [107.51579, 14.79282], [107.59285, 14.87795], [107.48277, 14.93751], [107.46516, 15.00982], [107.61486, 15.0566], [107.61926, 15.13949], [107.58844, 15.20111], [107.62587, 15.2266], [107.60605, 15.37524], [107.62367, 15.42193], [107.53341, 15.40496], [107.50699, 15.48771], [107.3815, 15.49832], [107.34408, 15.62345], [107.27583, 15.62769], [107.27143, 15.71459], [107.21859, 15.74638], [107.21419, 15.83747], [107.34188, 15.89464], [107.39471, 15.88829], [107.46296, 16.01106], [107.44975, 16.08511], [107.33968, 16.05549], [107.25822, 16.13587], [107.14595, 16.17816], [107.15035, 16.26271], [107.09091, 16.3092], [107.02597, 16.31132], [106.97385, 16.30204], [106.96638, 16.34938], [106.88067, 16.43594], [106.88727, 16.52671], [106.84104, 16.55415], [106.74418, 16.41904], [106.65832, 16.47816], [106.66052, 16.56892], [106.61477, 16.60713], [106.58267, 16.6012], [106.59013, 16.62259], [106.55485, 16.68704], [106.55265, 16.86831], [106.52183, 16.87884], [106.51963, 16.92097], [106.54824, 16.92729], [106.55045, 17.0031], [106.50862, 16.9673], [106.43597, 17.01362], [106.31929, 17.20509], [106.29287, 17.3018], [106.24444, 17.24714], [106.18991, 17.28227], [106.09019, 17.36399], [105.85744, 17.63221], [105.76612, 17.67147], [105.60381, 17.89356], [105.64784, 17.96687], [105.46292, 18.22008], [105.38366, 18.15315], [105.15942, 18.38691], [105.10408, 18.43533], [105.1327, 18.58355], [105.19654, 18.64196], [105.12829, 18.70453], [104.64617, 18.85668], [104.5361, 18.97747], [103.87125, 19.31854], [104.06058, 19.43484], [104.10832, 19.51575], [104.05617, 19.61743], [104.06498, 19.66926], [104.23229, 19.70242], [104.41281, 19.70035], [104.53169, 19.61743], [104.64837, 19.62365], [104.68359, 19.72729], [104.8355, 19.80395], [104.8465, 19.91783], [104.9874, 20.09573], [104.91695, 20.15567], [104.86852, 20.14121], [104.61315, 20.24452], [104.62195, 20.36633], [104.72102, 20.40554], [104.66158, 20.47774], [104.47886, 20.37459], [104.40621, 20.3849], [104.38199, 20.47155], [104.63957, 20.6653], [104.27412, 20.91433], [104.11121, 20.96779], [103.98024, 20.91531], [103.82282, 20.8732], [103.73478, 20.6669], [103.68633, 20.66324], [103.45737, 20.82382], [103.38032, 20.79501], [103.21497, 20.89832], [103.12055, 20.89994], [103.03469, 21.05821], [102.97745, 21.05821], [102.89825, 21.24707], [102.80794, 21.25736], [102.88939, 21.3107], [102.94223, 21.46034], [102.86297, 21.4255], [102.98846, 21.58936], [102.97965, 21.74076], [102.86077, 21.71213], [102.85637, 21.84501], [102.81894, 21.83888], [102.82115, 21.73667], [102.74189, 21.66713], [102.67145, 21.65894], [102.62301, 21.91447], [102.49092, 21.99002], [102.51734, 22.02676], [102.18712, 22.30403], [102.14099, 22.40092], [102.1245, 22.43372]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "LB",
+           iso1A3: "LBN",
+           iso1N3: "422",
+           wikidata: "Q822",
+           nameEn: "Lebanon",
+           aliases: ["RL"],
+           groups: ["145", "142", "UN"],
+           callingCodes: ["961"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[35.94816, 33.47886], [35.94465, 33.52774], [36.05723, 33.57904], [35.9341, 33.6596], [36.06778, 33.82927], [36.14517, 33.85118], [36.3967, 33.83365], [36.38263, 33.86579], [36.28589, 33.91981], [36.41078, 34.05253], [36.50576, 34.05982], [36.5128, 34.09916], [36.62537, 34.20251], [36.59195, 34.2316], [36.58667, 34.27667], [36.60778, 34.31009], [36.56556, 34.31881], [36.53039, 34.3798], [36.55853, 34.41609], [36.46179, 34.46541], [36.4442, 34.50165], [36.34745, 34.5002], [36.3369, 34.52629], [36.39846, 34.55672], [36.41429, 34.61175], [36.45299, 34.59438], [36.46003, 34.6378], [36.42941, 34.62505], [36.35384, 34.65447], [36.35135, 34.68516], [36.32399, 34.69334], [36.29165, 34.62991], [35.98718, 34.64977], [35.97386, 34.63322], [35.48515, 34.70851], [34.78515, 33.20368], [35.10645, 33.09318], [35.1924, 33.08743], [35.31429, 33.10515], [35.35223, 33.05617], [35.43059, 33.06659], [35.448, 33.09264], [35.50272, 33.09056], [35.50335, 33.114], [35.52573, 33.11921], [35.54228, 33.19865], [35.5362, 33.23196], [35.54808, 33.236], [35.54544, 33.25513], [35.55555, 33.25844], [35.56523, 33.28969], [35.58326, 33.28381], [35.58502, 33.26653], [35.62283, 33.24226], [35.62019, 33.27278], [35.77477, 33.33609], [35.81324, 33.36354], [35.82577, 33.40479], [35.88668, 33.43183], [35.94816, 33.47886]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "LC",
+           iso1A3: "LCA",
+           iso1N3: "662",
+           wikidata: "Q760",
+           nameEn: "St. Lucia",
+           aliases: ["WL"],
+           groups: ["029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           callingCodes: ["1 758"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-59.95997, 14.20285], [-61.69315, 14.26451], [-59.94058, 12.34011], [-59.95997, 14.20285]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "LI",
+           iso1A3: "LIE",
+           iso1N3: "438",
+           wikidata: "Q347",
+           nameEn: "Liechtenstein",
+           aliases: ["FL"],
+           groups: ["155", "150", "UN"],
+           callingCodes: ["423"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[9.60717, 47.06091], [9.61216, 47.07732], [9.63395, 47.08443], [9.62623, 47.14685], [9.56539, 47.17124], [9.58264, 47.20673], [9.56981, 47.21926], [9.55176, 47.22585], [9.56766, 47.24281], [9.53116, 47.27029], [9.52406, 47.24959], [9.50318, 47.22153], [9.4891, 47.19346], [9.48774, 47.17402], [9.51044, 47.13727], [9.52089, 47.10019], [9.51362, 47.08505], [9.47139, 47.06402], [9.47548, 47.05257], [9.54041, 47.06495], [9.55721, 47.04762], [9.60717, 47.06091]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "LK",
+           iso1A3: "LKA",
+           iso1N3: "144",
+           wikidata: "Q854",
+           nameEn: "Sri Lanka",
+           groups: ["034", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["94"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[76.59015, 5.591], [85.15017, 5.21497], [80.48418, 10.20786], [79.42124, 9.80115], [79.50447, 8.91876], [76.59015, 5.591]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "LR",
+           iso1A3: "LBR",
+           iso1N3: "430",
+           wikidata: "Q1014",
+           nameEn: "Liberia",
+           groups: ["011", "202", "002", "UN"],
+           callingCodes: ["231"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-8.47114, 7.55676], [-8.55874, 7.62525], [-8.55874, 7.70167], [-8.67814, 7.69428], [-8.72789, 7.51429], [-8.8448, 7.35149], [-8.85724, 7.26019], [-8.93435, 7.2824], [-9.09107, 7.1985], [-9.18311, 7.30461], [-9.20798, 7.38109], [-9.305, 7.42056], [-9.41943, 7.41809], [-9.48161, 7.37122], [-9.37465, 7.62032], [-9.35724, 7.74111], [-9.44928, 7.9284], [-9.41445, 8.02448], [-9.50898, 8.18455], [-9.47415, 8.35195], [-9.77763, 8.54633], [-10.05873, 8.42578], [-10.05375, 8.50697], [-10.14579, 8.52665], [-10.203, 8.47991], [-10.27575, 8.48711], [-10.30084, 8.30008], [-10.31635, 8.28554], [-10.29839, 8.21283], [-10.35227, 8.15223], [-10.45023, 8.15627], [-10.51554, 8.1393], [-10.57523, 8.04829], [-10.60492, 8.04072], [-10.60422, 7.7739], [-11.29417, 7.21576], [-11.4027, 6.97746], [-11.50429, 6.92704], [-12.15048, 6.15992], [-7.52774, 3.7105], [-7.53259, 4.35145], [-7.59349, 4.8909], [-7.53876, 4.94294], [-7.55369, 5.08667], [-7.48901, 5.14118], [-7.46165, 5.26256], [-7.36463, 5.32944], [-7.43428, 5.42355], [-7.37209, 5.61173], [-7.43926, 5.74787], [-7.43677, 5.84687], [-7.46165, 5.84934], [-7.48155, 5.80974], [-7.67309, 5.94337], [-7.70294, 5.90625], [-7.78254, 5.99037], [-7.79747, 6.07696], [-7.8497, 6.08932], [-7.83478, 6.20309], [-7.90692, 6.27728], [-8.00642, 6.31684], [-8.17557, 6.28222], [-8.3298, 6.36381], [-8.38453, 6.35887], [-8.45666, 6.49977], [-8.48652, 6.43797], [-8.59456, 6.50612], [-8.31736, 6.82837], [-8.29249, 7.1691], [-8.37458, 7.25794], [-8.41935, 7.51203], [-8.47114, 7.55676]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "LS",
+           iso1A3: "LSO",
+           iso1N3: "426",
+           wikidata: "Q1013",
+           nameEn: "Lesotho",
+           groups: ["018", "202", "002", "UN"],
+           driveSide: "left",
+           callingCodes: ["266"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[29.33204, -29.45598], [29.44883, -29.3772], [29.40524, -29.21246], [28.68043, -28.58744], [28.65091, -28.57025], [28.40612, -28.6215], [28.30518, -28.69531], [28.2348, -28.69471], [28.1317, -28.7293], [28.02503, -28.85991], [27.98675, -28.8787], [27.9392, -28.84864], [27.88933, -28.88156], [27.8907, -28.91612], [27.75458, -28.89839], [27.55974, -29.18954], [27.5158, -29.2261], [27.54258, -29.25575], [27.48679, -29.29349], [27.45125, -29.29708], [27.47254, -29.31968], [27.4358, -29.33465], [27.33464, -29.48161], [27.01016, -29.65439], [27.09489, -29.72796], [27.22719, -30.00718], [27.29603, -30.05473], [27.32555, -30.14785], [27.40778, -30.14577], [27.37293, -30.19401], [27.36649, -30.27246], [27.38108, -30.33456], [27.45452, -30.32239], [27.56901, -30.42504], [27.56781, -30.44562], [27.62137, -30.50509], [27.6521, -30.51707], [27.67819, -30.53437], [27.69467, -30.55862], [27.74814, -30.60635], [28.12073, -30.68072], [28.2319, -30.28476], [28.399, -30.1592], [28.68627, -30.12885], [28.80222, -30.10579], [28.9338, -30.05072], [29.16548, -29.91706], [29.12553, -29.76266], [29.28545, -29.58456], [29.33204, -29.45598]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "LT",
+           iso1A3: "LTU",
+           iso1N3: "440",
+           wikidata: "Q37",
+           nameEn: "Lithuania",
+           groups: ["EU", "154", "150", "UN"],
+           callingCodes: ["370"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[24.89005, 56.46666], [24.83686, 56.41565], [24.70022, 56.40483], [24.57353, 56.31525], [24.58143, 56.29125], [24.42746, 56.26522], [24.32334, 56.30226], [24.13139, 56.24881], [24.02657, 56.3231], [23.75726, 56.37282], [23.49803, 56.34307], [23.40486, 56.37689], [23.31606, 56.3827], [23.17312, 56.36795], [23.09531, 56.30511], [22.96988, 56.41213], [22.83048, 56.367], [22.69354, 56.36284], [22.56441, 56.39305], [22.3361, 56.4016], [22.09728, 56.42851], [22.00548, 56.41508], [21.74558, 56.33181], [21.57888, 56.31406], [21.49736, 56.29106], [21.24644, 56.16917], [21.15016, 56.07818], [20.68447, 56.04073], [20.60454, 55.40986], [20.95181, 55.27994], [21.26425, 55.24456], [21.35465, 55.28427], [21.38446, 55.29348], [21.46766, 55.21115], [21.51095, 55.18507], [21.55605, 55.20311], [21.64954, 55.1791], [21.85521, 55.09493], [21.96505, 55.07353], [21.99543, 55.08691], [22.03984, 55.07888], [22.02582, 55.05078], [22.06087, 55.02935], [22.11697, 55.02131], [22.14267, 55.05345], [22.31562, 55.0655], [22.47688, 55.04408], [22.58907, 55.07085], [22.60075, 55.01863], [22.65451, 54.97037], [22.68723, 54.9811], [22.76422, 54.92521], [22.85083, 54.88711], [22.87317, 54.79492], [22.73631, 54.72952], [22.73397, 54.66604], [22.75467, 54.6483], [22.74225, 54.64339], [22.7522, 54.63525], [22.68021, 54.58486], [22.71293, 54.56454], [22.67788, 54.532], [22.70208, 54.45312], [22.7253, 54.41732], [22.79705, 54.36264], [22.83756, 54.40827], [23.00584, 54.38514], [22.99649, 54.35927], [23.05726, 54.34565], [23.04323, 54.31567], [23.104, 54.29794], [23.13905, 54.31567], [23.15526, 54.31076], [23.15938, 54.29894], [23.24656, 54.25701], [23.3494, 54.25155], [23.39525, 54.21672], [23.42418, 54.17911], [23.45223, 54.17775], [23.49196, 54.14764], [23.52702, 54.04622], [23.48261, 53.98855], [23.51284, 53.95052], [23.61677, 53.92691], [23.71726, 53.93379], [23.80543, 53.89558], [23.81309, 53.94205], [23.95098, 53.9613], [23.98837, 53.92554], [24.19638, 53.96405], [24.34128, 53.90076], [24.44411, 53.90076], [24.62275, 54.00217], [24.69652, 54.01901], [24.69185, 53.96543], [24.74279, 53.96663], [24.85311, 54.02862], [24.77131, 54.11091], [24.96894, 54.17589], [24.991, 54.14241], [25.0728, 54.13419], [25.19199, 54.219], [25.22705, 54.26271], [25.35559, 54.26544], [25.509, 54.30267], [25.56823, 54.25212], [25.51452, 54.17799], [25.54724, 54.14925], [25.64875, 54.1259], [25.71084, 54.16704], [25.78563, 54.15747], [25.78553, 54.23327], [25.68513, 54.31727], [25.55425, 54.31591], [25.5376, 54.33158], [25.63371, 54.42075], [25.62203, 54.4656], [25.64813, 54.48704], [25.68045, 54.5321], [25.75977, 54.57252], [25.74122, 54.80108], [25.89462, 54.93438], [25.99129, 54.95705], [26.05907, 54.94631], [26.13386, 54.98924], [26.20397, 54.99729], [26.26941, 55.08032], [26.23202, 55.10439], [26.30628, 55.12536], [26.35121, 55.1525], [26.46249, 55.12814], [26.51481, 55.16051], [26.54753, 55.14181], [26.69243, 55.16718], [26.68075, 55.19787], [26.72983, 55.21788], [26.73017, 55.24226], [26.835, 55.28182], [26.83266, 55.30444], [26.80929, 55.31642], [26.6714, 55.33902], [26.5709, 55.32572], [26.44937, 55.34832], [26.5522, 55.40277], [26.55094, 55.5093], [26.63167, 55.57887], [26.63231, 55.67968], [26.58248, 55.6754], [26.46661, 55.70375], [26.39561, 55.71156], [26.18509, 55.86813], [26.03815, 55.95884], [25.90047, 56.0013], [25.85893, 56.00188], [25.81773, 56.05444], [25.69246, 56.08892], [25.68588, 56.14725], [25.53621, 56.16663], [25.39751, 56.15707], [25.23099, 56.19147], [25.09325, 56.1878], [25.05762, 56.26742], [24.89005, 56.46666]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "LU",
+           iso1A3: "LUX",
+           iso1N3: "442",
+           wikidata: "Q32",
+           nameEn: "Luxembourg",
+           groups: ["EU", "155", "150", "UN"],
+           callingCodes: ["352"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[6.1379, 50.12964], [6.1137, 50.13668], [6.12028, 50.16374], [6.08577, 50.17246], [6.06406, 50.15344], [6.03093, 50.16362], [6.02488, 50.18283], [5.96453, 50.17259], [5.95929, 50.13295], [5.89488, 50.11476], [5.8857, 50.07824], [5.85474, 50.06342], [5.86904, 50.04614], [5.8551, 50.02683], [5.81866, 50.01286], [5.82331, 49.99662], [5.83968, 49.9892], [5.83467, 49.97823], [5.81163, 49.97142], [5.80833, 49.96451], [5.77291, 49.96056], [5.77314, 49.93646], [5.73621, 49.89796], [5.78415, 49.87922], [5.75269, 49.8711], [5.75861, 49.85631], [5.74567, 49.85368], [5.75884, 49.84811], [5.74953, 49.84709], [5.74975, 49.83933], [5.74076, 49.83823], [5.7404, 49.83452], [5.74844, 49.82435], [5.74364, 49.82058], [5.74953, 49.81428], [5.75409, 49.79239], [5.78871, 49.7962], [5.82245, 49.75048], [5.83149, 49.74729], [5.82562, 49.72395], [5.84193, 49.72161], [5.86503, 49.72739], [5.88677, 49.70951], [5.86527, 49.69291], [5.86175, 49.67862], [5.9069, 49.66377], [5.90164, 49.6511], [5.90599, 49.63853], [5.88552, 49.63507], [5.88393, 49.62802], [5.87609, 49.62047], [5.8762, 49.60898], [5.84826, 49.5969], [5.84971, 49.58674], [5.86986, 49.58756], [5.87256, 49.57539], [5.8424, 49.56082], [5.84692, 49.55663], [5.84143, 49.5533], [5.81838, 49.54777], [5.80871, 49.5425], [5.81664, 49.53775], [5.83648, 49.5425], [5.84466, 49.53027], [5.83467, 49.52717], [5.83389, 49.52152], [5.86571, 49.50015], [5.94128, 49.50034], [5.94224, 49.49608], [5.96876, 49.49053], [5.97693, 49.45513], [6.02648, 49.45451], [6.02743, 49.44845], [6.04176, 49.44801], [6.05553, 49.46663], [6.07887, 49.46399], [6.08373, 49.45594], [6.10072, 49.45268], [6.09845, 49.46351], [6.10325, 49.4707], [6.12346, 49.4735], [6.12814, 49.49365], [6.14321, 49.48796], [6.16115, 49.49297], [6.15366, 49.50226], [6.17386, 49.50934], [6.19543, 49.50536], [6.2409, 49.51408], [6.25029, 49.50609], [6.27875, 49.503], [6.28818, 49.48465], [6.3687, 49.4593], [6.36778, 49.46937], [6.36907, 49.48931], [6.36788, 49.50377], [6.35666, 49.52931], [6.38072, 49.55171], [6.38228, 49.55855], [6.35825, 49.57053], [6.36676, 49.57813], [6.38024, 49.57593], [6.38342, 49.5799], [6.37464, 49.58886], [6.385, 49.59946], [6.39822, 49.60081], [6.41861, 49.61723], [6.4413, 49.65722], [6.43768, 49.66021], [6.42726, 49.66078], [6.42937, 49.66857], [6.44654, 49.67799], [6.46048, 49.69092], [6.48014, 49.69767], [6.49785, 49.71118], [6.50647, 49.71353], [6.5042, 49.71808], [6.49694, 49.72205], [6.49535, 49.72645], [6.50261, 49.72718], [6.51397, 49.72058], [6.51805, 49.72425], [6.50193, 49.73291], [6.50174, 49.75292], [6.51646, 49.75961], [6.51828, 49.76855], [6.51056, 49.77515], [6.51669, 49.78336], [6.50534, 49.78952], [6.52169, 49.79787], [6.53122, 49.80666], [6.52121, 49.81338], [6.51215, 49.80124], [6.50647, 49.80916], [6.48718, 49.81267], [6.47111, 49.82263], [6.45425, 49.81164], [6.44131, 49.81443], [6.42905, 49.81091], [6.42521, 49.81591], [6.40022, 49.82029], [6.36576, 49.85032], [6.34267, 49.84974], [6.33585, 49.83785], [6.32098, 49.83728], [6.32303, 49.85133], [6.30963, 49.87021], [6.29692, 49.86685], [6.28874, 49.87592], [6.26146, 49.88203], [6.23496, 49.89972], [6.22926, 49.92096], [6.21882, 49.92403], [6.22608, 49.929], [6.22094, 49.94955], [6.19856, 49.95053], [6.19089, 49.96991], [6.18045, 49.96611], [6.18554, 49.95622], [6.17872, 49.9537], [6.16466, 49.97086], [6.1701, 49.98518], [6.14147, 49.99563], [6.14948, 50.00908], [6.13806, 50.01056], [6.1295, 50.01849], [6.13273, 50.02019], [6.13794, 50.01466], [6.14666, 50.02207], [6.13044, 50.02929], [6.13458, 50.04141], [6.11274, 50.05916], [6.12055, 50.09171], [6.1379, 50.12964]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "LV",
+           iso1A3: "LVA",
+           iso1N3: "428",
+           wikidata: "Q211",
+           nameEn: "Latvia",
+           groups: ["EU", "154", "150", "UN"],
+           callingCodes: ["371"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[27.34698, 57.52242], [26.90364, 57.62823], [26.54675, 57.51813], [26.46527, 57.56885], [26.29253, 57.59244], [26.1866, 57.6849], [26.2029, 57.7206], [26.08098, 57.76619], [26.0543, 57.76105], [26.03332, 57.7718], [26.02415, 57.76865], [26.02069, 57.77169], [26.0266, 57.77441], [26.027, 57.78158], [26.02456, 57.78342], [26.0324, 57.79037], [26.05949, 57.84744], [25.73499, 57.90193], [25.29581, 58.08288], [25.28237, 57.98539], [25.19484, 58.0831], [24.3579, 57.87471], [24.26221, 57.91787], [23.20055, 57.56697], [22.80496, 57.87798], [19.84909, 57.57876], [19.64795, 57.06466], [20.68447, 56.04073], [21.15016, 56.07818], [21.24644, 56.16917], [21.49736, 56.29106], [21.57888, 56.31406], [21.74558, 56.33181], [22.00548, 56.41508], [22.09728, 56.42851], [22.3361, 56.4016], [22.56441, 56.39305], [22.69354, 56.36284], [22.83048, 56.367], [22.96988, 56.41213], [23.09531, 56.30511], [23.17312, 56.36795], [23.31606, 56.3827], [23.40486, 56.37689], [23.49803, 56.34307], [23.75726, 56.37282], [24.02657, 56.3231], [24.13139, 56.24881], [24.32334, 56.30226], [24.42746, 56.26522], [24.58143, 56.29125], [24.57353, 56.31525], [24.70022, 56.40483], [24.83686, 56.41565], [24.89005, 56.46666], [25.05762, 56.26742], [25.09325, 56.1878], [25.23099, 56.19147], [25.39751, 56.15707], [25.53621, 56.16663], [25.68588, 56.14725], [25.69246, 56.08892], [25.81773, 56.05444], [25.85893, 56.00188], [25.90047, 56.0013], [26.03815, 55.95884], [26.18509, 55.86813], [26.39561, 55.71156], [26.46661, 55.70375], [26.58248, 55.6754], [26.63231, 55.67968], [26.64888, 55.70515], [26.71802, 55.70645], [26.76872, 55.67658], [26.87448, 55.7172], [26.97153, 55.8102], [27.1559, 55.85032], [27.27804, 55.78299], [27.3541, 55.8089], [27.61683, 55.78558], [27.63065, 55.89687], [27.97865, 56.11849], [28.15217, 56.16964], [28.23716, 56.27588], [28.16599, 56.37806], [28.19057, 56.44637], [28.10069, 56.524], [28.13526, 56.57989], [28.04768, 56.59004], [27.86101, 56.88204], [27.66511, 56.83921], [27.86101, 57.29402], [27.52453, 57.42826], [27.56832, 57.53728], [27.34698, 57.52242]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "LY",
+           iso1A3: "LBY",
+           iso1N3: "434",
+           wikidata: "Q1016",
+           nameEn: "Libya",
+           groups: ["015", "002", "UN"],
+           callingCodes: ["218"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[26.92891, 33.39516], [11.58941, 33.36891], [11.55852, 33.1409], [11.51549, 33.09826], [11.46037, 32.6307], [11.57828, 32.48013], [11.53898, 32.4138], [11.04234, 32.2145], [10.7315, 31.97235], [10.62788, 31.96629], [10.48497, 31.72956], [10.31364, 31.72648], [10.12239, 31.42098], [10.29516, 30.90337], [9.88152, 30.34074], [9.76848, 30.34366], [9.55544, 30.23971], [9.3876, 30.16738], [9.78136, 29.40961], [9.89569, 26.57696], [9.51696, 26.39148], [9.38834, 26.19288], [10.03146, 25.35635], [10.02432, 24.98124], [10.33159, 24.5465], [10.85323, 24.5595], [11.41061, 24.21456], [11.62498, 24.26669], [11.96886, 23.51735], [13.5631, 23.16574], [14.22918, 22.61719], [14.99751, 23.00539], [15.99566, 23.49639], [23.99539, 19.49944], [23.99715, 20.00038], [24.99794, 19.99661], [24.99885, 21.99535], [24.99968, 29.24574], [24.71117, 30.17441], [25.01077, 30.73861], [24.8458, 31.39877], [26.92891, 33.39516]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MA",
+           iso1A3: "MAR",
+           iso1N3: "504",
+           wikidata: "Q1028",
+           nameEn: "Morocco",
+           groups: ["015", "002", "UN"],
+           callingCodes: ["212"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-2.27707, 35.35051], [-5.10878, 36.05227], [-7.2725, 35.73269], [-14.43883, 27.02969], [-17.27295, 21.93519], [-17.21511, 21.34226], [-17.02707, 21.34022], [-16.9978, 21.36239], [-16.44269, 21.39745], [-14.78487, 21.36587], [-14.47329, 21.63839], [-14.48112, 22.00886], [-14.1291, 22.41636], [-14.10361, 22.75501], [-13.75627, 23.77231], [-13.00628, 24.01923], [-12.92147, 24.39502], [-12.12281, 25.13682], [-12.06001, 26.04442], [-11.62052, 26.05229], [-11.38635, 26.611], [-11.23622, 26.72023], [-11.35695, 26.8505], [-10.68417, 26.90984], [-9.81998, 26.71379], [-9.56957, 26.90042], [-9.08698, 26.98639], [-8.71787, 26.9898], [-8.77527, 27.66663], [-8.66879, 27.6666], [-8.6715, 28.71194], [-7.61585, 29.36252], [-6.95824, 29.50924], [-6.78351, 29.44634], [-6.69965, 29.51623], [-5.75616, 29.61407], [-5.72121, 29.52322], [-5.58831, 29.48103], [-5.21671, 29.95253], [-4.6058, 30.28343], [-4.31774, 30.53229], [-3.64735, 30.67539], [-3.65418, 30.85566], [-3.54944, 31.0503], [-3.77103, 31.14984], [-3.77647, 31.31912], [-3.66386, 31.39202], [-3.66314, 31.6339], [-2.82784, 31.79459], [-2.93873, 32.06557], [-2.46166, 32.16603], [-1.22829, 32.07832], [-1.15735, 32.12096], [-1.24453, 32.1917], [-1.24998, 32.32993], [-0.9912, 32.52467], [-1.37794, 32.73628], [-1.54244, 32.95499], [-1.46249, 33.0499], [-1.67067, 33.27084], [-1.59508, 33.59929], [-1.73494, 33.71721], [-1.64666, 34.10405], [-1.78042, 34.39018], [-1.69788, 34.48056], [-1.84569, 34.61907], [-1.73707, 34.74226], [-1.97469, 34.886], [-1.97833, 34.93218], [-2.04734, 34.93218], [-2.21445, 35.04378], [-2.21248, 35.08532], [-2.27707, 35.35051]], [[-2.91909, 35.33927], [-2.92272, 35.27509], [-2.93893, 35.26737], [-2.95065, 35.26576], [-2.95431, 35.2728], [-2.96516, 35.27967], [-2.96826, 35.28296], [-2.96507, 35.28801], [-2.97035, 35.28852], [-2.96978, 35.29459], [-2.96648, 35.30475], [-2.96038, 35.31609], [-2.91909, 35.33927]], [[-3.90602, 35.21494], [-3.89343, 35.22728], [-3.88372, 35.20767], [-3.90602, 35.21494]], [[-4.30191, 35.17419], [-4.29436, 35.17149], [-4.30112, 35.17058], [-4.30191, 35.17419]], [[-2.40316, 35.16893], [-2.45965, 35.16527], [-2.43262, 35.20652], [-2.40316, 35.16893]], [[-5.38491, 35.92591], [-5.21179, 35.90091], [-5.34379, 35.8711], [-5.35844, 35.87375], [-5.37338, 35.88417], [-5.38491, 35.92591]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MC",
+           iso1A3: "MCO",
+           iso1N3: "492",
+           wikidata: "Q235",
+           nameEn: "Monaco",
+           groups: ["155", "150", "UN"],
+           callingCodes: ["377"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[7.47823, 43.73341], [7.4379, 43.74963], [7.4389, 43.75151], [7.43708, 43.75197], [7.43624, 43.75014], [7.43013, 43.74895], [7.42809, 43.74396], [7.42443, 43.74087], [7.42299, 43.74176], [7.42062, 43.73977], [7.41233, 43.73439], [7.41298, 43.73311], [7.41291, 43.73168], [7.41113, 43.73156], [7.40903, 43.7296], [7.42422, 43.72209], [7.47823, 43.73341]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MD",
+           iso1A3: "MDA",
+           iso1N3: "498",
+           wikidata: "Q217",
+           nameEn: "Moldova",
+           groups: ["151", "150", "UN"],
+           callingCodes: ["373"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[27.74422, 48.45926], [27.6658, 48.44034], [27.59027, 48.46311], [27.5889, 48.49224], [27.46942, 48.454], [27.44333, 48.41209], [27.37741, 48.41026], [27.37604, 48.44398], [27.32159, 48.4434], [27.27855, 48.37534], [27.13434, 48.37288], [27.08078, 48.43214], [27.0231, 48.42485], [27.03821, 48.37653], [26.93384, 48.36558], [26.85556, 48.41095], [26.71274, 48.40388], [26.82809, 48.31629], [26.79239, 48.29071], [26.6839, 48.35828], [26.62823, 48.25804], [26.81161, 48.25049], [26.87708, 48.19919], [26.94265, 48.1969], [26.98042, 48.15752], [26.96119, 48.13003], [27.04118, 48.12522], [27.02985, 48.09083], [27.15622, 47.98538], [27.1618, 47.92391], [27.29069, 47.73722], [27.25519, 47.71366], [27.32202, 47.64009], [27.3979, 47.59473], [27.47942, 47.48113], [27.55731, 47.46637], [27.60263, 47.32507], [27.68706, 47.28962], [27.73172, 47.29248], [27.81892, 47.1381], [28.09095, 46.97621], [28.12173, 46.82283], [28.24808, 46.64305], [28.22281, 46.50481], [28.25769, 46.43334], [28.18902, 46.35283], [28.19864, 46.31869], [28.10937, 46.22852], [28.13684, 46.18099], [28.08612, 46.01105], [28.13111, 45.92819], [28.16568, 45.6421], [28.08927, 45.6051], [28.18741, 45.47358], [28.21139, 45.46895], [28.30201, 45.54744], [28.41836, 45.51715], [28.43072, 45.48538], [28.51449, 45.49982], [28.49252, 45.56716], [28.54196, 45.58062], [28.51587, 45.6613], [28.47879, 45.66994], [28.52823, 45.73803], [28.70401, 45.78019], [28.69852, 45.81753], [28.78503, 45.83475], [28.74383, 45.96664], [28.98004, 46.00385], [29.00613, 46.04962], [28.94643, 46.09176], [29.06656, 46.19716], [28.94953, 46.25852], [28.98478, 46.31803], [29.004, 46.31495], [28.9306, 46.45699], [29.01241, 46.46177], [29.02409, 46.49582], [29.23547, 46.55435], [29.24886, 46.37912], [29.35357, 46.49505], [29.49914, 46.45889], [29.5939, 46.35472], [29.6763, 46.36041], [29.66359, 46.4215], [29.74496, 46.45605], [29.88329, 46.35851], [29.94114, 46.40114], [30.09103, 46.38694], [30.16794, 46.40967], [30.02511, 46.45132], [29.88916, 46.54302], [29.94409, 46.56002], [29.9743, 46.75325], [29.94522, 46.80055], [29.98814, 46.82358], [29.87405, 46.88199], [29.75458, 46.8604], [29.72986, 46.92234], [29.57056, 46.94766], [29.62137, 47.05069], [29.61038, 47.09932], [29.53044, 47.07851], [29.49732, 47.12878], [29.57696, 47.13581], [29.54996, 47.24962], [29.59665, 47.25521], [29.5733, 47.36508], [29.48678, 47.36043], [29.47854, 47.30366], [29.39889, 47.30179], [29.3261, 47.44664], [29.18603, 47.43387], [29.11743, 47.55001], [29.22414, 47.60012], [29.22242, 47.73607], [29.27255, 47.79953], [29.20663, 47.80367], [29.27804, 47.88893], [29.19839, 47.89261], [29.1723, 47.99013], [28.9306, 47.96255], [28.8414, 48.03392], [28.85232, 48.12506], [28.69896, 48.13106], [28.53921, 48.17453], [28.48428, 48.0737], [28.42454, 48.12047], [28.43701, 48.15832], [28.38712, 48.17567], [28.34009, 48.13147], [28.30609, 48.14018], [28.30586, 48.1597], [28.34912, 48.1787], [28.36996, 48.20543], [28.35519, 48.24957], [28.32508, 48.23384], [28.2856, 48.23202], [28.19314, 48.20749], [28.17666, 48.25963], [28.07504, 48.23494], [28.09873, 48.3124], [28.04527, 48.32661], [27.95883, 48.32368], [27.88391, 48.36699], [27.87533, 48.4037], [27.81902, 48.41874], [27.79225, 48.44244], [27.74422, 48.45926]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "ME",
+           iso1A3: "MNE",
+           iso1N3: "499",
+           wikidata: "Q236",
+           nameEn: "Montenegro",
+           groups: ["039", "150", "UN"],
+           callingCodes: ["382"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[19.22807, 43.5264], [19.15685, 43.53943], [19.13933, 43.5282], [19.04934, 43.50384], [19.01078, 43.55806], [18.91379, 43.50299], [18.95469, 43.49367], [18.96053, 43.45042], [19.01078, 43.43854], [19.04071, 43.397], [19.08673, 43.31453], [19.08206, 43.29668], [19.04233, 43.30008], [19.00844, 43.24988], [18.95001, 43.29327], [18.95819, 43.32899], [18.90911, 43.36383], [18.83912, 43.34795], [18.84794, 43.33735], [18.85342, 43.32426], [18.76538, 43.29838], [18.6976, 43.25243], [18.71747, 43.2286], [18.66605, 43.2056], [18.64735, 43.14766], [18.66254, 43.03928], [18.52232, 43.01451], [18.49076, 42.95553], [18.49661, 42.89306], [18.4935, 42.86433], [18.47633, 42.85829], [18.45921, 42.81682], [18.47324, 42.74992], [18.56789, 42.72074], [18.55221, 42.69045], [18.54603, 42.69171], [18.54841, 42.68328], [18.57373, 42.64429], [18.52232, 42.62279], [18.55504, 42.58409], [18.53751, 42.57376], [18.49778, 42.58409], [18.43735, 42.55921], [18.44307, 42.51077], [18.43588, 42.48556], [18.52152, 42.42302], [18.54128, 42.39171], [18.45131, 42.21682], [19.26406, 41.74971], [19.37597, 41.84849], [19.37451, 41.8842], [19.33812, 41.90669], [19.34601, 41.95675], [19.37691, 41.96977], [19.36867, 42.02564], [19.37548, 42.06835], [19.40687, 42.10024], [19.28623, 42.17745], [19.42, 42.33019], [19.42352, 42.36546], [19.4836, 42.40831], [19.65972, 42.62774], [19.73244, 42.66299], [19.77375, 42.58517], [19.74731, 42.57422], [19.76549, 42.50237], [19.82333, 42.46581], [19.9324, 42.51699], [20.00842, 42.5109], [20.01834, 42.54622], [20.07761, 42.55582], [20.0969, 42.65559], [20.02915, 42.71147], [20.02088, 42.74789], [20.04898, 42.77701], [20.2539, 42.76245], [20.27869, 42.81945], [20.35692, 42.8335], [20.34528, 42.90676], [20.16415, 42.97177], [20.14896, 42.99058], [20.12325, 42.96237], [20.05431, 42.99571], [20.04729, 43.02732], [19.98887, 43.0538], [19.96549, 43.11098], [19.92576, 43.08539], [19.79255, 43.11951], [19.76918, 43.16044], [19.64063, 43.19027], [19.62661, 43.2286], [19.54598, 43.25158], [19.52962, 43.31623], [19.48171, 43.32644], [19.44315, 43.38846], [19.22229, 43.47926], [19.22807, 43.5264]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MF",
+           iso1A3: "MAF",
+           iso1N3: "663",
+           wikidata: "Q126125",
+           nameEn: "Saint-Martin",
+           country: "FR",
+           groups: ["Q3320166", "EU", "029", "003", "419", "019", "UN"],
+           callingCodes: ["590"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-62.93924, 18.02904], [-62.62718, 18.26185], [-63.35989, 18.06012], [-63.33064, 17.9615], [-63.13502, 18.05445], [-63.11042, 18.05339], [-63.09686, 18.04608], [-63.07759, 18.04943], [-63.0579, 18.06614], [-63.04039, 18.05619], [-63.02323, 18.05757], [-62.93924, 18.02904]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MG",
+           iso1A3: "MDG",
+           iso1N3: "450",
+           wikidata: "Q1019",
+           nameEn: "Madagascar",
+           aliases: ["RM"],
+           groups: ["014", "202", "002", "UN"],
+           callingCodes: ["261"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[51.93891, -10.85085], [45.84651, -12.77177], [42.14681, -19.63341], [45.80092, -33.00974], [51.93891, -10.85085]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MH",
+           iso1A3: "MHL",
+           iso1N3: "584",
+           wikidata: "Q709",
+           nameEn: "Marshall Islands",
+           groups: ["057", "009", "UN"],
+           roadSpeedUnit: "mph",
+           callingCodes: ["692"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[169, 3.9], [173.53711, 5.70687], [169.29099, 15.77133], [159.04653, 10.59067], [169, 3.9]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MK",
+           iso1A3: "MKD",
+           iso1N3: "807",
+           wikidata: "Q221",
+           nameEn: "North Macedonia",
+           groups: ["039", "150", "UN"],
+           callingCodes: ["389"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[22.34773, 42.31725], [22.29275, 42.34913], [22.29605, 42.37477], [22.16384, 42.32103], [22.02908, 42.29848], [21.94405, 42.34669], [21.91595, 42.30392], [21.84654, 42.3247], [21.77176, 42.2648], [21.70111, 42.23789], [21.58992, 42.25915], [21.52145, 42.24465], [21.50823, 42.27156], [21.43882, 42.2789], [21.43882, 42.23609], [21.38428, 42.24465], [21.30496, 42.1418], [21.29913, 42.13954], [21.31983, 42.10993], [21.22728, 42.08909], [21.16614, 42.19815], [21.11491, 42.20794], [20.75464, 42.05229], [20.76786, 41.91839], [20.68523, 41.85318], [20.59524, 41.8818], [20.55976, 41.87068], [20.57144, 41.7897], [20.53405, 41.78099], [20.51301, 41.72433], [20.52937, 41.69292], [20.51769, 41.65975], [20.55508, 41.58113], [20.52103, 41.56473], [20.45809, 41.5549], [20.45331, 41.51436], [20.49039, 41.49277], [20.51301, 41.442], [20.55976, 41.4087], [20.52119, 41.34381], [20.49432, 41.33679], [20.51068, 41.2323], [20.59715, 41.13644], [20.58546, 41.11179], [20.59832, 41.09066], [20.63454, 41.0889], [20.65558, 41.08009], [20.71634, 40.91781], [20.73504, 40.9081], [20.81567, 40.89662], [20.83671, 40.92752], [20.94305, 40.92399], [20.97693, 40.90103], [20.97887, 40.85475], [21.15262, 40.85546], [21.21105, 40.8855], [21.25779, 40.86165], [21.35595, 40.87578], [21.41555, 40.9173], [21.53007, 40.90759], [21.57448, 40.86076], [21.69601, 40.9429], [21.7556, 40.92525], [21.91102, 41.04786], [21.90869, 41.09191], [22.06527, 41.15617], [22.1424, 41.12449], [22.17629, 41.15969], [22.26744, 41.16409], [22.42285, 41.11921], [22.5549, 41.13065], [22.58295, 41.11568], [22.62852, 41.14385], [22.65306, 41.18168], [22.71266, 41.13945], [22.74538, 41.16321], [22.76408, 41.32225], [22.81199, 41.3398], [22.93334, 41.34104], [22.96331, 41.35782], [22.95513, 41.63265], [23.03342, 41.71034], [23.01239, 41.76527], [22.96682, 41.77137], [22.90254, 41.87587], [22.86749, 42.02275], [22.67701, 42.06614], [22.51224, 42.15457], [22.50289, 42.19527], [22.47251, 42.20393], [22.38136, 42.30339], [22.34773, 42.31725]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "ML",
+           iso1A3: "MLI",
+           iso1N3: "466",
+           wikidata: "Q912",
+           nameEn: "Mali",
+           groups: ["011", "202", "002", "UN"],
+           callingCodes: ["223"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-4.83423, 24.99935], [-6.57191, 25.0002], [-5.60725, 16.49919], [-5.33435, 16.33354], [-5.50165, 15.50061], [-9.32979, 15.50032], [-9.31106, 15.69412], [-9.33314, 15.7044], [-9.44673, 15.60553], [-9.40447, 15.4396], [-10.71721, 15.4223], [-10.90932, 15.11001], [-11.43483, 15.62339], [-11.70705, 15.51558], [-11.94903, 14.76143], [-12.23936, 14.76324], [-11.93043, 13.84505], [-12.06897, 13.71049], [-11.83345, 13.33333], [-11.63025, 13.39174], [-11.39935, 12.97808], [-11.37536, 12.40788], [-11.50006, 12.17826], [-11.24136, 12.01286], [-10.99758, 12.24634], [-10.80355, 12.1053], [-10.71897, 11.91552], [-10.30604, 12.24634], [-9.714, 12.0226], [-9.63938, 12.18312], [-9.32097, 12.29009], [-9.38067, 12.48446], [-9.13689, 12.50875], [-8.94784, 12.34842], [-8.80854, 11.66715], [-8.40058, 11.37466], [-8.66923, 10.99397], [-8.35083, 11.06234], [-8.2667, 10.91762], [-8.32614, 10.69273], [-8.22711, 10.41722], [-8.10207, 10.44649], [-7.9578, 10.2703], [-7.97971, 10.17117], [-7.92107, 10.15577], [-7.63048, 10.46334], [-7.54462, 10.40921], [-7.52261, 10.4655], [-7.44555, 10.44602], [-7.3707, 10.24677], [-7.13331, 10.24877], [-7.0603, 10.14711], [-7.00966, 10.15794], [-6.97444, 10.21644], [-7.01186, 10.25111], [-6.93921, 10.35291], [-6.68164, 10.35074], [-6.63541, 10.66893], [-6.52974, 10.59104], [-6.42847, 10.5694], [-6.40646, 10.69922], [-6.325, 10.68624], [-6.24795, 10.74248], [-6.1731, 10.46983], [-6.18851, 10.24244], [-5.99478, 10.19694], [-5.78124, 10.43952], [-5.65135, 10.46767], [-5.51058, 10.43177], [-5.46643, 10.56074], [-5.47083, 10.75329], [-5.41579, 10.84628], [-5.49284, 11.07538], [-5.32994, 11.13371], [-5.32553, 11.21578], [-5.25949, 11.24816], [-5.25509, 11.36905], [-5.20665, 11.43811], [-5.22867, 11.60421], [-5.29251, 11.61715], [-5.26389, 11.75728], [-5.40258, 11.8327], [-5.26389, 11.84778], [-5.07897, 11.97918], [-4.72893, 12.01579], [-4.70692, 12.06746], [-4.62987, 12.06531], [-4.62546, 12.13204], [-4.54841, 12.1385], [-4.57703, 12.19875], [-4.41412, 12.31922], [-4.47356, 12.71252], [-4.238, 12.71467], [-4.21819, 12.95722], [-4.34477, 13.12927], [-3.96501, 13.49778], [-3.90558, 13.44375], [-3.96282, 13.38164], [-3.7911, 13.36665], [-3.54454, 13.1781], [-3.4313, 13.1588], [-3.43507, 13.27272], [-3.23599, 13.29035], [-3.28396, 13.5422], [-3.26407, 13.70699], [-2.88189, 13.64921], [-2.90831, 13.81174], [-2.84667, 14.05532], [-2.66175, 14.14713], [-2.47587, 14.29671], [-2.10223, 14.14878], [-1.9992, 14.19011], [-1.97945, 14.47709], [-1.68083, 14.50023], [-1.32166, 14.72774], [-1.05875, 14.7921], [-0.72004, 15.08655], [-0.24673, 15.07805], [0.06588, 14.96961], [0.23859, 15.00135], [0.72632, 14.95898], [0.96711, 14.98275], [1.31275, 15.27978], [3.01806, 15.34571], [3.03134, 15.42221], [3.50368, 15.35934], [4.19893, 16.39923], [4.21787, 17.00118], [4.26762, 17.00432], [4.26651, 19.14224], [3.36082, 18.9745], [3.12501, 19.1366], [3.24648, 19.81703], [1.20992, 20.73533], [1.15698, 21.12843], [-4.83423, 24.99935]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MM",
+           iso1A3: "MMR",
+           iso1N3: "104",
+           wikidata: "Q836",
+           nameEn: "Myanmar",
+           aliases: ["Burma", "BU"],
+           groups: ["035", "142", "UN"],
+           callingCodes: ["95"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[92.62187, 21.87037], [92.59775, 21.6092], [92.68152, 21.28454], [92.60187, 21.24615], [92.55105, 21.3856], [92.43158, 21.37025], [92.37939, 21.47764], [92.20087, 21.337], [92.17752, 21.17445], [92.26071, 21.05697], [92.47409, 20.38654], [92.61042, 13.76986], [94.6371, 13.81803], [97.63455, 9.60854], [98.12555, 9.44056], [98.33094, 9.91973], [98.47298, 9.95782], [98.52291, 9.92389], [98.55174, 9.92804], [98.7391, 10.31488], [98.81944, 10.52761], [98.77275, 10.62548], [98.78511, 10.68351], [98.86819, 10.78336], [99.0069, 10.85485], [98.99701, 10.92962], [99.02337, 10.97217], [99.06938, 10.94857], [99.32756, 11.28545], [99.31573, 11.32081], [99.39485, 11.3925], [99.47598, 11.62434], [99.5672, 11.62732], [99.64108, 11.78948], [99.64891, 11.82699], [99.53424, 12.02317], [99.56445, 12.14805], [99.47519, 12.1353], [99.409, 12.60603], [99.29254, 12.68921], [99.18905, 12.84799], [99.18748, 12.9898], [99.10646, 13.05804], [99.12225, 13.19847], [99.20617, 13.20575], [99.16695, 13.72621], [98.97356, 14.04868], [98.56762, 14.37701], [98.24874, 14.83013], [98.18821, 15.13125], [98.22, 15.21327], [98.30446, 15.30667], [98.40522, 15.25268], [98.41906, 15.27103], [98.39351, 15.34177], [98.4866, 15.39154], [98.56027, 15.33471], [98.58598, 15.46821], [98.541, 15.65406], [98.59853, 15.87197], [98.57019, 16.04578], [98.69585, 16.13353], [98.8376, 16.11706], [98.92656, 16.36425], [98.84485, 16.42354], [98.68074, 16.27068], [98.63817, 16.47424], [98.57912, 16.55983], [98.5695, 16.62826], [98.51113, 16.64503], [98.51833, 16.676], [98.51472, 16.68521], [98.51579, 16.69433], [98.51043, 16.70107], [98.49713, 16.69022], [98.50253, 16.7139], [98.46994, 16.73613], [98.53833, 16.81934], [98.49603, 16.8446], [98.52624, 16.89979], [98.39441, 17.06266], [98.34566, 17.04822], [98.10439, 17.33847], [98.11185, 17.36829], [97.91829, 17.54504], [97.76407, 17.71595], [97.66794, 17.88005], [97.73723, 17.97912], [97.60841, 18.23846], [97.64116, 18.29778], [97.56219, 18.33885], [97.50383, 18.26844], [97.34522, 18.54596], [97.36444, 18.57138], [97.5258, 18.4939], [97.76752, 18.58097], [97.73836, 18.88478], [97.66487, 18.9371], [97.73654, 18.9812], [97.73797, 19.04261], [97.83479, 19.09972], [97.84024, 19.22217], [97.78606, 19.26769], [97.84186, 19.29526], [97.78769, 19.39429], [97.88423, 19.5041], [97.84715, 19.55782], [98.04364, 19.65755], [98.03314, 19.80941], [98.13829, 19.78541], [98.24884, 19.67876], [98.51182, 19.71303], [98.56065, 19.67807], [98.83661, 19.80931], [98.98679, 19.7419], [99.0735, 20.10298], [99.20328, 20.12877], [99.416, 20.08614], [99.52943, 20.14811], [99.5569, 20.20676], [99.46077, 20.36198], [99.46008, 20.39673], [99.68255, 20.32077], [99.81096, 20.33687], [99.86383, 20.44371], [99.88211, 20.44488], [99.88451, 20.44596], [99.89168, 20.44548], [99.89301, 20.44311], [99.89692, 20.44789], [99.90499, 20.4487], [99.91616, 20.44986], [99.95721, 20.46301], [100.08404, 20.36626], [100.1957, 20.68247], [100.36375, 20.82783], [100.51079, 20.82194], [100.60112, 20.8347], [100.64628, 20.88279], [100.50974, 20.88574], [100.55281, 21.02796], [100.63578, 21.05639], [100.72716, 21.31786], [100.80173, 21.2934], [101.00234, 21.39612], [101.16198, 21.52808], [101.15156, 21.56129], [101.11744, 21.77659], [100.87265, 21.67396], [100.72143, 21.51898], [100.57861, 21.45637], [100.4811, 21.46148], [100.42892, 21.54325], [100.35201, 21.53176], [100.25863, 21.47043], [100.18447, 21.51898], [100.1625, 21.48704], [100.12542, 21.50365], [100.10757, 21.59945], [100.17486, 21.65306], [100.12679, 21.70539], [100.04956, 21.66843], [99.98654, 21.71064], [99.94003, 21.82782], [99.99084, 21.97053], [99.96612, 22.05965], [99.85351, 22.04183], [99.47585, 22.13345], [99.33166, 22.09656], [99.1552, 22.15874], [99.19176, 22.16983], [99.17318, 22.18025], [99.28771, 22.4105], [99.37972, 22.50188], [99.38247, 22.57544], [99.31243, 22.73893], [99.45654, 22.85726], [99.43537, 22.94086], [99.54218, 22.90014], [99.52214, 23.08218], [99.34127, 23.13099], [99.25741, 23.09025], [99.04601, 23.12215], [99.05975, 23.16382], [98.88597, 23.18656], [98.92515, 23.29535], [98.93958, 23.31414], [98.87573, 23.33038], [98.92104, 23.36946], [98.87683, 23.48995], [98.82877, 23.47908], [98.80294, 23.5345], [98.88396, 23.59555], [98.81775, 23.694], [98.82933, 23.72921], [98.79607, 23.77947], [98.68209, 23.80492], [98.67797, 23.9644], [98.89632, 24.10612], [98.87998, 24.15624], [98.85319, 24.13042], [98.59256, 24.08371], [98.54476, 24.13119], [98.20666, 24.11406], [98.07806, 24.07988], [98.06703, 24.08028], [98.0607, 24.07812], [98.05671, 24.07961], [98.05302, 24.07408], [98.04709, 24.07616], [97.99583, 24.04932], [97.98691, 24.03897], [97.93951, 24.01953], [97.90998, 24.02094], [97.88616, 24.00463], [97.88414, 23.99405], [97.88814, 23.98605], [97.89683, 23.98389], [97.89676, 23.97931], [97.8955, 23.97758], [97.88811, 23.97446], [97.86545, 23.97723], [97.84328, 23.97603], [97.79416, 23.95663], [97.79456, 23.94836], [97.72302, 23.89288], [97.64667, 23.84574], [97.5247, 23.94032], [97.62363, 24.00506], [97.72903, 24.12606], [97.75305, 24.16902], [97.72799, 24.18883], [97.72998, 24.2302], [97.76799, 24.26365], [97.71941, 24.29652], [97.66723, 24.30027], [97.65624, 24.33781], [97.7098, 24.35658], [97.66998, 24.45288], [97.60029, 24.4401], [97.52757, 24.43748], [97.56286, 24.54535], [97.56525, 24.72838], [97.54675, 24.74202], [97.5542, 24.74943], [97.56383, 24.75535], [97.56648, 24.76475], [97.64354, 24.79171], [97.70181, 24.84557], [97.73127, 24.83015], [97.76481, 24.8289], [97.79949, 24.85655], [97.72903, 24.91332], [97.72216, 25.08508], [97.77023, 25.11492], [97.83614, 25.2715], [97.92541, 25.20815], [98.14925, 25.41547], [98.12591, 25.50722], [98.18084, 25.56298], [98.16848, 25.62739], [98.25774, 25.6051], [98.31268, 25.55307], [98.40606, 25.61129], [98.54064, 25.85129], [98.63128, 25.79937], [98.70818, 25.86241], [98.60763, 26.01512], [98.57085, 26.11547], [98.63128, 26.15492], [98.66884, 26.09165], [98.7329, 26.17218], [98.67797, 26.24487], [98.72741, 26.36183], [98.77547, 26.61994], [98.7333, 26.85615], [98.69582, 27.56499], [98.43353, 27.67086], [98.42529, 27.55404], [98.32641, 27.51385], [98.13964, 27.9478], [98.15337, 28.12114], [97.90069, 28.3776], [97.79632, 28.33168], [97.70705, 28.5056], [97.56835, 28.55628], [97.50518, 28.49716], [97.47085, 28.2688], [97.41729, 28.29783], [97.34547, 28.21385], [97.31292, 28.06784], [97.35412, 28.06663], [97.38845, 28.01329], [97.35824, 27.87256], [97.29919, 27.92233], [96.90112, 27.62149], [96.91431, 27.45752], [97.17422, 27.14052], [97.14675, 27.09041], [96.89132, 27.17474], [96.85287, 27.2065], [96.88445, 27.25046], [96.73888, 27.36638], [96.55761, 27.29928], [96.40779, 27.29818], [96.15591, 27.24572], [96.04949, 27.19428], [95.93002, 27.04149], [95.81603, 27.01335], [95.437, 26.7083], [95.30339, 26.65372], [95.23513, 26.68499], [95.05798, 26.45408], [95.12801, 26.38397], [95.11428, 26.1019], [95.18556, 26.07338], [94.80117, 25.49359], [94.68032, 25.47003], [94.57458, 25.20318], [94.74212, 25.13606], [94.73937, 25.00545], [94.60204, 24.70889], [94.5526, 24.70764], [94.50729, 24.59281], [94.45279, 24.56656], [94.32362, 24.27692], [94.30215, 24.23752], [94.14081, 23.83333], [93.92089, 23.95812], [93.80279, 23.92549], [93.75952, 24.0003], [93.62871, 24.00922], [93.50616, 23.94432], [93.46633, 23.97067], [93.41415, 24.07854], [93.34735, 24.10151], [93.32351, 24.04468], [93.36059, 23.93176], [93.3908, 23.92925], [93.3908, 23.7622], [93.43475, 23.68299], [93.38805, 23.4728], [93.39981, 23.38828], [93.38781, 23.36139], [93.36862, 23.35426], [93.38478, 23.13698], [93.2878, 23.00464], [93.12988, 23.05772], [93.134, 22.92498], [93.09417, 22.69459], [93.134, 22.59573], [93.11477, 22.54374], [93.13537, 22.45873], [93.18206, 22.43716], [93.19991, 22.25425], [93.14224, 22.24535], [93.15734, 22.18687], [93.04885, 22.20595], [92.99255, 22.05965], [92.99804, 21.98964], [92.93899, 22.02656], [92.89504, 21.95143], [92.86208, 22.05456], [92.70416, 22.16017], [92.67532, 22.03547], [92.60949, 21.97638], [92.62187, 21.87037]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MN",
+           iso1A3: "MNG",
+           iso1N3: "496",
+           wikidata: "Q711",
+           nameEn: "Mongolia",
+           groups: ["030", "142", "UN"],
+           callingCodes: ["976"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[102.14032, 51.35566], [101.5044, 51.50467], [101.39085, 51.45753], [100.61116, 51.73028], [99.89203, 51.74903], [99.75578, 51.90108], [99.27888, 51.96876], [98.87768, 52.14563], [98.74142, 51.8637], [98.33222, 51.71832], [98.22053, 51.46579], [98.05257, 51.46696], [97.83305, 51.00248], [98.01472, 50.86652], [97.9693, 50.78044], [98.06393, 50.61262], [98.31373, 50.4996], [98.29481, 50.33561], [97.85197, 49.91339], [97.76871, 49.99861], [97.56432, 49.92801], [97.56811, 49.84265], [97.24639, 49.74737], [96.97388, 49.88413], [95.80056, 50.04239], [95.74757, 49.97915], [95.02465, 49.96941], [94.97166, 50.04725], [94.6121, 50.04239], [94.49477, 50.17832], [94.39258, 50.22193], [94.30823, 50.57498], [92.99595, 50.63183], [93.01109, 50.79001], [92.44714, 50.78762], [92.07173, 50.69585], [91.86048, 50.73734], [89.59711, 49.90851], [89.70687, 49.72535], [88.82499, 49.44808], [88.42449, 49.48821], [88.17223, 49.46934], [88.15543, 49.30314], [87.98977, 49.18147], [87.81333, 49.17354], [87.88171, 48.95853], [87.73822, 48.89582], [88.0788, 48.71436], [87.96361, 48.58478], [88.58939, 48.34531], [88.58316, 48.21893], [88.8011, 48.11302], [88.93186, 48.10263], [89.0711, 47.98528], [89.55453, 48.0423], [89.76624, 47.82745], [90.06512, 47.88177], [90.10871, 47.7375], [90.33598, 47.68303], [90.48854, 47.41826], [90.48542, 47.30438], [90.76108, 46.99399], [90.84035, 46.99525], [91.03649, 46.72916], [91.0147, 46.58171], [91.07696, 46.57315], [90.89639, 46.30711], [90.99672, 46.14207], [91.03026, 46.04194], [90.70907, 45.73437], [90.65114, 45.49314], [90.89169, 45.19667], [91.64048, 45.07408], [93.51161, 44.95964], [94.10003, 44.71016], [94.71959, 44.35284], [95.01191, 44.25274], [95.39772, 44.2805], [95.32891, 44.02407], [95.52594, 43.99353], [95.89543, 43.2528], [96.35658, 42.90363], [96.37926, 42.72055], [97.1777, 42.7964], [99.50671, 42.56535], [100.33297, 42.68231], [100.84979, 42.67087], [101.80515, 42.50074], [102.07645, 42.22519], [102.72403, 42.14675], [103.92804, 41.78246], [104.52258, 41.8706], [104.51667, 41.66113], [105.0123, 41.63188], [106.76517, 42.28741], [107.24774, 42.36107], [107.29755, 42.41395], [107.49681, 42.46221], [107.57258, 42.40898], [108.84489, 42.40246], [109.00679, 42.45302], [109.452, 42.44842], [109.89402, 42.63111], [110.08401, 42.6411], [110.4327, 42.78293], [111.0149, 43.3289], [111.59087, 43.51207], [111.79758, 43.6637], [111.93776, 43.68709], [111.96289, 43.81596], [111.40498, 44.3461], [111.76275, 44.98032], [111.98695, 45.09074], [112.4164, 45.06858], [112.74662, 44.86297], [113.70918, 44.72891], [114.5166, 45.27189], [114.54801, 45.38337], [114.74612, 45.43585], [114.94546, 45.37377], [115.60329, 45.44717], [116.16989, 45.68603], [116.27366, 45.78637], [116.24012, 45.8778], [116.26678, 45.96479], [116.58612, 46.30211], [116.75551, 46.33083], [116.83166, 46.38637], [117.36609, 46.36335], [117.41782, 46.57862], [117.60748, 46.59771], [117.69554, 46.50991], [118.30534, 46.73519], [118.78747, 46.68689], [118.8337, 46.77742], [118.89974, 46.77139], [118.92616, 46.72765], [119.00541, 46.74273], [119.10448, 46.65516], [119.24978, 46.64761], [119.32827, 46.61433], [119.42827, 46.63783], [119.65265, 46.62342], [119.68127, 46.59015], [119.77373, 46.62947], [119.80455, 46.67631], [119.89261, 46.66423], [119.91242, 46.90091], [119.85518, 46.92196], [119.71209, 47.19192], [119.62403, 47.24575], [119.56019, 47.24874], [119.54918, 47.29505], [119.31964, 47.42617], [119.35892, 47.48104], [119.13995, 47.53997], [119.12343, 47.66458], [118.7564, 47.76947], [118.55766, 47.99277], [118.29654, 48.00246], [118.22677, 48.03853], [118.11009, 48.04], [118.03676, 48.00982], [117.80196, 48.01661], [117.50181, 47.77216], [117.37875, 47.63627], [116.9723, 47.87285], [116.67405, 47.89039], [116.4465, 47.83662], [116.21879, 47.88505], [115.94296, 47.67741], [115.57128, 47.91988], [115.52082, 48.15367], [115.811, 48.25699], [115.78876, 48.51781], [116.06565, 48.81716], [116.03781, 48.87014], [116.71193, 49.83813], [116.62502, 49.92919], [116.22402, 50.04477], [115.73602, 49.87688], [115.26068, 49.97367], [114.9703, 50.19254], [114.325, 50.28098], [113.20216, 49.83356], [113.02647, 49.60772], [110.64493, 49.1816], [110.39891, 49.25083], [110.24373, 49.16676], [109.51325, 49.22859], [109.18017, 49.34709], [108.53969, 49.32325], [108.27937, 49.53167], [107.95387, 49.66659], [107.96116, 49.93191], [107.36407, 49.97612], [107.1174, 50.04239], [107.00007, 50.1977], [106.80326, 50.30177], [106.58373, 50.34044], [106.51122, 50.34408], [106.49628, 50.32436], [106.47156, 50.31909], [106.07865, 50.33474], [106.05562, 50.40582], [105.32528, 50.4648], [103.70343, 50.13952], [102.71178, 50.38873], [102.32194, 50.67982], [102.14032, 51.35566]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MO",
+           iso1A3: "MAC",
+           iso1N3: "446",
+           wikidata: "Q14773",
+           nameEn: "Macau",
+           aliases: ["Macao"],
+           country: "CN",
+           groups: ["030", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["853"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[113.54942, 22.14519], [113.54839, 22.10909], [113.57191, 22.07696], [113.63011, 22.10782], [113.60504, 22.20464], [113.57123, 22.20416], [113.56865, 22.20973], [113.5508, 22.21672], [113.54333, 22.21688], [113.54093, 22.21314], [113.53593, 22.2137], [113.53301, 22.21235], [113.53552, 22.20607], [113.52659, 22.18271], [113.54093, 22.15497], [113.54942, 22.14519]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MP",
+           iso1A3: "MNP",
+           iso1N3: "580",
+           wikidata: "Q16644",
+           nameEn: "Northern Mariana Islands",
+           aliases: ["US-MP"],
+           country: "US",
+           groups: ["Q1352230", "Q153732", "057", "009", "UN"],
+           roadSpeedUnit: "mph",
+           callingCodes: ["1 670"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[135.52896, 14.32623], [152.19114, 13.63487], [145.05972, 21.28731], [135.52896, 14.32623]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MQ",
+           iso1A3: "MTQ",
+           iso1N3: "474",
+           wikidata: "Q17054",
+           nameEn: "Martinique",
+           country: "FR",
+           groups: ["Q3320166", "EU", "029", "003", "419", "019", "UN"],
+           callingCodes: ["596"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-59.95997, 14.20285], [-61.07821, 15.25109], [-61.69315, 14.26451], [-59.95997, 14.20285]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MR",
+           iso1A3: "MRT",
+           iso1N3: "478",
+           wikidata: "Q1025",
+           nameEn: "Mauritania",
+           groups: ["011", "202", "002", "UN"],
+           callingCodes: ["222"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-5.60725, 16.49919], [-6.57191, 25.0002], [-4.83423, 24.99935], [-8.66674, 27.31569], [-8.66721, 25.99918], [-12.0002, 25.9986], [-12.00251, 23.4538], [-12.14969, 23.41935], [-12.36213, 23.3187], [-12.5741, 23.28975], [-13.00412, 23.02297], [-13.10753, 22.89493], [-13.15313, 22.75649], [-13.08438, 22.53866], [-13.01525, 21.33343], [-16.95474, 21.33997], [-16.99806, 21.12142], [-17.0357, 21.05368], [-17.0396, 20.9961], [-17.06781, 20.92697], [-17.0695, 20.85742], [-17.0471, 20.76408], [-17.15288, 16.07139], [-16.50854, 16.09032], [-16.48967, 16.0496], [-16.44814, 16.09753], [-16.4429, 16.20605], [-16.27016, 16.51565], [-15.6509, 16.50315], [-15.00557, 16.64997], [-14.32144, 16.61495], [-13.80075, 16.13961], [-13.43135, 16.09022], [-13.11029, 15.52116], [-12.23936, 14.76324], [-11.94903, 14.76143], [-11.70705, 15.51558], [-11.43483, 15.62339], [-10.90932, 15.11001], [-10.71721, 15.4223], [-9.40447, 15.4396], [-9.44673, 15.60553], [-9.33314, 15.7044], [-9.31106, 15.69412], [-9.32979, 15.50032], [-5.50165, 15.50061], [-5.33435, 16.33354], [-5.60725, 16.49919]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MS",
+           iso1A3: "MSR",
+           iso1N3: "500",
+           wikidata: "Q13353",
+           nameEn: "Montserrat",
+           country: "GB",
+           groups: ["BOTS", "029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["1 664"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-61.91508, 16.51165], [-62.1023, 16.97277], [-62.58307, 16.68909], [-61.91508, 16.51165]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MT",
+           iso1A3: "MLT",
+           iso1N3: "470",
+           wikidata: "Q233",
+           nameEn: "Malta",
+           groups: ["EU", "039", "150", "UN"],
+           driveSide: "left",
+           callingCodes: ["356"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[15.70991, 35.79901], [14.07544, 36.41525], [13.27636, 35.20764], [15.70991, 35.79901]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MU",
+           iso1A3: "MUS",
+           iso1N3: "480",
+           wikidata: "Q1027",
+           nameEn: "Mauritius",
+           groups: ["014", "202", "002", "UN"],
+           driveSide: "left",
+           callingCodes: ["230"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[56.09755, -9.55401], [57.50644, -31.92637], [68.4673, -19.15185], [56.09755, -9.55401]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MV",
+           iso1A3: "MDV",
+           iso1N3: "462",
+           wikidata: "Q826",
+           nameEn: "Maldives",
+           groups: ["034", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["960"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[71.9161, 8.55531], [72.57428, -3.7623], [76.59015, 5.591], [71.9161, 8.55531]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MW",
+           iso1A3: "MWI",
+           iso1N3: "454",
+           wikidata: "Q1020",
+           nameEn: "Malawi",
+           groups: ["014", "202", "002", "UN"],
+           driveSide: "left",
+           callingCodes: ["265"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[33.48052, -9.62442], [33.31581, -9.48554], [33.14925, -9.49322], [32.99397, -9.36712], [32.95389, -9.40138], [33.00476, -9.5133], [33.00256, -9.63053], [33.05485, -9.61316], [33.10163, -9.66525], [33.12144, -9.58929], [33.2095, -9.61099], [33.31517, -9.82364], [33.36581, -9.81063], [33.37902, -9.9104], [33.31297, -10.05133], [33.53863, -10.20148], [33.54797, -10.36077], [33.70675, -10.56896], [33.47636, -10.78465], [33.28022, -10.84428], [33.25998, -10.88862], [33.39697, -11.15296], [33.29267, -11.3789], [33.29267, -11.43536], [33.23663, -11.40637], [33.24252, -11.59302], [33.32692, -11.59248], [33.33937, -11.91252], [33.25998, -12.14242], [33.3705, -12.34931], [33.47636, -12.32498], [33.54485, -12.35996], [33.37517, -12.54085], [33.28177, -12.54692], [33.18837, -12.61377], [33.05917, -12.59554], [32.94397, -12.76868], [32.96733, -12.88251], [33.02181, -12.88707], [32.98289, -13.12671], [33.0078, -13.19492], [32.86113, -13.47292], [32.84176, -13.52794], [32.73683, -13.57682], [32.68436, -13.55769], [32.66468, -13.60019], [32.68654, -13.64268], [32.7828, -13.64805], [32.84528, -13.71576], [32.76962, -13.77224], [32.79015, -13.80755], [32.88985, -13.82956], [32.99042, -13.95689], [33.02977, -14.05022], [33.07568, -13.98447], [33.16749, -13.93992], [33.24249, -14.00019], [33.66677, -14.61306], [33.7247, -14.4989], [33.88503, -14.51652], [33.92898, -14.47929], [34.08588, -14.48893], [34.18733, -14.43823], [34.22355, -14.43607], [34.34453, -14.3985], [34.35843, -14.38652], [34.39277, -14.39467], [34.4192, -14.43191], [34.44641, -14.47746], [34.45053, -14.49873], [34.47628, -14.53363], [34.48932, -14.53646], [34.49636, -14.55091], [34.52366, -14.5667], [34.53962, -14.59776], [34.55112, -14.64494], [34.53516, -14.67782], [34.52057, -14.68263], [34.54503, -14.74672], [34.567, -14.77345], [34.61522, -14.99583], [34.57503, -15.30619], [34.43126, -15.44778], [34.44981, -15.60864], [34.25195, -15.90321], [34.43126, -16.04737], [34.40344, -16.20923], [35.04805, -16.83167], [35.13771, -16.81687], [35.17017, -16.93521], [35.04805, -17.00027], [35.0923, -17.13235], [35.3062, -17.1244], [35.27065, -16.93817], [35.30929, -16.82871], [35.27219, -16.69402], [35.14235, -16.56812], [35.25828, -16.4792], [35.30157, -16.2211], [35.43355, -16.11371], [35.52365, -16.15414], [35.70107, -16.10147], [35.80487, -16.03907], [35.85303, -15.41913], [35.78799, -15.17428], [35.91812, -14.89514], [35.87212, -14.89478], [35.86945, -14.67481], [35.5299, -14.27714], [35.47989, -14.15594], [34.86229, -13.48958], [34.60253, -13.48487], [34.37831, -12.17408], [34.46088, -12.0174], [34.70739, -12.15652], [34.82903, -12.04837], [34.57917, -11.87849], [34.64241, -11.57499], [34.96296, -11.57354], [34.91153, -11.39799], [34.79375, -11.32245], [34.63305, -11.11731], [34.61161, -11.01611], [34.67047, -10.93796], [34.65946, -10.6828], [34.57581, -10.56271], [34.51911, -10.12279], [34.54499, -10.0678], [34.03865, -9.49398], [33.95829, -9.54066], [33.9638, -9.62206], [33.93298, -9.71647], [33.76677, -9.58516], [33.48052, -9.62442]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MX",
+           iso1A3: "MEX",
+           iso1N3: "484",
+           wikidata: "Q96",
+           nameEn: "Mexico",
+           groups: ["013", "003", "419", "019", "UN"],
+           callingCodes: ["52"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-117.1243, 32.53427], [-118.48109, 32.5991], [-120.12904, 18.41089], [-92.37213, 14.39277], [-92.2261, 14.53423], [-92.1454, 14.6804], [-92.18161, 14.84147], [-92.1423, 14.88647], [-92.1454, 14.98143], [-92.0621, 15.07406], [-92.20983, 15.26077], [-91.73182, 16.07371], [-90.44567, 16.07573], [-90.40499, 16.40524], [-90.61212, 16.49832], [-90.69064, 16.70697], [-91.04436, 16.92175], [-91.43809, 17.25373], [-90.99199, 17.25192], [-90.98678, 17.81655], [-89.14985, 17.81563], [-89.15105, 17.95104], [-89.03839, 18.0067], [-88.8716, 17.89535], [-88.71505, 18.0707], [-88.48242, 18.49164], [-88.3268, 18.49048], [-88.29909, 18.47591], [-88.26593, 18.47617], [-88.03238, 18.41778], [-88.03165, 18.16657], [-87.90671, 18.15213], [-87.87604, 18.18313], [-87.86657, 18.19971], [-87.85693, 18.18266], [-87.84815, 18.18511], [-86.92368, 17.61462], [-85.9092, 21.8218], [-96.92418, 25.97377], [-97.13927, 25.96583], [-97.35946, 25.92189], [-97.37332, 25.83854], [-97.42511, 25.83969], [-97.45669, 25.86874], [-97.49828, 25.89877], [-97.52025, 25.88518], [-97.66511, 26.01708], [-97.95155, 26.0625], [-97.97017, 26.05232], [-98.24603, 26.07191], [-98.27075, 26.09457], [-98.30491, 26.10475], [-98.35126, 26.15129], [-99.00546, 26.3925], [-99.03053, 26.41249], [-99.08477, 26.39849], [-99.53573, 27.30926], [-99.49744, 27.43746], [-99.482, 27.47128], [-99.48045, 27.49016], [-99.50208, 27.50021], [-99.52955, 27.49747], [-99.51478, 27.55836], [-99.55409, 27.61314], [-100.50029, 28.66117], [-100.51222, 28.70679], [-100.5075, 28.74066], [-100.52313, 28.75598], [-100.59809, 28.88197], [-100.63689, 28.90812], [-100.67294, 29.09744], [-100.79696, 29.24688], [-100.87982, 29.296], [-100.94056, 29.33371], [-100.94579, 29.34523], [-100.96725, 29.3477], [-101.01128, 29.36947], [-101.05686, 29.44738], [-101.47277, 29.7744], [-102.60596, 29.8192], [-103.15787, 28.93865], [-104.37752, 29.54255], [-104.39363, 29.55396], [-104.3969, 29.57105], [-104.5171, 29.64671], [-104.77674, 30.4236], [-106.00363, 31.39181], [-106.09025, 31.40569], [-106.20346, 31.46305], [-106.23711, 31.51262], [-106.24612, 31.54193], [-106.28084, 31.56173], [-106.30305, 31.62154], [-106.33419, 31.66303], [-106.34864, 31.69663], [-106.3718, 31.71165], [-106.38003, 31.73151], [-106.41773, 31.75196], [-106.43419, 31.75478], [-106.45244, 31.76523], [-106.46726, 31.75998], [-106.47298, 31.75054], [-106.48815, 31.74769], [-106.50111, 31.75714], [-106.50962, 31.76155], [-106.51251, 31.76922], [-106.52266, 31.77509], [-106.529, 31.784], [-108.20899, 31.78534], [-108.20979, 31.33316], [-111.07523, 31.33232], [-114.82011, 32.49609], [-114.79524, 32.55731], [-114.81141, 32.55543], [-114.80584, 32.62028], [-114.76736, 32.64094], [-114.71871, 32.71894], [-115.88053, 32.63624], [-117.1243, 32.53427]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MY",
+           iso1A3: "MYS",
+           iso1N3: "458",
+           wikidata: "Q833",
+           nameEn: "Malaysia"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "MZ",
+           iso1A3: "MOZ",
+           iso1N3: "508",
+           wikidata: "Q1029",
+           nameEn: "Mozambique",
+           groups: ["014", "202", "002", "UN"],
+           driveSide: "left",
+           callingCodes: ["258"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[40.74206, -10.25691], [40.44265, -10.4618], [40.00295, -10.80255], [39.58249, -10.96043], [39.24395, -11.17433], [38.88996, -11.16978], [38.47258, -11.4199], [38.21598, -11.27289], [37.93618, -11.26228], [37.8388, -11.3123], [37.76614, -11.53352], [37.3936, -11.68949], [36.80309, -11.56836], [36.62068, -11.72884], [36.19094, -11.70008], [36.19094, -11.57593], [35.82767, -11.41081], [35.63599, -11.55927], [34.96296, -11.57354], [34.64241, -11.57499], [34.57917, -11.87849], [34.82903, -12.04837], [34.70739, -12.15652], [34.46088, -12.0174], [34.37831, -12.17408], [34.60253, -13.48487], [34.86229, -13.48958], [35.47989, -14.15594], [35.5299, -14.27714], [35.86945, -14.67481], [35.87212, -14.89478], [35.91812, -14.89514], [35.78799, -15.17428], [35.85303, -15.41913], [35.80487, -16.03907], [35.70107, -16.10147], [35.52365, -16.15414], [35.43355, -16.11371], [35.30157, -16.2211], [35.25828, -16.4792], [35.14235, -16.56812], [35.27219, -16.69402], [35.30929, -16.82871], [35.27065, -16.93817], [35.3062, -17.1244], [35.0923, -17.13235], [35.04805, -17.00027], [35.17017, -16.93521], [35.13771, -16.81687], [35.04805, -16.83167], [34.40344, -16.20923], [34.43126, -16.04737], [34.25195, -15.90321], [34.44981, -15.60864], [34.43126, -15.44778], [34.57503, -15.30619], [34.61522, -14.99583], [34.567, -14.77345], [34.54503, -14.74672], [34.52057, -14.68263], [34.53516, -14.67782], [34.55112, -14.64494], [34.53962, -14.59776], [34.52366, -14.5667], [34.49636, -14.55091], [34.48932, -14.53646], [34.47628, -14.53363], [34.45053, -14.49873], [34.44641, -14.47746], [34.4192, -14.43191], [34.39277, -14.39467], [34.35843, -14.38652], [34.34453, -14.3985], [34.22355, -14.43607], [34.18733, -14.43823], [34.08588, -14.48893], [33.92898, -14.47929], [33.88503, -14.51652], [33.7247, -14.4989], [33.66677, -14.61306], [33.24249, -14.00019], [30.22098, -14.99447], [30.41902, -15.62269], [30.42568, -15.9962], [30.91597, -15.99924], [30.97761, -16.05848], [31.13171, -15.98019], [31.30563, -16.01193], [31.42451, -16.15154], [31.67988, -16.19595], [31.90223, -16.34388], [31.91324, -16.41569], [32.02772, -16.43892], [32.28529, -16.43892], [32.42838, -16.4727], [32.71017, -16.59932], [32.69917, -16.66893], [32.78943, -16.70267], [32.97655, -16.70689], [32.91051, -16.89446], [32.84113, -16.92259], [32.96554, -17.11971], [33.00517, -17.30477], [33.0426, -17.3468], [32.96554, -17.48964], [32.98536, -17.55891], [33.0492, -17.60298], [32.94133, -17.99705], [33.03159, -18.35054], [33.02278, -18.4696], [32.88629, -18.51344], [32.88629, -18.58023], [32.95013, -18.69079], [32.9017, -18.7992], [32.82465, -18.77419], [32.70137, -18.84712], [32.73439, -18.92628], [32.69917, -18.94293], [32.72118, -19.02204], [32.84006, -19.0262], [32.87088, -19.09279], [32.85107, -19.29238], [32.77966, -19.36098], [32.78282, -19.47513], [32.84446, -19.48343], [32.84666, -19.68462], [32.95013, -19.67219], [33.06461, -19.77787], [33.01178, -20.02007], [32.93032, -20.03868], [32.85987, -20.16686], [32.85987, -20.27841], [32.66174, -20.56106], [32.55167, -20.56312], [32.48122, -20.63319], [32.51644, -20.91929], [32.37115, -21.133], [32.48236, -21.32873], [32.41234, -21.31246], [31.38336, -22.36919], [31.30611, -22.422], [31.55779, -23.176], [31.56539, -23.47268], [31.67942, -23.60858], [31.70223, -23.72695], [31.77445, -23.90082], [31.87707, -23.95293], [31.90368, -24.18892], [31.9835, -24.29983], [32.03196, -25.10785], [32.01676, -25.38117], [31.97875, -25.46356], [32.00631, -25.65044], [31.92649, -25.84216], [31.974, -25.95387], [32.00916, -25.999], [32.08599, -26.00978], [32.10435, -26.15656], [32.07352, -26.40185], [32.13409, -26.5317], [32.13315, -26.84345], [32.19409, -26.84032], [32.22302, -26.84136], [32.29584, -26.852], [32.35222, -26.86027], [34.51034, -26.91792], [42.99868, -12.65261], [40.74206, -10.25691]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "NA",
+           iso1A3: "NAM",
+           iso1N3: "516",
+           wikidata: "Q1030",
+           nameEn: "Namibia",
+           groups: ["018", "202", "002", "UN"],
+           driveSide: "left",
+           callingCodes: ["264"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[14.28743, -17.38814], [13.95896, -17.43141], [13.36212, -16.98048], [12.97145, -16.98567], [12.52111, -17.24495], [12.07076, -17.15165], [11.75063, -17.25013], [10.5065, -17.25284], [12.51595, -32.27486], [16.45332, -28.63117], [16.46592, -28.57126], [16.59922, -28.53246], [16.90446, -28.057], [17.15405, -28.08573], [17.4579, -28.68718], [18.99885, -28.89165], [19.99882, -28.42622], [19.99817, -24.76768], [19.99912, -21.99991], [20.99751, -22.00026], [20.99904, -18.31743], [21.45556, -18.31795], [23.0996, -18.00075], [23.29618, -17.99855], [23.61088, -18.4881], [24.19416, -18.01919], [24.40577, -17.95726], [24.57485, -18.07151], [24.6303, -17.9863], [24.71887, -17.9218], [24.73364, -17.89338], [24.95586, -17.79674], [25.05895, -17.84452], [25.16882, -17.78253], [25.26433, -17.79571], [25.00198, -17.58221], [24.70864, -17.49501], [24.5621, -17.52963], [24.38712, -17.46818], [24.32811, -17.49082], [24.23619, -17.47489], [23.47474, -17.62877], [21.42741, -18.02787], [21.14283, -17.94318], [18.84226, -17.80375], [18.39229, -17.38927], [14.28743, -17.38814]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "NC",
+           iso1A3: "NCL",
+           iso1N3: "540",
+           wikidata: "Q33788",
+           nameEn: "New Caledonia",
+           country: "FR",
+           groups: ["Q1451600", "054", "009", "UN"],
+           callingCodes: ["687"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[159.77159, -28.41151], [174.245, -23.1974], [156.73836, -14.50464], [159.77159, -28.41151]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "NE",
+           iso1A3: "NER",
+           iso1N3: "562",
+           wikidata: "Q1032",
+           nameEn: "Niger",
+           aliases: ["RN"],
+           groups: ["011", "202", "002", "UN"],
+           callingCodes: ["227"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[14.22918, 22.61719], [13.5631, 23.16574], [11.96886, 23.51735], [7.48273, 20.87258], [7.38361, 20.79165], [5.8153, 19.45101], [4.26651, 19.14224], [4.26762, 17.00432], [4.21787, 17.00118], [4.19893, 16.39923], [3.50368, 15.35934], [3.03134, 15.42221], [3.01806, 15.34571], [1.31275, 15.27978], [0.96711, 14.98275], [0.72632, 14.95898], [0.23859, 15.00135], [0.16936, 14.51654], [0.38051, 14.05575], [0.61924, 13.68491], [0.77377, 13.6866], [0.77637, 13.64442], [0.99514, 13.5668], [1.02813, 13.46635], [1.20088, 13.38951], [1.24429, 13.39373], [1.28509, 13.35488], [1.24516, 13.33968], [1.21217, 13.37853], [1.18873, 13.31771], [0.99253, 13.37515], [0.99167, 13.10727], [2.26349, 12.41915], [2.05785, 12.35539], [2.39723, 11.89473], [2.45824, 11.98672], [2.39657, 12.10952], [2.37783, 12.24804], [2.6593, 12.30631], [2.83978, 12.40585], [3.25352, 12.01467], [3.31613, 11.88495], [3.48187, 11.86092], [3.59375, 11.70269], [3.61075, 11.69181], [3.67988, 11.75429], [3.67122, 11.80865], [3.63063, 11.83042], [3.61955, 11.91847], [3.67775, 11.97599], [3.63136, 12.11826], [3.66364, 12.25884], [3.65111, 12.52223], [3.94339, 12.74979], [4.10006, 12.98862], [4.14367, 13.17189], [4.14186, 13.47586], [4.23456, 13.47725], [4.4668, 13.68286], [4.87425, 13.78], [4.9368, 13.7345], [5.07396, 13.75052], [5.21026, 13.73627], [5.27797, 13.75474], [5.35437, 13.83567], [5.52957, 13.8845], [6.15771, 13.64564], [6.27411, 13.67835], [6.43053, 13.6006], [6.69617, 13.34057], [6.94445, 12.99825], [7.0521, 13.00076], [7.12676, 13.02445], [7.22399, 13.1293], [7.39241, 13.09717], [7.81085, 13.34902], [8.07997, 13.30847], [8.25185, 13.20369], [8.41853, 13.06166], [8.49493, 13.07519], [8.60431, 13.01768], [8.64251, 12.93985], [8.97413, 12.83661], [9.65995, 12.80614], [10.00373, 13.18171], [10.19993, 13.27129], [10.46731, 13.28819], [10.66004, 13.36422], [11.4535, 13.37773], [11.88236, 13.2527], [12.04209, 13.14452], [12.16189, 13.10056], [12.19315, 13.12423], [12.47095, 13.06673], [12.58033, 13.27805], [12.6793, 13.29157], [12.87376, 13.48919], [13.05085, 13.53984], [13.19844, 13.52802], [13.33213, 13.71195], [13.6302, 13.71094], [13.47559, 14.40881], [13.48259, 14.46704], [13.68573, 14.55276], [13.67878, 14.64013], [13.809, 14.72915], [13.78991, 14.87519], [13.86301, 15.04043], [14.37425, 15.72591], [15.50373, 16.89649], [15.6032, 18.77402], [15.75098, 19.93002], [15.99632, 20.35364], [15.6721, 20.70069], [15.59841, 20.74039], [15.56004, 20.79488], [15.55382, 20.86507], [15.57248, 20.92138], [15.62515, 20.95395], [15.28332, 21.44557], [15.20213, 21.49365], [15.19692, 21.99339], [14.99751, 23.00539], [14.22918, 22.61719]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "NF",
+           iso1A3: "NFK",
+           iso1N3: "574",
+           wikidata: "Q31057",
+           nameEn: "Norfolk Island",
+           country: "AU",
+           groups: ["053", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["672 3"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[169.82316, -28.16667], [166.29505, -28.29175], [167.94076, -30.60745], [169.82316, -28.16667]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "NG",
+           iso1A3: "NGA",
+           iso1N3: "566",
+           wikidata: "Q1033",
+           nameEn: "Nigeria",
+           groups: ["011", "202", "002", "UN"],
+           callingCodes: ["234"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[6.15771, 13.64564], [5.52957, 13.8845], [5.35437, 13.83567], [5.27797, 13.75474], [5.21026, 13.73627], [5.07396, 13.75052], [4.9368, 13.7345], [4.87425, 13.78], [4.4668, 13.68286], [4.23456, 13.47725], [4.14186, 13.47586], [4.14367, 13.17189], [4.10006, 12.98862], [3.94339, 12.74979], [3.65111, 12.52223], [3.66364, 12.25884], [3.63136, 12.11826], [3.67775, 11.97599], [3.61955, 11.91847], [3.63063, 11.83042], [3.67122, 11.80865], [3.67988, 11.75429], [3.61075, 11.69181], [3.59375, 11.70269], [3.49175, 11.29765], [3.71505, 11.13015], [3.84243, 10.59316], [3.78292, 10.40538], [3.6844, 10.46351], [3.57275, 10.27185], [3.66908, 10.18136], [3.54429, 9.87739], [3.35383, 9.83641], [3.32099, 9.78032], [3.34726, 9.70696], [3.25093, 9.61632], [3.13928, 9.47167], [3.14147, 9.28375], [3.08017, 9.10006], [2.77907, 9.06924], [2.67523, 7.87825], [2.73095, 7.7755], [2.73405, 7.5423], [2.78668, 7.5116], [2.79442, 7.43486], [2.74489, 7.42565], [2.76965, 7.13543], [2.71702, 6.95722], [2.74024, 6.92802], [2.73405, 6.78508], [2.78823, 6.76356], [2.78204, 6.70514], [2.7325, 6.64057], [2.74334, 6.57291], [2.70464, 6.50831], [2.70566, 6.38038], [2.74181, 6.13349], [5.87055, 3.78489], [8.34397, 4.30689], [8.60302, 4.87353], [8.78027, 5.1243], [8.92029, 5.58403], [8.83687, 5.68483], [8.88156, 5.78857], [8.84209, 5.82562], [9.51757, 6.43874], [9.70674, 6.51717], [9.77824, 6.79088], [9.86314, 6.77756], [10.15135, 7.03781], [10.21466, 6.88996], [10.53639, 6.93432], [10.57214, 7.16345], [10.59746, 7.14719], [10.60789, 7.06885], [10.83727, 6.9358], [10.8179, 6.83377], [10.94302, 6.69325], [11.09644, 6.68437], [11.09495, 6.51717], [11.42041, 6.53789], [11.42264, 6.5882], [11.51499, 6.60892], [11.57755, 6.74059], [11.55818, 6.86186], [11.63117, 6.9905], [11.87396, 7.09398], [11.84864, 7.26098], [11.93205, 7.47812], [12.01844, 7.52981], [11.99908, 7.67302], [12.20909, 7.97553], [12.19271, 8.10826], [12.24782, 8.17904], [12.26123, 8.43696], [12.4489, 8.52536], [12.44146, 8.6152], [12.68722, 8.65938], [12.71701, 8.7595], [12.79, 8.75361], [12.81085, 8.91992], [12.90022, 9.11411], [12.91958, 9.33905], [12.85628, 9.36698], [13.02385, 9.49334], [13.22642, 9.57266], [13.25472, 9.76795], [13.29941, 9.8296], [13.25025, 9.86042], [13.24132, 9.91031], [13.27409, 9.93232], [13.286, 9.9822], [13.25323, 10.00127], [13.25025, 10.03647], [13.34111, 10.12299], [13.43644, 10.13326], [13.5705, 10.53183], [13.54964, 10.61236], [13.73434, 10.9255], [13.70753, 10.94451], [13.7403, 11.00593], [13.78945, 11.00154], [13.97489, 11.30258], [14.17821, 11.23831], [14.6124, 11.51283], [14.64591, 11.66166], [14.55207, 11.72001], [14.61612, 11.7798], [14.6474, 12.17466], [14.4843, 12.35223], [14.22215, 12.36533], [14.17523, 12.41916], [14.20204, 12.53405], [14.08251, 13.0797], [13.6302, 13.71094], [13.33213, 13.71195], [13.19844, 13.52802], [13.05085, 13.53984], [12.87376, 13.48919], [12.6793, 13.29157], [12.58033, 13.27805], [12.47095, 13.06673], [12.19315, 13.12423], [12.16189, 13.10056], [12.04209, 13.14452], [11.88236, 13.2527], [11.4535, 13.37773], [10.66004, 13.36422], [10.46731, 13.28819], [10.19993, 13.27129], [10.00373, 13.18171], [9.65995, 12.80614], [8.97413, 12.83661], [8.64251, 12.93985], [8.60431, 13.01768], [8.49493, 13.07519], [8.41853, 13.06166], [8.25185, 13.20369], [8.07997, 13.30847], [7.81085, 13.34902], [7.39241, 13.09717], [7.22399, 13.1293], [7.12676, 13.02445], [7.0521, 13.00076], [6.94445, 12.99825], [6.69617, 13.34057], [6.43053, 13.6006], [6.27411, 13.67835], [6.15771, 13.64564]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "NI",
+           iso1A3: "NIC",
+           iso1N3: "558",
+           wikidata: "Q811",
+           nameEn: "Nicaragua",
+           groups: ["013", "003", "419", "019", "UN"],
+           callingCodes: ["505"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-83.13724, 15.00002], [-83.49268, 15.01158], [-83.62101, 14.89448], [-83.89551, 14.76697], [-84.10584, 14.76353], [-84.48373, 14.63249], [-84.70119, 14.68078], [-84.82596, 14.82212], [-84.90082, 14.80489], [-85.1575, 14.53934], [-85.18602, 14.24929], [-85.32149, 14.2562], [-85.45762, 14.11304], [-85.73964, 13.9698], [-85.75477, 13.8499], [-86.03458, 13.99181], [-86.00685, 14.08474], [-86.14801, 14.04317], [-86.35219, 13.77157], [-86.76812, 13.79605], [-86.71267, 13.30348], [-86.87066, 13.30641], [-86.93383, 13.18677], [-86.93197, 13.05313], [-87.03785, 12.98682], [-87.06306, 13.00892], [-87.37107, 12.98646], [-87.55124, 13.12523], [-87.7346, 13.13228], [-88.11443, 12.63306], [-86.14524, 11.09059], [-85.71223, 11.06868], [-85.60529, 11.22607], [-84.92439, 10.9497], [-84.68197, 11.07568], [-83.90838, 10.71161], [-83.66597, 10.79916], [-83.68276, 11.01562], [-82.56142, 11.91792], [-82.06974, 14.49418], [-83.04763, 15.03256], [-83.13724, 15.00002]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "NL",
+           iso1A3: "NLD",
+           iso1N3: "528",
+           wikidata: "Q29999",
+           nameEn: "Kingdom of the Netherlands"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "NO",
+           iso1A3: "NOR",
+           iso1N3: "578",
+           wikidata: "Q20",
+           nameEn: "Norway"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "NP",
+           iso1A3: "NPL",
+           iso1N3: "524",
+           wikidata: "Q837",
+           nameEn: "Nepal",
+           groups: ["034", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["977"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[88.13378, 27.88015], [87.82681, 27.95248], [87.72718, 27.80938], [87.56996, 27.84517], [87.11696, 27.84104], [87.03757, 27.94835], [86.75582, 28.04182], [86.74181, 28.10638], [86.56265, 28.09569], [86.51609, 27.96623], [86.42736, 27.91122], [86.22966, 27.9786], [86.18607, 28.17364], [86.088, 28.09264], [86.08333, 28.02121], [86.12069, 27.93047], [86.06309, 27.90021], [85.94946, 27.9401], [85.97813, 27.99023], [85.90743, 28.05144], [85.84672, 28.18187], [85.74864, 28.23126], [85.71907, 28.38064], [85.69105, 28.38475], [85.60854, 28.25045], [85.59765, 28.30529], [85.4233, 28.32996], [85.38127, 28.28336], [85.10729, 28.34092], [85.18668, 28.54076], [85.19135, 28.62825], [85.06059, 28.68562], [84.85511, 28.58041], [84.62317, 28.73887], [84.47528, 28.74023], [84.2231, 28.89571], [84.24801, 29.02783], [84.18107, 29.23451], [83.97559, 29.33091], [83.82303, 29.30513], [83.63156, 29.16249], [83.44787, 29.30513], [83.28131, 29.56813], [83.07116, 29.61957], [82.73024, 29.81695], [82.5341, 29.9735], [82.38622, 30.02608], [82.16984, 30.0692], [82.19475, 30.16884], [82.10757, 30.23745], [82.10135, 30.35439], [81.99082, 30.33423], [81.62033, 30.44703], [81.5459, 30.37688], [81.41018, 30.42153], [81.39928, 30.21862], [81.33355, 30.15303], [81.2623, 30.14596], [81.29032, 30.08806], [81.24362, 30.0126], [81.12842, 30.01395], [81.03953, 30.20059], [80.92547, 30.17193], [80.91143, 30.22173], [80.86673, 30.17321], [80.8778, 30.13384], [80.67076, 29.95732], [80.60226, 29.95732], [80.57179, 29.91422], [80.56247, 29.86661], [80.48997, 29.79566], [80.43458, 29.80466], [80.41554, 29.79451], [80.36803, 29.73865], [80.38428, 29.68513], [80.41858, 29.63581], [80.37939, 29.57098], [80.24322, 29.44299], [80.31428, 29.30784], [80.28626, 29.20327], [80.24112, 29.21414], [80.26602, 29.13938], [80.23178, 29.11626], [80.18085, 29.13649], [80.05743, 28.91479], [80.06957, 28.82763], [80.12125, 28.82346], [80.37188, 28.63371], [80.44504, 28.63098], [80.52443, 28.54897], [80.50575, 28.6706], [80.55142, 28.69182], [81.03471, 28.40054], [81.19847, 28.36284], [81.32923, 28.13521], [81.38683, 28.17638], [81.48179, 28.12148], [81.47867, 28.08303], [81.91223, 27.84995], [81.97214, 27.93322], [82.06554, 27.92222], [82.46405, 27.6716], [82.70378, 27.72122], [82.74119, 27.49838], [82.93261, 27.50328], [82.94938, 27.46036], [83.19413, 27.45632], [83.27197, 27.38309], [83.2673, 27.36235], [83.29999, 27.32778], [83.35136, 27.33885], [83.38872, 27.39276], [83.39495, 27.4798], [83.61288, 27.47013], [83.85595, 27.35797], [83.86182, 27.4241], [83.93306, 27.44939], [84.02229, 27.43836], [84.10791, 27.52399], [84.21376, 27.45218], [84.25735, 27.44941], [84.29315, 27.39], [84.62161, 27.33885], [84.69166, 27.21294], [84.64496, 27.04669], [84.793, 26.9968], [84.82913, 27.01989], [84.85754, 26.98984], [84.96687, 26.95599], [84.97186, 26.9149], [85.00536, 26.89523], [85.05592, 26.88991], [85.02635, 26.85381], [85.15883, 26.86966], [85.19291, 26.86909], [85.18046, 26.80519], [85.21159, 26.75933], [85.34302, 26.74954], [85.47752, 26.79292], [85.56471, 26.84133], [85.5757, 26.85955], [85.59461, 26.85161], [85.61621, 26.86721], [85.66239, 26.84822], [85.73483, 26.79613], [85.72315, 26.67471], [85.76907, 26.63076], [85.83126, 26.61134], [85.85126, 26.60866], [85.8492, 26.56667], [86.02729, 26.66756], [86.13596, 26.60651], [86.22513, 26.58863], [86.26235, 26.61886], [86.31564, 26.61925], [86.49726, 26.54218], [86.54258, 26.53819], [86.57073, 26.49825], [86.61313, 26.48658], [86.62686, 26.46891], [86.69124, 26.45169], [86.74025, 26.42386], [86.76797, 26.45892], [86.82898, 26.43919], [86.94543, 26.52076], [86.95912, 26.52076], [87.01559, 26.53228], [87.04691, 26.58685], [87.0707, 26.58571], [87.09147, 26.45039], [87.14751, 26.40542], [87.18863, 26.40558], [87.24682, 26.4143], [87.26587, 26.40592], [87.26568, 26.37294], [87.34568, 26.34787], [87.37314, 26.40815], [87.46566, 26.44058], [87.51571, 26.43106], [87.55274, 26.40596], [87.59175, 26.38342], [87.66803, 26.40294], [87.67893, 26.43501], [87.76004, 26.40711], [87.7918, 26.46737], [87.84193, 26.43663], [87.89085, 26.48565], [87.90115, 26.44923], [88.00895, 26.36029], [88.09414, 26.43732], [88.09963, 26.54195], [88.16452, 26.64111], [88.1659, 26.68177], [88.19107, 26.75516], [88.12302, 26.95324], [88.13422, 26.98705], [88.11719, 26.98758], [87.9887, 27.11045], [88.01587, 27.21388], [88.01646, 27.21612], [88.07277, 27.43007], [88.04008, 27.49223], [88.19107, 27.79285], [88.1973, 27.85067], [88.13378, 27.88015]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "NR",
+           iso1A3: "NRU",
+           iso1N3: "520",
+           wikidata: "Q697",
+           nameEn: "Nauru",
+           groups: ["057", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["674"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[166.95155, 0.14829], [166.21778, -0.7977], [167.60042, -0.88259], [166.95155, 0.14829]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "NU",
+           iso1A3: "NIU",
+           iso1N3: "570",
+           wikidata: "Q34020",
+           nameEn: "Niue",
+           country: "NZ",
+           groups: ["061", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["683"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-170.83899, -18.53439], [-170.82274, -20.44429], [-168.63096, -18.60489], [-170.83899, -18.53439]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "NZ",
+           iso1A3: "NZL",
+           iso1N3: "554",
+           wikidata: "Q664",
+           nameEn: "New Zealand"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "OM",
+           iso1A3: "OMN",
+           iso1N3: "512",
+           wikidata: "Q842",
+           nameEn: "Oman",
+           groups: ["145", "142", "UN"],
+           callingCodes: ["968"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[56.82555, 25.7713], [56.79239, 26.41236], [56.68954, 26.76645], [56.2644, 26.58649], [55.81777, 26.18798], [56.08666, 26.05038], [56.15498, 26.06828], [56.19334, 25.9795], [56.13963, 25.82765], [56.17416, 25.77239], [56.13579, 25.73524], [56.14826, 25.66351], [56.18363, 25.65508], [56.20473, 25.61119], [56.25365, 25.60211], [56.26636, 25.60643], [56.25341, 25.61443], [56.26534, 25.62825], [56.82555, 25.7713]]], [[[56.26062, 25.33108], [56.23362, 25.31253], [56.25008, 25.28843], [56.24465, 25.27505], [56.20838, 25.25668], [56.20872, 25.24104], [56.24341, 25.22867], [56.27628, 25.23404], [56.34438, 25.26653], [56.35172, 25.30681], [56.3111, 25.30107], [56.3005, 25.31815], [56.26062, 25.33108]], [[56.28423, 25.26344], [56.27086, 25.26128], [56.2716, 25.27916], [56.28102, 25.28486], [56.29379, 25.2754], [56.28423, 25.26344]]], [[[61.45114, 22.55394], [56.86325, 25.03856], [56.3227, 24.97284], [56.34873, 24.93205], [56.30269, 24.88334], [56.20568, 24.85063], [56.20062, 24.78565], [56.13684, 24.73699], [56.06128, 24.74457], [56.03535, 24.81161], [55.97836, 24.87673], [55.97467, 24.89639], [56.05106, 24.87461], [56.05715, 24.95727], [55.96316, 25.00857], [55.90849, 24.96771], [55.85094, 24.96858], [55.81116, 24.9116], [55.81348, 24.80102], [55.83408, 24.77858], [55.83271, 24.68567], [55.76461, 24.5287], [55.83271, 24.41521], [55.83395, 24.32776], [55.80747, 24.31069], [55.79145, 24.27914], [55.76781, 24.26209], [55.75939, 24.26114], [55.75382, 24.2466], [55.75257, 24.23466], [55.76558, 24.23227], [55.77658, 24.23476], [55.83367, 24.20193], [55.95472, 24.2172], [56.01799, 24.07426], [55.8308, 24.01633], [55.73301, 24.05994], [55.48677, 23.94946], [55.57358, 23.669], [55.22634, 23.10378], [55.2137, 22.71065], [55.66469, 21.99658], [54.99756, 20.00083], [52.00311, 19.00083], [52.78009, 17.35124], [52.74267, 17.29519], [52.81185, 17.28568], [57.49095, 8.14549], [61.45114, 22.55394]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "PA",
+           iso1A3: "PAN",
+           iso1N3: "591",
+           wikidata: "Q804",
+           nameEn: "Panama",
+           groups: ["013", "003", "419", "019", "UN"],
+           callingCodes: ["507"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-77.32389, 8.81247], [-77.58292, 9.22278], [-78.79327, 9.93766], [-82.51044, 9.65379], [-82.56507, 9.57279], [-82.61345, 9.49881], [-82.66667, 9.49746], [-82.77206, 9.59573], [-82.87919, 9.62645], [-82.84871, 9.4973], [-82.93516, 9.46741], [-82.93516, 9.07687], [-82.72126, 8.97125], [-82.88253, 8.83331], [-82.91377, 8.774], [-82.92068, 8.74832], [-82.8794, 8.6981], [-82.82739, 8.60153], [-82.83975, 8.54755], [-82.83322, 8.52464], [-82.8382, 8.48117], [-82.8679, 8.44042], [-82.93056, 8.43465], [-83.05209, 8.33394], [-82.9388, 8.26634], [-82.88641, 8.10219], [-82.89137, 8.05755], [-82.89978, 8.04083], [-82.94503, 7.93865], [-82.13751, 6.97312], [-78.06168, 7.07793], [-77.89178, 7.22681], [-77.81426, 7.48319], [-77.72157, 7.47612], [-77.72514, 7.72348], [-77.57185, 7.51147], [-77.17257, 7.97422], [-77.45064, 8.49991], [-77.32389, 8.81247]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "PE",
+           iso1A3: "PER",
+           iso1N3: "604",
+           wikidata: "Q419",
+           nameEn: "Peru",
+           groups: ["005", "419", "019", "UN"],
+           callingCodes: ["51"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-74.26675, -0.97229], [-74.42701, -0.50218], [-75.18513, -0.0308], [-75.25764, -0.11943], [-75.40192, -0.17196], [-75.61997, -0.10012], [-75.60169, -0.18708], [-75.53615, -0.19213], [-75.22862, -0.60048], [-75.22862, -0.95588], [-75.3872, -0.9374], [-75.57429, -1.55961], [-76.05203, -2.12179], [-76.6324, -2.58397], [-77.94147, -3.05454], [-78.19369, -3.36431], [-78.14324, -3.47653], [-78.22642, -3.51113], [-78.24589, -3.39907], [-78.34362, -3.38633], [-78.68394, -4.60754], [-78.85149, -4.66795], [-79.01659, -5.01481], [-79.1162, -4.97774], [-79.26248, -4.95167], [-79.59402, -4.46848], [-79.79722, -4.47558], [-80.13945, -4.29786], [-80.39256, -4.48269], [-80.46386, -4.41516], [-80.32114, -4.21323], [-80.45023, -4.20938], [-80.4822, -4.05477], [-80.46386, -4.01342], [-80.13232, -3.90317], [-80.19926, -3.68894], [-80.18741, -3.63994], [-80.19848, -3.59249], [-80.21642, -3.5888], [-80.20535, -3.51667], [-80.22629, -3.501], [-80.23651, -3.48652], [-80.24586, -3.48677], [-80.24123, -3.46124], [-80.20647, -3.431], [-80.30602, -3.39149], [-84.52388, -3.36941], [-85.71054, -21.15413], [-70.59118, -18.35072], [-70.378, -18.3495], [-70.31267, -18.31258], [-70.16394, -18.31737], [-69.96732, -18.25992], [-69.81607, -18.12582], [-69.75305, -17.94605], [-69.82868, -17.72048], [-69.79087, -17.65563], [-69.66483, -17.65083], [-69.46897, -17.4988], [-69.46863, -17.37466], [-69.62883, -17.28142], [-69.16896, -16.72233], [-69.00853, -16.66769], [-69.04027, -16.57214], [-68.98358, -16.42165], [-68.79464, -16.33272], [-68.96238, -16.194], [-69.09986, -16.22693], [-69.20291, -16.16668], [-69.40336, -15.61358], [-69.14856, -15.23478], [-69.36254, -14.94634], [-68.88135, -14.18639], [-69.05265, -13.68546], [-68.8864, -13.40792], [-68.85615, -12.87769], [-68.65044, -12.50689], [-68.98115, -11.8979], [-69.57156, -10.94555], [-69.57835, -10.94051], [-69.90896, -10.92744], [-70.38791, -11.07096], [-70.51395, -10.92249], [-70.64134, -11.0108], [-70.62487, -9.80666], [-70.55429, -9.76692], [-70.58453, -9.58303], [-70.53373, -9.42628], [-71.23394, -9.9668], [-72.14742, -9.98049], [-72.31883, -9.5184], [-72.72216, -9.41397], [-73.21498, -9.40904], [-72.92886, -9.04074], [-73.76576, -7.89884], [-73.65485, -7.77897], [-73.96938, -7.58465], [-73.77011, -7.28944], [-73.73986, -6.87919], [-73.12983, -6.43852], [-73.24579, -6.05764], [-72.83973, -5.14765], [-72.64391, -5.0391], [-71.87003, -4.51661], [-70.96814, -4.36915], [-70.77601, -4.15717], [-70.33236, -4.15214], [-70.19582, -4.3607], [-70.11305, -4.27281], [-70.00888, -4.37833], [-69.94708, -4.2431], [-70.3374, -3.79505], [-70.52393, -3.87553], [-70.71396, -3.7921], [-70.04609, -2.73906], [-70.94377, -2.23142], [-71.75223, -2.15058], [-72.92587, -2.44514], [-73.65312, -1.26222], [-74.26675, -0.97229]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "PF",
+           iso1A3: "PYF",
+           iso1N3: "258",
+           wikidata: "Q30971",
+           nameEn: "French Polynesia",
+           country: "FR",
+           groups: ["Q1451600", "061", "009", "UN"],
+           callingCodes: ["689"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-135.59706, -4.70473], [-156.48634, -15.52824], [-156.45576, -31.75456], [-133.59543, -28.4709], [-135.59706, -4.70473]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "PG",
+           iso1A3: "PNG",
+           iso1N3: "598",
+           wikidata: "Q691",
+           nameEn: "Papua New Guinea",
+           groups: ["054", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["675"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[141.03157, 2.12829], [140.99813, -6.3233], [140.85295, -6.72996], [140.90448, -6.85033], [141.01763, -6.90181], [141.01842, -9.35091], [141.88934, -9.36111], [142.19246, -9.15378], [142.48658, -9.36754], [143.29772, -9.33993], [143.87386, -9.02382], [145.2855, -9.62524], [156.73836, -14.50464], [154.74815, -7.33315], [155.60735, -6.92266], [155.69784, -6.92661], [155.92557, -6.84664], [156.03993, -6.65703], [156.03296, -6.55528], [160.43769, -4.17974], [141.03157, 2.12829]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "PH",
+           iso1A3: "PHL",
+           iso1N3: "608",
+           wikidata: "Q928",
+           nameEn: "Philippines",
+           aliases: ["PI", "RP"],
+           groups: ["035", "142", "UN"],
+           callingCodes: ["63"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[129.19694, 7.84182], [121.8109, 21.77688], [120.69238, 21.52331], [118.82252, 14.67191], [115.39742, 10.92666], [116.79524, 7.43869], [117.17735, 7.52841], [117.93857, 6.89845], [117.98544, 6.27477], [119.52945, 5.35672], [118.93936, 4.09009], [118.06469, 4.16638], [121.14448, 2.12444], [129.19694, 7.84182]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "PK",
+           iso1A3: "PAK",
+           iso1N3: "586",
+           wikidata: "Q843",
+           nameEn: "Pakistan",
+           groups: ["034", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["92"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[75.72737, 36.7529], [75.45562, 36.71971], [75.40481, 36.95382], [75.13839, 37.02622], [74.56453, 37.03023], [74.53739, 36.96224], [74.43389, 37.00977], [74.04856, 36.82648], [73.82685, 36.91421], [72.6323, 36.84601], [72.18135, 36.71838], [71.80267, 36.49924], [71.60491, 36.39429], [71.19505, 36.04134], [71.37969, 35.95865], [71.55273, 35.71483], [71.49917, 35.6267], [71.65435, 35.4479], [71.54294, 35.31037], [71.5541, 35.28776], [71.67495, 35.21262], [71.52938, 35.09023], [71.55273, 35.02615], [71.49917, 35.00478], [71.50329, 34.97328], [71.29472, 34.87728], [71.28356, 34.80882], [71.08718, 34.69034], [71.11602, 34.63047], [71.0089, 34.54568], [71.02401, 34.44835], [71.17662, 34.36769], [71.12815, 34.26619], [71.13078, 34.16503], [71.09453, 34.13524], [71.09307, 34.11961], [71.06933, 34.10564], [71.07345, 34.06242], [70.88119, 33.97933], [70.54336, 33.9463], [69.90203, 34.04194], [69.87307, 33.9689], [69.85671, 33.93719], [70.00503, 33.73528], [70.14236, 33.71701], [70.14785, 33.6553], [70.20141, 33.64387], [70.17062, 33.53535], [70.32775, 33.34496], [70.13686, 33.21064], [70.07369, 33.22557], [70.02563, 33.14282], [69.85259, 33.09451], [69.79766, 33.13247], [69.71526, 33.09911], [69.57656, 33.09911], [69.49004, 33.01509], [69.49854, 32.88843], [69.5436, 32.8768], [69.47082, 32.85834], [69.38018, 32.76601], [69.43649, 32.7302], [69.44747, 32.6678], [69.38155, 32.56601], [69.2868, 32.53938], [69.23599, 32.45946], [69.27932, 32.29119], [69.27032, 32.14141], [69.3225, 31.93186], [69.20577, 31.85957], [69.11514, 31.70782], [69.00939, 31.62249], [68.95995, 31.64822], [68.91078, 31.59687], [68.79997, 31.61665], [68.6956, 31.75687], [68.57475, 31.83158], [68.44222, 31.76446], [68.27605, 31.75863], [68.25614, 31.80357], [68.1655, 31.82691], [68.00071, 31.6564], [67.86887, 31.63536], [67.72056, 31.52304], [67.58323, 31.52772], [67.62374, 31.40473], [67.7748, 31.4188], [67.78854, 31.33203], [67.29964, 31.19586], [67.03323, 31.24519], [67.04147, 31.31561], [66.83273, 31.26867], [66.72561, 31.20526], [66.68166, 31.07597], [66.58175, 30.97532], [66.42645, 30.95309], [66.39194, 30.9408], [66.28413, 30.57001], [66.34869, 30.404], [66.23609, 30.06321], [66.36042, 29.9583], [66.24175, 29.85181], [65.04005, 29.53957], [64.62116, 29.58903], [64.19796, 29.50407], [64.12966, 29.39157], [63.5876, 29.50456], [62.47751, 29.40782], [60.87231, 29.86514], [61.31508, 29.38903], [61.53765, 29.00507], [61.65978, 28.77937], [61.93581, 28.55284], [62.40259, 28.42703], [62.59499, 28.24842], [62.79412, 28.28108], [62.7638, 28.02992], [62.84905, 27.47627], [62.79684, 27.34381], [62.80604, 27.22412], [63.19649, 27.25674], [63.32283, 27.14437], [63.25005, 27.08692], [63.25005, 26.84212], [63.18688, 26.83844], [63.1889, 26.65072], [62.77352, 26.64099], [62.31484, 26.528], [62.21304, 26.26601], [62.05117, 26.31647], [61.89391, 26.26251], [61.83831, 26.07249], [61.83968, 25.7538], [61.683, 25.66638], [61.6433, 25.27541], [61.46682, 24.57869], [68.11329, 23.53945], [68.20763, 23.85849], [68.39339, 23.96838], [68.74643, 23.97027], [68.7416, 24.31904], [68.90914, 24.33156], [68.97781, 24.26021], [69.07806, 24.29777], [69.19341, 24.25646], [69.29778, 24.28712], [69.59579, 24.29777], [69.73335, 24.17007], [70.03428, 24.172], [70.11712, 24.30915], [70.5667, 24.43787], [70.57906, 24.27774], [70.71502, 24.23517], [70.88393, 24.27398], [70.85784, 24.30903], [70.94985, 24.3791], [71.04461, 24.34657], [71.12838, 24.42662], [71.00341, 24.46038], [70.97594, 24.60904], [71.09405, 24.69017], [70.94002, 24.92843], [70.89148, 25.15064], [70.66695, 25.39314], [70.67382, 25.68186], [70.60378, 25.71898], [70.53649, 25.68928], [70.37444, 25.67443], [70.2687, 25.71156], [70.0985, 25.93238], [70.08193, 26.08094], [70.17532, 26.24118], [70.17532, 26.55362], [70.05584, 26.60398], [69.88555, 26.56836], [69.50904, 26.74892], [69.58519, 27.18109], [70.03136, 27.56627], [70.12502, 27.8057], [70.37307, 28.01208], [70.60927, 28.02178], [70.79054, 27.68423], [71.89921, 27.96035], [71.9244, 28.11555], [72.20329, 28.3869], [72.29495, 28.66367], [72.40402, 28.78283], [72.94272, 29.02487], [73.01337, 29.16422], [73.05886, 29.1878], [73.28094, 29.56646], [73.3962, 29.94707], [73.58665, 30.01848], [73.80299, 30.06969], [73.97225, 30.19829], [73.95736, 30.28466], [73.88993, 30.36305], [74.5616, 31.04153], [74.67971, 31.05479], [74.6852, 31.12771], [74.60006, 31.13711], [74.60281, 31.10419], [74.56023, 31.08303], [74.51629, 31.13829], [74.53223, 31.30321], [74.59773, 31.4136], [74.64713, 31.45605], [74.59319, 31.50197], [74.61517, 31.55698], [74.57498, 31.60382], [74.47771, 31.72227], [74.58907, 31.87824], [74.79919, 31.95983], [74.86236, 32.04485], [74.9269, 32.0658], [75.00793, 32.03786], [75.25649, 32.10187], [75.38046, 32.26836], [75.28259, 32.36556], [75.03265, 32.49538], [74.97634, 32.45367], [74.84725, 32.49075], [74.68362, 32.49298], [74.67431, 32.56676], [74.65251, 32.56416], [74.64424, 32.60985], [74.69542, 32.66792], [74.65345, 32.71225], [74.7113, 32.84219], [74.64675, 32.82604], [74.6289, 32.75561], [74.45312, 32.77755], [74.41467, 32.90563], [74.31227, 32.92795], [74.34875, 32.97823], [74.31854, 33.02891], [74.17571, 33.07495], [74.15374, 33.13477], [74.02144, 33.18908], [74.01366, 33.25199], [74.08782, 33.26232], [74.17983, 33.3679], [74.18121, 33.4745], [74.10115, 33.56392], [74.03576, 33.56718], [73.97367, 33.64061], [73.98968, 33.66155], [73.96423, 33.73071], [74.00891, 33.75437], [74.05898, 33.82089], [74.14001, 33.83002], [74.26086, 33.92237], [74.25262, 34.01577], [74.21554, 34.03853], [73.91341, 34.01235], [73.88732, 34.05105], [73.90677, 34.10504], [73.98208, 34.2522], [73.90517, 34.35317], [73.8475, 34.32935], [73.74862, 34.34183], [73.74999, 34.3781], [73.88732, 34.48911], [73.89419, 34.54568], [73.93951, 34.57169], [73.93401, 34.63386], [73.96423, 34.68244], [74.12897, 34.70073], [74.31239, 34.79626], [74.58083, 34.77386], [74.6663, 34.703], [75.01479, 34.64629], [75.38009, 34.55021], [75.75438, 34.51827], [76.04614, 34.67566], [76.15463, 34.6429], [76.47186, 34.78965], [76.67648, 34.76371], [76.74377, 34.84039], [76.74514, 34.92488], [76.87193, 34.96906], [76.99251, 34.93349], [77.11796, 35.05419], [76.93465, 35.39866], [76.85088, 35.39754], [76.75475, 35.52617], [76.77323, 35.66062], [76.50961, 35.8908], [76.33453, 35.84296], [76.14913, 35.82848], [76.15325, 35.9264], [75.93028, 36.13136], [76.00906, 36.17511], [76.0324, 36.41198], [75.92391, 36.56986], [75.72737, 36.7529]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "PL",
+           iso1A3: "POL",
+           iso1N3: "616",
+           wikidata: "Q36",
+           nameEn: "Poland",
+           groups: ["EU", "151", "150", "UN"],
+           callingCodes: ["48"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[18.57853, 55.25302], [14.20811, 54.12784], [14.22634, 53.9291], [14.20647, 53.91671], [14.18544, 53.91258], [14.20823, 53.90776], [14.21323, 53.8664], [14.27249, 53.74464], [14.26782, 53.69866], [14.2836, 53.67721], [14.27133, 53.66613], [14.28477, 53.65955], [14.2853, 53.63392], [14.31904, 53.61581], [14.30416, 53.55499], [14.3273, 53.50587], [14.35209, 53.49506], [14.4215, 53.27724], [14.44133, 53.27427], [14.45125, 53.26241], [14.40662, 53.21098], [14.37853, 53.20405], [14.36696, 53.16444], [14.38679, 53.13669], [14.35044, 53.05829], [14.25954, 53.00264], [14.14056, 52.95786], [14.15873, 52.87715], [14.12256, 52.84311], [14.13806, 52.82392], [14.22071, 52.81175], [14.61073, 52.59847], [14.6289, 52.57136], [14.60081, 52.53116], [14.63056, 52.48993], [14.54423, 52.42568], [14.55228, 52.35264], [14.56378, 52.33838], [14.58149, 52.28007], [14.70139, 52.25038], [14.71319, 52.22144], [14.68344, 52.19612], [14.70616, 52.16927], [14.67683, 52.13936], [14.6917, 52.10283], [14.72971, 52.09167], [14.76026, 52.06624], [14.71339, 52.00337], [14.70488, 51.97679], [14.7139, 51.95643], [14.71836, 51.95606], [14.72163, 51.95188], [14.7177, 51.94048], [14.70601, 51.92944], [14.6933, 51.9044], [14.6588, 51.88359], [14.59089, 51.83302], [14.60493, 51.80473], [14.64625, 51.79472], [14.66386, 51.73282], [14.69065, 51.70842], [14.75392, 51.67445], [14.75759, 51.62318], [14.7727, 51.61263], [14.71125, 51.56209], [14.73047, 51.54606], [14.72652, 51.53902], [14.73219, 51.52922], [14.94749, 51.47155], [14.9652, 51.44793], [14.96899, 51.38367], [14.98008, 51.33449], [15.04288, 51.28387], [15.01242, 51.21285], [15.0047, 51.16874], [14.99311, 51.16249], [14.99414, 51.15813], [15.00083, 51.14974], [14.99646, 51.14365], [14.99079, 51.14284], [14.99689, 51.12205], [14.98229, 51.11354], [14.97938, 51.07742], [14.95529, 51.04552], [14.92942, 50.99744], [14.89252, 50.94999], [14.89681, 50.9422], [14.81664, 50.88148], [14.82803, 50.86966], [14.99852, 50.86817], [15.01088, 50.97984], [14.96419, 50.99108], [15.02433, 51.0242], [15.03895, 51.0123], [15.06218, 51.02269], [15.10152, 51.01095], [15.11937, 50.99021], [15.16744, 51.01959], [15.1743, 50.9833], [15.2361, 50.99886], [15.27043, 50.97724], [15.2773, 50.8907], [15.36656, 50.83956], [15.3803, 50.77187], [15.43798, 50.80833], [15.73186, 50.73885], [15.81683, 50.75666], [15.87331, 50.67188], [15.97219, 50.69799], [16.0175, 50.63009], [15.98317, 50.61528], [16.02437, 50.60046], [16.10265, 50.66405], [16.20839, 50.63096], [16.23174, 50.67101], [16.33611, 50.66579], [16.44597, 50.58041], [16.34572, 50.49575], [16.31413, 50.50274], [16.19526, 50.43291], [16.21585, 50.40627], [16.22821, 50.41054], [16.28118, 50.36891], [16.30289, 50.38292], [16.36495, 50.37679], [16.3622, 50.34875], [16.39379, 50.3207], [16.42674, 50.32509], [16.56407, 50.21009], [16.55446, 50.16613], [16.63137, 50.1142], [16.7014, 50.09659], [16.8456, 50.20834], [16.98018, 50.24172], [17.00353, 50.21449], [17.02825, 50.23118], [16.99803, 50.25753], [17.02138, 50.27772], [16.99803, 50.30316], [16.94448, 50.31281], [16.90877, 50.38642], [16.85933, 50.41093], [16.89229, 50.45117], [17.1224, 50.39494], [17.14498, 50.38117], [17.19579, 50.38817], [17.19991, 50.3654], [17.27681, 50.32246], [17.34273, 50.32947], [17.34548, 50.2628], [17.3702, 50.28123], [17.58889, 50.27837], [17.67764, 50.28977], [17.69292, 50.32859], [17.74648, 50.29966], [17.72176, 50.25665], [17.76296, 50.23382], [17.70528, 50.18812], [17.59404, 50.16437], [17.66683, 50.10275], [17.6888, 50.12037], [17.7506, 50.07896], [17.77669, 50.02253], [17.86886, 49.97452], [18.00191, 50.01723], [18.04585, 50.01194], [18.04585, 50.03311], [18.00396, 50.04954], [18.03212, 50.06574], [18.07898, 50.04535], [18.10628, 50.00223], [18.20241, 49.99958], [18.21752, 49.97309], [18.27107, 49.96779], [18.27794, 49.93863], [18.31914, 49.91565], [18.33278, 49.92415], [18.33562, 49.94747], [18.41604, 49.93498], [18.53423, 49.89906], [18.54495, 49.9079], [18.54299, 49.92537], [18.57697, 49.91565], [18.57045, 49.87849], [18.60341, 49.86256], [18.57183, 49.83334], [18.61278, 49.7618], [18.61368, 49.75426], [18.62645, 49.75002], [18.62943, 49.74603], [18.62676, 49.71983], [18.69817, 49.70473], [18.72838, 49.68163], [18.80479, 49.6815], [18.84786, 49.5446], [18.84521, 49.51672], [18.94536, 49.52143], [18.97283, 49.49914], [18.9742, 49.39557], [19.18019, 49.41165], [19.25435, 49.53391], [19.36009, 49.53747], [19.37795, 49.574], [19.45348, 49.61583], [19.52626, 49.57311], [19.53313, 49.52856], [19.57845, 49.46077], [19.64162, 49.45184], [19.6375, 49.40897], [19.72127, 49.39288], [19.78581, 49.41701], [19.82237, 49.27806], [19.75286, 49.20751], [19.86409, 49.19316], [19.90529, 49.23532], [19.98494, 49.22904], [20.08238, 49.1813], [20.13738, 49.31685], [20.21977, 49.35265], [20.31453, 49.34817], [20.31728, 49.39914], [20.39939, 49.3896], [20.46422, 49.41612], [20.5631, 49.375], [20.61666, 49.41791], [20.72274, 49.41813], [20.77971, 49.35383], [20.9229, 49.29626], [20.98733, 49.30774], [21.09799, 49.37176], [21.041, 49.41791], [21.12477, 49.43666], [21.19756, 49.4054], [21.27858, 49.45988], [21.43376, 49.41433], [21.62328, 49.4447], [21.77983, 49.35443], [21.82927, 49.39467], [21.96385, 49.3437], [22.04427, 49.22136], [22.56155, 49.08865], [22.89122, 49.00725], [22.86336, 49.10513], [22.72009, 49.20288], [22.748, 49.32759], [22.69444, 49.49378], [22.64534, 49.53094], [22.78304, 49.65543], [22.80261, 49.69098], [22.83179, 49.69875], [22.99329, 49.84249], [23.28221, 50.0957], [23.67635, 50.33385], [23.71382, 50.38248], [23.79445, 50.40481], [23.99563, 50.41289], [24.03668, 50.44507], [24.07048, 50.5071], [24.0996, 50.60752], [24.0595, 50.71625], [23.95925, 50.79271], [23.99254, 50.83847], [24.0952, 50.83262], [24.14524, 50.86128], [24.04576, 50.90196], [23.92217, 51.00836], [23.90376, 51.07697], [23.80678, 51.18405], [23.63858, 51.32182], [23.69905, 51.40871], [23.62751, 51.50512], [23.56236, 51.53673], [23.57053, 51.55938], [23.53198, 51.74298], [23.62691, 51.78208], [23.61523, 51.92066], [23.68733, 51.9906], [23.64066, 52.07626], [23.61, 52.11264], [23.54314, 52.12148], [23.47859, 52.18215], [23.20071, 52.22848], [23.18196, 52.28812], [23.34141, 52.44845], [23.45112, 52.53774], [23.58296, 52.59868], [23.73615, 52.6149], [23.93763, 52.71332], [23.91805, 52.94016], [23.94689, 52.95919], [23.92184, 53.02079], [23.87548, 53.0831], [23.91393, 53.16469], [23.85657, 53.22923], [23.81995, 53.24131], [23.62004, 53.60942], [23.51284, 53.95052], [23.48261, 53.98855], [23.52702, 54.04622], [23.49196, 54.14764], [23.45223, 54.17775], [23.42418, 54.17911], [23.39525, 54.21672], [23.3494, 54.25155], [23.24656, 54.25701], [23.15938, 54.29894], [23.15526, 54.31076], [23.13905, 54.31567], [23.104, 54.29794], [23.04323, 54.31567], [23.05726, 54.34565], [22.99649, 54.35927], [23.00584, 54.38514], [22.83756, 54.40827], [22.79705, 54.36264], [21.41123, 54.32395], [20.63871, 54.3706], [19.8038, 54.44203], [19.64312, 54.45423], [18.57853, 55.25302]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "PM",
+           iso1A3: "SPM",
+           iso1N3: "666",
+           wikidata: "Q34617",
+           nameEn: "Saint Pierre and Miquelon",
+           country: "FR",
+           groups: ["Q1451600", "021", "003", "019", "UN"],
+           callingCodes: ["508"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-56.72993, 46.65575], [-55.90758, 46.6223], [-56.27503, 47.39728], [-56.72993, 46.65575]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "PN",
+           iso1A3: "PCN",
+           iso1N3: "612",
+           wikidata: "Q35672",
+           nameEn: "Pitcairn Islands",
+           country: "GB",
+           groups: ["BOTS", "061", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["64"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-133.59543, -28.4709], [-122.0366, -24.55017], [-133.61511, -21.93325], [-133.59543, -28.4709]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "PR",
+           iso1A3: "PRI",
+           iso1N3: "630",
+           wikidata: "Q1183",
+           nameEn: "Puerto Rico",
+           aliases: ["US-PR"],
+           country: "US",
+           groups: ["Q1352230", "029", "003", "419", "019", "UN"],
+           roadSpeedUnit: "mph",
+           callingCodes: ["1 787", "1 939"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-65.27974, 17.56928], [-65.02435, 18.73231], [-67.99519, 18.97186], [-68.23894, 17.84663], [-65.27974, 17.56928]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "PS",
+           iso1A3: "PSE",
+           iso1N3: "275",
+           wikidata: "Q219060",
+           nameEn: "Palestine"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "PT",
+           iso1A3: "PRT",
+           iso1N3: "620",
+           wikidata: "Q45",
+           nameEn: "Portugal"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "PW",
+           iso1A3: "PLW",
+           iso1N3: "585",
+           wikidata: "Q695",
+           nameEn: "Palau",
+           groups: ["057", "009", "UN"],
+           roadSpeedUnit: "mph",
+           callingCodes: ["680"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[128.97621, 3.08804], [136.39296, 1.54187], [136.04605, 12.45908], [128.97621, 3.08804]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "PY",
+           iso1A3: "PRY",
+           iso1N3: "600",
+           wikidata: "Q733",
+           nameEn: "Paraguay",
+           groups: ["005", "419", "019", "UN"],
+           callingCodes: ["595"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-58.16225, -20.16193], [-58.23216, -19.80058], [-59.06965, -19.29148], [-60.00638, -19.2981], [-61.73723, -19.63958], [-61.93912, -20.10053], [-62.26883, -20.55311], [-62.2757, -21.06657], [-62.64455, -22.25091], [-62.51761, -22.37684], [-62.22768, -22.55807], [-61.9756, -23.0507], [-61.0782, -23.62932], [-60.99754, -23.80934], [-60.28163, -24.04436], [-60.03367, -24.00701], [-59.45482, -24.34787], [-59.33886, -24.49935], [-58.33055, -24.97099], [-58.25492, -24.92528], [-57.80821, -25.13863], [-57.57431, -25.47269], [-57.87176, -25.93604], [-58.1188, -26.16704], [-58.3198, -26.83443], [-58.65321, -27.14028], [-58.59549, -27.29973], [-58.04205, -27.2387], [-56.85337, -27.5165], [-56.18313, -27.29851], [-55.89195, -27.3467], [-55.74475, -27.44485], [-55.59094, -27.32444], [-55.62322, -27.1941], [-55.39611, -26.97679], [-55.25243, -26.93808], [-55.16948, -26.96068], [-55.06351, -26.80195], [-55.00584, -26.78754], [-54.80868, -26.55669], [-54.70732, -26.45099], [-54.69333, -26.37705], [-54.67359, -25.98607], [-54.60664, -25.9691], [-54.62063, -25.91213], [-54.59398, -25.59224], [-54.59509, -25.53696], [-54.60196, -25.48397], [-54.62033, -25.46026], [-54.4423, -25.13381], [-54.28207, -24.07305], [-54.32807, -24.01865], [-54.6238, -23.83078], [-55.02691, -23.97317], [-55.0518, -23.98666], [-55.12292, -23.99669], [-55.41784, -23.9657], [-55.44117, -23.9185], [-55.43585, -23.87157], [-55.5555, -23.28237], [-55.52288, -23.2595], [-55.5446, -23.22811], [-55.63849, -22.95122], [-55.62493, -22.62765], [-55.68742, -22.58407], [-55.6986, -22.56268], [-55.72366, -22.5519], [-55.741, -22.52018], [-55.74941, -22.46436], [-55.8331, -22.29008], [-56.23206, -22.25347], [-56.45893, -22.08072], [-56.5212, -22.11556], [-56.6508, -22.28387], [-57.98625, -22.09157], [-57.94642, -21.73799], [-57.88239, -21.6868], [-57.93492, -21.65505], [-57.84536, -20.93155], [-58.16225, -20.16193]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "QA",
+           iso1A3: "QAT",
+           iso1N3: "634",
+           wikidata: "Q846",
+           nameEn: "Qatar",
+           groups: ["145", "142", "UN"],
+           callingCodes: ["974"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[50.92992, 24.54396], [51.09638, 24.46907], [51.29972, 24.50747], [51.39468, 24.62785], [51.58834, 24.66608], [51.83108, 24.71675], [51.83682, 26.70231], [50.93865, 26.30758], [50.81266, 25.88946], [50.86149, 25.6965], [50.7801, 25.595], [50.80824, 25.54641], [50.57069, 25.57887], [50.8133, 24.74049], [50.92992, 24.54396]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "RE",
+           iso1A3: "REU",
+           iso1N3: "638",
+           wikidata: "Q17070",
+           nameEn: "R\xE9union",
+           country: "FR",
+           groups: ["Q3320166", "EU", "014", "202", "002", "UN"],
+           callingCodes: ["262"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[53.37984, -21.23941], [56.73473, -21.9174], [56.62373, -20.2711], [53.37984, -21.23941]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "RO",
+           iso1A3: "ROU",
+           iso1N3: "642",
+           wikidata: "Q218",
+           nameEn: "Romania",
+           groups: ["EU", "151", "150", "UN"],
+           callingCodes: ["40"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[27.15622, 47.98538], [27.02985, 48.09083], [27.04118, 48.12522], [26.96119, 48.13003], [26.98042, 48.15752], [26.94265, 48.1969], [26.87708, 48.19919], [26.81161, 48.25049], [26.62823, 48.25804], [26.55202, 48.22445], [26.33504, 48.18418], [26.17711, 47.99246], [26.05901, 47.9897], [25.77723, 47.93919], [25.63878, 47.94924], [25.23778, 47.89403], [25.11144, 47.75203], [24.88896, 47.7234], [24.81893, 47.82031], [24.70632, 47.84428], [24.61994, 47.95062], [24.43578, 47.97131], [24.34926, 47.9244], [24.22566, 47.90231], [24.11281, 47.91487], [24.06466, 47.95317], [24.02999, 47.95087], [24.00801, 47.968], [23.98553, 47.96076], [23.96337, 47.96672], [23.94192, 47.94868], [23.89352, 47.94512], [23.8602, 47.9329], [23.80904, 47.98142], [23.75188, 47.99705], [23.66262, 47.98786], [23.63894, 48.00293], [23.5653, 48.00499], [23.52803, 48.01818], [23.4979, 47.96858], [23.33577, 48.0237], [23.27397, 48.08245], [23.15999, 48.12188], [23.1133, 48.08061], [23.08858, 48.00716], [23.0158, 47.99338], [22.92241, 48.02002], [22.94301, 47.96672], [22.89849, 47.95851], [22.77991, 47.87211], [22.76617, 47.8417], [22.67247, 47.7871], [22.46559, 47.76583], [22.41979, 47.7391], [22.31816, 47.76126], [22.00917, 47.50492], [22.03389, 47.42508], [22.01055, 47.37767], [21.94463, 47.38046], [21.78395, 47.11104], [21.648, 47.03902], [21.68645, 46.99595], [21.59581, 46.91628], [21.59307, 46.86935], [21.52028, 46.84118], [21.48935, 46.7577], [21.5151, 46.72147], [21.43926, 46.65109], [21.33214, 46.63035], [21.26929, 46.4993], [21.28061, 46.44941], [21.16872, 46.30118], [21.06572, 46.24897], [20.86797, 46.28884], [20.74574, 46.25467], [20.76085, 46.21002], [20.63863, 46.12728], [20.49718, 46.18721], [20.45377, 46.14405], [20.35573, 46.16629], [20.28324, 46.1438], [20.26068, 46.12332], [20.35862, 45.99356], [20.54818, 45.89939], [20.65645, 45.82801], [20.70069, 45.7493], [20.77416, 45.75601], [20.78446, 45.78522], [20.82364, 45.77738], [20.80361, 45.65875], [20.76798, 45.60969], [20.83321, 45.53567], [20.77217, 45.49788], [20.86026, 45.47295], [20.87948, 45.42743], [21.09894, 45.30144], [21.17612, 45.32566], [21.20392, 45.2677], [21.29398, 45.24148], [21.48278, 45.19557], [21.51299, 45.15345], [21.4505, 45.04294], [21.35855, 45.01941], [21.54938, 44.9327], [21.56328, 44.89502], [21.48202, 44.87199], [21.44013, 44.87613], [21.35643, 44.86364], [21.38802, 44.78133], [21.55007, 44.77304], [21.60019, 44.75208], [21.61942, 44.67059], [21.67504, 44.67107], [21.71692, 44.65349], [21.7795, 44.66165], [21.99364, 44.63395], [22.08016, 44.49844], [22.13234, 44.47444], [22.18315, 44.48179], [22.30844, 44.6619], [22.45301, 44.7194], [22.61917, 44.61489], [22.69196, 44.61587], [22.76749, 44.54446], [22.70981, 44.51852], [22.61368, 44.55719], [22.56493, 44.53419], [22.54021, 44.47836], [22.45436, 44.47258], [22.56012, 44.30712], [22.68166, 44.28206], [22.67173, 44.21564], [23.04988, 44.07694], [23.01674, 44.01946], [22.87873, 43.9844], [22.83753, 43.88055], [22.85314, 43.84452], [23.05288, 43.79494], [23.26772, 43.84843], [23.4507, 43.84936], [23.61687, 43.79289], [23.73978, 43.80627], [24.18149, 43.68218], [24.35364, 43.70211], [24.50264, 43.76314], [24.62281, 43.74082], [24.73542, 43.68523], [24.96682, 43.72693], [25.10718, 43.6831], [25.17144, 43.70261], [25.39528, 43.61866], [25.72792, 43.69263], [25.94911, 43.85745], [26.05584, 43.90925], [26.10115, 43.96908], [26.38764, 44.04356], [26.62712, 44.05698], [26.95141, 44.13555], [27.26845, 44.12602], [27.39757, 44.0141], [27.60834, 44.01206], [27.64542, 44.04958], [27.73468, 43.95326], [27.92008, 44.00761], [27.99558, 43.84193], [28.23293, 43.76], [29.24336, 43.70874], [30.04414, 45.08461], [29.69272, 45.19227], [29.65428, 45.25629], [29.68175, 45.26885], [29.59798, 45.38857], [29.42632, 45.44545], [29.24779, 45.43388], [28.96077, 45.33164], [28.94292, 45.28045], [28.81383, 45.3384], [28.78911, 45.24179], [28.71358, 45.22631], [28.5735, 45.24759], [28.34554, 45.32102], [28.28504, 45.43907], [28.21139, 45.46895], [28.18741, 45.47358], [28.08927, 45.6051], [28.16568, 45.6421], [28.13111, 45.92819], [28.08612, 46.01105], [28.13684, 46.18099], [28.10937, 46.22852], [28.19864, 46.31869], [28.18902, 46.35283], [28.25769, 46.43334], [28.22281, 46.50481], [28.24808, 46.64305], [28.12173, 46.82283], [28.09095, 46.97621], [27.81892, 47.1381], [27.73172, 47.29248], [27.68706, 47.28962], [27.60263, 47.32507], [27.55731, 47.46637], [27.47942, 47.48113], [27.3979, 47.59473], [27.32202, 47.64009], [27.25519, 47.71366], [27.29069, 47.73722], [27.1618, 47.92391], [27.15622, 47.98538]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "RS",
+           iso1A3: "SRB",
+           iso1N3: "688",
+           wikidata: "Q403",
+           nameEn: "Serbia",
+           groups: ["039", "150", "UN"],
+           callingCodes: ["381"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[19.66007, 46.19005], [19.56113, 46.16824], [19.52473, 46.1171], [19.28826, 45.99694], [19.14543, 45.9998], [19.10388, 46.04015], [19.0791, 45.96458], [19.01284, 45.96529], [18.99712, 45.93537], [18.81394, 45.91329], [18.85783, 45.85493], [18.90305, 45.71863], [18.96691, 45.66731], [18.88776, 45.57253], [18.94562, 45.53712], [19.07471, 45.53086], [19.08364, 45.48804], [18.99918, 45.49333], [18.97446, 45.37528], [19.10774, 45.29547], [19.28208, 45.23813], [19.41941, 45.23475], [19.43589, 45.17137], [19.19144, 45.17863], [19.14063, 45.12972], [19.07952, 45.14668], [19.1011, 45.01191], [19.05205, 44.97692], [19.15573, 44.95409], [19.06853, 44.89915], [19.02871, 44.92541], [18.98957, 44.90645], [19.01994, 44.85493], [19.18183, 44.92055], [19.36722, 44.88164], [19.32543, 44.74058], [19.26388, 44.65412], [19.16699, 44.52197], [19.13369, 44.52521], [19.12278, 44.50132], [19.14837, 44.45253], [19.14681, 44.41463], [19.11785, 44.40313], [19.10749, 44.39421], [19.10704, 44.38249], [19.10365, 44.37795], [19.10298, 44.36924], [19.11865, 44.36712], [19.1083, 44.3558], [19.11547, 44.34218], [19.13556, 44.338], [19.13332, 44.31492], [19.16741, 44.28648], [19.18328, 44.28383], [19.20508, 44.2917], [19.23306, 44.26097], [19.26945, 44.26957], [19.32464, 44.27185], [19.34773, 44.23244], [19.3588, 44.18353], [19.40927, 44.16722], [19.43905, 44.13088], [19.47338, 44.15034], [19.48386, 44.14332], [19.47321, 44.1193], [19.51167, 44.08158], [19.55999, 44.06894], [19.57467, 44.04716], [19.61991, 44.05254], [19.61836, 44.01464], [19.56498, 43.99922], [19.52515, 43.95573], [19.38439, 43.96611], [19.24363, 44.01502], [19.23465, 43.98764], [19.3986, 43.79668], [19.5176, 43.71403], [19.50455, 43.58385], [19.42696, 43.57987], [19.41941, 43.54056], [19.36653, 43.60921], [19.33426, 43.58833], [19.2553, 43.5938], [19.24774, 43.53061], [19.22807, 43.5264], [19.22229, 43.47926], [19.44315, 43.38846], [19.48171, 43.32644], [19.52962, 43.31623], [19.54598, 43.25158], [19.62661, 43.2286], [19.64063, 43.19027], [19.76918, 43.16044], [19.79255, 43.11951], [19.92576, 43.08539], [19.96549, 43.11098], [19.98887, 43.0538], [20.04729, 43.02732], [20.05431, 42.99571], [20.12325, 42.96237], [20.14896, 42.99058], [20.16415, 42.97177], [20.34528, 42.90676], [20.35692, 42.8335], [20.40594, 42.84853], [20.43734, 42.83157], [20.53484, 42.8885], [20.48692, 42.93208], [20.59929, 43.01067], [20.64557, 43.00826], [20.69515, 43.09641], [20.59929, 43.20492], [20.68688, 43.21335], [20.73811, 43.25068], [20.82145, 43.26769], [20.88685, 43.21697], [20.83727, 43.17842], [20.96287, 43.12416], [21.00749, 43.13984], [21.05378, 43.10707], [21.08952, 43.13471], [21.14465, 43.11089], [21.16734, 42.99694], [21.2041, 43.02277], [21.23877, 43.00848], [21.23534, 42.95523], [21.2719, 42.8994], [21.32974, 42.90424], [21.36941, 42.87397], [21.44047, 42.87276], [21.39045, 42.74888], [21.47498, 42.74695], [21.59154, 42.72643], [21.58755, 42.70418], [21.6626, 42.67813], [21.75025, 42.70125], [21.79413, 42.65923], [21.75672, 42.62695], [21.7327, 42.55041], [21.70522, 42.54176], [21.7035, 42.51899], [21.62556, 42.45106], [21.64209, 42.41081], [21.62887, 42.37664], [21.59029, 42.38042], [21.57021, 42.3647], [21.53467, 42.36809], [21.5264, 42.33634], [21.56772, 42.30946], [21.58992, 42.25915], [21.70111, 42.23789], [21.77176, 42.2648], [21.84654, 42.3247], [21.91595, 42.30392], [21.94405, 42.34669], [22.02908, 42.29848], [22.16384, 42.32103], [22.29605, 42.37477], [22.29275, 42.34913], [22.34773, 42.31725], [22.45919, 42.33822], [22.47498, 42.3915], [22.51961, 42.3991], [22.55669, 42.50144], [22.43983, 42.56851], [22.4997, 42.74144], [22.43309, 42.82057], [22.54302, 42.87774], [22.74826, 42.88701], [22.78397, 42.98253], [22.89521, 43.03625], [22.98104, 43.11199], [23.00806, 43.19279], [22.89727, 43.22417], [22.82036, 43.33665], [22.53397, 43.47225], [22.47582, 43.6558], [22.41043, 43.69566], [22.35558, 43.81281], [22.41449, 44.00514], [22.61688, 44.06534], [22.61711, 44.16938], [22.67173, 44.21564], [22.68166, 44.28206], [22.56012, 44.30712], [22.45436, 44.47258], [22.54021, 44.47836], [22.56493, 44.53419], [22.61368, 44.55719], [22.70981, 44.51852], [22.76749, 44.54446], [22.69196, 44.61587], [22.61917, 44.61489], [22.45301, 44.7194], [22.30844, 44.6619], [22.18315, 44.48179], [22.13234, 44.47444], [22.08016, 44.49844], [21.99364, 44.63395], [21.7795, 44.66165], [21.71692, 44.65349], [21.67504, 44.67107], [21.61942, 44.67059], [21.60019, 44.75208], [21.55007, 44.77304], [21.38802, 44.78133], [21.35643, 44.86364], [21.44013, 44.87613], [21.48202, 44.87199], [21.56328, 44.89502], [21.54938, 44.9327], [21.35855, 45.01941], [21.4505, 45.04294], [21.51299, 45.15345], [21.48278, 45.19557], [21.29398, 45.24148], [21.20392, 45.2677], [21.17612, 45.32566], [21.09894, 45.30144], [20.87948, 45.42743], [20.86026, 45.47295], [20.77217, 45.49788], [20.83321, 45.53567], [20.76798, 45.60969], [20.80361, 45.65875], [20.82364, 45.77738], [20.78446, 45.78522], [20.77416, 45.75601], [20.70069, 45.7493], [20.65645, 45.82801], [20.54818, 45.89939], [20.35862, 45.99356], [20.26068, 46.12332], [20.09713, 46.17315], [20.03533, 46.14509], [20.01816, 46.17696], [19.93508, 46.17553], [19.81491, 46.1313], [19.66007, 46.19005]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "RU",
+           iso1A3: "RUS",
+           iso1N3: "643",
+           wikidata: "Q159",
+           nameEn: "Russia"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "RW",
+           iso1A3: "RWA",
+           iso1N3: "646",
+           wikidata: "Q1037",
+           nameEn: "Rwanda",
+           groups: ["014", "202", "002", "UN"],
+           callingCodes: ["250"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[30.47194, -1.0555], [30.35212, -1.06896], [30.16369, -1.34303], [29.912, -1.48269], [29.82657, -1.31187], [29.59061, -1.39016], [29.53062, -1.40499], [29.45038, -1.5054], [29.36322, -1.50887], [29.24323, -1.66826], [29.24458, -1.69663], [29.11847, -1.90576], [29.17562, -2.12278], [29.105, -2.27043], [29.00051, -2.29001], [28.95642, -2.37321], [28.89601, -2.37321], [28.86826, -2.41888], [28.86846, -2.44866], [28.89132, -2.47557], [28.89342, -2.49017], [28.88846, -2.50493], [28.87497, -2.50887], [28.86209, -2.5231], [28.86193, -2.53185], [28.87943, -2.55165], [28.89288, -2.55848], [28.90226, -2.62385], [28.89793, -2.66111], [28.94346, -2.69124], [29.00357, -2.70596], [29.04081, -2.7416], [29.0562, -2.58632], [29.32234, -2.6483], [29.36805, -2.82933], [29.88237, -2.75105], [29.95911, -2.33348], [30.14034, -2.43626], [30.42933, -2.31064], [30.54501, -2.41404], [30.83915, -2.35795], [30.89303, -2.08223], [30.80802, -1.91477], [30.84079, -1.64652], [30.71974, -1.43244], [30.57123, -1.33264], [30.50889, -1.16412], [30.45116, -1.10641], [30.47194, -1.0555]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SA",
+           iso1A3: "SAU",
+           iso1N3: "682",
+           wikidata: "Q851",
+           nameEn: "Saudi Arabia",
+           groups: ["145", "142", "UN"],
+           callingCodes: ["966"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[40.01521, 32.05667], [39.29903, 32.23259], [38.99233, 31.99721], [36.99791, 31.50081], [37.99354, 30.49998], [37.66395, 30.33245], [37.4971, 29.99949], [36.75083, 29.86903], [36.50005, 29.49696], [36.07081, 29.18469], [34.8812, 29.36878], [34.4454, 27.91479], [37.8565, 22.00903], [39.63762, 18.37348], [40.99158, 15.81743], [42.15205, 16.40211], [42.76801, 16.40371], [42.94625, 16.39721], [42.94351, 16.49467], [42.97215, 16.51093], [43.11601, 16.53166], [43.15274, 16.67248], [43.22066, 16.65179], [43.21325, 16.74416], [43.25857, 16.75304], [43.26303, 16.79479], [43.24801, 16.80613], [43.22956, 16.80613], [43.22012, 16.83932], [43.18338, 16.84852], [43.1398, 16.90696], [43.19328, 16.94703], [43.1813, 16.98438], [43.18233, 17.02673], [43.23967, 17.03428], [43.17787, 17.14717], [43.20156, 17.25901], [43.32653, 17.31179], [43.22533, 17.38343], [43.29185, 17.53224], [43.43005, 17.56148], [43.70631, 17.35762], [44.50126, 17.47475], [46.31018, 17.20464], [46.76494, 17.29151], [47.00571, 16.94765], [47.48245, 17.10808], [47.58351, 17.50366], [48.19996, 18.20584], [49.04884, 18.59899], [52.00311, 19.00083], [54.99756, 20.00083], [55.66469, 21.99658], [55.2137, 22.71065], [55.13599, 22.63334], [52.56622, 22.94341], [51.59617, 24.12041], [51.58871, 24.27256], [51.41644, 24.39615], [51.58834, 24.66608], [51.39468, 24.62785], [51.29972, 24.50747], [51.09638, 24.46907], [50.92992, 24.54396], [50.8133, 24.74049], [50.57069, 25.57887], [50.302, 25.87592], [50.26923, 26.08243], [50.38162, 26.53976], [50.71771, 26.73086], [50.37726, 27.89227], [49.98877, 27.87827], [49.00421, 28.81495], [48.42991, 28.53628], [47.70561, 28.5221], [47.59863, 28.66798], [47.58376, 28.83382], [47.46202, 29.0014], [46.5527, 29.10283], [46.42415, 29.05947], [44.72255, 29.19736], [42.97796, 30.48295], [42.97601, 30.72204], [40.01521, 32.05667]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SB",
+           iso1A3: "SLB",
+           iso1N3: "090",
+           wikidata: "Q685",
+           nameEn: "Solomon Islands",
+           groups: ["054", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["677"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[172.71443, -12.01327], [160.43769, -4.17974], [156.03296, -6.55528], [156.03993, -6.65703], [155.92557, -6.84664], [155.69784, -6.92661], [155.60735, -6.92266], [154.74815, -7.33315], [156.73836, -14.50464], [172.71443, -12.01327]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SC",
+           iso1A3: "SYC",
+           iso1N3: "690",
+           wikidata: "Q1042",
+           nameEn: "Seychelles",
+           groups: ["014", "202", "002", "UN"],
+           driveSide: "left",
+           callingCodes: ["248"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[43.75112, -10.38913], [54.83239, -10.93575], [66.3222, 5.65313], [43.75112, -10.38913]]]]
          }
-
-         function updateForCurrentLocale() {
-           if (!_localeCode) return;
-           _languageCode = _localeCode.split('-')[0];
-           var currentData = _dataLocales[_localeCode] || _dataLocales[_languageCode];
-           var hash = utilStringQs(window.location.hash);
-
-           if (hash.rtl === 'true') {
-             _textDirection = 'rtl';
-           } else if (hash.rtl === 'false') {
-             _textDirection = 'ltr';
-           } else {
-             _textDirection = currentData && currentData.rtl ? 'rtl' : 'ltr';
-           }
-
-           var locale = _localeCode;
-           if (locale.toLowerCase() === 'en-us') locale = 'en';
-           _languageNames = _localeStrings[locale].languageNames;
-           _scriptNames = _localeStrings[locale].scriptNames;
-           _usesMetric = _localeCode.slice(-3).toLowerCase() !== '-us';
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SD",
+           iso1A3: "SDN",
+           iso1N3: "729",
+           wikidata: "Q1049",
+           nameEn: "Sudan",
+           groups: ["015", "002", "UN"],
+           callingCodes: ["249"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[37.8565, 22.00903], [34.0765, 22.00501], [33.99686, 21.76784], [33.57251, 21.72406], [33.17563, 22.00405], [24.99885, 21.99535], [24.99794, 19.99661], [23.99715, 20.00038], [23.99539, 19.49944], [23.99997, 15.69575], [23.62785, 15.7804], [23.38812, 15.69649], [23.10792, 15.71297], [22.93201, 15.55107], [22.92579, 15.47007], [22.99584, 15.40105], [22.99584, 15.22989], [22.66115, 14.86308], [22.70474, 14.69149], [22.38562, 14.58907], [22.44944, 14.24986], [22.55997, 14.23024], [22.5553, 14.11704], [22.22995, 13.96754], [22.08674, 13.77863], [22.29689, 13.3731], [22.1599, 13.19281], [22.02914, 13.13976], [21.94819, 13.05637], [21.81432, 12.81362], [21.89371, 12.68001], [21.98711, 12.63292], [22.15679, 12.66634], [22.22684, 12.74682], [22.46345, 12.61925], [22.38873, 12.45514], [22.50548, 12.16769], [22.48369, 12.02766], [22.64092, 12.07485], [22.54907, 11.64372], [22.7997, 11.40424], [22.93124, 11.41645], [22.97249, 11.21955], [22.87758, 10.91915], [23.02221, 10.69235], [23.3128, 10.45214], [23.67164, 9.86923], [23.69155, 9.67566], [24.09319, 9.66572], [24.12744, 9.73784], [24.49389, 9.79962], [24.84653, 9.80643], [24.97739, 9.9081], [25.05688, 10.06776], [25.0918, 10.33718], [25.78141, 10.42599], [25.93163, 10.38159], [25.93241, 10.17941], [26.21338, 9.91545], [26.35815, 9.57946], [26.70685, 9.48735], [27.14427, 9.62858], [27.90704, 9.61323], [28.99983, 9.67155], [29.06988, 9.74826], [29.53844, 9.75133], [29.54, 10.07949], [29.94629, 10.29245], [30.00389, 10.28633], [30.53005, 9.95992], [30.82893, 9.71451], [30.84605, 9.7498], [31.28504, 9.75287], [31.77539, 10.28939], [31.99177, 10.65065], [32.46967, 11.04662], [32.39358, 11.18207], [32.39578, 11.70208], [32.10079, 11.95203], [32.73921, 11.95203], [32.73921, 12.22757], [33.25876, 12.22111], [33.13988, 11.43248], [33.26977, 10.83632], [33.24645, 10.77913], [33.52294, 10.64382], [33.66604, 10.44254], [33.80913, 10.32994], [33.90159, 10.17179], [33.96984, 10.15446], [33.99185, 9.99623], [33.96323, 9.80972], [33.9082, 9.762], [33.87958, 9.49937], [34.10229, 9.50238], [34.08717, 9.55243], [34.13186, 9.7492], [34.20484, 9.9033], [34.22718, 10.02506], [34.32102, 10.11599], [34.34783, 10.23914], [34.2823, 10.53508], [34.4372, 10.781], [34.59062, 10.89072], [34.77383, 10.74588], [34.77532, 10.69027], [34.86618, 10.74588], [34.86916, 10.78832], [34.97491, 10.86147], [34.97789, 10.91559], [34.93172, 10.95946], [35.01215, 11.19626], [34.95704, 11.24448], [35.09556, 11.56278], [35.05832, 11.71158], [35.11492, 11.85156], [35.24302, 11.91132], [35.70476, 12.67101], [36.01458, 12.72478], [36.14268, 12.70879], [36.16651, 12.88019], [36.13374, 12.92665], [36.24545, 13.36759], [36.38993, 13.56459], [36.48824, 13.83954], [36.44653, 13.95666], [36.54376, 14.25597], [36.44337, 15.14963], [36.54276, 15.23478], [36.69761, 15.75323], [36.76371, 15.80831], [36.92193, 16.23451], [36.99777, 17.07172], [37.42694, 17.04041], [37.50967, 17.32199], [38.13362, 17.53906], [38.37133, 17.66269], [38.45916, 17.87167], [38.57727, 17.98125], [39.63762, 18.37348], [37.8565, 22.00903]]]]
          }
-         /* Locales */
-         // Returns a Promise to load the strings for the requested locale
-
-
-         localizer.loadLocale = function (requested) {
-           if (!_dataLocales) {
-             return Promise.reject('loadLocale called before init');
-           }
-
-           var locale = requested; // US English is the default
-
-           if (locale.toLowerCase() === 'en-us') locale = 'en';
-
-           if (!_dataLocales[locale]) {
-             return Promise.reject("Unsupported locale: ".concat(requested));
-           }
-
-           if (_localeStrings[locale]) {
-             // already loaded
-             return Promise.resolve(locale);
-           }
-
-           var fileMap = _mainFileFetcher.fileMap();
-           var key = "locale_".concat(locale);
-           fileMap[key] = "locales/".concat(locale, ".json");
-           return _mainFileFetcher.get(key).then(function (d) {
-             _localeStrings[locale] = d[locale];
-             return locale;
-           });
-         };
-
-         localizer.pluralRule = function (number) {
-           return pluralRule(number, _localeCode);
-         }; // Returns the plural rule for the given `number` with the given `localeCode`.
-         // One of: `zero`, `one`, `two`, `few`, `many`, `other`
-
-
-         function pluralRule(number, localeCode) {
-           // modern browsers have this functionality built-in
-           var rules = 'Intl' in window && Intl.PluralRules && new Intl.PluralRules(localeCode);
-
-           if (rules) {
-             return rules.select(number);
-           } // fallback to basic one/other, as in English
-
-
-           if (number === 1) return 'one';
-           return 'other';
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SE",
+           iso1A3: "SWE",
+           iso1N3: "752",
+           wikidata: "Q34",
+           nameEn: "Sweden",
+           groups: ["EU", "154", "150", "UN"],
+           callingCodes: ["46"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[24.15791, 65.85385], [23.90497, 66.15802], [23.71339, 66.21299], [23.64982, 66.30603], [23.67591, 66.3862], [23.63776, 66.43568], [23.85959, 66.56434], [23.89488, 66.772], [23.98059, 66.79585], [23.98563, 66.84149], [23.56214, 67.17038], [23.58735, 67.20752], [23.54701, 67.25435], [23.75372, 67.29914], [23.75372, 67.43688], [23.39577, 67.46974], [23.54701, 67.59306], [23.45627, 67.85297], [23.65793, 67.9497], [23.40081, 68.05545], [23.26469, 68.15134], [23.15377, 68.14759], [23.10336, 68.26551], [22.73028, 68.40881], [22.00429, 68.50692], [21.03001, 68.88969], [20.90649, 68.89696], [20.85104, 68.93142], [20.91658, 68.96764], [20.78802, 69.03087], [20.55258, 69.06069], [20.0695, 69.04469], [20.28444, 68.93283], [20.33435, 68.80174], [20.22027, 68.67246], [19.95647, 68.55546], [20.22027, 68.48759], [19.93508, 68.35911], [18.97255, 68.52416], [18.63032, 68.50849], [18.39503, 68.58672], [18.1241, 68.53721], [18.13836, 68.20874], [17.90787, 67.96537], [17.30416, 68.11591], [16.7409, 67.91037], [16.38441, 67.52923], [16.12774, 67.52106], [16.09922, 67.4364], [16.39154, 67.21653], [16.35589, 67.06419], [15.37197, 66.48217], [15.49318, 66.28509], [15.05113, 66.15572], [14.53778, 66.12399], [14.50926, 65.31786], [13.64276, 64.58402], [14.11117, 64.46674], [14.16051, 64.18725], [13.98222, 64.00953], [13.23411, 64.09087], [12.74105, 64.02171], [12.14928, 63.59373], [12.19919, 63.47935], [11.98529, 63.27487], [12.19919, 63.00104], [12.07085, 62.6297], [12.29187, 62.25699], [12.14746, 61.7147], [12.40595, 61.57226], [12.57707, 61.56547], [12.86939, 61.35427], [12.69115, 61.06584], [12.2277, 61.02442], [12.59133, 60.50559], [12.52003, 60.13846], [12.36317, 59.99259], [12.15641, 59.8926], [11.87121, 59.86039], [11.92112, 59.69531], [11.69297, 59.59442], [11.8213, 59.24985], [11.65732, 58.90177], [11.45199, 58.89604], [11.4601, 58.99022], [11.34459, 59.11672], [11.15367, 59.07862], [11.08911, 58.98745], [10.64958, 58.89391], [10.40861, 58.38489], [12.16597, 56.60205], [12.07466, 56.29488], [12.65312, 56.04345], [12.6372, 55.91371], [12.88472, 55.63369], [12.60345, 55.42675], [12.84405, 55.13257], [14.28399, 55.1553], [14.89259, 55.5623], [15.79951, 55.54655], [19.64795, 57.06466], [19.84909, 57.57876], [20.5104, 59.15546], [19.08191, 60.19152], [19.23413, 60.61414], [20.15877, 63.06556], [24.14112, 65.39731], [24.15107, 65.81427], [24.14798, 65.83466], [24.15791, 65.85385]]]]
          }
-         /**
-         * Try to find that string in `locale` or the current `_localeCode` matching
-         * the given `stringId`. If no string can be found in the requested locale,
-         * we'll recurse down all the `_localeCodes` until one is found.
-         *
-         * @param  {string}   stringId      string identifier
-         * @param  {object?}  replacements  token replacements and default string
-         * @param  {string?}  locale        locale to use (defaults to currentLocale)
-         * @return {string?}  localized string
-         */
-
-
-         localizer.tInfo = function (stringId, replacements, locale) {
-           locale = locale || _localeCode;
-           var path = stringId.split('.').map(function (s) {
-             return s.replace(/<TX_DOT>/g, '.');
-           }).reverse();
-           var stringsKey = locale; // US English is the default
-
-           if (stringsKey.toLowerCase() === 'en-us') stringsKey = 'en';
-           var result = _localeStrings[stringsKey];
-
-           while (result !== undefined && path.length) {
-             result = result[path.pop()];
-           }
-
-           if (result !== undefined) {
-             if (replacements) {
-               if (_typeof(result) === 'object' && Object.keys(result).length) {
-                 // If plural forms are provided, dig one level deeper based on the
-                 // first numeric token replacement provided.
-                 var number = Object.values(replacements).find(function (value) {
-                   return typeof value === 'number';
-                 });
-
-                 if (number !== undefined) {
-                   var rule = pluralRule(number, locale);
-
-                   if (result[rule]) {
-                     result = result[rule];
-                   } else {
-                     // We're pretty sure this should be a plural but no string
-                     // could be found for the given rule. Just pick the first
-                     // string and hope it makes sense.
-                     result = Object.values(result)[0];
-                   }
-                 }
-               }
-
-               if (typeof result === 'string') {
-                 for (var key in replacements) {
-                   var value = replacements[key];
-
-                   if (typeof value === 'number' && value.toLocaleString) {
-                     // format numbers for the locale
-                     value = value.toLocaleString(locale, {
-                       style: 'decimal',
-                       useGrouping: true,
-                       minimumFractionDigits: 0
-                     });
-                   }
-
-                   var token = "{".concat(key, "}");
-                   var regex = new RegExp(token, 'g');
-                   result = result.replace(regex, value);
-                 }
-               }
-             }
-
-             if (typeof result === 'string') {
-               // found a localized string!
-               return {
-                 text: result,
-                 locale: locale
-               };
-             }
-           } // no localized string found...
-           // attempt to fallback to a lower-priority language
-
-
-           var index = _localeCodes.indexOf(locale);
-
-           if (index >= 0 && index < _localeCodes.length - 1) {
-             // eventually this will be 'en' or another locale with 100% coverage
-             var fallback = _localeCodes[index + 1];
-             return localizer.tInfo(stringId, replacements, fallback);
-           }
-
-           if (replacements && 'default' in replacements) {
-             // Fallback to a default value if one is specified in `replacements`
-             return {
-               text: replacements["default"],
-               locale: null
-             };
-           }
-
-           var missing = "Missing ".concat(locale, " translation: ").concat(stringId);
-           if (typeof console !== 'undefined') console.error(missing); // eslint-disable-line
-
-           return {
-             text: missing,
-             locale: 'en'
-           };
-         }; // Returns only the localized text, discarding the locale info
-
-
-         localizer.t = function (stringId, replacements, locale) {
-           return localizer.tInfo(stringId, replacements, locale).text;
-         }; // Returns the localized text wrapped in an HTML element encoding the locale info
-
-
-         localizer.t.html = function (stringId, replacements, locale) {
-           var info = localizer.tInfo(stringId, replacements, locale); // text may be empty or undefined if `replacements.default` is
-
-           return info.text ? localizer.htmlForLocalizedText(info.text, info.locale) : '';
-         };
-
-         localizer.htmlForLocalizedText = function (text, localeCode) {
-           return "<span class=\"localized-text\" lang=\"".concat(localeCode || 'unknown', "\">").concat(text, "</span>");
-         };
-
-         localizer.languageName = function (code, options) {
-           if (_languageNames[code]) {
-             // name in locale language
-             // e.g. "German"
-             return _languageNames[code];
-           } // sometimes we only want the local name
-
-
-           if (options && options.localOnly) return null;
-           var langInfo = _dataLanguages[code];
-
-           if (langInfo) {
-             if (langInfo.nativeName) {
-               // name in native language
-               // e.g. "Deutsch (de)"
-               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 language this is based on
-
-               if (_languageNames[base]) {
-                 // base language name in locale language
-                 var scriptCode = langInfo.script;
-                 var script = _scriptNames[scriptCode] || scriptCode; // e.g. "Serbian (Cyrillic)"
-
-                 return localizer.t('translate.language_and_code', {
-                   language: _languageNames[base],
-                   code: script
-                 });
-               } else if (_dataLanguages[base] && _dataLanguages[base].nativeName) {
-                 // e.g. "српски (sr-Cyrl)"
-                 return localizer.t('translate.language_and_code', {
-                   language: _dataLanguages[base].nativeName,
-                   code: code
-                 });
-               }
-             }
-           }
-
-           return code; // if not found, use the code
-         };
-
-         return localizer;
-       }
-
-       // `presetCollection` is a wrapper around an `Array` of presets `collection`,
-       // and decorated with some extra methods for searching and matching geometry
-       //
-
-       function presetCollection(collection) {
-         var MAXRESULTS = 50;
-         var _this = {};
-         var _memo = {};
-         _this.collection = collection;
-
-         _this.item = function (id) {
-           if (_memo[id]) return _memo[id];
-
-           var found = _this.collection.find(function (d) {
-             return d.id === id;
-           });
-
-           if (found) _memo[id] = found;
-           return found;
-         };
-
-         _this.index = function (id) {
-           return _this.collection.findIndex(function (d) {
-             return d.id === id;
-           });
-         };
-
-         _this.matchGeometry = function (geometry) {
-           return presetCollection(_this.collection.filter(function (d) {
-             return d.matchGeometry(geometry);
-           }));
-         };
-
-         _this.matchAllGeometry = function (geometries) {
-           return presetCollection(_this.collection.filter(function (d) {
-             return d && d.matchAllGeometry(geometries);
-           }));
-         };
-
-         _this.matchAnyGeometry = function (geometries) {
-           return presetCollection(_this.collection.filter(function (d) {
-             return geometries.some(function (geom) {
-               return d.matchGeometry(geom);
-             });
-           }));
-         };
-
-         _this.fallback = function (geometry) {
-           var id = geometry;
-           if (id === 'vertex') id = 'point';
-           return _this.item(id);
-         };
-
-         _this.search = function (value, geometry, countryCode) {
-           if (!value) return _this;
-           value = value.toLowerCase().trim(); // match at name beginning or just after a space (e.g. "office" -> match "Law Office")
-
-           function leading(a) {
-             var index = a.indexOf(value);
-             return index === 0 || a[index - 1] === ' ';
-           } // match at name beginning only
-
-
-           function leadingStrict(a) {
-             var index = a.indexOf(value);
-             return index === 0;
-           }
-
-           function sortNames(a, b) {
-             var aCompare = (a.suggestion ? a.originalName : a.name()).toLowerCase();
-             var bCompare = (b.suggestion ? b.originalName : b.name()).toLowerCase(); // priority if search string matches preset name exactly - #4325
-
-             if (value === aCompare) return -1;
-             if (value === bCompare) return 1; // priority for higher matchScore
-
-             var i = b.originalScore - a.originalScore;
-             if (i !== 0) return i; // priority if search string appears earlier in preset name
-
-             i = aCompare.indexOf(value) - bCompare.indexOf(value);
-             if (i !== 0) return i; // priority for shorter preset names
-
-             return aCompare.length - bCompare.length;
-           }
-
-           var pool = _this.collection;
-
-           if (countryCode) {
-             pool = pool.filter(function (a) {
-               if (a.countryCodes && a.countryCodes.indexOf(countryCode) === -1) return false;
-               if (a.notCountryCodes && a.notCountryCodes.indexOf(countryCode) !== -1) return false;
-               return true;
-             });
-           }
-
-           var searchable = pool.filter(function (a) {
-             return a.searchable !== false && a.suggestion !== true;
-           });
-           var suggestions = pool.filter(function (a) {
-             return a.suggestion === true;
-           }); // matches value to preset.name
-
-           var leading_name = searchable.filter(function (a) {
-             return leading(a.name().toLowerCase());
-           }).sort(sortNames); // matches value to preset suggestion name (original name is unhyphenated)
-
-           var leading_suggestions = suggestions.filter(function (a) {
-             return leadingStrict(a.originalName.toLowerCase());
-           }).sort(sortNames); // matches value to preset.terms values
-
-           var leading_terms = searchable.filter(function (a) {
-             return (a.terms() || []).some(leading);
-           }); // matches value to preset.tags values
-
-           var leading_tag_values = searchable.filter(function (a) {
-             return Object.values(a.tags || {}).filter(function (val) {
-               return val !== '*';
-             }).some(leading);
-           }); // finds close matches to value in preset.name
-
-           var similar_name = searchable.map(function (a) {
-             return {
-               preset: a,
-               dist: utilEditDistance(value, a.name())
-             };
-           }).filter(function (a) {
-             return a.dist + Math.min(value.length - a.preset.name().length, 0) < 3;
-           }).sort(function (a, b) {
-             return a.dist - b.dist;
-           }).map(function (a) {
-             return a.preset;
-           }); // finds close matches to value to preset suggestion name (original name is unhyphenated)
-
-           var similar_suggestions = suggestions.map(function (a) {
-             return {
-               preset: a,
-               dist: utilEditDistance(value, a.originalName.toLowerCase())
-             };
-           }).filter(function (a) {
-             return a.dist + Math.min(value.length - a.preset.originalName.length, 0) < 1;
-           }).sort(function (a, b) {
-             return a.dist - b.dist;
-           }).map(function (a) {
-             return a.preset;
-           }); // finds close matches to value in preset.terms
-
-           var similar_terms = searchable.filter(function (a) {
-             return (a.terms() || []).some(function (b) {
-               return utilEditDistance(value, b) + Math.min(value.length - b.length, 0) < 3;
-             });
-           });
-           var results = leading_name.concat(leading_suggestions, leading_terms, leading_tag_values, similar_name, similar_suggestions, similar_terms).slice(0, MAXRESULTS - 1);
-
-           if (geometry) {
-             if (typeof geometry === 'string') {
-               results.push(_this.fallback(geometry));
-             } else {
-               geometry.forEach(function (geom) {
-                 return results.push(_this.fallback(geom));
-               });
-             }
-           }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SG",
+           iso1A3: "SGP",
+           iso1N3: "702",
+           wikidata: "Q334",
+           nameEn: "Singapore",
+           groups: ["035", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["65"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[104.00131, 1.42405], [103.93384, 1.42926], [103.89565, 1.42841], [103.86383, 1.46288], [103.81181, 1.47953], [103.76395, 1.45183], [103.74161, 1.4502], [103.7219, 1.46108], [103.67468, 1.43166], [103.62738, 1.35255], [103.56591, 1.19719], [103.66049, 1.18825], [103.74084, 1.12902], [104.03085, 1.26954], [104.12282, 1.27714], [104.08072, 1.35998], [104.09162, 1.39694], [104.08871, 1.42015], [104.07348, 1.43322], [104.04622, 1.44691], [104.02277, 1.4438], [104.00131, 1.42405]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SH",
+           iso1A3: "SHN",
+           iso1N3: "654",
+           wikidata: "Q192184",
+           nameEn: "Saint Helena, Ascension and Tristan da Cunha",
+           country: "GB"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SI",
+           iso1A3: "SVN",
+           iso1N3: "705",
+           wikidata: "Q215",
+           nameEn: "Slovenia",
+           groups: ["EU", "039", "150", "UN"],
+           callingCodes: ["386"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[16.50139, 46.56684], [16.39217, 46.63673], [16.38594, 46.6549], [16.41863, 46.66238], [16.42641, 46.69228], [16.37816, 46.69975], [16.30966, 46.7787], [16.31303, 46.79838], [16.3408, 46.80641], [16.34547, 46.83836], [16.2941, 46.87137], [16.2365, 46.87775], [16.21892, 46.86961], [16.15711, 46.85434], [16.14365, 46.8547], [16.10983, 46.867], [16.05786, 46.83927], [15.99054, 46.82772], [15.99126, 46.78199], [15.98432, 46.74991], [15.99769, 46.7266], [16.02808, 46.71094], [16.04347, 46.68694], [16.04036, 46.6549], [15.99988, 46.67947], [15.98512, 46.68463], [15.94864, 46.68769], [15.87691, 46.7211], [15.8162, 46.71897], [15.78518, 46.70712], [15.76771, 46.69863], [15.73823, 46.70011], [15.72279, 46.69548], [15.69523, 46.69823], [15.67411, 46.70735], [15.6543, 46.70616], [15.6543, 46.69228], [15.6365, 46.6894], [15.63255, 46.68069], [15.62317, 46.67947], [15.59826, 46.68908], [15.54533, 46.66985], [15.55333, 46.64988], [15.54431, 46.6312], [15.46906, 46.61321], [15.45514, 46.63697], [15.41235, 46.65556], [15.23711, 46.63994], [15.14215, 46.66131], [15.01451, 46.641], [14.98024, 46.6009], [14.96002, 46.63459], [14.92283, 46.60848], [14.87129, 46.61], [14.86419, 46.59411], [14.83549, 46.56614], [14.81836, 46.51046], [14.72185, 46.49974], [14.66892, 46.44936], [14.5942, 46.43434], [14.56463, 46.37208], [14.52176, 46.42617], [14.45877, 46.41717], [14.42608, 46.44614], [14.314, 46.43327], [14.28326, 46.44315], [14.15989, 46.43327], [14.12097, 46.47724], [14.04002, 46.49117], [14.00422, 46.48474], [13.89837, 46.52331], [13.7148, 46.5222], [13.68684, 46.43881], [13.59777, 46.44137], [13.5763, 46.42613], [13.5763, 46.40915], [13.47019, 46.3621], [13.43418, 46.35992], [13.44808, 46.33507], [13.37671, 46.29668], [13.42218, 46.20758], [13.47587, 46.22725], [13.56114, 46.2054], [13.56682, 46.18703], [13.64451, 46.18966], [13.66472, 46.17392], [13.64053, 46.13587], [13.57072, 46.09022], [13.50104, 46.05986], [13.49568, 46.04839], [13.50998, 46.04498], [13.49702, 46.01832], [13.47474, 46.00546], [13.50104, 45.98078], [13.52963, 45.96588], [13.56759, 45.96991], [13.58903, 45.99009], [13.62074, 45.98388], [13.63458, 45.98947], [13.64307, 45.98326], [13.6329, 45.94894], [13.63815, 45.93607], [13.61931, 45.91782], [13.60857, 45.89907], [13.59565, 45.89446], [13.58644, 45.88173], [13.57563, 45.8425], [13.58858, 45.83503], [13.59784, 45.8072], [13.66986, 45.79955], [13.8235, 45.7176], [13.83332, 45.70855], [13.83422, 45.68703], [13.87933, 45.65207], [13.9191, 45.6322], [13.8695, 45.60835], [13.86771, 45.59898], [13.84106, 45.58185], [13.78445, 45.5825], [13.74587, 45.59811], [13.7198, 45.59352], [13.6076, 45.64761], [13.45644, 45.59464], [13.56979, 45.4895], [13.62902, 45.45898], [13.67398, 45.4436], [13.7785, 45.46787], [13.81742, 45.43729], [13.88124, 45.42637], [13.90771, 45.45149], [13.97309, 45.45258], [13.99488, 45.47551], [13.96063, 45.50825], [14.00578, 45.52352], [14.07116, 45.48752], [14.20348, 45.46896], [14.22371, 45.50388], [14.24239, 45.50607], [14.26611, 45.48239], [14.27681, 45.4902], [14.32487, 45.47142], [14.36693, 45.48642], [14.49769, 45.54424], [14.5008, 45.60852], [14.53816, 45.6205], [14.57397, 45.67165], [14.60977, 45.66403], [14.59576, 45.62812], [14.69694, 45.57366], [14.68605, 45.53006], [14.71718, 45.53442], [14.80124, 45.49515], [14.81992, 45.45913], [14.90554, 45.47769], [14.92266, 45.52788], [15.02385, 45.48533], [15.05187, 45.49079], [15.16862, 45.42309], [15.27758, 45.46678], [15.33051, 45.45258], [15.38188, 45.48752], [15.30249, 45.53224], [15.29837, 45.5841], [15.27747, 45.60504], [15.31027, 45.6303], [15.34695, 45.63382], [15.34214, 45.64702], [15.38952, 45.63682], [15.4057, 45.64727], [15.34919, 45.71623], [15.30872, 45.69014], [15.25423, 45.72275], [15.40836, 45.79491], [15.47531, 45.79802], [15.47325, 45.8253], [15.52234, 45.82195], [15.57952, 45.84953], [15.64185, 45.82915], [15.66662, 45.84085], [15.70411, 45.8465], [15.68232, 45.86819], [15.68383, 45.88867], [15.67967, 45.90455], [15.70636, 45.92116], [15.70327, 46.00015], [15.71246, 46.01196], [15.72977, 46.04682], [15.62317, 46.09103], [15.6083, 46.11992], [15.59909, 46.14761], [15.64904, 46.19229], [15.6434, 46.21396], [15.67395, 46.22478], [15.75436, 46.21969], [15.75479, 46.20336], [15.78817, 46.21719], [15.79284, 46.25811], [15.97965, 46.30652], [16.07616, 46.3463], [16.07314, 46.36458], [16.05065, 46.3833], [16.05281, 46.39141], [16.14859, 46.40547], [16.18824, 46.38282], [16.30233, 46.37837], [16.30162, 46.40437], [16.27329, 46.41467], [16.27398, 46.42875], [16.25124, 46.48067], [16.23961, 46.49653], [16.26759, 46.50566], [16.26733, 46.51505], [16.29793, 46.5121], [16.37193, 46.55008], [16.38771, 46.53608], [16.44036, 46.5171], [16.5007, 46.49644], [16.52604, 46.47831], [16.59527, 46.47524], [16.52604, 46.5051], [16.52885, 46.53303], [16.50139, 46.56684]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SJ",
+           iso1A3: "SJM",
+           iso1N3: "744",
+           wikidata: "Q842829",
+           nameEn: "Svalbard and Jan Mayen",
+           country: "NO"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SK",
+           iso1A3: "SVK",
+           iso1N3: "703",
+           wikidata: "Q214",
+           nameEn: "Slovakia",
+           groups: ["EU", "151", "150", "UN"],
+           callingCodes: ["421"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[19.82237, 49.27806], [19.78581, 49.41701], [19.72127, 49.39288], [19.6375, 49.40897], [19.64162, 49.45184], [19.57845, 49.46077], [19.53313, 49.52856], [19.52626, 49.57311], [19.45348, 49.61583], [19.37795, 49.574], [19.36009, 49.53747], [19.25435, 49.53391], [19.18019, 49.41165], [18.9742, 49.39557], [18.97283, 49.49914], [18.94536, 49.52143], [18.84521, 49.51672], [18.74761, 49.492], [18.67757, 49.50895], [18.6144, 49.49824], [18.57183, 49.51162], [18.53063, 49.49022], [18.54848, 49.47059], [18.44686, 49.39467], [18.4084, 49.40003], [18.4139, 49.36517], [18.36446, 49.3267], [18.18456, 49.28909], [18.15022, 49.24518], [18.1104, 49.08624], [18.06885, 49.03157], [17.91814, 49.01784], [17.87831, 48.92679], [17.77944, 48.92318], [17.73126, 48.87885], [17.7094, 48.86721], [17.5295, 48.81117], [17.45671, 48.85004], [17.3853, 48.80936], [17.29054, 48.85546], [17.19355, 48.87602], [17.11202, 48.82925], [17.00215, 48.70887], [16.93955, 48.60371], [16.94611, 48.53614], [16.85204, 48.44968], [16.8497, 48.38321], [16.83588, 48.3844], [16.83317, 48.38138], [16.84243, 48.35258], [16.90903, 48.32519], [16.89461, 48.31332], [16.97701, 48.17385], [17.02919, 48.13996], [17.05735, 48.14179], [17.09168, 48.09366], [17.07039, 48.0317], [17.16001, 48.00636], [17.23699, 48.02094], [17.71215, 47.7548], [18.02938, 47.75665], [18.29305, 47.73541], [18.56496, 47.76588], [18.66521, 47.76772], [18.74074, 47.8157], [18.8506, 47.82308], [18.76821, 47.87469], [18.76134, 47.97499], [18.82176, 48.04206], [19.01952, 48.07052], [19.23924, 48.0595], [19.28182, 48.08336], [19.47957, 48.09437], [19.52489, 48.19791], [19.63338, 48.25006], [19.92452, 48.1283], [20.24312, 48.2784], [20.29943, 48.26104], [20.5215, 48.53336], [20.83248, 48.5824], [21.11516, 48.49546], [21.44063, 48.58456], [21.6068, 48.50365], [21.67134, 48.3989], [21.72525, 48.34628], [21.8279, 48.33321], [21.83339, 48.36242], [22.14689, 48.4005], [22.16023, 48.56548], [22.21379, 48.6218], [22.34151, 48.68893], [22.42934, 48.92857], [22.48296, 48.99172], [22.54338, 49.01424], [22.56155, 49.08865], [22.04427, 49.22136], [21.96385, 49.3437], [21.82927, 49.39467], [21.77983, 49.35443], [21.62328, 49.4447], [21.43376, 49.41433], [21.27858, 49.45988], [21.19756, 49.4054], [21.12477, 49.43666], [21.041, 49.41791], [21.09799, 49.37176], [20.98733, 49.30774], [20.9229, 49.29626], [20.77971, 49.35383], [20.72274, 49.41813], [20.61666, 49.41791], [20.5631, 49.375], [20.46422, 49.41612], [20.39939, 49.3896], [20.31728, 49.39914], [20.31453, 49.34817], [20.21977, 49.35265], [20.13738, 49.31685], [20.08238, 49.1813], [19.98494, 49.22904], [19.90529, 49.23532], [19.86409, 49.19316], [19.75286, 49.20751], [19.82237, 49.27806]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SL",
+           iso1A3: "SLE",
+           iso1N3: "694",
+           wikidata: "Q1044",
+           nameEn: "Sierra Leone",
+           groups: ["011", "202", "002", "UN"],
+           callingCodes: ["232"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-10.27575, 8.48711], [-10.37257, 8.48941], [-10.54891, 8.31174], [-10.63934, 8.35326], [-10.70565, 8.29235], [-10.61422, 8.5314], [-10.47707, 8.67669], [-10.56197, 8.81225], [-10.5783, 9.06386], [-10.74484, 9.07998], [-10.6534, 9.29919], [-11.2118, 10.00098], [-11.89624, 9.99763], [-11.91023, 9.93927], [-12.12634, 9.87203], [-12.24262, 9.92386], [-12.47254, 9.86834], [-12.76788, 9.3133], [-12.94095, 9.26335], [-13.08953, 9.0409], [-13.18586, 9.0925], [-13.29911, 9.04245], [-14.36218, 8.64107], [-12.15048, 6.15992], [-11.50429, 6.92704], [-11.4027, 6.97746], [-11.29417, 7.21576], [-10.60422, 7.7739], [-10.60492, 8.04072], [-10.57523, 8.04829], [-10.51554, 8.1393], [-10.45023, 8.15627], [-10.35227, 8.15223], [-10.29839, 8.21283], [-10.31635, 8.28554], [-10.30084, 8.30008], [-10.27575, 8.48711]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SM",
+           iso1A3: "SMR",
+           iso1N3: "674",
+           wikidata: "Q238",
+           nameEn: "San Marino",
+           groups: ["039", "150", "UN"],
+           callingCodes: ["378"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[12.45648, 43.89369], [12.48771, 43.89706], [12.49429, 43.90973], [12.49247, 43.91774], [12.49724, 43.92248], [12.50269, 43.92363], [12.50496, 43.93017], [12.51553, 43.94096], [12.51427, 43.94897], [12.50655, 43.95796], [12.50875, 43.96198], [12.50622, 43.97131], [12.51109, 43.97201], [12.51064, 43.98165], [12.5154, 43.98508], [12.51463, 43.99122], [12.50678, 43.99113], [12.49406, 43.98492], [12.47853, 43.98052], [12.46205, 43.97463], [12.44684, 43.96597], [12.43662, 43.95698], [12.42005, 43.9578], [12.41414, 43.95273], [12.40415, 43.95485], [12.40506, 43.94325], [12.41165, 43.93769], [12.41551, 43.92984], [12.40733, 43.92379], [12.41233, 43.90956], [12.40935, 43.9024], [12.41641, 43.89991], [12.44184, 43.90498], [12.45648, 43.89369]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SN",
+           iso1A3: "SEN",
+           iso1N3: "686",
+           wikidata: "Q1041",
+           nameEn: "Senegal",
+           groups: ["011", "202", "002", "UN"],
+           callingCodes: ["221"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-14.32144, 16.61495], [-15.00557, 16.64997], [-15.6509, 16.50315], [-16.27016, 16.51565], [-16.4429, 16.20605], [-16.44814, 16.09753], [-16.48967, 16.0496], [-16.50854, 16.09032], [-17.15288, 16.07139], [-18.35085, 14.63444], [-17.43598, 13.59273], [-15.47902, 13.58758], [-15.36504, 13.79313], [-14.93719, 13.80173], [-14.34721, 13.46578], [-13.8955, 13.59126], [-13.79409, 13.34472], [-14.36795, 13.23033], [-15.14917, 13.57989], [-15.26908, 13.37768], [-15.80478, 13.34832], [-15.80355, 13.16729], [-16.69343, 13.16791], [-16.74676, 13.06025], [-17.43966, 13.04579], [-17.4623, 11.92379], [-16.70562, 12.34803], [-16.38191, 12.36449], [-16.20591, 12.46157], [-15.67302, 12.42974], [-15.17582, 12.6847], [-13.70523, 12.68013], [-13.05296, 12.64003], [-13.06603, 12.49342], [-12.87336, 12.51892], [-12.35415, 12.32758], [-11.91331, 12.42008], [-11.46267, 12.44559], [-11.37536, 12.40788], [-11.39935, 12.97808], [-11.63025, 13.39174], [-11.83345, 13.33333], [-12.06897, 13.71049], [-11.93043, 13.84505], [-12.23936, 14.76324], [-13.11029, 15.52116], [-13.43135, 16.09022], [-13.80075, 16.13961], [-14.32144, 16.61495]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SO",
+           iso1A3: "SOM",
+           iso1N3: "706",
+           wikidata: "Q1045",
+           nameEn: "Somalia",
+           groups: ["014", "202", "002", "UN"],
+           callingCodes: ["252"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[51.12877, 12.56479], [43.90659, 12.3823], [42.95776, 10.98533], [42.69452, 10.62672], [42.87643, 10.18441], [43.0937, 9.90579], [43.23518, 9.84605], [43.32613, 9.59205], [44.19222, 8.93028], [46.99339, 7.9989], [47.92477, 8.00111], [47.97917, 8.00124], [44.98104, 4.91821], [44.02436, 4.9451], [43.40263, 4.79289], [43.04177, 4.57923], [42.97746, 4.44032], [42.84526, 4.28357], [42.55853, 4.20518], [42.07619, 4.17667], [41.89488, 3.97375], [41.31368, 3.14314], [40.98767, 2.82959], [41.00099, -0.83068], [41.56, -1.59812], [41.56362, -1.66375], [41.75542, -1.85308], [57.49095, 8.14549], [51.12877, 12.56479]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SR",
+           iso1A3: "SUR",
+           iso1N3: "740",
+           wikidata: "Q730",
+           nameEn: "Suriname",
+           groups: ["005", "419", "019", "UN"],
+           driveSide: "left",
+           callingCodes: ["597"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-54.26916, 5.26909], [-54.01877, 5.52789], [-54.01074, 5.68785], [-53.7094, 6.2264], [-56.84822, 6.73257], [-57.31629, 5.33714], [-57.22536, 5.15605], [-57.37442, 5.0208], [-57.8699, 4.89394], [-58.0307, 3.95513], [-57.35891, 3.32121], [-56.70519, 2.02964], [-56.55439, 2.02003], [-56.47045, 1.95135], [-55.99278, 1.83137], [-55.89863, 1.89861], [-55.92159, 2.05236], [-56.13054, 2.27723], [-55.96292, 2.53188], [-55.71493, 2.40342], [-55.01919, 2.564], [-54.6084, 2.32856], [-54.42864, 2.42442], [-54.28534, 2.67798], [-53.9849, 3.58697], [-53.98914, 3.627], [-54.05128, 3.63557], [-54.19367, 3.84387], [-54.38444, 4.13222], [-54.4717, 4.91964], [-54.26916, 5.26909]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SS",
+           iso1A3: "SSD",
+           iso1N3: "728",
+           wikidata: "Q958",
+           nameEn: "South Sudan",
+           groups: ["014", "202", "002", "UN"],
+           callingCodes: ["211"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[34.10229, 9.50238], [33.87958, 9.49937], [33.9082, 9.762], [33.96323, 9.80972], [33.99185, 9.99623], [33.96984, 10.15446], [33.90159, 10.17179], [33.80913, 10.32994], [33.66604, 10.44254], [33.52294, 10.64382], [33.24645, 10.77913], [33.26977, 10.83632], [33.13988, 11.43248], [33.25876, 12.22111], [32.73921, 12.22757], [32.73921, 11.95203], [32.10079, 11.95203], [32.39578, 11.70208], [32.39358, 11.18207], [32.46967, 11.04662], [31.99177, 10.65065], [31.77539, 10.28939], [31.28504, 9.75287], [30.84605, 9.7498], [30.82893, 9.71451], [30.53005, 9.95992], [30.00389, 10.28633], [29.94629, 10.29245], [29.54, 10.07949], [29.53844, 9.75133], [29.06988, 9.74826], [28.99983, 9.67155], [27.90704, 9.61323], [27.14427, 9.62858], [26.70685, 9.48735], [26.35815, 9.57946], [26.21338, 9.91545], [25.93241, 10.17941], [25.93163, 10.38159], [25.78141, 10.42599], [25.0918, 10.33718], [25.05688, 10.06776], [24.97739, 9.9081], [24.84653, 9.80643], [24.49389, 9.79962], [24.12744, 9.73784], [24.09319, 9.66572], [23.69155, 9.67566], [23.62179, 9.53823], [23.64981, 9.44303], [23.64358, 9.28637], [23.56263, 9.19418], [23.4848, 9.16959], [23.44744, 8.99128], [23.59065, 8.99743], [23.51905, 8.71749], [24.25691, 8.69288], [24.13238, 8.36959], [24.35965, 8.26177], [24.85156, 8.16933], [24.98855, 7.96588], [25.25319, 7.8487], [25.29214, 7.66675], [25.20649, 7.61115], [25.20337, 7.50312], [25.35281, 7.42595], [25.37461, 7.33024], [25.90076, 7.09549], [26.38022, 6.63493], [26.32729, 6.36272], [26.58259, 6.1987], [26.51721, 6.09655], [27.22705, 5.71254], [27.22705, 5.62889], [27.28621, 5.56382], [27.23017, 5.37167], [27.26886, 5.25876], [27.44012, 5.07349], [27.56656, 4.89375], [27.65462, 4.89375], [27.76469, 4.79284], [27.79551, 4.59976], [28.20719, 4.35614], [28.6651, 4.42638], [28.8126, 4.48784], [29.03054, 4.48784], [29.22207, 4.34297], [29.43341, 4.50101], [29.49726, 4.7007], [29.82087, 4.56246], [29.79666, 4.37809], [30.06964, 4.13221], [30.1621, 4.10586], [30.22374, 3.93896], [30.27658, 3.95653], [30.47691, 3.83353], [30.55396, 3.84451], [30.57378, 3.74567], [30.56277, 3.62703], [30.78512, 3.67097], [30.80713, 3.60506], [30.85997, 3.5743], [30.85153, 3.48867], [30.97601, 3.693], [31.16666, 3.79853], [31.29476, 3.8015], [31.50478, 3.67814], [31.50776, 3.63652], [31.72075, 3.74354], [31.81459, 3.82083], [31.86821, 3.78664], [31.96205, 3.6499], [31.95907, 3.57408], [32.05187, 3.589], [32.08491, 3.56287], [32.08866, 3.53543], [32.19888, 3.50867], [32.20782, 3.6053], [32.41337, 3.748], [32.72021, 3.77327], [32.89746, 3.81339], [33.02852, 3.89296], [33.18356, 3.77812], [33.51264, 3.75068], [33.9873, 4.23316], [34.47601, 4.72162], [35.34151, 5.02364], [35.30992, 4.90402], [35.47843, 4.91872], [35.42366, 4.76969], [35.51424, 4.61643], [35.9419, 4.61933], [35.82118, 4.77382], [35.81968, 5.10757], [35.8576, 5.33413], [35.50792, 5.42431], [35.29938, 5.34042], [35.31188, 5.50106], [35.13058, 5.62118], [35.12611, 5.68937], [35.00546, 5.89387], [34.96227, 6.26415], [35.01738, 6.46991], [34.87736, 6.60161], [34.77459, 6.5957], [34.65096, 6.72589], [34.53776, 6.74808], [34.53925, 6.82794], [34.47669, 6.91076], [34.35753, 6.91963], [34.19369, 7.04382], [34.19369, 7.12807], [34.01495, 7.25664], [34.03878, 7.27437], [34.02984, 7.36449], [33.87642, 7.5491], [33.71407, 7.65983], [33.44745, 7.7543], [33.32531, 7.71297], [33.24637, 7.77939], [33.04944, 7.78989], [33.0006, 7.90333], [33.08401, 8.05822], [33.18083, 8.13047], [33.1853, 8.29264], [33.19721, 8.40317], [33.3119, 8.45474], [33.54575, 8.47094], [33.66938, 8.44442], [33.71407, 8.3678], [33.87195, 8.41938], [33.89579, 8.4842], [34.01346, 8.50041], [34.14453, 8.60204], [34.14304, 9.04654], [34.10229, 9.50238]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "ST",
+           iso1A3: "STP",
+           iso1N3: "678",
+           wikidata: "Q1039",
+           nameEn: "S\xE3o Tom\xE9 and Principe",
+           groups: ["017", "202", "002", "UN"],
+           callingCodes: ["239"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[4.34149, 1.91417], [6.6507, -0.28606], [7.9035, 1.92304], [4.34149, 1.91417]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SV",
+           iso1A3: "SLV",
+           iso1N3: "222",
+           wikidata: "Q792",
+           nameEn: "El Salvador",
+           groups: ["013", "003", "419", "019", "UN"],
+           callingCodes: ["503"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-89.34776, 14.43013], [-89.39028, 14.44561], [-89.57441, 14.41637], [-89.58814, 14.33165], [-89.50614, 14.26084], [-89.52397, 14.22628], [-89.61844, 14.21937], [-89.70756, 14.1537], [-89.75569, 14.07073], [-89.73251, 14.04133], [-89.76103, 14.02923], [-89.81807, 14.07073], [-89.88937, 14.0396], [-90.10505, 13.85104], [-90.11344, 13.73679], [-90.55276, 12.8866], [-88.11443, 12.63306], [-87.7346, 13.13228], [-87.55124, 13.12523], [-87.69751, 13.25228], [-87.73714, 13.32715], [-87.80177, 13.35689], [-87.84675, 13.41078], [-87.83467, 13.44655], [-87.77354, 13.45767], [-87.73841, 13.44169], [-87.72115, 13.46083], [-87.71657, 13.50577], [-87.78148, 13.52906], [-87.73106, 13.75443], [-87.68821, 13.80829], [-87.7966, 13.91353], [-88.00331, 13.86948], [-88.07641, 13.98447], [-88.23018, 13.99915], [-88.25791, 13.91108], [-88.48982, 13.86458], [-88.49738, 13.97224], [-88.70661, 14.04317], [-88.73182, 14.10919], [-88.815, 14.11652], [-88.85785, 14.17763], [-88.94608, 14.20207], [-89.04187, 14.33644], [-89.34776, 14.43013]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SX",
+           iso1A3: "SXM",
+           iso1N3: "534",
+           wikidata: "Q26273",
+           nameEn: "Sint Maarten",
+           aliases: ["NL-SX"],
+           country: "NL",
+           groups: ["Q1451600", "029", "003", "419", "019", "UN"],
+           callingCodes: ["1 721"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-63.33064, 17.9615], [-63.1055, 17.86651], [-62.93924, 18.02904], [-63.02323, 18.05757], [-63.04039, 18.05619], [-63.0579, 18.06614], [-63.07759, 18.04943], [-63.09686, 18.04608], [-63.11042, 18.05339], [-63.13502, 18.05445], [-63.33064, 17.9615]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SY",
+           iso1A3: "SYR",
+           iso1N3: "760",
+           wikidata: "Q858",
+           nameEn: "Syria",
+           groups: ["145", "142", "UN"],
+           callingCodes: ["963"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[42.23683, 37.2863], [42.21548, 37.28026], [42.20454, 37.28715], [42.22381, 37.30238], [42.22257, 37.31395], [42.2112, 37.32491], [42.19301, 37.31323], [42.18225, 37.28569], [42.00894, 37.17209], [41.515, 37.08084], [41.21937, 37.07665], [40.90856, 37.13147], [40.69136, 37.0996], [39.81589, 36.75538], [39.21538, 36.66834], [39.03217, 36.70911], [38.74042, 36.70629], [38.55908, 36.84429], [38.38859, 36.90064], [38.21064, 36.91842], [37.81974, 36.76055], [37.68048, 36.75065], [37.49103, 36.66904], [37.47253, 36.63243], [37.21988, 36.6736], [37.16177, 36.66069], [37.10894, 36.6704], [37.08279, 36.63495], [37.02088, 36.66422], [37.01647, 36.69512], [37.04619, 36.71101], [37.04399, 36.73483], [36.99886, 36.74012], [36.99557, 36.75997], [36.66727, 36.82901], [36.61581, 36.74629], [36.62681, 36.71189], [36.57398, 36.65186], [36.58829, 36.58295], [36.54206, 36.49539], [36.6081, 36.33772], [36.65653, 36.33861], [36.68672, 36.23677], [36.6125, 36.22592], [36.50463, 36.2419], [36.4617, 36.20461], [36.39206, 36.22088], [36.37474, 36.01163], [36.33956, 35.98687], [36.30099, 36.00985], [36.28338, 36.00273], [36.29769, 35.96086], [36.27678, 35.94839], [36.25366, 35.96264], [36.19973, 35.95195], [36.17441, 35.92076], [36.1623, 35.80925], [36.14029, 35.81015], [36.13919, 35.83692], [36.11827, 35.85923], [35.99829, 35.88242], [36.01844, 35.92403], [36.00514, 35.94113], [35.98499, 35.94107], [35.931, 35.92109], [35.51152, 36.10954], [35.48515, 34.70851], [35.97386, 34.63322], [35.98718, 34.64977], [36.29165, 34.62991], [36.32399, 34.69334], [36.35135, 34.68516], [36.35384, 34.65447], [36.42941, 34.62505], [36.46003, 34.6378], [36.45299, 34.59438], [36.41429, 34.61175], [36.39846, 34.55672], [36.3369, 34.52629], [36.34745, 34.5002], [36.4442, 34.50165], [36.46179, 34.46541], [36.55853, 34.41609], [36.53039, 34.3798], [36.56556, 34.31881], [36.60778, 34.31009], [36.58667, 34.27667], [36.59195, 34.2316], [36.62537, 34.20251], [36.5128, 34.09916], [36.50576, 34.05982], [36.41078, 34.05253], [36.28589, 33.91981], [36.38263, 33.86579], [36.3967, 33.83365], [36.14517, 33.85118], [36.06778, 33.82927], [35.9341, 33.6596], [36.05723, 33.57904], [35.94465, 33.52774], [35.94816, 33.47886], [35.88668, 33.43183], [35.82577, 33.40479], [35.81324, 33.36354], [35.77477, 33.33609], [35.813, 33.3172], [35.77513, 33.27342], [35.81295, 33.24841], [35.81647, 33.2028], [35.83846, 33.19397], [35.84285, 33.16673], [35.81911, 33.1336], [35.81911, 33.11077], [35.84802, 33.1031], [35.87188, 32.98028], [35.89298, 32.9456], [35.87012, 32.91976], [35.84021, 32.8725], [35.83758, 32.82817], [35.78745, 32.77938], [35.75983, 32.74803], [35.88405, 32.71321], [35.93307, 32.71966], [35.96633, 32.66237], [36.02239, 32.65911], [36.08074, 32.51463], [36.20379, 32.52751], [36.20875, 32.49529], [36.23948, 32.50108], [36.40959, 32.37908], [36.83946, 32.31293], [38.79171, 33.37328], [40.64314, 34.31604], [40.97676, 34.39788], [41.12388, 34.65742], [41.2345, 34.80049], [41.21654, 35.1508], [41.26569, 35.42708], [41.38184, 35.62502], [41.37027, 35.84095], [41.2564, 36.06012], [41.28864, 36.35368], [41.40058, 36.52502], [41.81736, 36.58782], [42.36697, 37.0627], [42.35724, 37.10998], [42.32313, 37.17814], [42.34735, 37.22548], [42.2824, 37.2798], [42.26039, 37.27017], [42.23683, 37.2863]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "SZ",
+           iso1A3: "SWZ",
+           iso1N3: "748",
+           wikidata: "Q1050",
+           nameEn: "Eswatini",
+           aliases: ["Swaziland"],
+           groups: ["018", "202", "002", "UN"],
+           driveSide: "left",
+           callingCodes: ["268"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[31.86881, -25.99973], [31.4175, -25.71886], [31.31237, -25.7431], [31.13073, -25.91558], [30.95819, -26.26303], [30.78927, -26.48271], [30.81101, -26.84722], [30.88826, -26.79622], [30.97757, -26.92706], [30.96088, -27.0245], [31.15027, -27.20151], [31.49834, -27.31549], [31.97592, -27.31675], [31.97463, -27.11057], [32.00893, -26.8096], [32.09664, -26.80721], [32.13315, -26.84345], [32.13409, -26.5317], [32.07352, -26.40185], [32.10435, -26.15656], [32.08599, -26.00978], [32.00916, -25.999], [31.974, -25.95387], [31.86881, -25.99973]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TA",
+           iso1A3: "TAA",
+           wikidata: "Q220982",
+           nameEn: "Tristan da Cunha",
+           aliases: ["SH-TA"],
+           country: "GB",
+           groups: ["SH", "BOTS", "011", "202", "002", "UN"],
+           isoStatus: "excRes",
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["290 8", "44 20"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-13.38232, -34.07258], [-16.67337, -41.9188], [-5.88482, -41.4829], [-13.38232, -34.07258]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TC",
+           iso1A3: "TCA",
+           iso1N3: "796",
+           wikidata: "Q18221",
+           nameEn: "Turks and Caicos Islands",
+           country: "GB",
+           groups: ["BOTS", "029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["1 649"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-71.70065, 25.7637], [-72.98446, 20.4801], [-69.80718, 21.35956], [-71.70065, 25.7637]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TD",
+           iso1A3: "TCD",
+           iso1N3: "148",
+           wikidata: "Q657",
+           nameEn: "Chad",
+           groups: ["017", "202", "002", "UN"],
+           callingCodes: ["235"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[23.99539, 19.49944], [15.99566, 23.49639], [14.99751, 23.00539], [15.19692, 21.99339], [15.20213, 21.49365], [15.28332, 21.44557], [15.62515, 20.95395], [15.57248, 20.92138], [15.55382, 20.86507], [15.56004, 20.79488], [15.59841, 20.74039], [15.6721, 20.70069], [15.99632, 20.35364], [15.75098, 19.93002], [15.6032, 18.77402], [15.50373, 16.89649], [14.37425, 15.72591], [13.86301, 15.04043], [13.78991, 14.87519], [13.809, 14.72915], [13.67878, 14.64013], [13.68573, 14.55276], [13.48259, 14.46704], [13.47559, 14.40881], [13.6302, 13.71094], [14.08251, 13.0797], [14.46881, 13.08259], [14.56101, 12.91036], [14.55058, 12.78256], [14.83314, 12.62963], [14.90827, 12.3269], [14.89019, 12.16593], [14.96952, 12.0925], [15.00146, 12.1223], [15.0349, 12.10698], [15.05786, 12.0608], [15.04808, 11.8731], [15.11579, 11.79313], [15.06595, 11.71126], [15.13149, 11.5537], [15.0585, 11.40481], [15.10021, 11.04101], [15.04957, 11.02347], [15.09127, 10.87431], [15.06737, 10.80921], [15.15532, 10.62846], [15.14936, 10.53915], [15.23724, 10.47764], [15.30874, 10.31063], [15.50535, 10.1098], [15.68761, 9.99344], [15.41408, 9.92876], [15.24618, 9.99246], [15.14043, 9.99246], [15.05999, 9.94845], [14.95722, 9.97926], [14.80082, 9.93818], [14.4673, 10.00264], [14.20411, 10.00055], [14.1317, 9.82413], [14.01793, 9.73169], [13.97544, 9.6365], [14.37094, 9.2954], [14.35707, 9.19611], [14.83566, 8.80557], [15.09484, 8.65982], [15.20426, 8.50892], [15.50743, 7.79302], [15.59272, 7.7696], [15.56964, 7.58936], [15.49743, 7.52179], [15.73118, 7.52006], [15.79942, 7.44149], [16.40703, 7.68809], [16.41583, 7.77971], [16.58315, 7.88657], [16.59415, 7.76444], [16.658, 7.75353], [16.6668, 7.67281], [16.8143, 7.53971], [17.67288, 7.98905], [17.93926, 7.95853], [18.02731, 8.01085], [18.6085, 8.05009], [18.64153, 8.08714], [18.62612, 8.14163], [18.67455, 8.22226], [18.79783, 8.25929], [19.11044, 8.68172], [18.86388, 8.87971], [19.06421, 9.00367], [20.36748, 9.11019], [20.82979, 9.44696], [21.26348, 9.97642], [21.34934, 9.95907], [21.52766, 10.2105], [21.63553, 10.217], [21.71479, 10.29932], [21.72139, 10.64136], [22.45889, 11.00246], [22.87758, 10.91915], [22.97249, 11.21955], [22.93124, 11.41645], [22.7997, 11.40424], [22.54907, 11.64372], [22.64092, 12.07485], [22.48369, 12.02766], [22.50548, 12.16769], [22.38873, 12.45514], [22.46345, 12.61925], [22.22684, 12.74682], [22.15679, 12.66634], [21.98711, 12.63292], [21.89371, 12.68001], [21.81432, 12.81362], [21.94819, 13.05637], [22.02914, 13.13976], [22.1599, 13.19281], [22.29689, 13.3731], [22.08674, 13.77863], [22.22995, 13.96754], [22.5553, 14.11704], [22.55997, 14.23024], [22.44944, 14.24986], [22.38562, 14.58907], [22.70474, 14.69149], [22.66115, 14.86308], [22.99584, 15.22989], [22.99584, 15.40105], [22.92579, 15.47007], [22.93201, 15.55107], [23.10792, 15.71297], [23.38812, 15.69649], [23.62785, 15.7804], [23.99997, 15.69575], [23.99539, 19.49944]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TF",
+           iso1A3: "ATF",
+           iso1N3: "260",
+           wikidata: "Q129003",
+           nameEn: "French Southern Territories",
+           country: "FR"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TG",
+           iso1A3: "TGO",
+           iso1N3: "768",
+           wikidata: "Q945",
+           nameEn: "Togo",
+           groups: ["011", "202", "002", "UN"],
+           callingCodes: ["228"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[0.50388, 11.01011], [-0.13493, 11.14075], [-0.14462, 11.10811], [-0.05733, 11.08628], [-0.0275, 11.11202], [-514e-5, 11.10763], [342e-5, 11.08317], [0.02395, 11.06229], [0.03355, 10.9807], [-63e-4, 10.96417], [-908e-5, 10.91644], [-0.02685, 10.8783], [-0.0228, 10.81916], [-0.07183, 10.76794], [-0.07327, 10.71845], [-0.09141, 10.7147], [-0.05945, 10.63458], [0.12886, 10.53149], [0.18846, 10.4096], [0.29453, 10.41546], [0.33028, 10.30408], [0.39584, 10.31112], [0.35293, 10.09412], [0.41371, 10.06361], [0.41252, 10.02018], [0.36366, 10.03309], [0.32075, 9.72781], [0.34816, 9.71607], [0.34816, 9.66907], [0.32313, 9.6491], [0.28261, 9.69022], [0.26712, 9.66437], [0.29334, 9.59387], [0.36008, 9.6256], [0.38153, 9.58682], [0.23851, 9.57389], [0.2409, 9.52335], [0.30406, 9.521], [0.31241, 9.50337], [0.2254, 9.47869], [0.25758, 9.42696], [0.33148, 9.44812], [0.36485, 9.49749], [0.49118, 9.48339], [0.56388, 9.40697], [0.45424, 9.04581], [0.52455, 8.87746], [0.37319, 8.75262], [0.47211, 8.59945], [0.64731, 8.48866], [0.73432, 8.29529], [0.63897, 8.25873], [0.5913, 8.19622], [0.61156, 8.18324], [0.6056, 8.13959], [0.58891, 8.12779], [0.62943, 7.85751], [0.58295, 7.62368], [0.51979, 7.58706], [0.52455, 7.45354], [0.57223, 7.39326], [0.62943, 7.41099], [0.65327, 7.31643], [0.59606, 7.01252], [0.52217, 6.9723], [0.52098, 6.94391], [0.56508, 6.92971], [0.52853, 6.82921], [0.57406, 6.80348], [0.58176, 6.76049], [0.6497, 6.73682], [0.63659, 6.63857], [0.74862, 6.56517], [0.71048, 6.53083], [0.89283, 6.33779], [0.99652, 6.33779], [1.03108, 6.24064], [1.05969, 6.22998], [1.09187, 6.17074], [1.19966, 6.17069], [1.19771, 6.11522], [1.27574, 5.93551], [1.67336, 6.02702], [1.62913, 6.24075], [1.79826, 6.28221], [1.76906, 6.43189], [1.58105, 6.68619], [1.61812, 6.74843], [1.55877, 6.99737], [1.64249, 6.99562], [1.61838, 9.0527], [1.5649, 9.16941], [1.41746, 9.3226], [1.33675, 9.54765], [1.36624, 9.5951], [1.35507, 9.99525], [0.77666, 10.37665], [0.80358, 10.71459], [0.8804, 10.803], [0.91245, 10.99597], [0.66104, 10.99964], [0.4958, 10.93269], [0.50521, 10.98035], [0.48852, 10.98561], [0.50388, 11.01011]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TH",
+           iso1A3: "THA",
+           iso1N3: "764",
+           wikidata: "Q869",
+           nameEn: "Thailand",
+           groups: ["035", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["66"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[100.08404, 20.36626], [99.95721, 20.46301], [99.91616, 20.44986], [99.90499, 20.4487], [99.89692, 20.44789], [99.89301, 20.44311], [99.89168, 20.44548], [99.88451, 20.44596], [99.88211, 20.44488], [99.86383, 20.44371], [99.81096, 20.33687], [99.68255, 20.32077], [99.46008, 20.39673], [99.46077, 20.36198], [99.5569, 20.20676], [99.52943, 20.14811], [99.416, 20.08614], [99.20328, 20.12877], [99.0735, 20.10298], [98.98679, 19.7419], [98.83661, 19.80931], [98.56065, 19.67807], [98.51182, 19.71303], [98.24884, 19.67876], [98.13829, 19.78541], [98.03314, 19.80941], [98.04364, 19.65755], [97.84715, 19.55782], [97.88423, 19.5041], [97.78769, 19.39429], [97.84186, 19.29526], [97.78606, 19.26769], [97.84024, 19.22217], [97.83479, 19.09972], [97.73797, 19.04261], [97.73654, 18.9812], [97.66487, 18.9371], [97.73836, 18.88478], [97.76752, 18.58097], [97.5258, 18.4939], [97.36444, 18.57138], [97.34522, 18.54596], [97.50383, 18.26844], [97.56219, 18.33885], [97.64116, 18.29778], [97.60841, 18.23846], [97.73723, 17.97912], [97.66794, 17.88005], [97.76407, 17.71595], [97.91829, 17.54504], [98.11185, 17.36829], [98.10439, 17.33847], [98.34566, 17.04822], [98.39441, 17.06266], [98.52624, 16.89979], [98.49603, 16.8446], [98.53833, 16.81934], [98.46994, 16.73613], [98.50253, 16.7139], [98.49713, 16.69022], [98.51043, 16.70107], [98.51579, 16.69433], [98.51472, 16.68521], [98.51833, 16.676], [98.51113, 16.64503], [98.5695, 16.62826], [98.57912, 16.55983], [98.63817, 16.47424], [98.68074, 16.27068], [98.84485, 16.42354], [98.92656, 16.36425], [98.8376, 16.11706], [98.69585, 16.13353], [98.57019, 16.04578], [98.59853, 15.87197], [98.541, 15.65406], [98.58598, 15.46821], [98.56027, 15.33471], [98.4866, 15.39154], [98.39351, 15.34177], [98.41906, 15.27103], [98.40522, 15.25268], [98.30446, 15.30667], [98.22, 15.21327], [98.18821, 15.13125], [98.24874, 14.83013], [98.56762, 14.37701], [98.97356, 14.04868], [99.16695, 13.72621], [99.20617, 13.20575], [99.12225, 13.19847], [99.10646, 13.05804], [99.18748, 12.9898], [99.18905, 12.84799], [99.29254, 12.68921], [99.409, 12.60603], [99.47519, 12.1353], [99.56445, 12.14805], [99.53424, 12.02317], [99.64891, 11.82699], [99.64108, 11.78948], [99.5672, 11.62732], [99.47598, 11.62434], [99.39485, 11.3925], [99.31573, 11.32081], [99.32756, 11.28545], [99.06938, 10.94857], [99.02337, 10.97217], [98.99701, 10.92962], [99.0069, 10.85485], [98.86819, 10.78336], [98.78511, 10.68351], [98.77275, 10.62548], [98.81944, 10.52761], [98.7391, 10.31488], [98.55174, 9.92804], [98.52291, 9.92389], [98.47298, 9.95782], [98.33094, 9.91973], [98.12555, 9.44056], [97.63455, 9.60854], [97.19814, 8.18901], [99.31854, 5.99868], [99.50117, 6.44501], [99.91873, 6.50233], [100.0756, 6.4045], [100.12, 6.42105], [100.19511, 6.72559], [100.29651, 6.68439], [100.30828, 6.66462], [100.31618, 6.66781], [100.31884, 6.66423], [100.32671, 6.66526], [100.32607, 6.65933], [100.31929, 6.65413], [100.35413, 6.54932], [100.41152, 6.52299], [100.41791, 6.5189], [100.42351, 6.51762], [100.43027, 6.52389], [100.66986, 6.45086], [100.74361, 6.50811], [100.74822, 6.46231], [100.81045, 6.45086], [100.85884, 6.24929], [101.10313, 6.25617], [101.12618, 6.19431], [101.06165, 6.14161], [101.12388, 6.11411], [101.087, 5.9193], [101.02708, 5.91013], [100.98815, 5.79464], [101.14062, 5.61613], [101.25755, 5.71065], [101.25524, 5.78633], [101.58019, 5.93534], [101.69773, 5.75881], [101.75074, 5.79091], [101.80144, 5.74505], [101.89188, 5.8386], [101.91776, 5.84269], [101.92819, 5.85511], [101.94712, 5.98421], [101.9714, 6.00575], [101.97114, 6.01992], [101.99209, 6.04075], [102.01835, 6.05407], [102.09182, 6.14161], [102.07732, 6.193], [102.08127, 6.22679], [102.09086, 6.23546], [102.46318, 7.22462], [102.47649, 9.66162], [102.52395, 11.25257], [102.91449, 11.65512], [102.90973, 11.75613], [102.83957, 11.8519], [102.78427, 11.98746], [102.77026, 12.06815], [102.70176, 12.1686], [102.73134, 12.37091], [102.78116, 12.40284], [102.7796, 12.43781], [102.57567, 12.65358], [102.51963, 12.66117], [102.4994, 12.71736], [102.53053, 12.77506], [102.49335, 12.92711], [102.48694, 12.97537], [102.52275, 12.99813], [102.46011, 13.08057], [102.43422, 13.09061], [102.36146, 13.26006], [102.36001, 13.31142], [102.34611, 13.35618], [102.35692, 13.38274], [102.35563, 13.47307], [102.361, 13.50551], [102.33828, 13.55613], [102.36859, 13.57488], [102.44601, 13.5637], [102.5358, 13.56933], [102.57573, 13.60461], [102.62483, 13.60883], [102.58635, 13.6286], [102.5481, 13.6589], [102.56848, 13.69366], [102.72727, 13.77806], [102.77864, 13.93374], [102.91251, 14.01531], [102.93275, 14.19044], [103.16469, 14.33075], [103.39353, 14.35639], [103.53518, 14.42575], [103.71109, 14.4348], [103.70175, 14.38052], [103.93836, 14.3398], [104.27616, 14.39861], [104.55014, 14.36091], [104.69335, 14.42726], [104.97667, 14.38806], [105.02804, 14.23722], [105.08408, 14.20402], [105.14012, 14.23873], [105.17748, 14.34432], [105.20894, 14.34967], [105.43783, 14.43865], [105.53864, 14.55731], [105.5121, 14.80802], [105.61162, 15.00037], [105.46661, 15.13132], [105.58043, 15.32724], [105.50662, 15.32054], [105.4692, 15.33709], [105.47635, 15.3796], [105.58191, 15.41031], [105.60446, 15.53301], [105.61756, 15.68792], [105.46573, 15.74742], [105.42285, 15.76971], [105.37959, 15.84074], [105.34115, 15.92737], [105.38508, 15.987], [105.42001, 16.00657], [105.06204, 16.09792], [105.00262, 16.25627], [104.88057, 16.37311], [104.73349, 16.565], [104.76099, 16.69302], [104.7397, 16.81005], [104.76442, 16.84752], [104.7373, 16.91125], [104.73712, 17.01404], [104.80716, 17.19025], [104.80061, 17.39367], [104.69867, 17.53038], [104.45404, 17.66788], [104.35432, 17.82871], [104.2757, 17.86139], [104.21776, 17.99335], [104.10927, 18.10826], [104.06533, 18.21656], [103.97725, 18.33631], [103.93916, 18.33914], [103.85642, 18.28666], [103.82449, 18.33979], [103.699, 18.34125], [103.60957, 18.40528], [103.47773, 18.42841], [103.41044, 18.4486], [103.30977, 18.4341], [103.24779, 18.37807], [103.23818, 18.34875], [103.29757, 18.30475], [103.17093, 18.2618], [103.14994, 18.23172], [103.1493, 18.17799], [103.07343, 18.12351], [103.07823, 18.03833], [103.0566, 18.00144], [103.01998, 17.97095], [102.9912, 17.9949], [102.95812, 18.0054], [102.86323, 17.97531], [102.81988, 17.94233], [102.79044, 17.93612], [102.75954, 17.89561], [102.68538, 17.86653], [102.67543, 17.84529], [102.69946, 17.81686], [102.68194, 17.80151], [102.59485, 17.83537], [102.5896, 17.84889], [102.61432, 17.92273], [102.60971, 17.95411], [102.59234, 17.96127], [102.45523, 17.97106], [102.11359, 18.21532], [101.88485, 18.02474], [101.78087, 18.07559], [101.72294, 17.92867], [101.44667, 17.7392], [101.15108, 17.47586], [100.96541, 17.57926], [101.02185, 17.87637], [101.1793, 18.0544], [101.19118, 18.2125], [101.15108, 18.25624], [101.18227, 18.34367], [101.06047, 18.43247], [101.27585, 18.68875], [101.22832, 18.73377], [101.25803, 18.89545], [101.35606, 19.04716], [101.261, 19.12717], [101.24911, 19.33334], [101.20604, 19.35296], [101.21347, 19.46223], [101.26991, 19.48324], [101.26545, 19.59242], [101.08928, 19.59748], [100.90302, 19.61901], [100.77231, 19.48324], [100.64606, 19.55884], [100.58219, 19.49164], [100.49604, 19.53504], [100.398, 19.75047], [100.5094, 19.87904], [100.58808, 20.15791], [100.55218, 20.17741], [100.51052, 20.14928], [100.47567, 20.19133], [100.4537, 20.19971], [100.44992, 20.23644], [100.41473, 20.25625], [100.37439, 20.35156], [100.33383, 20.4028], [100.25769, 20.3992], [100.22076, 20.31598], [100.16668, 20.2986], [100.1712, 20.24324], [100.11785, 20.24787], [100.09337, 20.26293], [100.09999, 20.31614], [100.08404, 20.36626]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TJ",
+           iso1A3: "TJK",
+           iso1N3: "762",
+           wikidata: "Q863",
+           nameEn: "Tajikistan",
+           groups: ["143", "142", "UN"],
+           callingCodes: ["992"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[70.45251, 41.04438], [70.38028, 41.02014], [70.36655, 40.90296], [69.69434, 40.62615], [69.59441, 40.70181], [69.53021, 40.77621], [69.38327, 40.7918], [69.32834, 40.70233], [69.3455, 40.57988], [69.2643, 40.57506], [69.21063, 40.54469], [69.27066, 40.49274], [69.28525, 40.41894], [69.30774, 40.36102], [69.33794, 40.34819], [69.32833, 40.29794], [69.30808, 40.2821], [69.24817, 40.30357], [69.25229, 40.26362], [69.30104, 40.24502], [69.30448, 40.18774], [69.2074, 40.21488], [69.15659, 40.2162], [69.04544, 40.22904], [68.85832, 40.20885], [68.84357, 40.18604], [68.79276, 40.17555], [68.77902, 40.20492], [68.5332, 40.14826], [68.52771, 40.11676], [68.62796, 40.07789], [69.01523, 40.15771], [69.01935, 40.11466], [68.96579, 40.06949], [68.84906, 40.04952], [68.93695, 39.91167], [68.88889, 39.87163], [68.63071, 39.85265], [68.61972, 39.68905], [68.54166, 39.53929], [68.12053, 39.56317], [67.70992, 39.66156], [67.62889, 39.60234], [67.44899, 39.57799], [67.46547, 39.53564], [67.39681, 39.52505], [67.46822, 39.46146], [67.45998, 39.315], [67.36522, 39.31287], [67.33226, 39.23739], [67.67833, 39.14479], [67.68915, 39.00775], [68.09704, 39.02589], [68.19743, 38.85985], [68.06948, 38.82115], [68.12877, 38.73677], [68.05598, 38.71641], [68.0807, 38.64136], [68.05873, 38.56087], [68.11366, 38.47169], [68.06274, 38.39435], [68.13289, 38.40822], [68.40343, 38.19484], [68.27159, 37.91477], [68.12635, 37.93], [67.81566, 37.43107], [67.8474, 37.31594], [67.78329, 37.1834], [67.7803, 37.08978], [67.87917, 37.0591], [68.02194, 36.91923], [68.18542, 37.02074], [68.27605, 37.00977], [68.29253, 37.10621], [68.41201, 37.10402], [68.41888, 37.13906], [68.61851, 37.19815], [68.6798, 37.27906], [68.81438, 37.23862], [68.80889, 37.32494], [68.91189, 37.26704], [68.88168, 37.33368], [68.96407, 37.32603], [69.03274, 37.25174], [69.25152, 37.09426], [69.39529, 37.16752], [69.45022, 37.23315], [69.36645, 37.40462], [69.44954, 37.4869], [69.51888, 37.5844], [69.80041, 37.5746], [69.84435, 37.60616], [69.93362, 37.61378], [69.95971, 37.5659], [70.15015, 37.52519], [70.28243, 37.66706], [70.27694, 37.81258], [70.1863, 37.84296], [70.17206, 37.93276], [70.4898, 38.12546], [70.54673, 38.24541], [70.60407, 38.28046], [70.61526, 38.34774], [70.64966, 38.34999], [70.69189, 38.37031], [70.6761, 38.39144], [70.67438, 38.40597], [70.69807, 38.41861], [70.72485, 38.4131], [70.75455, 38.4252], [70.77132, 38.45548], [70.78581, 38.45502], [70.78702, 38.45031], [70.79766, 38.44944], [70.80521, 38.44447], [70.81697, 38.44507], [70.82538, 38.45394], [70.84376, 38.44688], [70.88719, 38.46826], [70.92728, 38.43021], [70.98693, 38.48862], [71.03545, 38.44779], [71.0556, 38.40176], [71.09542, 38.42517], [71.10592, 38.42077], [71.10957, 38.40671], [71.1451, 38.40106], [71.21291, 38.32797], [71.33114, 38.30339], [71.33869, 38.27335], [71.37803, 38.25641], [71.36444, 38.15358], [71.29878, 38.04429], [71.28922, 38.01272], [71.27622, 37.99946], [71.27278, 37.96496], [71.24969, 37.93031], [71.2809, 37.91995], [71.296, 37.93403], [71.32871, 37.88564], [71.51565, 37.95349], [71.58843, 37.92425], [71.59255, 37.79956], [71.55752, 37.78677], [71.54324, 37.77104], [71.53053, 37.76534], [71.55234, 37.73209], [71.54186, 37.69691], [71.51972, 37.61945], [71.5065, 37.60912], [71.49693, 37.53527], [71.50616, 37.50733], [71.5256, 37.47971], [71.49612, 37.4279], [71.47685, 37.40281], [71.4862, 37.33405], [71.49821, 37.31975], [71.50674, 37.31502], [71.48536, 37.26017], [71.4824, 37.24921], [71.48339, 37.23937], [71.47386, 37.2269], [71.4555, 37.21418], [71.4494, 37.18137], [71.44127, 37.11856], [71.43097, 37.05855], [71.45578, 37.03094], [71.46923, 36.99925], [71.48481, 36.93218], [71.51502, 36.89128], [71.57195, 36.74943], [71.67083, 36.67346], [71.83229, 36.68084], [72.31676, 36.98115], [72.54095, 37.00007], [72.66381, 37.02014], [72.79693, 37.22222], [73.06884, 37.31729], [73.29633, 37.46495], [73.77197, 37.4417], [73.76647, 37.33913], [73.61129, 37.27469], [73.64974, 37.23643], [73.82552, 37.22659], [73.8564, 37.26158], [74.20308, 37.34208], [74.23339, 37.41116], [74.41055, 37.3948], [74.56161, 37.37734], [74.68383, 37.3948], [74.8294, 37.3435], [74.88887, 37.23275], [75.12328, 37.31839], [75.09719, 37.37297], [75.15899, 37.41443], [75.06011, 37.52779], [74.94338, 37.55501], [74.8912, 37.67576], [75.00935, 37.77486], [74.92416, 37.83428], [74.9063, 38.03033], [74.82665, 38.07359], [74.80331, 38.19889], [74.69894, 38.22155], [74.69619, 38.42947], [74.51217, 38.47034], [74.17022, 38.65504], [73.97933, 38.52945], [73.79806, 38.61106], [73.80656, 38.66449], [73.7033, 38.84782], [73.7445, 38.93867], [73.82964, 38.91517], [73.81728, 39.04007], [73.75823, 39.023], [73.60638, 39.24534], [73.54572, 39.27567], [73.55396, 39.3543], [73.5004, 39.38402], [73.59241, 39.40843], [73.59831, 39.46425], [73.45096, 39.46677], [73.31912, 39.38615], [73.18454, 39.35536], [72.85934, 39.35116], [72.62027, 39.39696], [72.33173, 39.33093], [72.23834, 39.17248], [72.17242, 39.2661], [72.09689, 39.26823], [72.04059, 39.36704], [71.90601, 39.27674], [71.79202, 39.27355], [71.7522, 39.32031], [71.80164, 39.40631], [71.76816, 39.45456], [71.62688, 39.44056], [71.5517, 39.45722], [71.55856, 39.57588], [71.49814, 39.61397], [71.08752, 39.50704], [71.06418, 39.41586], [70.7854, 39.38933], [70.64087, 39.58792], [70.44757, 39.60128], [70.2869, 39.53141], [70.11111, 39.58223], [69.87491, 39.53882], [69.68677, 39.59281], [69.3594, 39.52516], [69.26938, 39.8127], [69.35649, 40.01994], [69.43134, 39.98431], [69.43557, 39.92877], [69.53615, 39.93991], [69.5057, 40.03277], [69.53855, 40.0887], [69.53794, 40.11833], [69.55555, 40.12296], [69.57615, 40.10524], [69.64704, 40.12165], [69.67001, 40.10639], [70.01283, 40.23288], [70.58297, 40.00891], [70.57384, 39.99394], [70.47557, 39.93216], [70.55033, 39.96619], [70.58912, 39.95211], [70.65946, 39.9878], [70.65827, 40.0981], [70.7928, 40.12797], [70.80495, 40.16813], [70.9818, 40.22392], [70.8607, 40.217], [70.62342, 40.17396], [70.56394, 40.26421], [70.57149, 40.3442], [70.37511, 40.38605], [70.32626, 40.45174], [70.49871, 40.52503], [70.80009, 40.72825], [70.45251, 41.04438]]], [[[70.68112, 40.90612], [70.6158, 40.97661], [70.56077, 41.00642], [70.54223, 40.98787], [70.57501, 40.98941], [70.6721, 40.90555], [70.68112, 40.90612]]], [[[70.74189, 39.86319], [70.53651, 39.89155], [70.52631, 39.86989], [70.54998, 39.85137], [70.59667, 39.83542], [70.63105, 39.77923], [70.74189, 39.86319]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TK",
+           iso1A3: "TKL",
+           iso1N3: "772",
+           wikidata: "Q36823",
+           nameEn: "Tokelau",
+           country: "NZ",
+           groups: ["061", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["690"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-168.251, -9.44289], [-174.18635, -7.80441], [-174.17993, -10.13616], [-168.251, -9.44289]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TL",
+           iso1A3: "TLS",
+           iso1N3: "626",
+           wikidata: "Q574",
+           nameEn: "East Timor",
+           aliases: ["Timor-Leste", "TP"],
+           groups: ["035", "142", "UN"],
+           driveSide: "left",
+           callingCodes: ["670"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[124.46701, -9.13002], [124.94011, -8.85617], [124.97742, -9.08128], [125.11764, -8.96359], [125.18632, -9.03142], [125.18907, -9.16434], [125.09434, -9.19669], [125.04044, -9.17093], [124.97892, -9.19281], [125.09025, -9.46406], [125.68138, -9.85176], [127.55165, -9.05052], [127.42116, -8.22471], [125.87691, -8.31789], [125.58506, -7.95311], [124.92337, -8.75859], [124.33472, -9.11416], [124.04628, -9.22671], [124.04286, -9.34243], [124.10539, -9.41206], [124.14517, -9.42324], [124.21247, -9.36904], [124.28115, -9.42189], [124.28115, -9.50453], [124.3535, -9.48493], [124.35258, -9.43002], [124.38554, -9.3582], [124.45971, -9.30263], [124.46701, -9.13002]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TM",
+           iso1A3: "TKM",
+           iso1N3: "795",
+           wikidata: "Q874",
+           nameEn: "Turkmenistan",
+           groups: ["143", "142", "UN"],
+           callingCodes: ["993"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[60.5078, 41.21694], [60.06581, 41.4363], [60.18117, 41.60082], [60.06032, 41.76287], [60.08504, 41.80997], [60.33223, 41.75058], [59.95046, 41.97966], [60.0356, 42.01028], [60.04659, 42.08982], [59.96419, 42.1428], [60.00539, 42.212], [59.94633, 42.27655], [59.4341, 42.29738], [59.2955, 42.37064], [59.17317, 42.52248], [58.93422, 42.5407], [58.6266, 42.79314], [58.57991, 42.64988], [58.27504, 42.69632], [58.14321, 42.62159], [58.29427, 42.56497], [58.51674, 42.30348], [58.40688, 42.29535], [58.3492, 42.43335], [57.99214, 42.50021], [57.90975, 42.4374], [57.92897, 42.24047], [57.84932, 42.18555], [57.6296, 42.16519], [57.30275, 42.14076], [57.03633, 41.92043], [56.96218, 41.80383], [57.03359, 41.41777], [57.13796, 41.36625], [57.03423, 41.25435], [56.00314, 41.32584], [55.45471, 41.25609], [54.95182, 41.92424], [54.20635, 42.38477], [52.97575, 42.1308], [52.47884, 41.78034], [52.26048, 41.69249], [51.7708, 40.29239], [53.89734, 37.3464], [54.24565, 37.32047], [54.36211, 37.34912], [54.58664, 37.45809], [54.67247, 37.43532], [54.77822, 37.51597], [54.81804, 37.61285], [54.77684, 37.62264], [54.851, 37.75739], [55.13412, 37.94705], [55.44152, 38.08564], [55.76561, 38.12238], [55.97847, 38.08024], [56.33278, 38.08132], [56.32454, 38.18502], [56.43303, 38.26054], [56.62255, 38.24005], [56.73928, 38.27887], [57.03453, 38.18717], [57.21169, 38.28965], [57.37236, 38.09321], [57.35042, 37.98546], [57.79534, 37.89299], [58.21399, 37.77281], [58.22999, 37.6856], [58.39959, 37.63134], [58.47786, 37.6433], [58.5479, 37.70526], [58.6921, 37.64548], [58.9338, 37.67374], [59.22905, 37.51161], [59.33507, 37.53146], [59.39797, 37.47892], [59.39385, 37.34257], [59.55178, 37.13594], [59.74678, 37.12499], [60.00768, 37.04102], [60.34767, 36.63214], [61.14516, 36.64644], [61.18187, 36.55348], [61.1393, 36.38782], [61.22719, 36.12759], [61.12007, 35.95992], [61.22444, 35.92879], [61.26152, 35.80749], [61.22719, 35.67038], [61.27371, 35.61482], [61.58742, 35.43803], [61.77693, 35.41341], [61.97743, 35.4604], [62.05709, 35.43803], [62.15871, 35.33278], [62.29191, 35.25964], [62.29878, 35.13312], [62.48006, 35.28796], [62.62288, 35.22067], [62.74098, 35.25432], [62.90853, 35.37086], [63.0898, 35.43131], [63.12276, 35.53196], [63.10079, 35.63024], [63.23262, 35.67487], [63.10318, 35.81782], [63.12276, 35.86208], [63.29579, 35.85985], [63.53475, 35.90881], [63.56496, 35.95106], [63.98519, 36.03773], [64.05385, 36.10433], [64.43288, 36.24401], [64.57295, 36.34362], [64.62514, 36.44311], [64.61141, 36.6351], [64.97945, 37.21913], [65.51778, 37.23881], [65.64263, 37.34388], [65.64137, 37.45061], [65.72274, 37.55438], [66.30993, 37.32409], [66.55743, 37.35409], [66.52303, 37.39827], [66.65761, 37.45497], [66.52852, 37.58568], [66.53676, 37.80084], [66.67684, 37.96776], [66.56697, 38.0435], [66.41042, 38.02403], [66.24013, 38.16238], [65.83913, 38.25733], [65.55873, 38.29052], [64.32576, 38.98691], [64.19086, 38.95561], [63.70778, 39.22349], [63.6913, 39.27666], [62.43337, 39.98528], [62.34273, 40.43206], [62.11751, 40.58242], [61.87856, 41.12257], [61.4446, 41.29407], [61.39732, 41.19873], [61.33199, 41.14946], [61.22212, 41.14946], [61.03261, 41.25691], [60.5078, 41.21694]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TN",
+           iso1A3: "TUN",
+           iso1N3: "788",
+           wikidata: "Q948",
+           nameEn: "Tunisia",
+           groups: ["015", "002", "UN"],
+           callingCodes: ["216"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[11.2718, 37.6713], [7.89009, 38.19924], [8.59123, 37.14286], [8.64044, 36.9401], [8.62972, 36.86499], [8.67706, 36.8364], [8.57613, 36.78062], [8.46537, 36.7706], [8.47609, 36.66607], [8.16167, 36.48817], [8.18936, 36.44939], [8.40731, 36.42208], [8.2626, 35.91733], [8.26472, 35.73669], [8.35371, 35.66373], [8.36086, 35.47774], [8.30329, 35.29884], [8.47318, 35.23376], [8.3555, 35.10007], [8.30727, 34.95378], [8.25189, 34.92009], [8.29655, 34.72798], [8.20482, 34.57575], [7.86264, 34.3987], [7.81242, 34.21841], [7.74207, 34.16492], [7.66174, 34.20167], [7.52851, 34.06493], [7.54088, 33.7726], [7.73687, 33.42114], [7.83028, 33.18851], [8.11433, 33.10175], [8.1179, 33.05086], [8.31895, 32.83483], [8.35999, 32.50101], [9.07483, 32.07865], [9.55544, 30.23971], [9.76848, 30.34366], [9.88152, 30.34074], [10.29516, 30.90337], [10.12239, 31.42098], [10.31364, 31.72648], [10.48497, 31.72956], [10.62788, 31.96629], [10.7315, 31.97235], [11.04234, 32.2145], [11.53898, 32.4138], [11.57828, 32.48013], [11.46037, 32.6307], [11.51549, 33.09826], [11.55852, 33.1409], [11.58941, 33.36891], [11.2718, 37.6713]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TO",
+           iso1A3: "TON",
+           iso1N3: "776",
+           wikidata: "Q678",
+           nameEn: "Tonga",
+           groups: ["061", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["676"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-176.74538, -22.89767], [-180, -22.90585], [-180, -24.21376], [-173.10761, -24.19665], [-173.13438, -14.94228], [-176.76826, -14.95183], [-176.74538, -22.89767]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TR",
+           iso1A3: "TUR",
+           iso1N3: "792",
+           wikidata: "Q43",
+           nameEn: "Turkey",
+           groups: ["145", "142", "UN"],
+           callingCodes: ["90"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[41.54366, 41.52185], [40.89217, 41.72528], [34.8305, 42.4581], [28.32297, 41.98371], [28.02971, 41.98066], [27.91479, 41.97902], [27.83492, 41.99709], [27.81235, 41.94803], [27.69949, 41.97515], [27.55191, 41.90928], [27.52379, 41.93756], [27.45478, 41.96591], [27.27411, 42.10409], [27.22376, 42.10152], [27.19251, 42.06028], [27.08486, 42.08735], [27.03277, 42.0809], [26.95638, 42.00741], [26.79143, 41.97386], [26.62996, 41.97644], [26.56051, 41.92995], [26.57961, 41.90024], [26.53968, 41.82653], [26.36952, 41.82265], [26.33589, 41.76802], [26.32952, 41.73637], [26.35957, 41.71149], [26.47958, 41.67037], [26.5209, 41.62592], [26.59196, 41.60491], [26.59742, 41.48058], [26.61767, 41.42281], [26.62997, 41.34613], [26.5837, 41.32131], [26.5209, 41.33993], [26.39861, 41.25053], [26.32259, 41.24929], [26.31928, 41.07386], [26.3606, 41.02027], [26.33297, 40.98388], [26.35894, 40.94292], [26.32259, 40.94042], [26.28623, 40.93005], [26.29441, 40.89119], [26.26169, 40.9168], [26.20856, 40.86048], [26.21351, 40.83298], [26.15685, 40.80709], [26.12854, 40.77339], [26.12495, 40.74283], [26.08638, 40.73214], [26.0754, 40.72772], [26.03489, 40.73051], [25.94795, 40.72797], [26.04292, 40.3958], [25.61285, 40.17161], [25.94257, 39.39358], [26.43357, 39.43096], [26.70773, 39.0312], [26.61814, 38.81372], [26.21136, 38.65436], [26.32173, 38.48731], [26.24183, 38.44695], [26.21136, 38.17558], [27.05537, 37.9131], [27.16428, 37.72343], [26.99377, 37.69034], [26.95583, 37.64989], [27.14757, 37.32], [27.20312, 36.94571], [27.45627, 36.9008], [27.24613, 36.71622], [27.46117, 36.53789], [27.89482, 36.69898], [27.95037, 36.46155], [28.23708, 36.56812], [29.30783, 36.01033], [29.48192, 36.18377], [29.61002, 36.1731], [29.61805, 36.14179], [29.69611, 36.10365], [29.73302, 35.92555], [32.82353, 35.70297], [35.51152, 36.10954], [35.931, 35.92109], [35.98499, 35.94107], [36.00514, 35.94113], [36.01844, 35.92403], [35.99829, 35.88242], [36.11827, 35.85923], [36.13919, 35.83692], [36.14029, 35.81015], [36.1623, 35.80925], [36.17441, 35.92076], [36.19973, 35.95195], [36.25366, 35.96264], [36.27678, 35.94839], [36.29769, 35.96086], [36.28338, 36.00273], [36.30099, 36.00985], [36.33956, 35.98687], [36.37474, 36.01163], [36.39206, 36.22088], [36.4617, 36.20461], [36.50463, 36.2419], [36.6125, 36.22592], [36.68672, 36.23677], [36.65653, 36.33861], [36.6081, 36.33772], [36.54206, 36.49539], [36.58829, 36.58295], [36.57398, 36.65186], [36.62681, 36.71189], [36.61581, 36.74629], [36.66727, 36.82901], [36.99557, 36.75997], [36.99886, 36.74012], [37.04399, 36.73483], [37.04619, 36.71101], [37.01647, 36.69512], [37.02088, 36.66422], [37.08279, 36.63495], [37.10894, 36.6704], [37.16177, 36.66069], [37.21988, 36.6736], [37.47253, 36.63243], [37.49103, 36.66904], [37.68048, 36.75065], [37.81974, 36.76055], [38.21064, 36.91842], [38.38859, 36.90064], [38.55908, 36.84429], [38.74042, 36.70629], [39.03217, 36.70911], [39.21538, 36.66834], [39.81589, 36.75538], [40.69136, 37.0996], [40.90856, 37.13147], [41.21937, 37.07665], [41.515, 37.08084], [42.00894, 37.17209], [42.18225, 37.28569], [42.19301, 37.31323], [42.2112, 37.32491], [42.22257, 37.31395], [42.22381, 37.30238], [42.20454, 37.28715], [42.21548, 37.28026], [42.23683, 37.2863], [42.26039, 37.27017], [42.2824, 37.2798], [42.34735, 37.22548], [42.32313, 37.17814], [42.35724, 37.10998], [42.56725, 37.14878], [42.78887, 37.38615], [42.93705, 37.32015], [43.11403, 37.37436], [43.30083, 37.30629], [43.33508, 37.33105], [43.50787, 37.24436], [43.56702, 37.25675], [43.63085, 37.21957], [43.7009, 37.23692], [43.8052, 37.22825], [43.82699, 37.19477], [43.84878, 37.22205], [43.90949, 37.22453], [44.02002, 37.33229], [44.13521, 37.32486], [44.2613, 37.25055], [44.27998, 37.16501], [44.22239, 37.15756], [44.18503, 37.09551], [44.25975, 36.98119], [44.30645, 36.97373], [44.35937, 37.02843], [44.35315, 37.04955], [44.38117, 37.05825], [44.42631, 37.05825], [44.63179, 37.19229], [44.76698, 37.16162], [44.78319, 37.1431], [44.7868, 37.16644], [44.75986, 37.21549], [44.81021, 37.2915], [44.58449, 37.45018], [44.61401, 37.60165], [44.56887, 37.6429], [44.62096, 37.71985], [44.55498, 37.783], [44.45948, 37.77065], [44.3883, 37.85433], [44.22509, 37.88859], [44.42476, 38.25763], [44.50115, 38.33939], [44.44386, 38.38295], [44.38309, 38.36117], [44.3119, 38.37887], [44.3207, 38.49799], [44.32058, 38.62752], [44.28065, 38.6465], [44.26155, 38.71427], [44.30322, 38.81581], [44.18863, 38.93881], [44.20946, 39.13975], [44.1043, 39.19842], [44.03667, 39.39223], [44.22452, 39.4169], [44.29818, 39.378], [44.37921, 39.4131], [44.42832, 39.4131], [44.41849, 39.56659], [44.48111, 39.61579], [44.47298, 39.68788], [44.6137, 39.78393], [44.65422, 39.72163], [44.71806, 39.71124], [44.81043, 39.62677], [44.80977, 39.65768], [44.75779, 39.7148], [44.61845, 39.8281], [44.46635, 39.97733], [44.26973, 40.04866], [44.1778, 40.02845], [44.1057, 40.03555], [43.92307, 40.01787], [43.65688, 40.11199], [43.65221, 40.14889], [43.71136, 40.16673], [43.59928, 40.34019], [43.60862, 40.43267], [43.54791, 40.47413], [43.63664, 40.54159], [43.7425, 40.66805], [43.74872, 40.7365], [43.67712, 40.84846], [43.67712, 40.93084], [43.58683, 40.98961], [43.47319, 41.02251], [43.44984, 41.0988], [43.4717, 41.12611], [43.44973, 41.17666], [43.36118, 41.2028], [43.23096, 41.17536], [43.1945, 41.25242], [43.13373, 41.25503], [43.21707, 41.30331], [43.02956, 41.37891], [42.8785, 41.50516], [42.84899, 41.47265], [42.78995, 41.50126], [42.84471, 41.58912], [42.72794, 41.59714], [42.59202, 41.58183], [42.51772, 41.43606], [42.26387, 41.49346], [41.95134, 41.52466], [41.81939, 41.43621], [41.7124, 41.47417], [41.7148, 41.4932], [41.54366, 41.52185]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TT",
+           iso1A3: "TTO",
+           iso1N3: "780",
+           wikidata: "Q754",
+           nameEn: "Trinidad and Tobago",
+           groups: ["029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           callingCodes: ["1 868"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-61.62505, 11.18974], [-62.08693, 10.04435], [-60.89962, 9.81445], [-60.07172, 11.77667], [-61.62505, 11.18974]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TV",
+           iso1A3: "TUV",
+           iso1N3: "798",
+           wikidata: "Q672",
+           nameEn: "Tuvalu",
+           groups: ["061", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["688"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[174, -5], [174, -11.5], [179.99999, -11.5], [179.99999, -5], [174, -5]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TW",
+           iso1A3: "TWN",
+           iso1N3: "158",
+           wikidata: "Q865",
+           nameEn: "Taiwan",
+           aliases: ["RC"],
+           groups: ["030", "142"],
+           callingCodes: ["886"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[121.8109, 21.77688], [122.26612, 25.98197], [120.49232, 25.22863], [118.56434, 24.49266], [118.42453, 24.54644], [118.35291, 24.51645], [118.28244, 24.51231], [118.11703, 24.39734], [120.69238, 21.52331], [121.8109, 21.77688]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "TZ",
+           iso1A3: "TZA",
+           iso1N3: "834",
+           wikidata: "Q924",
+           nameEn: "Tanzania",
+           groups: ["014", "202", "002", "UN"],
+           driveSide: "left",
+           callingCodes: ["255"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[30.80408, -0.99911], [30.76635, -0.9852], [30.70631, -1.01175], [30.64166, -1.06601], [30.47194, -1.0555], [30.45116, -1.10641], [30.50889, -1.16412], [30.57123, -1.33264], [30.71974, -1.43244], [30.84079, -1.64652], [30.80802, -1.91477], [30.89303, -2.08223], [30.83915, -2.35795], [30.54501, -2.41404], [30.41789, -2.66266], [30.52747, -2.65841], [30.40662, -2.86151], [30.4987, -2.9573], [30.57926, -2.89791], [30.6675, -2.98987], [30.83823, -2.97837], [30.84165, -3.25152], [30.45915, -3.56532], [30.22042, -4.01738], [30.03323, -4.26631], [29.88172, -4.35743], [29.82885, -4.36153], [29.77289, -4.41733], [29.75109, -4.45836], [29.63827, -4.44681], [29.43673, -4.44845], [29.52552, -6.2731], [30.2567, -7.14121], [30.79243, -8.27382], [31.00796, -8.58615], [31.37533, -8.60769], [31.57147, -8.70619], [31.57147, -8.81388], [31.71158, -8.91386], [31.81587, -8.88618], [31.94663, -8.93846], [31.94196, -9.02303], [31.98866, -9.07069], [32.08206, -9.04609], [32.16146, -9.05993], [32.25486, -9.13371], [32.43543, -9.11988], [32.49147, -9.14754], [32.53661, -9.24281], [32.75611, -9.28583], [32.76233, -9.31963], [32.95389, -9.40138], [32.99397, -9.36712], [33.14925, -9.49322], [33.31581, -9.48554], [33.48052, -9.62442], [33.76677, -9.58516], [33.93298, -9.71647], [33.9638, -9.62206], [33.95829, -9.54066], [34.03865, -9.49398], [34.54499, -10.0678], [34.51911, -10.12279], [34.57581, -10.56271], [34.65946, -10.6828], [34.67047, -10.93796], [34.61161, -11.01611], [34.63305, -11.11731], [34.79375, -11.32245], [34.91153, -11.39799], [34.96296, -11.57354], [35.63599, -11.55927], [35.82767, -11.41081], [36.19094, -11.57593], [36.19094, -11.70008], [36.62068, -11.72884], [36.80309, -11.56836], [37.3936, -11.68949], [37.76614, -11.53352], [37.8388, -11.3123], [37.93618, -11.26228], [38.21598, -11.27289], [38.47258, -11.4199], [38.88996, -11.16978], [39.24395, -11.17433], [39.58249, -10.96043], [40.00295, -10.80255], [40.44265, -10.4618], [40.74206, -10.25691], [40.14328, -4.64201], [39.62121, -4.68136], [39.44306, -4.93877], [39.21631, -4.67835], [37.81321, -3.69179], [37.75036, -3.54243], [37.63099, -3.50723], [37.5903, -3.42735], [37.71745, -3.304], [37.67199, -3.06222], [34.0824, -1.02264], [34.03084, -1.05101], [34.02286, -1.00779], [33.93107, -0.99298], [30.80408, -0.99911]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "UA",
+           iso1A3: "UKR",
+           iso1N3: "804",
+           wikidata: "Q212",
+           nameEn: "Ukraine",
+           groups: ["151", "150", "UN"],
+           callingCodes: ["380"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[33.57318, 46.10317], [33.61467, 46.13561], [33.63854, 46.14147], [33.61517, 46.22615], [33.646, 46.23028], [33.74047, 46.18555], [33.79715, 46.20482], [33.85234, 46.19863], [33.91549, 46.15938], [34.05272, 46.10838], [34.07311, 46.11769], [34.12929, 46.10494], [34.181, 46.06804], [34.25111, 46.0532], [34.33912, 46.06114], [34.41221, 46.00245], [34.44155, 45.95995], [34.48729, 45.94267], [34.52011, 45.95097], [34.55889, 45.99347], [34.60861, 45.99347], [34.66679, 45.97136], [34.75479, 45.90705], [34.80153, 45.90047], [34.79905, 45.81009], [34.96015, 45.75634], [35.23066, 45.79231], [37.62608, 46.82615], [38.12112, 46.86078], [38.3384, 46.98085], [38.22955, 47.12069], [38.23049, 47.2324], [38.32112, 47.2585], [38.33074, 47.30508], [38.22225, 47.30788], [38.28954, 47.39255], [38.28679, 47.53552], [38.35062, 47.61631], [38.76379, 47.69346], [38.79628, 47.81109], [38.87979, 47.87719], [39.73935, 47.82876], [39.82213, 47.96396], [39.77544, 48.04206], [39.88256, 48.04482], [39.83724, 48.06501], [39.94847, 48.22811], [40.00752, 48.22445], [39.99241, 48.31768], [39.97325, 48.31399], [39.9693, 48.29904], [39.95248, 48.29972], [39.91465, 48.26743], [39.90041, 48.3049], [39.84273, 48.30947], [39.84136, 48.33321], [39.94847, 48.35055], [39.88794, 48.44226], [39.86196, 48.46633], [39.84548, 48.57821], [39.79764, 48.58668], [39.67226, 48.59368], [39.71765, 48.68673], [39.73104, 48.7325], [39.79466, 48.83739], [39.97182, 48.79398], [40.08168, 48.87443], [40.03636, 48.91957], [39.98967, 48.86901], [39.78368, 48.91596], [39.74874, 48.98675], [39.72649, 48.9754], [39.71353, 48.98959], [39.6683, 48.99454], [39.6836, 49.05121], [39.93437, 49.05709], [40.01988, 49.1761], [40.22176, 49.25683], [40.18331, 49.34996], [40.14912, 49.37681], [40.1141, 49.38798], [40.03087, 49.45452], [40.03636, 49.52321], [40.16683, 49.56865], [40.13249, 49.61672], [39.84548, 49.56064], [39.65047, 49.61761], [39.59142, 49.73758], [39.44496, 49.76067], [39.27968, 49.75976], [39.1808, 49.88911], [38.9391, 49.79524], [38.90477, 49.86787], [38.73311, 49.90238], [38.68677, 50.00904], [38.65688, 49.97176], [38.35408, 50.00664], [38.32524, 50.08866], [38.18517, 50.08161], [38.21675, 49.98104], [38.02999, 49.90592], [38.02999, 49.94482], [37.90776, 50.04194], [37.79515, 50.08425], [37.75807, 50.07896], [37.61113, 50.21976], [37.62879, 50.24481], [37.62486, 50.29966], [37.47243, 50.36277], [37.48204, 50.46079], [37.08468, 50.34935], [36.91762, 50.34963], [36.69377, 50.26982], [36.64571, 50.218], [36.56655, 50.2413], [36.58371, 50.28563], [36.47817, 50.31457], [36.30101, 50.29088], [36.20763, 50.3943], [36.06893, 50.45205], [35.8926, 50.43829], [35.80388, 50.41356], [35.73659, 50.35489], [35.61711, 50.35707], [35.58003, 50.45117], [35.47463, 50.49247], [35.39464, 50.64751], [35.48116, 50.66405], [35.47704, 50.77274], [35.41367, 50.80227], [35.39307, 50.92145], [35.32598, 50.94524], [35.40837, 51.04119], [35.31774, 51.08434], [35.20375, 51.04723], [35.12685, 51.16191], [35.14058, 51.23162], [34.97304, 51.2342], [34.82472, 51.17483], [34.6874, 51.18], [34.6613, 51.25053], [34.38802, 51.2746], [34.31661, 51.23936], [34.23009, 51.26429], [34.33446, 51.363], [34.22048, 51.4187], [34.30562, 51.5205], [34.17599, 51.63253], [34.07765, 51.67065], [34.42922, 51.72852], [34.41136, 51.82793], [34.09413, 52.00835], [34.11199, 52.14087], [34.05239, 52.20132], [33.78789, 52.37204], [33.55718, 52.30324], [33.48027, 52.31499], [33.51323, 52.35779], [33.18913, 52.3754], [32.89937, 52.2461], [32.85405, 52.27888], [32.69475, 52.25535], [32.54781, 52.32423], [32.3528, 52.32842], [32.38988, 52.24946], [32.33083, 52.23685], [32.34044, 52.1434], [32.2777, 52.10266], [32.23331, 52.08085], [32.08813, 52.03319], [31.92159, 52.05144], [31.96141, 52.08015], [31.85018, 52.11305], [31.81722, 52.09955], [31.7822, 52.11406], [31.38326, 52.12991], [31.25142, 52.04131], [31.13332, 52.1004], [30.95589, 52.07775], [30.90897, 52.00699], [30.76443, 51.89739], [30.68804, 51.82806], [30.51946, 51.59649], [30.64992, 51.35014], [30.56203, 51.25655], [30.36153, 51.33984], [30.34642, 51.42555], [30.17888, 51.51025], [29.77376, 51.4461], [29.7408, 51.53417], [29.54372, 51.48372], [29.49773, 51.39814], [29.42357, 51.4187], [29.32881, 51.37843], [29.25191, 51.49828], [29.25603, 51.57089], [29.20659, 51.56918], [29.16402, 51.64679], [29.1187, 51.65872], [28.99098, 51.56833], [28.95528, 51.59222], [28.81795, 51.55552], [28.76027, 51.48802], [28.78224, 51.45294], [28.75615, 51.41442], [28.73143, 51.46236], [28.69161, 51.44695], [28.64429, 51.5664], [28.47051, 51.59734], [28.37592, 51.54505], [28.23452, 51.66988], [28.10658, 51.57857], [27.95827, 51.56065], [27.91844, 51.61952], [27.85253, 51.62293], [27.76052, 51.47604], [27.67125, 51.50854], [27.71932, 51.60672], [27.55727, 51.63486], [27.51058, 51.5854], [27.47212, 51.61184], [27.24828, 51.60161], [27.26613, 51.65957], [27.20948, 51.66713], [27.20602, 51.77291], [26.99422, 51.76933], [26.9489, 51.73788], [26.80043, 51.75777], [26.69759, 51.82284], [26.46962, 51.80501], [26.39367, 51.87315], [26.19084, 51.86781], [26.00408, 51.92967], [25.83217, 51.92587], [25.80574, 51.94556], [25.73673, 51.91973], [25.46163, 51.92205], [25.20228, 51.97143], [24.98784, 51.91273], [24.37123, 51.88222], [24.29021, 51.80841], [24.3163, 51.75063], [24.13075, 51.66979], [23.99907, 51.58369], [23.8741, 51.59734], [23.91118, 51.63316], [23.7766, 51.66809], [23.60906, 51.62122], [23.6736, 51.50255], [23.62751, 51.50512], [23.69905, 51.40871], [23.63858, 51.32182], [23.80678, 51.18405], [23.90376, 51.07697], [23.92217, 51.00836], [24.04576, 50.90196], [24.14524, 50.86128], [24.0952, 50.83262], [23.99254, 50.83847], [23.95925, 50.79271], [24.0595, 50.71625], [24.0996, 50.60752], [24.07048, 50.5071], [24.03668, 50.44507], [23.99563, 50.41289], [23.79445, 50.40481], [23.71382, 50.38248], [23.67635, 50.33385], [23.28221, 50.0957], [22.99329, 49.84249], [22.83179, 49.69875], [22.80261, 49.69098], [22.78304, 49.65543], [22.64534, 49.53094], [22.69444, 49.49378], [22.748, 49.32759], [22.72009, 49.20288], [22.86336, 49.10513], [22.89122, 49.00725], [22.56155, 49.08865], [22.54338, 49.01424], [22.48296, 48.99172], [22.42934, 48.92857], [22.34151, 48.68893], [22.21379, 48.6218], [22.16023, 48.56548], [22.14689, 48.4005], [22.2083, 48.42534], [22.38133, 48.23726], [22.49806, 48.25189], [22.59007, 48.15121], [22.58733, 48.10813], [22.66835, 48.09162], [22.73427, 48.12005], [22.81804, 48.11363], [22.87847, 48.04665], [22.84276, 47.98602], [22.89849, 47.95851], [22.94301, 47.96672], [22.92241, 48.02002], [23.0158, 47.99338], [23.08858, 48.00716], [23.1133, 48.08061], [23.15999, 48.12188], [23.27397, 48.08245], [23.33577, 48.0237], [23.4979, 47.96858], [23.52803, 48.01818], [23.5653, 48.00499], [23.63894, 48.00293], [23.66262, 47.98786], [23.75188, 47.99705], [23.80904, 47.98142], [23.8602, 47.9329], [23.89352, 47.94512], [23.94192, 47.94868], [23.96337, 47.96672], [23.98553, 47.96076], [24.00801, 47.968], [24.02999, 47.95087], [24.06466, 47.95317], [24.11281, 47.91487], [24.22566, 47.90231], [24.34926, 47.9244], [24.43578, 47.97131], [24.61994, 47.95062], [24.70632, 47.84428], [24.81893, 47.82031], [24.88896, 47.7234], [25.11144, 47.75203], [25.23778, 47.89403], [25.63878, 47.94924], [25.77723, 47.93919], [26.05901, 47.9897], [26.17711, 47.99246], [26.33504, 48.18418], [26.55202, 48.22445], [26.62823, 48.25804], [26.6839, 48.35828], [26.79239, 48.29071], [26.82809, 48.31629], [26.71274, 48.40388], [26.85556, 48.41095], [26.93384, 48.36558], [27.03821, 48.37653], [27.0231, 48.42485], [27.08078, 48.43214], [27.13434, 48.37288], [27.27855, 48.37534], [27.32159, 48.4434], [27.37604, 48.44398], [27.37741, 48.41026], [27.44333, 48.41209], [27.46942, 48.454], [27.5889, 48.49224], [27.59027, 48.46311], [27.6658, 48.44034], [27.74422, 48.45926], [27.79225, 48.44244], [27.81902, 48.41874], [27.87533, 48.4037], [27.88391, 48.36699], [27.95883, 48.32368], [28.04527, 48.32661], [28.09873, 48.3124], [28.07504, 48.23494], [28.17666, 48.25963], [28.19314, 48.20749], [28.2856, 48.23202], [28.32508, 48.23384], [28.35519, 48.24957], [28.36996, 48.20543], [28.34912, 48.1787], [28.30586, 48.1597], [28.30609, 48.14018], [28.34009, 48.13147], [28.38712, 48.17567], [28.43701, 48.15832], [28.42454, 48.12047], [28.48428, 48.0737], [28.53921, 48.17453], [28.69896, 48.13106], [28.85232, 48.12506], [28.8414, 48.03392], [28.9306, 47.96255], [29.1723, 47.99013], [29.19839, 47.89261], [29.27804, 47.88893], [29.20663, 47.80367], [29.27255, 47.79953], [29.22242, 47.73607], [29.22414, 47.60012], [29.11743, 47.55001], [29.18603, 47.43387], [29.3261, 47.44664], [29.39889, 47.30179], [29.47854, 47.30366], [29.48678, 47.36043], [29.5733, 47.36508], [29.59665, 47.25521], [29.54996, 47.24962], [29.57696, 47.13581], [29.49732, 47.12878], [29.53044, 47.07851], [29.61038, 47.09932], [29.62137, 47.05069], [29.57056, 46.94766], [29.72986, 46.92234], [29.75458, 46.8604], [29.87405, 46.88199], [29.98814, 46.82358], [29.94522, 46.80055], [29.9743, 46.75325], [29.94409, 46.56002], [29.88916, 46.54302], [30.02511, 46.45132], [30.16794, 46.40967], [30.09103, 46.38694], [29.94114, 46.40114], [29.88329, 46.35851], [29.74496, 46.45605], [29.66359, 46.4215], [29.6763, 46.36041], [29.5939, 46.35472], [29.49914, 46.45889], [29.35357, 46.49505], [29.24886, 46.37912], [29.23547, 46.55435], [29.02409, 46.49582], [29.01241, 46.46177], [28.9306, 46.45699], [29.004, 46.31495], [28.98478, 46.31803], [28.94953, 46.25852], [29.06656, 46.19716], [28.94643, 46.09176], [29.00613, 46.04962], [28.98004, 46.00385], [28.74383, 45.96664], [28.78503, 45.83475], [28.69852, 45.81753], [28.70401, 45.78019], [28.52823, 45.73803], [28.47879, 45.66994], [28.51587, 45.6613], [28.54196, 45.58062], [28.49252, 45.56716], [28.51449, 45.49982], [28.43072, 45.48538], [28.41836, 45.51715], [28.30201, 45.54744], [28.21139, 45.46895], [28.28504, 45.43907], [28.34554, 45.32102], [28.5735, 45.24759], [28.71358, 45.22631], [28.78911, 45.24179], [28.81383, 45.3384], [28.94292, 45.28045], [28.96077, 45.33164], [29.24779, 45.43388], [29.42632, 45.44545], [29.59798, 45.38857], [29.68175, 45.26885], [29.65428, 45.25629], [29.69272, 45.19227], [30.04414, 45.08461], [31.62627, 45.50633], [33.54017, 46.0123], [33.59087, 46.06013], [33.57318, 46.10317]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "UG",
+           iso1A3: "UGA",
+           iso1N3: "800",
+           wikidata: "Q1036",
+           nameEn: "Uganda",
+           groups: ["014", "202", "002", "UN"],
+           driveSide: "left",
+           callingCodes: ["256"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[33.93107, -0.99298], [33.9264, -0.54188], [33.98449, -0.13079], [33.90936, 0.10581], [34.10067, 0.36372], [34.08727, 0.44713], [34.11408, 0.48884], [34.13493, 0.58118], [34.20196, 0.62289], [34.27345, 0.63182], [34.31516, 0.75693], [34.40041, 0.80266], [34.43349, 0.85254], [34.52369, 1.10692], [34.57427, 1.09868], [34.58029, 1.14712], [34.67562, 1.21265], [34.80223, 1.22754], [34.82606, 1.26626], [34.82606, 1.30944], [34.7918, 1.36752], [34.87819, 1.5596], [34.92734, 1.56109], [34.9899, 1.6668], [34.98692, 1.97348], [34.90947, 2.42447], [34.95267, 2.47209], [34.77244, 2.70272], [34.78137, 2.76223], [34.73967, 2.85447], [34.65774, 2.8753], [34.60114, 2.93034], [34.56242, 3.11478], [34.45815, 3.18319], [34.40006, 3.37949], [34.41794, 3.44342], [34.39112, 3.48802], [34.44922, 3.51627], [34.45815, 3.67385], [34.15429, 3.80464], [34.06046, 4.15235], [33.9873, 4.23316], [33.51264, 3.75068], [33.18356, 3.77812], [33.02852, 3.89296], [32.89746, 3.81339], [32.72021, 3.77327], [32.41337, 3.748], [32.20782, 3.6053], [32.19888, 3.50867], [32.08866, 3.53543], [32.08491, 3.56287], [32.05187, 3.589], [31.95907, 3.57408], [31.96205, 3.6499], [31.86821, 3.78664], [31.81459, 3.82083], [31.72075, 3.74354], [31.50776, 3.63652], [31.50478, 3.67814], [31.29476, 3.8015], [31.16666, 3.79853], [30.97601, 3.693], [30.85153, 3.48867], [30.94081, 3.50847], [30.93486, 3.40737], [30.84251, 3.26908], [30.77101, 3.04897], [30.8574, 2.9508], [30.8857, 2.83923], [30.75612, 2.5863], [30.74271, 2.43601], [30.83059, 2.42559], [30.91102, 2.33332], [30.96911, 2.41071], [31.06593, 2.35862], [31.07934, 2.30207], [31.12104, 2.27676], [31.1985, 2.29462], [31.20148, 2.2217], [31.28042, 2.17853], [31.30127, 2.11006], [30.48503, 1.21675], [30.24671, 1.14974], [30.22139, 0.99635], [30.1484, 0.89805], [29.98307, 0.84295], [29.95477, 0.64486], [29.97413, 0.52124], [29.87284, 0.39166], [29.81922, 0.16824], [29.77454, 0.16675], [29.7224, 0.07291], [29.72687, -0.08051], [29.65091, -0.46777], [29.67474, -0.47969], [29.67176, -0.55714], [29.62708, -0.71055], [29.63006, -0.8997], [29.58388, -0.89821], [29.59061, -1.39016], [29.82657, -1.31187], [29.912, -1.48269], [30.16369, -1.34303], [30.35212, -1.06896], [30.47194, -1.0555], [30.64166, -1.06601], [30.70631, -1.01175], [30.76635, -0.9852], [30.80408, -0.99911], [33.93107, -0.99298]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "UM",
+           iso1A3: "UMI",
+           iso1N3: "581",
+           wikidata: "Q16645",
+           nameEn: "United States Minor Outlying Islands",
+           country: "US"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "UN",
+           wikidata: "Q1065",
+           nameEn: "United Nations",
+           level: "unitedNations",
+           isoStatus: "excRes"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "US",
+           iso1A3: "USA",
+           iso1N3: "840",
+           wikidata: "Q30",
+           nameEn: "United States of America"
+         },
+         geometry: null
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "UY",
+           iso1A3: "URY",
+           iso1N3: "858",
+           wikidata: "Q77",
+           nameEn: "Uruguay",
+           groups: ["005", "419", "019", "UN"],
+           callingCodes: ["598"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-57.65132, -30.19229], [-57.61478, -30.25165], [-57.64859, -30.35095], [-57.89115, -30.49572], [-57.8024, -30.77193], [-57.89476, -30.95994], [-57.86729, -31.06352], [-57.9908, -31.34924], [-57.98127, -31.3872], [-58.07569, -31.44916], [-58.0023, -31.53084], [-58.00076, -31.65016], [-58.20252, -31.86966], [-58.10036, -32.25338], [-58.22362, -32.52416], [-58.1224, -32.98842], [-58.40475, -33.11777], [-58.44442, -33.84033], [-58.34425, -34.15035], [-57.83001, -34.69099], [-54.78916, -36.21945], [-52.83257, -34.01481], [-53.37138, -33.74313], [-53.39593, -33.75169], [-53.44031, -33.69344], [-53.52794, -33.68908], [-53.53459, -33.16843], [-53.1111, -32.71147], [-53.37671, -32.57005], [-53.39572, -32.58596], [-53.76024, -32.0751], [-54.17384, -31.86168], [-55.50821, -30.91349], [-55.50841, -30.9027], [-55.51862, -30.89828], [-55.52712, -30.89997], [-55.53276, -30.90218], [-55.53431, -30.89714], [-55.54572, -30.89051], [-55.55218, -30.88193], [-55.55373, -30.8732], [-55.5634, -30.8686], [-55.58866, -30.84117], [-55.87388, -31.05053], [-56.4619, -30.38457], [-56.4795, -30.3899], [-56.49267, -30.39471], [-56.90236, -30.02578], [-57.22502, -30.26121], [-57.65132, -30.19229]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "UZ",
+           iso1A3: "UZB",
+           iso1N3: "860",
+           wikidata: "Q265",
+           nameEn: "Uzbekistan",
+           groups: ["143", "142", "UN"],
+           callingCodes: ["998"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[65.85194, 42.85481], [65.53277, 43.31856], [65.18666, 43.48835], [64.96464, 43.74748], [64.53885, 43.56941], [63.34656, 43.64003], [62.01711, 43.51008], [61.01475, 44.41383], [58.59711, 45.58671], [55.97842, 44.99622], [55.97832, 44.99622], [55.97822, 44.99617], [55.97811, 44.99617], [55.97801, 44.99612], [55.97801, 44.99607], [55.97791, 44.99607], [55.9778, 44.99607], [55.9777, 44.99601], [55.9777, 44.99596], [55.9776, 44.99591], [55.97749, 44.99591], [55.97739, 44.99591], [55.97739, 44.99586], [55.97729, 44.99586], [55.97718, 44.99581], [55.97708, 44.99576], [55.97698, 44.9957], [55.97698, 44.99565], [55.97687, 44.9956], [55.97677, 44.9956], [55.97677, 44.99555], [55.97677, 44.9955], [55.97667, 44.99545], [55.97656, 44.99539], [55.97646, 44.99534], [55.97646, 44.99529], [55.97636, 44.99524], [55.97636, 44.99519], [55.97625, 44.99514], [55.97615, 44.99508], [55.97615, 44.99503], [55.97615, 44.99498], [55.97615, 44.99493], [55.97615, 44.99483], [55.97615, 44.99477], [55.97605, 44.99477], [55.97605, 44.99467], [55.97605, 44.99462], [55.97605, 44.99457], [55.97605, 44.99452], [55.97594, 44.99446], [55.97584, 44.99441], [55.97584, 44.99436], [55.97584, 44.99431], [55.97584, 44.99426], [55.97584, 44.99421], [55.97584, 44.99415], [55.97584, 44.99405], [55.97584, 44.994], [55.97584, 44.9939], [55.97584, 44.99384], [55.97584, 44.99374], [55.97584, 44.99369], [55.97584, 44.99359], [55.97584, 44.99353], [55.97584, 44.99348], [55.97584, 44.99343], [55.97584, 44.99338], [55.97584, 44.99328], [55.97584, 44.99322], [56.00314, 41.32584], [57.03423, 41.25435], [57.13796, 41.36625], [57.03359, 41.41777], [56.96218, 41.80383], [57.03633, 41.92043], [57.30275, 42.14076], [57.6296, 42.16519], [57.84932, 42.18555], [57.92897, 42.24047], [57.90975, 42.4374], [57.99214, 42.50021], [58.3492, 42.43335], [58.40688, 42.29535], [58.51674, 42.30348], [58.29427, 42.56497], [58.14321, 42.62159], [58.27504, 42.69632], [58.57991, 42.64988], [58.6266, 42.79314], [58.93422, 42.5407], [59.17317, 42.52248], [59.2955, 42.37064], [59.4341, 42.29738], [59.94633, 42.27655], [60.00539, 42.212], [59.96419, 42.1428], [60.04659, 42.08982], [60.0356, 42.01028], [59.95046, 41.97966], [60.33223, 41.75058], [60.08504, 41.80997], [60.06032, 41.76287], [60.18117, 41.60082], [60.06581, 41.4363], [60.5078, 41.21694], [61.03261, 41.25691], [61.22212, 41.14946], [61.33199, 41.14946], [61.39732, 41.19873], [61.4446, 41.29407], [61.87856, 41.12257], [62.11751, 40.58242], [62.34273, 40.43206], [62.43337, 39.98528], [63.6913, 39.27666], [63.70778, 39.22349], [64.19086, 38.95561], [64.32576, 38.98691], [65.55873, 38.29052], [65.83913, 38.25733], [66.24013, 38.16238], [66.41042, 38.02403], [66.56697, 38.0435], [66.67684, 37.96776], [66.53676, 37.80084], [66.52852, 37.58568], [66.65761, 37.45497], [66.52303, 37.39827], [66.55743, 37.35409], [66.64699, 37.32958], [66.95598, 37.40162], [67.08232, 37.35469], [67.13039, 37.27168], [67.2224, 37.24545], [67.2581, 37.17216], [67.51868, 37.26102], [67.78329, 37.1834], [67.8474, 37.31594], [67.81566, 37.43107], [68.12635, 37.93], [68.27159, 37.91477], [68.40343, 38.19484], [68.13289, 38.40822], [68.06274, 38.39435], [68.11366, 38.47169], [68.05873, 38.56087], [68.0807, 38.64136], [68.05598, 38.71641], [68.12877, 38.73677], [68.06948, 38.82115], [68.19743, 38.85985], [68.09704, 39.02589], [67.68915, 39.00775], [67.67833, 39.14479], [67.33226, 39.23739], [67.36522, 39.31287], [67.45998, 39.315], [67.46822, 39.46146], [67.39681, 39.52505], [67.46547, 39.53564], [67.44899, 39.57799], [67.62889, 39.60234], [67.70992, 39.66156], [68.12053, 39.56317], [68.54166, 39.53929], [68.61972, 39.68905], [68.63071, 39.85265], [68.88889, 39.87163], [68.93695, 39.91167], [68.84906, 40.04952], [68.96579, 40.06949], [69.01935, 40.11466], [69.01523, 40.15771], [68.62796, 40.07789], [68.52771, 40.11676], [68.5332, 40.14826], [68.77902, 40.20492], [68.79276, 40.17555], [68.84357, 40.18604], [68.85832, 40.20885], [69.04544, 40.22904], [69.15659, 40.2162], [69.2074, 40.21488], [69.30448, 40.18774], [69.30104, 40.24502], [69.25229, 40.26362], [69.24817, 40.30357], [69.30808, 40.2821], [69.32833, 40.29794], [69.33794, 40.34819], [69.30774, 40.36102], [69.28525, 40.41894], [69.27066, 40.49274], [69.21063, 40.54469], [69.2643, 40.57506], [69.3455, 40.57988], [69.32834, 40.70233], [69.38327, 40.7918], [69.53021, 40.77621], [69.59441, 40.70181], [69.69434, 40.62615], [70.36655, 40.90296], [70.38028, 41.02014], [70.45251, 41.04438], [70.80009, 40.72825], [70.49871, 40.52503], [70.32626, 40.45174], [70.37511, 40.38605], [70.57149, 40.3442], [70.56394, 40.26421], [70.62342, 40.17396], [70.8607, 40.217], [70.9818, 40.22392], [70.95789, 40.28761], [71.05901, 40.28765], [71.13042, 40.34106], [71.36663, 40.31593], [71.4246, 40.28619], [71.51215, 40.26943], [71.51549, 40.22986], [71.61725, 40.20615], [71.61931, 40.26775], [71.68386, 40.26984], [71.70569, 40.20391], [71.69621, 40.18492], [71.71719, 40.17886], [71.73054, 40.14818], [71.82646, 40.21872], [71.85002, 40.25647], [72.05464, 40.27586], [71.96401, 40.31907], [72.18648, 40.49893], [72.24368, 40.46091], [72.40346, 40.4007], [72.44191, 40.48222], [72.41513, 40.50856], [72.38384, 40.51535], [72.41714, 40.55736], [72.34406, 40.60144], [72.40517, 40.61917], [72.47795, 40.5532], [72.66713, 40.5219], [72.66713, 40.59076], [72.69579, 40.59778], [72.73995, 40.58409], [72.74768, 40.58051], [72.74862, 40.57131], [72.75982, 40.57273], [72.74894, 40.59592], [72.74866, 40.60873], [72.80137, 40.67856], [72.84754, 40.67229], [72.85372, 40.7116], [72.8722, 40.71111], [72.93296, 40.73089], [72.99133, 40.76457], [73.0612, 40.76678], [73.13412, 40.79122], [73.13267, 40.83512], [73.01869, 40.84681], [72.94454, 40.8094], [72.84291, 40.85512], [72.68157, 40.84942], [72.59136, 40.86947], [72.55109, 40.96046], [72.48742, 40.97136], [72.45206, 41.03018], [72.38511, 41.02785], [72.36138, 41.04384], [72.34757, 41.06104], [72.34026, 41.04539], [72.324, 41.03381], [72.18339, 40.99571], [72.17594, 41.02377], [72.21061, 41.05607], [72.1792, 41.10621], [72.14864, 41.13363], [72.17594, 41.15522], [72.16433, 41.16483], [72.10745, 41.15483], [72.07249, 41.11739], [71.85964, 41.19081], [71.91457, 41.2982], [71.83914, 41.3546], [71.76625, 41.4466], [71.71132, 41.43012], [71.73054, 41.54713], [71.65914, 41.49599], [71.6787, 41.42111], [71.57227, 41.29175], [71.46688, 41.31883], [71.43814, 41.19644], [71.46148, 41.13958], [71.40198, 41.09436], [71.34877, 41.16807], [71.27187, 41.11015], [71.25813, 41.18796], [71.11806, 41.15359], [71.02193, 41.19494], [70.9615, 41.16393], [70.86263, 41.23833], [70.77885, 41.24813], [70.78572, 41.36419], [70.67586, 41.47953], [70.48909, 41.40335], [70.17682, 41.5455], [70.69777, 41.92554], [71.28719, 42.18033], [71.13263, 42.28356], [70.94483, 42.26238], [69.49545, 41.545], [69.45751, 41.56863], [69.39485, 41.51518], [69.45081, 41.46246], [69.37468, 41.46555], [69.35554, 41.47211], [69.29778, 41.43673], [69.25059, 41.46693], [69.23332, 41.45847], [69.22671, 41.46298], [69.20439, 41.45391], [69.18528, 41.45175], [69.17701, 41.43769], [69.15137, 41.43078], [69.05006, 41.36183], [69.01308, 41.22804], [68.7217, 41.05025], [68.73945, 40.96989], [68.65662, 40.93861], [68.62221, 41.03019], [68.49983, 40.99669], [68.58444, 40.91447], [68.63, 40.59358], [68.49983, 40.56437], [67.96736, 40.83798], [68.1271, 41.0324], [68.08273, 41.08148], [67.98511, 41.02794], [67.9644, 41.14611], [66.69129, 41.1311], [66.53302, 41.87388], [66.00546, 41.94455], [66.09482, 42.93426], [65.85194, 42.85481]], [[70.68112, 40.90612], [70.6721, 40.90555], [70.57501, 40.98941], [70.54223, 40.98787], [70.56077, 41.00642], [70.6158, 40.97661], [70.68112, 40.90612]]], [[[71.21139, 40.03369], [71.12218, 40.03052], [71.06305, 40.1771], [71.00236, 40.18154], [71.01035, 40.05481], [71.11037, 40.01984], [71.11668, 39.99291], [71.09063, 39.99], [71.10501, 39.95568], [71.04979, 39.89808], [71.10531, 39.91354], [71.16101, 39.88423], [71.23067, 39.93581], [71.1427, 39.95026], [71.21139, 40.03369]]], [[[71.86463, 39.98598], [71.78838, 40.01404], [71.71511, 39.96348], [71.7504, 39.93701], [71.84316, 39.95582], [71.86463, 39.98598]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "VA",
+           iso1A3: "VAT",
+           iso1N3: "336",
+           wikidata: "Q237",
+           nameEn: "Vatican City",
+           aliases: ["Holy See"],
+           groups: ["039", "150"],
+           callingCodes: ["379", "39 06"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[12.45181, 41.90056], [12.45446, 41.90028], [12.45435, 41.90143], [12.45626, 41.90172], [12.45691, 41.90125], [12.4577, 41.90115], [12.45834, 41.90174], [12.45826, 41.90281], [12.45755, 41.9033], [12.45762, 41.9058], [12.45561, 41.90629], [12.45543, 41.90738], [12.45091, 41.90625], [12.44984, 41.90545], [12.44815, 41.90326], [12.44582, 41.90194], [12.44834, 41.90095], [12.45181, 41.90056]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "VC",
+           iso1A3: "VCT",
+           iso1N3: "670",
+           wikidata: "Q757",
+           nameEn: "St. Vincent and the Grenadines",
+           aliases: ["WV"],
+           groups: ["029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           callingCodes: ["1 784"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-62.64026, 12.69984], [-59.94058, 12.34011], [-61.69315, 14.26451], [-62.64026, 12.69984]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "VE",
+           iso1A3: "VEN",
+           iso1N3: "862",
+           wikidata: "Q717",
+           nameEn: "Venezuela",
+           aliases: ["YV"],
+           groups: ["005", "419", "019", "UN"],
+           callingCodes: ["58"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-71.22331, 13.01387], [-70.92579, 11.96275], [-71.3275, 11.85], [-71.9675, 11.65536], [-72.24983, 11.14138], [-72.4767, 11.1117], [-72.88002, 10.44309], [-72.98085, 9.85253], [-73.36905, 9.16636], [-73.02119, 9.27584], [-72.94052, 9.10663], [-72.77415, 9.10165], [-72.65474, 8.61428], [-72.4042, 8.36513], [-72.36987, 8.19976], [-72.35163, 8.01163], [-72.39137, 8.03534], [-72.47213, 7.96106], [-72.48801, 7.94329], [-72.48183, 7.92909], [-72.47042, 7.92306], [-72.45806, 7.91141], [-72.46183, 7.90682], [-72.44454, 7.86031], [-72.46763, 7.79518], [-72.47827, 7.65604], [-72.45321, 7.57232], [-72.47415, 7.48928], [-72.43132, 7.40034], [-72.19437, 7.37034], [-72.04895, 7.03837], [-71.82441, 7.04314], [-71.44118, 7.02116], [-71.42212, 7.03854], [-71.37234, 7.01588], [-71.03941, 6.98163], [-70.7596, 7.09799], [-70.10716, 6.96516], [-69.41843, 6.1072], [-67.60654, 6.2891], [-67.4625, 6.20625], [-67.43513, 5.98835], [-67.58558, 5.84537], [-67.63914, 5.64963], [-67.59141, 5.5369], [-67.83341, 5.31104], [-67.85358, 4.53249], [-67.62671, 3.74303], [-67.50067, 3.75812], [-67.30945, 3.38393], [-67.85862, 2.86727], [-67.85862, 2.79173], [-67.65696, 2.81691], [-67.21967, 2.35778], [-66.85795, 1.22998], [-66.28507, 0.74585], [-65.6727, 1.01353], [-65.50158, 0.92086], [-65.57288, 0.62856], [-65.11657, 1.12046], [-64.38932, 1.5125], [-64.34654, 1.35569], [-64.08274, 1.64792], [-64.06135, 1.94722], [-63.39827, 2.16098], [-63.39114, 2.4317], [-64.0257, 2.48156], [-64.02908, 2.79797], [-64.48379, 3.7879], [-64.84028, 4.24665], [-64.72977, 4.28931], [-64.57648, 4.12576], [-64.14512, 4.12932], [-63.99183, 3.90172], [-63.86082, 3.94796], [-63.70218, 3.91417], [-63.67099, 4.01731], [-63.50611, 3.83592], [-63.42233, 3.89995], [-63.4464, 3.9693], [-63.21111, 3.96219], [-62.98296, 3.59935], [-62.7655, 3.73099], [-62.74411, 4.03331], [-62.57656, 4.04754], [-62.44822, 4.18621], [-62.13094, 4.08309], [-61.54629, 4.2822], [-61.48569, 4.43149], [-61.29675, 4.44216], [-61.31457, 4.54167], [-61.15703, 4.49839], [-60.98303, 4.54167], [-60.86539, 4.70512], [-60.5802, 4.94312], [-60.73204, 5.20931], [-61.4041, 5.95304], [-61.15058, 6.19558], [-61.20762, 6.58174], [-61.13632, 6.70922], [-60.54873, 6.8631], [-60.39419, 6.94847], [-60.28074, 7.1162], [-60.44116, 7.20817], [-60.54098, 7.14804], [-60.63367, 7.25061], [-60.59802, 7.33194], [-60.71923, 7.55817], [-60.64793, 7.56877], [-60.51959, 7.83373], [-60.38056, 7.8302], [-60.02407, 8.04557], [-59.97059, 8.20791], [-59.83156, 8.23261], [-59.80661, 8.28906], [-59.85562, 8.35213], [-59.98508, 8.53046], [-59.54058, 8.6862], [-60.89962, 9.81445], [-62.08693, 10.04435], [-61.62505, 11.18974], [-63.73917, 11.92623], [-63.19938, 16.44103], [-67.89186, 12.4116], [-68.01417, 11.77722], [-68.33524, 11.78151], [-68.99639, 11.79035], [-71.22331, 13.01387]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "VG",
+           iso1A3: "VGB",
+           iso1N3: "092",
+           wikidata: "Q25305",
+           nameEn: "British Virgin Islands",
+           country: "GB",
+           groups: ["BOTS", "029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["1 284"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-64.47127, 17.55688], [-63.88746, 19.15706], [-65.02435, 18.73231], [-64.86027, 18.39056], [-64.64673, 18.36549], [-64.47127, 17.55688]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "VI",
+           iso1A3: "VIR",
+           iso1N3: "850",
+           wikidata: "Q11703",
+           nameEn: "United States Virgin Islands",
+           aliases: ["US-VI"],
+           country: "US",
+           groups: ["Q1352230", "029", "003", "419", "019", "UN"],
+           driveSide: "left",
+           roadSpeedUnit: "mph",
+           roadHeightUnit: "ft",
+           callingCodes: ["1 340"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-65.02435, 18.73231], [-65.27974, 17.56928], [-64.47127, 17.55688], [-64.64673, 18.36549], [-64.86027, 18.39056], [-65.02435, 18.73231]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "VN",
+           iso1A3: "VNM",
+           iso1N3: "704",
+           wikidata: "Q881",
+           nameEn: "Vietnam",
+           groups: ["035", "142", "UN"],
+           callingCodes: ["84"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[108.10003, 21.47338], [108.0569, 21.53604], [108.02926, 21.54997], [107.97932, 21.54503], [107.97383, 21.53961], [107.97074, 21.54072], [107.96774, 21.53601], [107.95232, 21.5388], [107.92652, 21.58906], [107.90006, 21.5905], [107.86114, 21.65128], [107.80355, 21.66141], [107.66967, 21.60787], [107.56537, 21.61945], [107.54047, 21.5934], [107.49065, 21.59774], [107.49532, 21.62958], [107.47197, 21.6672], [107.41593, 21.64839], [107.38636, 21.59774], [107.35989, 21.60063], [107.35834, 21.6672], [107.29296, 21.74674], [107.24625, 21.7077], [107.20734, 21.71493], [107.10771, 21.79879], [107.02615, 21.81981], [107.00964, 21.85948], [107.06101, 21.88982], [107.05634, 21.92303], [106.99252, 21.95191], [106.97228, 21.92592], [106.92714, 21.93459], [106.9178, 21.97357], [106.81038, 21.97934], [106.74345, 22.00965], [106.72551, 21.97923], [106.69276, 21.96013], [106.68274, 21.99811], [106.70142, 22.02409], [106.6983, 22.15102], [106.67495, 22.1885], [106.69986, 22.22309], [106.6516, 22.33977], [106.55976, 22.34841], [106.57221, 22.37], [106.55665, 22.46498], [106.58395, 22.474], [106.61269, 22.60301], [106.65316, 22.5757], [106.71698, 22.58432], [106.72321, 22.63606], [106.76293, 22.73491], [106.82404, 22.7881], [106.83685, 22.8098], [106.81271, 22.8226], [106.78422, 22.81532], [106.71128, 22.85982], [106.71387, 22.88296], [106.6734, 22.89587], [106.6516, 22.86862], [106.60179, 22.92884], [106.55976, 22.92311], [106.51306, 22.94891], [106.49749, 22.91164], [106.34961, 22.86718], [106.27022, 22.87722], [106.19705, 22.98475], [106.00179, 22.99049], [105.99568, 22.94178], [105.90119, 22.94168], [105.8726, 22.92756], [105.72382, 23.06641], [105.57594, 23.075], [105.56037, 23.16806], [105.49966, 23.20669], [105.42805, 23.30824], [105.40782, 23.28107], [105.32376, 23.39684], [105.22569, 23.27249], [105.17276, 23.28679], [105.11672, 23.25247], [105.07002, 23.26248], [104.98712, 23.19176], [104.96532, 23.20463], [104.9486, 23.17235], [104.91435, 23.18666], [104.87992, 23.17141], [104.87382, 23.12854], [104.79478, 23.12934], [104.8334, 23.01484], [104.86765, 22.95178], [104.84942, 22.93631], [104.77114, 22.90017], [104.72755, 22.81984], [104.65283, 22.83419], [104.60457, 22.81841], [104.58122, 22.85571], [104.47225, 22.75813], [104.35593, 22.69353], [104.25683, 22.76534], [104.27084, 22.8457], [104.11384, 22.80363], [104.03734, 22.72945], [104.01088, 22.51823], [103.99247, 22.51958], [103.97384, 22.50634], [103.96783, 22.51173], [103.96352, 22.50584], [103.95191, 22.5134], [103.94513, 22.52553], [103.93286, 22.52703], [103.87904, 22.56683], [103.64506, 22.79979], [103.56255, 22.69499], [103.57812, 22.65764], [103.52675, 22.59155], [103.43646, 22.70648], [103.43179, 22.75816], [103.32282, 22.8127], [103.28079, 22.68063], [103.18895, 22.64471], [103.15782, 22.59873], [103.17961, 22.55705], [103.07843, 22.50097], [103.0722, 22.44775], [102.9321, 22.48659], [102.8636, 22.60735], [102.60675, 22.73376], [102.57095, 22.7036], [102.51802, 22.77969], [102.46665, 22.77108], [102.42618, 22.69212], [102.38415, 22.67919], [102.41061, 22.64184], [102.25339, 22.4607], [102.26428, 22.41321], [102.16621, 22.43336], [102.14099, 22.40092], [102.18712, 22.30403], [102.51734, 22.02676], [102.49092, 21.99002], [102.62301, 21.91447], [102.67145, 21.65894], [102.74189, 21.66713], [102.82115, 21.73667], [102.81894, 21.83888], [102.85637, 21.84501], [102.86077, 21.71213], [102.97965, 21.74076], [102.98846, 21.58936], [102.86297, 21.4255], [102.94223, 21.46034], [102.88939, 21.3107], [102.80794, 21.25736], [102.89825, 21.24707], [102.97745, 21.05821], [103.03469, 21.05821], [103.12055, 20.89994], [103.21497, 20.89832], [103.38032, 20.79501], [103.45737, 20.82382], [103.68633, 20.66324], [103.73478, 20.6669], [103.82282, 20.8732], [103.98024, 20.91531], [104.11121, 20.96779], [104.27412, 20.91433], [104.63957, 20.6653], [104.38199, 20.47155], [104.40621, 20.3849], [104.47886, 20.37459], [104.66158, 20.47774], [104.72102, 20.40554], [104.62195, 20.36633], [104.61315, 20.24452], [104.86852, 20.14121], [104.91695, 20.15567], [104.9874, 20.09573], [104.8465, 19.91783], [104.8355, 19.80395], [104.68359, 19.72729], [104.64837, 19.62365], [104.53169, 19.61743], [104.41281, 19.70035], [104.23229, 19.70242], [104.06498, 19.66926], [104.05617, 19.61743], [104.10832, 19.51575], [104.06058, 19.43484], [103.87125, 19.31854], [104.5361, 18.97747], [104.64617, 18.85668], [105.12829, 18.70453], [105.19654, 18.64196], [105.1327, 18.58355], [105.10408, 18.43533], [105.15942, 18.38691], [105.38366, 18.15315], [105.46292, 18.22008], [105.64784, 17.96687], [105.60381, 17.89356], [105.76612, 17.67147], [105.85744, 17.63221], [106.09019, 17.36399], [106.18991, 17.28227], [106.24444, 17.24714], [106.29287, 17.3018], [106.31929, 17.20509], [106.43597, 17.01362], [106.50862, 16.9673], [106.55045, 17.0031], [106.54824, 16.92729], [106.51963, 16.92097], [106.52183, 16.87884], [106.55265, 16.86831], [106.55485, 16.68704], [106.59013, 16.62259], [106.58267, 16.6012], [106.61477, 16.60713], [106.66052, 16.56892], [106.65832, 16.47816], [106.74418, 16.41904], [106.84104, 16.55415], [106.88727, 16.52671], [106.88067, 16.43594], [106.96638, 16.34938], [106.97385, 16.30204], [107.02597, 16.31132], [107.09091, 16.3092], [107.15035, 16.26271], [107.14595, 16.17816], [107.25822, 16.13587], [107.33968, 16.05549], [107.44975, 16.08511], [107.46296, 16.01106], [107.39471, 15.88829], [107.34188, 15.89464], [107.21419, 15.83747], [107.21859, 15.74638], [107.27143, 15.71459], [107.27583, 15.62769], [107.34408, 15.62345], [107.3815, 15.49832], [107.50699, 15.48771], [107.53341, 15.40496], [107.62367, 15.42193], [107.60605, 15.37524], [107.62587, 15.2266], [107.58844, 15.20111], [107.61926, 15.13949], [107.61486, 15.0566], [107.46516, 15.00982], [107.48277, 14.93751], [107.59285, 14.87795], [107.51579, 14.79282], [107.54361, 14.69092], [107.55371, 14.628], [107.52102, 14.59034], [107.52569, 14.54665], [107.48521, 14.40346], [107.44941, 14.41552], [107.39493, 14.32655], [107.40427, 14.24509], [107.33577, 14.11832], [107.37158, 14.07906], [107.35757, 14.02319], [107.38247, 13.99147], [107.44318, 13.99751], [107.46498, 13.91593], [107.45252, 13.78897], [107.53503, 13.73908], [107.61909, 13.52577], [107.62843, 13.3668], [107.49144, 13.01215], [107.49611, 12.88926], [107.55993, 12.7982], [107.5755, 12.52177], [107.55059, 12.36824], [107.4463, 12.29373], [107.42917, 12.24657], [107.34511, 12.33327], [107.15831, 12.27547], [106.99953, 12.08983], [106.92325, 12.06548], [106.79405, 12.0807], [106.70687, 11.96956], [106.4111, 11.97413], [106.4687, 11.86751], [106.44068, 11.86294], [106.44535, 11.8279], [106.41577, 11.76999], [106.45158, 11.68616], [106.44691, 11.66787], [106.37219, 11.69836], [106.30525, 11.67549], [106.26478, 11.72122], [106.18539, 11.75171], [106.13158, 11.73283], [106.06708, 11.77761], [106.02038, 11.77457], [106.00792, 11.7197], [105.95188, 11.63738], [105.88962, 11.67854], [105.8507, 11.66635], [105.80867, 11.60536], [105.81645, 11.56876], [105.87328, 11.55953], [105.88962, 11.43605], [105.86782, 11.28343], [106.10444, 11.07879], [106.1527, 11.10476], [106.1757, 11.07301], [106.20095, 10.97795], [106.14301, 10.98176], [106.18539, 10.79451], [106.06708, 10.8098], [105.94535, 10.9168], [105.93403, 10.83853], [105.84603, 10.85873], [105.86376, 10.89839], [105.77751, 11.03671], [105.50045, 10.94586], [105.42884, 10.96878], [105.34011, 10.86179], [105.11449, 10.96332], [105.08326, 10.95656], [105.02722, 10.89236], [105.09571, 10.72722], [104.95094, 10.64003], [104.87933, 10.52833], [104.59018, 10.53073], [104.49869, 10.4057], [104.47963, 10.43046], [104.43778, 10.42386], [103.99198, 10.48391], [102.47649, 9.66162], [104.81582, 8.03101], [109.55486, 8.10026], [111.60491, 13.57105], [108.00365, 17.98159], [108.10003, 21.47338]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "VU",
+           iso1A3: "VUT",
+           iso1N3: "548",
+           wikidata: "Q686",
+           nameEn: "Vanuatu",
+           groups: ["054", "009", "UN"],
+           callingCodes: ["678"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[156.73836, -14.50464], [174.245, -23.1974], [172.71443, -12.01327], [156.73836, -14.50464]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "WF",
+           iso1A3: "WLF",
+           iso1N3: "876",
+           wikidata: "Q35555",
+           nameEn: "Wallis and Futuna",
+           country: "FR",
+           groups: ["Q1451600", "061", "009", "UN"],
+           callingCodes: ["681"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-178.66551, -14.32452], [-176.76826, -14.95183], [-175.59809, -12.61507], [-178.66551, -14.32452]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "WS",
+           iso1A3: "WSM",
+           iso1N3: "882",
+           wikidata: "Q683",
+           nameEn: "Samoa",
+           groups: ["061", "009", "UN"],
+           driveSide: "left",
+           callingCodes: ["685"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[-173.74402, -14.26669], [-170.99605, -15.1275], [-171.39864, -10.21587], [-173.74402, -14.26669]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "XK",
+           iso1A3: "XKX",
+           wikidata: "Q1246",
+           nameEn: "Kosovo",
+           aliases: ["KV"],
+           groups: ["039", "150"],
+           isoStatus: "usrAssn",
+           callingCodes: ["383"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[21.39045, 42.74888], [21.44047, 42.87276], [21.36941, 42.87397], [21.32974, 42.90424], [21.2719, 42.8994], [21.23534, 42.95523], [21.23877, 43.00848], [21.2041, 43.02277], [21.16734, 42.99694], [21.14465, 43.11089], [21.08952, 43.13471], [21.05378, 43.10707], [21.00749, 43.13984], [20.96287, 43.12416], [20.83727, 43.17842], [20.88685, 43.21697], [20.82145, 43.26769], [20.73811, 43.25068], [20.68688, 43.21335], [20.59929, 43.20492], [20.69515, 43.09641], [20.64557, 43.00826], [20.59929, 43.01067], [20.48692, 42.93208], [20.53484, 42.8885], [20.43734, 42.83157], [20.40594, 42.84853], [20.35692, 42.8335], [20.27869, 42.81945], [20.2539, 42.76245], [20.04898, 42.77701], [20.02088, 42.74789], [20.02915, 42.71147], [20.0969, 42.65559], [20.07761, 42.55582], [20.17127, 42.50469], [20.21797, 42.41237], [20.24399, 42.32168], [20.34479, 42.32656], [20.3819, 42.3029], [20.48857, 42.25444], [20.56955, 42.12097], [20.55633, 42.08173], [20.59434, 42.03879], [20.63069, 41.94913], [20.57946, 41.91593], [20.59524, 41.8818], [20.68523, 41.85318], [20.76786, 41.91839], [20.75464, 42.05229], [21.11491, 42.20794], [21.16614, 42.19815], [21.22728, 42.08909], [21.31983, 42.10993], [21.29913, 42.13954], [21.30496, 42.1418], [21.38428, 42.24465], [21.43882, 42.23609], [21.43882, 42.2789], [21.50823, 42.27156], [21.52145, 42.24465], [21.58992, 42.25915], [21.56772, 42.30946], [21.5264, 42.33634], [21.53467, 42.36809], [21.57021, 42.3647], [21.59029, 42.38042], [21.62887, 42.37664], [21.64209, 42.41081], [21.62556, 42.45106], [21.7035, 42.51899], [21.70522, 42.54176], [21.7327, 42.55041], [21.75672, 42.62695], [21.79413, 42.65923], [21.75025, 42.70125], [21.6626, 42.67813], [21.58755, 42.70418], [21.59154, 42.72643], [21.47498, 42.74695], [21.39045, 42.74888]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "YE",
+           iso1A3: "YEM",
+           iso1N3: "887",
+           wikidata: "Q805",
+           nameEn: "Yemen",
+           groups: ["145", "142", "UN"],
+           callingCodes: ["967"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[57.49095, 8.14549], [52.81185, 17.28568], [52.74267, 17.29519], [52.78009, 17.35124], [52.00311, 19.00083], [49.04884, 18.59899], [48.19996, 18.20584], [47.58351, 17.50366], [47.48245, 17.10808], [47.00571, 16.94765], [46.76494, 17.29151], [46.31018, 17.20464], [44.50126, 17.47475], [43.70631, 17.35762], [43.43005, 17.56148], [43.29185, 17.53224], [43.22533, 17.38343], [43.32653, 17.31179], [43.20156, 17.25901], [43.17787, 17.14717], [43.23967, 17.03428], [43.18233, 17.02673], [43.1813, 16.98438], [43.19328, 16.94703], [43.1398, 16.90696], [43.18338, 16.84852], [43.22012, 16.83932], [43.22956, 16.80613], [43.24801, 16.80613], [43.26303, 16.79479], [43.25857, 16.75304], [43.21325, 16.74416], [43.22066, 16.65179], [43.15274, 16.67248], [43.11601, 16.53166], [42.97215, 16.51093], [42.94351, 16.49467], [42.94625, 16.39721], [42.76801, 16.40371], [42.15205, 16.40211], [40.99158, 15.81743], [43.29075, 12.79154], [43.32909, 12.59711], [43.90659, 12.3823], [51.12877, 12.56479], [57.49095, 8.14549]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "YT",
+           iso1A3: "MYT",
+           iso1N3: "175",
+           wikidata: "Q17063",
+           nameEn: "Mayotte",
+           country: "FR",
+           groups: ["Q3320166", "EU", "014", "202", "002", "UN"],
+           callingCodes: ["262"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[43.28731, -13.97126], [45.54824, -13.22353], [45.4971, -11.75965], [43.28731, -13.97126]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "ZA",
+           iso1A3: "ZAF",
+           iso1N3: "710",
+           wikidata: "Q258",
+           nameEn: "South Africa",
+           groups: ["018", "202", "002", "UN"],
+           driveSide: "left",
+           callingCodes: ["27"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[31.30611, -22.422], [31.16344, -22.32599], [31.08932, -22.34884], [30.86696, -22.28907], [30.6294, -22.32599], [30.48686, -22.31368], [30.38614, -22.34533], [30.28351, -22.35587], [30.2265, -22.2961], [30.13147, -22.30841], [29.92242, -22.19408], [29.76848, -22.14128], [29.64609, -22.12917], [29.37703, -22.19581], [29.21955, -22.17771], [29.18974, -22.18599], [29.15268, -22.21399], [29.10881, -22.21202], [29.0151, -22.22907], [28.91889, -22.44299], [28.63287, -22.55887], [28.34874, -22.5694], [28.04562, -22.8394], [28.04752, -22.90243], [27.93729, -22.96194], [27.93539, -23.04941], [27.74154, -23.2137], [27.6066, -23.21894], [27.52393, -23.37952], [27.33768, -23.40917], [26.99749, -23.65486], [26.84165, -24.24885], [26.51667, -24.47219], [26.46346, -24.60358], [26.39409, -24.63468], [25.8515, -24.75727], [25.84295, -24.78661], [25.88571, -24.87802], [25.72702, -25.25503], [25.69661, -25.29284], [25.6643, -25.4491], [25.58543, -25.6343], [25.33076, -25.76616], [25.12266, -25.75931], [25.01718, -25.72507], [24.8946, -25.80723], [24.67319, -25.81749], [24.44703, -25.73021], [24.36531, -25.773], [24.18287, -25.62916], [23.9244, -25.64286], [23.47588, -25.29971], [23.03497, -25.29971], [22.86012, -25.50572], [22.70808, -25.99186], [22.56365, -26.19668], [22.41921, -26.23078], [22.21206, -26.3773], [22.06192, -26.61882], [21.90703, -26.66808], [21.83291, -26.65959], [21.77114, -26.69015], [21.7854, -26.79199], [21.69322, -26.86152], [21.37869, -26.82083], [21.13353, -26.86661], [20.87031, -26.80047], [20.68596, -26.9039], [20.63275, -26.78181], [20.61754, -26.4692], [20.86081, -26.14892], [20.64795, -25.47827], [20.29826, -24.94869], [20.03678, -24.81004], [20.02809, -24.78725], [19.99817, -24.76768], [19.99882, -28.42622], [18.99885, -28.89165], [17.4579, -28.68718], [17.15405, -28.08573], [16.90446, -28.057], [16.59922, -28.53246], [16.46592, -28.57126], [16.45332, -28.63117], [12.51595, -32.27486], [38.88176, -48.03306], [34.51034, -26.91792], [32.35222, -26.86027], [32.29584, -26.852], [32.22302, -26.84136], [32.19409, -26.84032], [32.13315, -26.84345], [32.09664, -26.80721], [32.00893, -26.8096], [31.97463, -27.11057], [31.97592, -27.31675], [31.49834, -27.31549], [31.15027, -27.20151], [30.96088, -27.0245], [30.97757, -26.92706], [30.88826, -26.79622], [30.81101, -26.84722], [30.78927, -26.48271], [30.95819, -26.26303], [31.13073, -25.91558], [31.31237, -25.7431], [31.4175, -25.71886], [31.86881, -25.99973], [31.974, -25.95387], [31.92649, -25.84216], [32.00631, -25.65044], [31.97875, -25.46356], [32.01676, -25.38117], [32.03196, -25.10785], [31.9835, -24.29983], [31.90368, -24.18892], [31.87707, -23.95293], [31.77445, -23.90082], [31.70223, -23.72695], [31.67942, -23.60858], [31.56539, -23.47268], [31.55779, -23.176], [31.30611, -22.422]], [[29.33204, -29.45598], [29.28545, -29.58456], [29.12553, -29.76266], [29.16548, -29.91706], [28.9338, -30.05072], [28.80222, -30.10579], [28.68627, -30.12885], [28.399, -30.1592], [28.2319, -30.28476], [28.12073, -30.68072], [27.74814, -30.60635], [27.69467, -30.55862], [27.67819, -30.53437], [27.6521, -30.51707], [27.62137, -30.50509], [27.56781, -30.44562], [27.56901, -30.42504], [27.45452, -30.32239], [27.38108, -30.33456], [27.36649, -30.27246], [27.37293, -30.19401], [27.40778, -30.14577], [27.32555, -30.14785], [27.29603, -30.05473], [27.22719, -30.00718], [27.09489, -29.72796], [27.01016, -29.65439], [27.33464, -29.48161], [27.4358, -29.33465], [27.47254, -29.31968], [27.45125, -29.29708], [27.48679, -29.29349], [27.54258, -29.25575], [27.5158, -29.2261], [27.55974, -29.18954], [27.75458, -28.89839], [27.8907, -28.91612], [27.88933, -28.88156], [27.9392, -28.84864], [27.98675, -28.8787], [28.02503, -28.85991], [28.1317, -28.7293], [28.2348, -28.69471], [28.30518, -28.69531], [28.40612, -28.6215], [28.65091, -28.57025], [28.68043, -28.58744], [29.40524, -29.21246], [29.44883, -29.3772], [29.33204, -29.45598]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "ZM",
+           iso1A3: "ZMB",
+           iso1N3: "894",
+           wikidata: "Q953",
+           nameEn: "Zambia",
+           groups: ["014", "202", "002", "UN"],
+           driveSide: "left",
+           callingCodes: ["260"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[32.95389, -9.40138], [32.76233, -9.31963], [32.75611, -9.28583], [32.53661, -9.24281], [32.49147, -9.14754], [32.43543, -9.11988], [32.25486, -9.13371], [32.16146, -9.05993], [32.08206, -9.04609], [31.98866, -9.07069], [31.94196, -9.02303], [31.94663, -8.93846], [31.81587, -8.88618], [31.71158, -8.91386], [31.57147, -8.81388], [31.57147, -8.70619], [31.37533, -8.60769], [31.00796, -8.58615], [30.79243, -8.27382], [28.88917, -8.4831], [28.9711, -8.66935], [28.38526, -9.23393], [28.36562, -9.30091], [28.52636, -9.35379], [28.51627, -9.44726], [28.56208, -9.49122], [28.68532, -9.78], [28.62795, -9.92942], [28.65032, -10.65133], [28.37241, -11.57848], [28.48357, -11.87532], [29.18592, -12.37921], [29.4992, -12.43843], [29.48404, -12.23604], [29.8139, -12.14898], [29.81551, -13.44683], [29.65078, -13.41844], [29.60531, -13.21685], [29.01918, -13.41353], [28.33199, -12.41375], [27.59932, -12.22123], [27.21025, -11.76157], [27.22541, -11.60323], [27.04351, -11.61312], [26.88687, -12.01868], [26.01777, -11.91488], [25.33058, -11.65767], [25.34069, -11.19707], [24.42612, -11.44975], [24.34528, -11.06816], [24.00027, -10.89356], [24.02603, -11.15368], [23.98804, -12.13149], [24.06672, -12.29058], [23.90937, -12.844], [24.03339, -12.99091], [21.97988, -13.00148], [22.00323, -16.18028], [22.17217, -16.50269], [23.20038, -17.47563], [23.47474, -17.62877], [24.23619, -17.47489], [24.32811, -17.49082], [24.38712, -17.46818], [24.5621, -17.52963], [24.70864, -17.49501], [25.00198, -17.58221], [25.26433, -17.79571], [25.51646, -17.86232], [25.6827, -17.81987], [25.85738, -17.91403], [25.85892, -17.97726], [26.08925, -17.98168], [26.0908, -17.93021], [26.21601, -17.88608], [26.55918, -17.99638], [26.68403, -18.07411], [26.74314, -18.0199], [26.89926, -17.98756], [27.14196, -17.81398], [27.30736, -17.60487], [27.61377, -17.34378], [27.62795, -17.24365], [27.83141, -16.96274], [28.73725, -16.5528], [28.76199, -16.51575], [28.81454, -16.48611], [28.8501, -16.04537], [28.9243, -15.93987], [29.01298, -15.93805], [29.21955, -15.76589], [29.4437, -15.68702], [29.8317, -15.6126], [30.35574, -15.6513], [30.41902, -15.62269], [30.22098, -14.99447], [33.24249, -14.00019], [33.16749, -13.93992], [33.07568, -13.98447], [33.02977, -14.05022], [32.99042, -13.95689], [32.88985, -13.82956], [32.79015, -13.80755], [32.76962, -13.77224], [32.84528, -13.71576], [32.7828, -13.64805], [32.68654, -13.64268], [32.66468, -13.60019], [32.68436, -13.55769], [32.73683, -13.57682], [32.84176, -13.52794], [32.86113, -13.47292], [33.0078, -13.19492], [32.98289, -13.12671], [33.02181, -12.88707], [32.96733, -12.88251], [32.94397, -12.76868], [33.05917, -12.59554], [33.18837, -12.61377], [33.28177, -12.54692], [33.37517, -12.54085], [33.54485, -12.35996], [33.47636, -12.32498], [33.3705, -12.34931], [33.25998, -12.14242], [33.33937, -11.91252], [33.32692, -11.59248], [33.24252, -11.59302], [33.23663, -11.40637], [33.29267, -11.43536], [33.29267, -11.3789], [33.39697, -11.15296], [33.25998, -10.88862], [33.28022, -10.84428], [33.47636, -10.78465], [33.70675, -10.56896], [33.54797, -10.36077], [33.53863, -10.20148], [33.31297, -10.05133], [33.37902, -9.9104], [33.36581, -9.81063], [33.31517, -9.82364], [33.2095, -9.61099], [33.12144, -9.58929], [33.10163, -9.66525], [33.05485, -9.61316], [33.00256, -9.63053], [33.00476, -9.5133], [32.95389, -9.40138]]]]
+         }
+       }, {
+         type: "Feature",
+         properties: {
+           iso1A2: "ZW",
+           iso1A3: "ZWE",
+           iso1N3: "716",
+           wikidata: "Q954",
+           nameEn: "Zimbabwe",
+           groups: ["014", "202", "002", "UN"],
+           driveSide: "left",
+           callingCodes: ["263"]
+         },
+         geometry: {
+           type: "MultiPolygon",
+           coordinates: [[[[30.41902, -15.62269], [30.35574, -15.6513], [29.8317, -15.6126], [29.4437, -15.68702], [29.21955, -15.76589], [29.01298, -15.93805], [28.9243, -15.93987], [28.8501, -16.04537], [28.81454, -16.48611], [28.76199, -16.51575], [28.73725, -16.5528], [27.83141, -16.96274], [27.62795, -17.24365], [27.61377, -17.34378], [27.30736, -17.60487], [27.14196, -17.81398], [26.89926, -17.98756], [26.74314, -18.0199], [26.68403, -18.07411], [26.55918, -17.99638], [26.21601, -17.88608], [26.0908, -17.93021], [26.08925, -17.98168], [25.85892, -17.97726], [25.85738, -17.91403], [25.6827, -17.81987], [25.51646, -17.86232], [25.26433, -17.79571], [25.23909, -17.90832], [25.31799, -18.07091], [25.39972, -18.12691], [25.53465, -18.39041], [25.68859, -18.56165], [25.79217, -18.6355], [25.82353, -18.82808], [25.94326, -18.90362], [25.99837, -19.02943], [25.96226, -19.08152], [26.17227, -19.53709], [26.72246, -19.92707], [27.21278, -20.08244], [27.29831, -20.28935], [27.28865, -20.49873], [27.69361, -20.48531], [27.72972, -20.51735], [27.69171, -21.08409], [27.91407, -21.31621], [28.01669, -21.57624], [28.29416, -21.59037], [28.49942, -21.66634], [28.58114, -21.63455], [29.07763, -21.81877], [29.04023, -21.85864], [29.02191, -21.90647], [29.02191, -21.95665], [29.04108, -22.00563], [29.08495, -22.04867], [29.14501, -22.07275], [29.1974, -22.07472], [29.24648, -22.05967], [29.3533, -22.18363], [29.37703, -22.19581], [29.64609, -22.12917], [29.76848, -22.14128], [29.92242, -22.19408], [30.13147, -22.30841], [30.2265, -22.2961], [30.28351, -22.35587], [30.38614, -22.34533], [30.48686, -22.31368], [30.6294, -22.32599], [30.86696, -22.28907], [31.08932, -22.34884], [31.16344, -22.32599], [31.30611, -22.422], [31.38336, -22.36919], [32.41234, -21.31246], [32.48236, -21.32873], [32.37115, -21.133], [32.51644, -20.91929], [32.48122, -20.63319], [32.55167, -20.56312], [32.66174, -20.56106], [32.85987, -20.27841], [32.85987, -20.16686], [32.93032, -20.03868], [33.01178, -20.02007], [33.06461, -19.77787], [32.95013, -19.67219], [32.84666, -19.68462], [32.84446, -19.48343], [32.78282, -19.47513], [32.77966, -19.36098], [32.85107, -19.29238], [32.87088, -19.09279], [32.84006, -19.0262], [32.72118, -19.02204], [32.69917, -18.94293], [32.73439, -18.92628], [32.70137, -18.84712], [32.82465, -18.77419], [32.9017, -18.7992], [32.95013, -18.69079], [32.88629, -18.58023], [32.88629, -18.51344], [33.02278, -18.4696], [33.03159, -18.35054], [32.94133, -17.99705], [33.0492, -17.60298], [32.98536, -17.55891], [32.96554, -17.48964], [33.0426, -17.3468], [33.00517, -17.30477], [32.96554, -17.11971], [32.84113, -16.92259], [32.91051, -16.89446], [32.97655, -16.70689], [32.78943, -16.70267], [32.69917, -16.66893], [32.71017, -16.59932], [32.42838, -16.4727], [32.28529, -16.43892], [32.02772, -16.43892], [31.91324, -16.41569], [31.90223, -16.34388], [31.67988, -16.19595], [31.42451, -16.15154], [31.30563, -16.01193], [31.13171, -15.98019], [30.97761, -16.05848], [30.91597, -15.99924], [30.42568, -15.9962], [30.41902, -15.62269]]]]
+         }
+       }];
+       var borders_default = {
+         type: type,
+         features: features
+       }; // src/country-coder.ts
 
-           return presetCollection(utilArrayUniq(results));
-         };
+       var borders = borders_default;
+       var whichPolygonGetter = {};
+       var featuresByCode = {};
+       var idFilterRegex = /(?=(?!^(and|the|of|el|la|de)$))(\b(and|the|of|el|la|de)\b)|[-_ .,'()&[\]/]/gi;
 
-         return _this;
-       }
+       function canonicalID(id) {
+         var s = id || "";
 
-       // `presetCategory` builds a `presetCollection` of member presets,
-       // decorated with some extra methods for searching and matching geometry
-       //
+         if (s.charAt(0) === ".") {
+           return s.toUpperCase();
+         } else {
+           return s.replace(idFilterRegex, "").toUpperCase();
+         }
+       }
 
-       function presetCategory(categoryID, category, all) {
-         var _this = Object.assign({}, category); // shallow copy
+       var levels = ["subterritory", "territory", "subcountryGroup", "country", "sharedLandform", "intermediateRegion", "subregion", "region", "subunion", "union", "unitedNations", "world"];
+       loadDerivedDataAndCaches(borders);
 
+       function loadDerivedDataAndCaches(borders2) {
+         var identifierProps = ["iso1A2", "iso1A3", "m49", "wikidata", "emojiFlag", "ccTLD", "nameEn"];
+         var geometryFeatures = [];
 
-         _this.id = categoryID;
-         _this.members = presetCollection(category.members.map(function (presetID) {
-           return all.item(presetID);
-         }).filter(Boolean));
-         _this.geometry = _this.members.collection.reduce(function (acc, preset) {
-           for (var i in preset.geometry) {
-             var geometry = preset.geometry[i];
+         for (var i in borders2.features) {
+           var feature2 = borders2.features[i];
+           feature2.properties.id = feature2.properties.iso1A2 || feature2.properties.m49 || feature2.properties.wikidata;
+           loadM49(feature2);
+           loadTLD(feature2);
+           loadIsoStatus(feature2);
+           loadLevel(feature2);
+           loadGroups(feature2);
+           loadFlag(feature2);
+           cacheFeatureByIDs(feature2);
+           if (feature2.geometry) geometryFeatures.push(feature2);
+         }
 
-             if (acc.indexOf(geometry) === -1) {
-               acc.push(geometry);
-             }
-           }
+         for (var _i in borders2.features) {
+           var _feature = borders2.features[_i];
+           _feature.properties.groups = _feature.properties.groups.map(function (groupID) {
+             return featuresByCode[groupID].properties.id;
+           });
+           loadMembersForGroupsOf(_feature);
+         }
 
-           return acc;
-         }, []);
+         for (var _i2 in borders2.features) {
+           var _feature2 = borders2.features[_i2];
+           loadRoadSpeedUnit(_feature2);
+           loadRoadHeightUnit(_feature2);
+           loadDriveSide(_feature2);
+           loadCallingCodes(_feature2);
+           loadGroupGroups(_feature2);
+         }
 
-         _this.matchGeometry = function (geom) {
-           return _this.geometry.indexOf(geom) >= 0;
-         };
+         for (var _i3 in borders2.features) {
+           var _feature3 = borders2.features[_i3];
 
-         _this.matchAllGeometry = function (geometries) {
-           return _this.members.collection.some(function (preset) {
-             return preset.matchAllGeometry(geometries);
+           _feature3.properties.groups.sort(function (groupID1, groupID2) {
+             return levels.indexOf(featuresByCode[groupID1].properties.level) - levels.indexOf(featuresByCode[groupID2].properties.level);
            });
-         };
 
-         _this.matchScore = function () {
-           return -1;
-         };
+           if (_feature3.properties.members) _feature3.properties.members.sort(function (id1, id2) {
+             var diff = levels.indexOf(featuresByCode[id1].properties.level) - levels.indexOf(featuresByCode[id2].properties.level);
 
-         _this.name = function () {
-           return _t("presets.categories.".concat(categoryID, ".name"), {
-             'default': categoryID
-           });
-         };
+             if (diff === 0) {
+               return borders2.features.indexOf(featuresByCode[id1]) - borders2.features.indexOf(featuresByCode[id2]);
+             }
 
-         _this.nameLabel = function () {
-           return _t.html("presets.categories.".concat(categoryID, ".name"), {
-             'default': categoryID
+             return diff;
            });
-         };
+         }
 
-         _this.terms = function () {
-           return [];
+         var geometryOnlyCollection = {
+           type: "FeatureCollection",
+           features: geometryFeatures
          };
+         whichPolygonGetter = whichPolygon_1(geometryOnlyCollection);
 
-         return _this;
-       }
+         function loadGroups(feature2) {
+           var props = feature2.properties;
 
-       // `presetField` decorates a given `field` Object
-       // with some extra methods for searching and matching geometry
-       //
+           if (!props.groups) {
+             props.groups = [];
+           }
 
-       function presetField(fieldID, field) {
-         var _this = Object.assign({}, field); // shallow copy
+           if (feature2.geometry && props.country) {
+             props.groups.push(props.country);
+           }
 
+           if (props.m49 !== "001") {
+             props.groups.push("001");
+           }
+         }
 
-         _this.id = fieldID; // for use in classes, element ids, css selectors
+         function loadM49(feature2) {
+           var props = feature2.properties;
 
-         _this.safeid = utilSafeClassName(fieldID);
+           if (!props.m49 && props.iso1N3) {
+             props.m49 = props.iso1N3;
+           }
+         }
 
-         _this.matchGeometry = function (geom) {
-           return !_this.geometry || _this.geometry.indexOf(geom) !== -1;
-         };
+         function loadTLD(feature2) {
+           var props = feature2.properties;
+           if (props.level === "unitedNations") return;
 
-         _this.matchAllGeometry = function (geometries) {
-           return !_this.geometry || geometries.every(function (geom) {
-             return _this.geometry.indexOf(geom) !== -1;
-           });
-         };
+           if (!props.ccTLD && props.iso1A2) {
+             props.ccTLD = "." + props.iso1A2.toLowerCase();
+           }
+         }
 
-         _this.t = function (scope, options) {
-           return _t("presets.fields.".concat(fieldID, ".").concat(scope), options);
-         };
+         function loadIsoStatus(feature2) {
+           var props = feature2.properties;
 
-         _this.t.html = function (scope, options) {
-           return _t.html("presets.fields.".concat(fieldID, ".").concat(scope), options);
-         };
+           if (!props.isoStatus && props.iso1A2) {
+             props.isoStatus = "official";
+           }
+         }
 
-         _this.title = function () {
-           return _this.overrideLabel || _this.t('label', {
-             'default': fieldID
-           });
-         };
+         function loadLevel(feature2) {
+           var props = feature2.properties;
+           if (props.level) return;
 
-         _this.label = function () {
-           return _this.overrideLabel || _this.t.html('label', {
-             'default': fieldID
-           });
-         };
+           if (!props.country) {
+             props.level = "country";
+           } else if (!props.iso1A2 || props.isoStatus === "official") {
+             props.level = "territory";
+           } else {
+             props.level = "subterritory";
+           }
+         }
 
-         var _placeholder = _this.placeholder;
+         function loadGroupGroups(feature2) {
+           var props = feature2.properties;
+           if (feature2.geometry || !props.members) return;
+           var featureLevelIndex = levels.indexOf(props.level);
+           var sharedGroups = [];
 
-         _this.placeholder = function () {
-           return _this.t('placeholder', {
-             'default': _placeholder
-           });
-         };
+           var _loop = function _loop(_i4) {
+             var memberID = props.members[_i4];
+             var member = featuresByCode[memberID];
+             var memberGroups = member.properties.groups.filter(function (groupID) {
+               return groupID !== feature2.properties.id && featureLevelIndex < levels.indexOf(featuresByCode[groupID].properties.level);
+             });
 
-         _this.originalTerms = (_this.terms || []).join();
+             if (_i4 === "0") {
+               sharedGroups = memberGroups;
+             } else {
+               sharedGroups = sharedGroups.filter(function (groupID) {
+                 return memberGroups.indexOf(groupID) !== -1;
+               });
+             }
+           };
 
-         _this.terms = function () {
-           return _this.t('terms', {
-             'default': _this.originalTerms
-           }).toLowerCase().trim().split(/\s*,+\s*/);
-         };
+           for (var _i4 in props.members) {
+             _loop(_i4);
+           }
 
-         _this.increment = _this.type === 'number' ? _this.increment || 1 : undefined;
-         return _this;
-       }
+           props.groups = props.groups.concat(sharedGroups.filter(function (groupID) {
+             return props.groups.indexOf(groupID) === -1;
+           }));
 
-       // `Array.prototype.lastIndexOf` method
-       // https://tc39.es/ecma262/#sec-array.prototype.lastindexof
-       _export({ target: 'Array', proto: true, forced: arrayLastIndexOf !== [].lastIndexOf }, {
-         lastIndexOf: arrayLastIndexOf
-       });
+           for (var j in sharedGroups) {
+             var groupFeature = featuresByCode[sharedGroups[j]];
 
-       // `presetPreset` decorates a given `preset` Object
-       // with some extra methods for searching and matching geometry
-       //
+             if (groupFeature.properties.members.indexOf(props.id) === -1) {
+               groupFeature.properties.members.push(props.id);
+             }
+           }
+         }
 
-       function presetPreset(presetID, preset, addable, allFields, allPresets) {
-         allFields = allFields || {};
-         allPresets = allPresets || {};
+         function loadRoadSpeedUnit(feature2) {
+           var props = feature2.properties;
 
-         var _this = Object.assign({}, preset); // shallow copy
+           if (feature2.geometry) {
+             if (!props.roadSpeedUnit) props.roadSpeedUnit = "km/h";
+           } else if (props.members) {
+             var vals = Array.from(new Set(props.members.map(function (id) {
+               var member = featuresByCode[id];
+               if (member.geometry) return member.properties.roadSpeedUnit || "km/h";
+             }).filter(Boolean)));
+             if (vals.length === 1) props.roadSpeedUnit = vals[0];
+           }
+         }
 
+         function loadRoadHeightUnit(feature2) {
+           var props = feature2.properties;
 
-         var _addable = addable || false;
+           if (feature2.geometry) {
+             if (!props.roadHeightUnit) props.roadHeightUnit = "m";
+           } else if (props.members) {
+             var vals = Array.from(new Set(props.members.map(function (id) {
+               var member = featuresByCode[id];
+               if (member.geometry) return member.properties.roadHeightUnit || "m";
+             }).filter(Boolean)));
+             if (vals.length === 1) props.roadHeightUnit = vals[0];
+           }
+         }
 
-         var _resolvedFields; // cache
+         function loadDriveSide(feature2) {
+           var props = feature2.properties;
 
+           if (feature2.geometry) {
+             if (!props.driveSide) props.driveSide = "right";
+           } else if (props.members) {
+             var vals = Array.from(new Set(props.members.map(function (id) {
+               var member = featuresByCode[id];
+               if (member.geometry) return member.properties.driveSide || "right";
+             }).filter(Boolean)));
+             if (vals.length === 1) props.driveSide = vals[0];
+           }
+         }
 
-         var _resolvedMoreFields; // cache
+         function loadCallingCodes(feature2) {
+           var props = feature2.properties;
 
+           if (!feature2.geometry && props.members) {
+             props.callingCodes = Array.from(new Set(props.members.reduce(function (array, id) {
+               var member = featuresByCode[id];
+               if (member.geometry && member.properties.callingCodes) return array.concat(member.properties.callingCodes);
+               return array;
+             }, [])));
+           }
+         }
 
-         _this.id = presetID;
-         _this.safeid = utilSafeClassName(presetID); // for use in css classes, selectors, element ids
+         function loadFlag(feature2) {
+           if (!feature2.properties.iso1A2) return;
+           var flag = feature2.properties.iso1A2.replace(/./g, function (_char) {
+             return String.fromCodePoint(_char.charCodeAt(0) + 127397);
+           });
+           feature2.properties.emojiFlag = flag;
+         }
 
-         _this.originalTerms = (_this.terms || []).join();
-         _this.originalName = _this.name || '';
-         _this.originalScore = _this.matchScore || 1;
-         _this.originalReference = _this.reference || {};
-         _this.originalFields = _this.fields || [];
-         _this.originalMoreFields = _this.moreFields || [];
+         function loadMembersForGroupsOf(feature2) {
+           for (var j in feature2.properties.groups) {
+             var groupID = feature2.properties.groups[j];
+             var groupFeature = featuresByCode[groupID];
+             if (!groupFeature.properties.members) groupFeature.properties.members = [];
+             groupFeature.properties.members.push(feature2.properties.id);
+           }
+         }
 
-         _this.fields = function () {
-           return _resolvedFields || (_resolvedFields = resolve('fields'));
-         };
+         function cacheFeatureByIDs(feature2) {
+           var ids = [];
 
-         _this.moreFields = function () {
-           return _resolvedMoreFields || (_resolvedMoreFields = resolve('moreFields'));
-         };
+           for (var k in identifierProps) {
+             var prop = identifierProps[k];
+             var id = feature2.properties[prop];
+             if (id) ids.push(id);
+           }
 
-         _this.resetFields = function () {
-           return _resolvedFields = _resolvedMoreFields = null;
-         };
+           if (feature2.properties.aliases) {
+             for (var j in feature2.properties.aliases) {
+               ids.push(feature2.properties.aliases[j]);
+             }
+           }
 
-         _this.tags = _this.tags || {};
-         _this.addTags = _this.addTags || _this.tags;
-         _this.removeTags = _this.removeTags || _this.addTags;
-         _this.geometry = _this.geometry || [];
+           for (var _i5 in ids) {
+             var _id = canonicalID(ids[_i5]);
 
-         _this.matchGeometry = function (geom) {
-           return _this.geometry.indexOf(geom) >= 0;
-         };
+             featuresByCode[_id] = feature2;
+           }
+         }
+       }
 
-         _this.matchAllGeometry = function (geoms) {
-           return geoms.every(_this.matchGeometry);
-         };
+       function locArray(loc) {
+         if (Array.isArray(loc)) {
+           return loc;
+         } else if (loc.coordinates) {
+           return loc.coordinates;
+         }
 
-         _this.matchScore = function (entityTags) {
-           var tags = _this.tags;
-           var seen = {};
-           var score = 0; // match on tags
+         return loc.geometry.coordinates;
+       }
 
-           for (var k in tags) {
-             seen[k] = true;
+       function smallestFeature(loc) {
+         var query = locArray(loc);
+         var featureProperties = whichPolygonGetter(query);
+         if (!featureProperties) return null;
+         return featuresByCode[featureProperties.id];
+       }
 
-             if (entityTags[k] === tags[k]) {
-               score += _this.originalScore;
-             } else if (tags[k] === '*' && k in entityTags) {
-               score += _this.originalScore / 2;
-             } else {
-               return -1;
-             }
-           } // boost score for additional matches in addTags - #6802
+       function countryFeature(loc) {
+         var feature2 = smallestFeature(loc);
+         if (!feature2) return null;
+         var countryCode = feature2.properties.country || feature2.properties.iso1A2;
+         return featuresByCode[countryCode] || null;
+       }
 
+       var defaultOpts = {
+         level: void 0,
+         maxLevel: void 0,
+         withProp: void 0
+       };
 
-           var addTags = _this.addTags;
+       function featureForLoc(loc, opts) {
+         var targetLevel = opts.level || "country";
+         var maxLevel = opts.maxLevel || "world";
+         var withProp = opts.withProp;
+         var targetLevelIndex = levels.indexOf(targetLevel);
+         if (targetLevelIndex === -1) return null;
+         var maxLevelIndex = levels.indexOf(maxLevel);
+         if (maxLevelIndex === -1) return null;
+         if (maxLevelIndex < targetLevelIndex) return null;
 
-           for (var _k in addTags) {
-             if (!seen[_k] && entityTags[_k] === addTags[_k]) {
-               score += _this.originalScore;
+         if (targetLevel === "country") {
+           var fastFeature = countryFeature(loc);
+
+           if (fastFeature) {
+             if (!withProp || fastFeature.properties[withProp]) {
+               return fastFeature;
              }
            }
+         }
 
-           return score;
-         };
+         var features2 = featuresContaining(loc);
 
-         _this.t = function (scope, options) {
-           var textID = "presets.presets.".concat(presetID, ".").concat(scope);
-           return _t(textID, options);
-         };
+         for (var i in features2) {
+           var feature2 = features2[i];
+           var levelIndex = levels.indexOf(feature2.properties.level);
 
-         _this.t.html = function (scope, options) {
-           var textID = "presets.presets.".concat(presetID, ".").concat(scope);
-           return _t.html(textID, options);
-         };
+           if (feature2.properties.level === targetLevel || levelIndex > targetLevelIndex && levelIndex <= maxLevelIndex) {
+             if (!withProp || feature2.properties[withProp]) {
+               return feature2;
+             }
+           }
+         }
 
-         _this.name = function () {
-           return _this.t('name', {
-             'default': _this.originalName
-           });
-         };
+         return null;
+       }
 
-         _this.nameLabel = function () {
-           return _this.t.html('name', {
-             'default': _this.originalName
-           });
-         };
+       function featureForID(id) {
+         var stringID;
 
-         _this.subtitle = function () {
-           if (_this.suggestion) {
-             var path = presetID.split('/');
-             path.pop(); // remove brand name
+         if (typeof id === "number") {
+           stringID = id.toString();
 
-             return _t('presets.presets.' + path.join('/') + '.name');
+           if (stringID.length === 1) {
+             stringID = "00" + stringID;
+           } else if (stringID.length === 2) {
+             stringID = "0" + stringID;
            }
+         } else {
+           stringID = canonicalID(id);
+         }
 
-           return null;
-         };
+         return featuresByCode[stringID] || null;
+       }
 
-         _this.subtitleLabel = function () {
-           if (_this.suggestion) {
-             var path = presetID.split('/');
-             path.pop(); // remove brand name
+       function smallestFeaturesForBbox(bbox) {
+         return whichPolygonGetter.bbox(bbox).map(function (props) {
+           return featuresByCode[props.id];
+         });
+       }
 
-             return _t.html('presets.presets.' + path.join('/') + '.name');
-           }
+       function smallestOrMatchingFeature(query) {
+         if (_typeof(query) === "object") {
+           return smallestFeature(query);
+         }
 
-           return null;
-         };
+         return featureForID(query);
+       }
 
-         _this.terms = function () {
-           return _this.t('terms', {
-             'default': _this.originalTerms
-           }).toLowerCase().trim().split(/\s*,+\s*/);
-         };
+       function feature$1(query) {
+         var opts = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : defaultOpts;
 
-         _this.isFallback = function () {
-           var tagCount = Object.keys(_this.tags).length;
-           return tagCount === 0 || tagCount === 1 && _this.tags.hasOwnProperty('area');
-         };
+         if (_typeof(query) === "object") {
+           return featureForLoc(query, opts);
+         }
 
-         _this.addable = function (val) {
-           if (!arguments.length) return _addable;
-           _addable = val;
-           return _this;
-         };
+         return featureForID(query);
+       }
 
-         _this.reference = function () {
-           // Lookup documentation on Wikidata...
-           var qid = _this.tags.wikidata || _this.tags['brand:wikidata'] || _this.tags['operator:wikidata'];
+       function iso1A2Code(query) {
+         var opts = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : defaultOpts;
+         opts.withProp = "iso1A2";
+         var match = feature$1(query, opts);
+         if (!match) return null;
+         return match.properties.iso1A2 || null;
+       }
 
-           if (qid) {
-             return {
-               qid: qid
-             };
-           } // Lookup documentation on OSM Wikibase...
+       function featuresContaining(query, strict) {
+         var matchingFeatures;
 
+         if (Array.isArray(query) && query.length === 4) {
+           matchingFeatures = smallestFeaturesForBbox(query);
+         } else {
+           var smallestOrMatching = smallestOrMatchingFeature(query);
+           matchingFeatures = smallestOrMatching ? [smallestOrMatching] : [];
+         }
 
-           var key = _this.originalReference.key || Object.keys(utilObjectOmit(_this.tags, 'name'))[0];
-           var value = _this.originalReference.value || _this.tags[key];
+         if (!matchingFeatures.length) return [];
+         var returnFeatures;
 
-           if (value === '*') {
-             return {
-               key: key
-             };
-           } else {
-             return {
-               key: key,
-               value: value
-             };
-           }
-         };
+         if (!strict || _typeof(query) === "object") {
+           returnFeatures = matchingFeatures.slice();
+         } else {
+           returnFeatures = [];
+         }
 
-         _this.unsetTags = function (tags, geometry, skipFieldDefaults) {
-           tags = utilObjectOmit(tags, Object.keys(_this.removeTags));
+         for (var j in matchingFeatures) {
+           var properties = matchingFeatures[j].properties;
 
-           if (geometry && !skipFieldDefaults) {
-             _this.fields().forEach(function (field) {
-               if (field.matchGeometry(geometry) && field.key && field["default"] === tags[field.key]) {
-                 delete tags[field.key];
-               }
-             });
+           for (var i in properties.groups) {
+             var groupID = properties.groups[i];
+             var groupFeature = featuresByCode[groupID];
+
+             if (returnFeatures.indexOf(groupFeature) === -1) {
+               returnFeatures.push(groupFeature);
+             }
            }
+         }
 
-           delete tags.area;
-           return tags;
-         };
+         return returnFeatures;
+       }
 
-         _this.setTags = function (tags, geometry, skipFieldDefaults) {
-           var addTags = _this.addTags;
-           tags = Object.assign({}, tags); // shallow copy
+       function featuresIn(id, strict) {
+         var feature2 = featureForID(id);
+         if (!feature2) return [];
+         var features2 = [];
 
-           for (var k in addTags) {
-             if (addTags[k] === '*') {
-               tags[k] = 'yes';
-             } else {
-               tags[k] = addTags[k];
-             }
-           } // Add area=yes if necessary.
-           // This is necessary if the geometry is already an area (e.g. user drew an area) AND any of:
-           // 1. chosen preset could be either an area or a line (`barrier=city_wall`)
-           // 2. chosen preset doesn't have a key in osmAreaKeys (`railway=station`)
+         if (!strict) {
+           features2.push(feature2);
+         }
 
+         var properties = feature2.properties;
 
-           if (!addTags.hasOwnProperty('area')) {
-             delete tags.area;
+         if (properties.members) {
+           for (var i in properties.members) {
+             var memberID = properties.members[i];
+             features2.push(featuresByCode[memberID]);
+           }
+         }
 
-             if (geometry === 'area') {
-               var needsAreaTag = true;
+         return features2;
+       }
 
-               if (_this.geometry.indexOf('line') === -1) {
-                 for (var _k2 in addTags) {
-                   if (_k2 in osmAreaKeys) {
-                     needsAreaTag = false;
-                     break;
-                   }
-                 }
-               }
+       function aggregateFeature(id) {
+         var features2 = featuresIn(id, false);
+         if (features2.length === 0) return null;
+         var aggregateCoordinates = [];
 
-               if (needsAreaTag) {
-                 tags.area = 'yes';
-               }
-             }
-           }
+         for (var i in features2) {
+           var feature2 = features2[i];
 
-           if (geometry && !skipFieldDefaults) {
-             _this.fields().forEach(function (field) {
-               if (field.matchGeometry(geometry) && field.key && !tags[field.key] && field["default"]) {
-                 tags[field.key] = field["default"];
-               }
-             });
+           if (feature2.geometry && feature2.geometry.type === "MultiPolygon" && feature2.geometry.coordinates) {
+             aggregateCoordinates = aggregateCoordinates.concat(feature2.geometry.coordinates);
            }
+         }
 
-           return tags;
-         }; // For a preset without fields, use the fields of the parent preset.
-         // Replace {preset} placeholders with the fields of the specified presets.
+         return {
+           type: "Feature",
+           properties: features2[0].properties,
+           geometry: {
+             type: "MultiPolygon",
+             coordinates: aggregateCoordinates
+           }
+         };
+       }
 
+       function roadSpeedUnit(query) {
+         var feature2 = smallestOrMatchingFeature(query);
+         return feature2 && feature2.properties.roadSpeedUnit || null;
+       }
 
-         function resolve(which) {
-           var fieldIDs = which === 'fields' ? _this.originalFields : _this.originalMoreFields;
-           var resolved = [];
-           fieldIDs.forEach(function (fieldID) {
-             var match = fieldID.match(/\{(.*)\}/);
+       var RADIUS = 6378137;
+       var FLATTENING = 1 / 298.257223563;
+       var POLAR_RADIUS = 6356752.3142;
+       var wgs84 = {
+         RADIUS: RADIUS,
+         FLATTENING: FLATTENING,
+         POLAR_RADIUS: POLAR_RADIUS
+       };
 
-             if (match !== null) {
-               // a presetID wrapped in braces {}
-               resolved = resolved.concat(inheritFields(match[1], which));
-             } else if (allFields[fieldID]) {
-               // a normal fieldID
-               resolved.push(allFields[fieldID]);
-             } else {
-               console.log("Cannot resolve \"".concat(fieldID, "\" found in ").concat(_this.id, ".").concat(which)); // eslint-disable-line no-console
-             }
-           }); // no fields resolved, so use the parent's if possible
+       var geometry_1 = geometry;
+       var ring = ringArea;
 
-           if (!resolved.length) {
-             var endIndex = _this.id.lastIndexOf('/');
+       function geometry(_) {
+         var area = 0,
+             i;
 
-             var parentID = endIndex && _this.id.substring(0, endIndex);
+         switch (_.type) {
+           case 'Polygon':
+             return polygonArea(_.coordinates);
 
-             if (parentID) {
-               resolved = inheritFields(parentID, which);
+           case 'MultiPolygon':
+             for (i = 0; i < _.coordinates.length; i++) {
+               area += polygonArea(_.coordinates[i]);
              }
-           }
 
-           return utilArrayUniq(resolved); // returns an array of fields to inherit from the given presetID, if found
+             return area;
 
-           function inheritFields(presetID, which) {
-             var parent = allPresets[presetID];
-             if (!parent) return [];
+           case 'Point':
+           case 'MultiPoint':
+           case 'LineString':
+           case 'MultiLineString':
+             return 0;
 
-             if (which === 'fields') {
-               return parent.fields().filter(shouldInherit);
-             } else if (which === 'moreFields') {
-               return parent.moreFields();
-             } else {
-               return [];
+           case 'GeometryCollection':
+             for (i = 0; i < _.geometries.length; i++) {
+               area += geometry(_.geometries[i]);
              }
-           } // Skip `fields` for the keys which define the preset.
-           // These are usually `typeCombo` fields like `shop=*`
-
 
-           function shouldInherit(f) {
-             if (f.key && _this.tags[f.key] !== undefined && // inherit anyway if multiple values are allowed or just a checkbox
-             f.type !== 'multiCombo' && f.type !== 'semiCombo' && f.type !== 'manyCombo' && f.type !== 'check') return false;
-             return true;
-           }
+             return area;
          }
-
-         return _this;
        }
 
-       var _mainPresetIndex = presetIndex(); // singleton
-       // `presetIndex` wraps a `presetCollection`
-       // with methods for loading new data and returning defaults
-       //
-
-       function presetIndex() {
-         var dispatch$1 = dispatch('favoritePreset', 'recentsChange');
-         var MAXRECENTS = 30; // seed the preset lists with geometry fallbacks
-
-         var POINT = presetPreset('point', {
-           name: 'Point',
-           tags: {},
-           geometry: ['point', 'vertex'],
-           matchScore: 0.1
-         });
-         var LINE = presetPreset('line', {
-           name: 'Line',
-           tags: {},
-           geometry: ['line'],
-           matchScore: 0.1
-         });
-         var AREA = presetPreset('area', {
-           name: 'Area',
-           tags: {
-             area: 'yes'
-           },
-           geometry: ['area'],
-           matchScore: 0.1
-         });
-         var RELATION = presetPreset('relation', {
-           name: 'Relation',
-           tags: {},
-           geometry: ['relation'],
-           matchScore: 0.1
-         });
-
-         var _this = presetCollection([POINT, LINE, AREA, RELATION]);
-
-         var _presets = {
-           point: POINT,
-           line: LINE,
-           area: AREA,
-           relation: RELATION
-         };
-         var _defaults = {
-           point: presetCollection([POINT]),
-           vertex: presetCollection([POINT]),
-           line: presetCollection([LINE]),
-           area: presetCollection([AREA]),
-           relation: presetCollection([RELATION])
-         };
-         var _fields = {};
-         var _categories = {};
-         var _universal = [];
-         var _addablePresetIDs = null; // Set of preset IDs that the user can add
+       function polygonArea(coords) {
+         var area = 0;
 
-         var _recents;
+         if (coords && coords.length > 0) {
+           area += Math.abs(ringArea(coords[0]));
 
-         var _favorites; // Index of presets by (geometry, tag key).
+           for (var i = 1; i < coords.length; i++) {
+             area -= Math.abs(ringArea(coords[i]));
+           }
+         }
 
+         return area;
+       }
+       /**
+        * Calculate the approximate area of the polygon were it projected onto
+        *     the earth.  Note that this area will be positive if ring is oriented
+        *     clockwise, otherwise it will be negative.
+        *
+        * Reference:
+        * Robert. G. Chamberlain and William H. Duquette, "Some Algorithms for
+        *     Polygons on a Sphere", JPL Publication 07-03, Jet Propulsion
+        *     Laboratory, Pasadena, CA, June 2007 http://trs-new.jpl.nasa.gov/dspace/handle/2014/40409
+        *
+        * Returns:
+        * {float} The approximate signed geodesic area of the polygon in square
+        *     meters.
+        */
 
-         var _geometryIndex = {
-           point: {},
-           vertex: {},
-           line: {},
-           area: {},
-           relation: {}
-         };
 
-         var _loadPromise;
+       function ringArea(coords) {
+         var p1,
+             p2,
+             p3,
+             lowerIndex,
+             middleIndex,
+             upperIndex,
+             i,
+             area = 0,
+             coordsLength = coords.length;
 
-         _this.ensureLoaded = function () {
-           if (_loadPromise) return _loadPromise;
-           return _loadPromise = Promise.all([_mainFileFetcher.get('preset_categories'), _mainFileFetcher.get('preset_defaults'), _mainFileFetcher.get('preset_presets'), _mainFileFetcher.get('preset_fields')]).then(function (vals) {
-             _this.merge({
-               categories: vals[0],
-               defaults: vals[1],
-               presets: vals[2],
-               fields: vals[3]
-             });
+         if (coordsLength > 2) {
+           for (i = 0; i < coordsLength; i++) {
+             if (i === coordsLength - 2) {
+               // i = N-2
+               lowerIndex = coordsLength - 2;
+               middleIndex = coordsLength - 1;
+               upperIndex = 0;
+             } else if (i === coordsLength - 1) {
+               // i = N-1
+               lowerIndex = coordsLength - 1;
+               middleIndex = 0;
+               upperIndex = 1;
+             } else {
+               // i = 0 to N-3
+               lowerIndex = i;
+               middleIndex = i + 1;
+               upperIndex = i + 2;
+             }
 
-             osmSetAreaKeys(_this.areaKeys());
-             osmSetPointTags(_this.pointTags());
-             osmSetVertexTags(_this.vertexTags());
-           });
-         };
+             p1 = coords[lowerIndex];
+             p2 = coords[middleIndex];
+             p3 = coords[upperIndex];
+             area += (rad(p3[0]) - rad(p1[0])) * Math.sin(rad(p2[1]));
+           }
 
-         _this.merge = function (d) {
-           // Merge Fields
-           if (d.fields) {
-             Object.keys(d.fields).forEach(function (fieldID) {
-               var f = d.fields[fieldID];
+           area = area * wgs84.RADIUS * wgs84.RADIUS / 2;
+         }
 
-               if (f) {
-                 // add or replace
-                 _fields[fieldID] = presetField(fieldID, f);
-               } else {
-                 // remove
-                 delete _fields[fieldID];
-               }
-             });
-           } // Merge Presets
+         return area;
+       }
 
+       function rad(_) {
+         return _ * Math.PI / 180;
+       }
 
-           if (d.presets) {
-             Object.keys(d.presets).forEach(function (presetID) {
-               var p = d.presets[presetID];
+       var geojsonArea = {
+         geometry: geometry_1,
+         ring: ring
+       };
 
-               if (p) {
-                 // add or replace
-                 var isAddable = !_addablePresetIDs || _addablePresetIDs.has(presetID);
+       var $includes = arrayIncludes.includes;
 
-                 _presets[presetID] = presetPreset(presetID, p, isAddable, _fields, _presets);
-               } else {
-                 // remove (but not if it's a fallback)
-                 var existing = _presets[presetID];
 
-                 if (existing && !existing.isFallback()) {
-                   delete _presets[presetID];
-                 }
-               }
-             });
-           } // Need to rebuild _this.collection before loading categories
+       // `Array.prototype.includes` method
+       // https://tc39.es/ecma262/#sec-array.prototype.includes
+       _export({ target: 'Array', proto: true }, {
+         includes: function includes(el /* , fromIndex = 0 */) {
+           return $includes(this, el, arguments.length > 1 ? arguments[1] : undefined);
+         }
+       });
 
+       // https://tc39.es/ecma262/#sec-array.prototype-@@unscopables
+       addToUnscopables('includes');
 
-           _this.collection = Object.values(_presets).concat(Object.values(_categories)); // Merge Categories
+       var validateCenter_1$1 = function validateCenter(center) {
+         var validCenterLengths = [2, 3];
 
-           if (d.categories) {
-             Object.keys(d.categories).forEach(function (categoryID) {
-               var c = d.categories[categoryID];
+         if (!Array.isArray(center) || !validCenterLengths.includes(center.length)) {
+           throw new Error("ERROR! Center has to be an array of length two or three");
+         }
 
-               if (c) {
-                 // add or replace
-                 _categories[categoryID] = presetCategory(categoryID, c, _this);
-               } else {
-                 // remove
-                 delete _categories[categoryID];
-               }
-             });
-           } // Rebuild _this.collection after loading categories
+         var _center = _slicedToArray(center, 2),
+             lng = _center[0],
+             lat = _center[1];
 
+         if (typeof lng !== "number" || typeof lat !== "number") {
+           throw new Error("ERROR! Longitude and Latitude has to be numbers but where ".concat(_typeof(lng), " and ").concat(_typeof(lat)));
+         }
 
-           _this.collection = Object.values(_presets).concat(Object.values(_categories)); // Merge Defaults
+         if (lng > 180 || lng < -180) {
+           throw new Error("ERROR! Longitude has to be between -180 and 180 but was ".concat(lng));
+         }
 
-           if (d.defaults) {
-             Object.keys(d.defaults).forEach(function (geometry) {
-               var def = d.defaults[geometry];
+         if (lat > 90 || lat < -90) {
+           throw new Error("ERROR! Latitude has to be between -90 and 90 but was ".concat(lat));
+         }
+       };
 
-               if (Array.isArray(def)) {
-                 // add or replace
-                 _defaults[geometry] = presetCollection(def.map(function (id) {
-                   return _presets[id] || _categories[id];
-                 }).filter(Boolean));
-               } else {
-                 // remove
-                 delete _defaults[geometry];
-               }
-             });
-           } // Rebuild universal fields array
+       var validateCenter$1 = {
+         validateCenter: validateCenter_1$1
+       };
 
+       var validateRadius_1$1 = function validateRadius(radius) {
+         if (typeof radius !== "number") {
+           throw new Error("ERROR! Radius has to be a positive number but was: ".concat(_typeof(radius)));
+         }
 
-           _universal = Object.values(_fields).filter(function (field) {
-             return field.universal;
-           }); // Reset all the preset fields - they'll need to be resolved again
+         if (radius <= 0) {
+           throw new Error("ERROR! Radius has to be a positive number but was: ".concat(radius));
+         }
+       };
 
-           Object.values(_presets).forEach(function (preset) {
-             return preset.resetFields();
-           }); // Rebuild geometry index
+       var validateRadius$1 = {
+         validateRadius: validateRadius_1$1
+       };
 
-           _geometryIndex = {
-             point: {},
-             vertex: {},
-             line: {},
-             area: {},
-             relation: {}
-           };
+       var validateNumberOfEdges_1$1 = function validateNumberOfEdges(numberOfEdges) {
+         if (typeof numberOfEdges !== "number") {
+           var ARGUMENT_TYPE = Array.isArray(numberOfEdges) ? "array" : _typeof(numberOfEdges);
+           throw new Error("ERROR! Number of edges has to be a number but was: ".concat(ARGUMENT_TYPE));
+         }
 
-           _this.collection.forEach(function (preset) {
-             (preset.geometry || []).forEach(function (geometry) {
-               var g = _geometryIndex[geometry];
+         if (numberOfEdges < 3) {
+           throw new Error("ERROR! Number of edges has to be at least 3 but was: ".concat(numberOfEdges));
+         }
+       };
 
-               for (var key in preset.tags) {
-                 (g[key] = g[key] || []).push(preset);
-               }
-             });
-           });
+       var validateNumberOfEdges$1 = {
+         validateNumberOfEdges: validateNumberOfEdges_1$1
+       };
 
-           return _this;
-         };
+       var validateEarthRadius_1$1 = function validateEarthRadius(earthRadius) {
+         if (typeof earthRadius !== "number") {
+           var ARGUMENT_TYPE = Array.isArray(earthRadius) ? "array" : _typeof(earthRadius);
+           throw new Error("ERROR! Earth radius has to be a number but was: ".concat(ARGUMENT_TYPE));
+         }
 
-         _this.match = function (entity, resolver) {
-           return resolver["transient"](entity, 'presetMatch', function () {
-             var geometry = entity.geometry(resolver); // Treat entities on addr:interpolation lines as points, not vertices - #3241
+         if (earthRadius <= 0) {
+           throw new Error("ERROR! Earth radius has to be a positive number but was: ".concat(earthRadius));
+         }
+       };
 
-             if (geometry === 'vertex' && entity.isOnAddressLine(resolver)) {
-               geometry = 'point';
-             }
+       var validateEarthRadius$1 = {
+         validateEarthRadius: validateEarthRadius_1$1
+       };
 
-             return _this.matchTags(entity.tags, geometry);
-           });
-         };
+       var validateBearing_1$1 = function validateBearing(bearing) {
+         if (typeof bearing !== "number") {
+           var ARGUMENT_TYPE = Array.isArray(bearing) ? "array" : _typeof(bearing);
+           throw new Error("ERROR! Bearing has to be a number but was: ".concat(ARGUMENT_TYPE));
+         }
+       };
 
-         _this.matchTags = function (tags, geometry) {
-           var geometryMatches = _geometryIndex[geometry];
-           var address;
-           var best = -1;
-           var match;
+       var validateBearing$1 = {
+         validateBearing: validateBearing_1$1
+       };
 
-           for (var k in tags) {
-             // If any part of an address is present, allow fallback to "Address" preset - #4353
-             if (/^addr:/.test(k) && geometryMatches['addr:*']) {
-               address = geometryMatches['addr:*'][0];
-             }
+       var validateCenter = validateCenter$1.validateCenter;
+       var validateRadius = validateRadius$1.validateRadius;
+       var validateNumberOfEdges = validateNumberOfEdges$1.validateNumberOfEdges;
+       var validateEarthRadius = validateEarthRadius$1.validateEarthRadius;
+       var validateBearing = validateBearing$1.validateBearing;
 
-             var keyMatches = geometryMatches[k];
-             if (!keyMatches) continue;
+       function validateInput$1(_ref) {
+         var center = _ref.center,
+             radius = _ref.radius,
+             numberOfEdges = _ref.numberOfEdges,
+             earthRadius = _ref.earthRadius,
+             bearing = _ref.bearing;
+         validateCenter(center);
+         validateRadius(radius);
+         validateNumberOfEdges(numberOfEdges);
+         validateEarthRadius(earthRadius);
+         validateBearing(bearing);
+       }
+
+       var validateCenter_1 = validateCenter;
+       var validateRadius_1 = validateRadius;
+       var validateNumberOfEdges_1 = validateNumberOfEdges;
+       var validateEarthRadius_1 = validateEarthRadius;
+       var validateBearing_1 = validateBearing;
+       var validateInput_1 = validateInput$1;
+       var inputValidation = {
+         validateCenter: validateCenter_1,
+         validateRadius: validateRadius_1,
+         validateNumberOfEdges: validateNumberOfEdges_1,
+         validateEarthRadius: validateEarthRadius_1,
+         validateBearing: validateBearing_1,
+         validateInput: validateInput_1
+       };
 
-             for (var i = 0; i < keyMatches.length; i++) {
-               var score = keyMatches[i].matchScore(tags);
+       var validateInput = inputValidation.validateInput;
+       var defaultEarthRadius = 6378137; // equatorial Earth radius
 
-               if (score > best) {
-                 best = score;
-                 match = keyMatches[i];
-               }
-             }
-           }
+       function toRadians(angleInDegrees) {
+         return angleInDegrees * Math.PI / 180;
+       }
 
-           if (address && (!match || match.isFallback())) {
-             match = address;
-           }
+       function toDegrees(angleInRadians) {
+         return angleInRadians * 180 / Math.PI;
+       }
 
-           return match || _this.fallback(geometry);
-         };
+       function offset(c1, distance, earthRadius, bearing) {
+         var lat1 = toRadians(c1[1]);
+         var lon1 = toRadians(c1[0]);
+         var dByR = distance / earthRadius;
+         var lat = Math.asin(Math.sin(lat1) * Math.cos(dByR) + Math.cos(lat1) * Math.sin(dByR) * Math.cos(bearing));
+         var lon = lon1 + Math.atan2(Math.sin(bearing) * Math.sin(dByR) * Math.cos(lat1), Math.cos(dByR) - Math.sin(lat1) * Math.sin(lat));
+         return [toDegrees(lon), toDegrees(lat)];
+       }
 
-         _this.allowsVertex = function (entity, resolver) {
-           if (entity.type !== 'node') return false;
-           if (Object.keys(entity.tags).length === 0) return true;
-           return resolver["transient"](entity, 'vertexMatch', function () {
-             // address lines allow vertices to act as standalone points
-             if (entity.isOnAddressLine(resolver)) return true;
-             var geometries = osmNodeGeometriesForTags(entity.tags);
-             if (geometries.vertex) return true;
-             if (geometries.point) return false; // allow vertices for unspecified points
+       var circleToPolygon = function circleToPolygon(center, radius, options) {
+         var n = getNumberOfEdges(options);
+         var earthRadius = getEarthRadius(options);
+         var bearing = getBearing(options);
+         var direction = getDirection(options); // validateInput() throws error on invalid input and do nothing on valid input
 
-             return true;
-           });
-         }; // Because of the open nature of tagging, iD will never have a complete
-         // list of tags used in OSM, so we want it to have logic like "assume
-         // that a closed way with an amenity tag is an area, unless the amenity
-         // is one of these specific types". This function computes a structure
-         // that allows testing of such conditions, based on the presets designated
-         // as as supporting (or not supporting) the area geometry.
-         //
-         // The returned object L is a keeplist/discardlist of tags. A closed way
-         // with a tag (k, v) is considered to be an area if `k in L && !(v in L[k])`
-         // (see `Way#isArea()`). In other words, the keys of L form the keeplist,
-         // and the subkeys form the discardlist.
+         validateInput({
+           center: center,
+           radius: radius,
+           numberOfEdges: n,
+           earthRadius: earthRadius,
+           bearing: bearing
+         });
+         var start = toRadians(bearing);
+         var coordinates = [];
 
+         for (var i = 0; i < n; ++i) {
+           coordinates.push(offset(center, radius, earthRadius, start + direction * 2 * Math.PI * -i / n));
+         }
 
-         _this.areaKeys = function () {
-           // The ignore list is for keys that imply lines. (We always add `area=yes` for exceptions)
-           var ignore = ['barrier', 'highway', 'footway', 'railway', 'junction', 'type'];
-           var areaKeys = {}; // ignore name-suggestion-index and deprecated presets
+         coordinates.push(coordinates[0]);
+         return {
+           type: "Polygon",
+           coordinates: [coordinates]
+         };
+       };
 
-           var presets = _this.collection.filter(function (p) {
-             return !p.suggestion && !p.replacement;
-           }); // keeplist
+       function getNumberOfEdges(options) {
+         if (isUndefinedOrNull(options)) {
+           return 32;
+         } else if (isObjectNotArray(options)) {
+           var numberOfEdges = options.numberOfEdges;
+           return numberOfEdges === undefined ? 32 : numberOfEdges;
+         }
 
+         return options;
+       }
 
-           presets.forEach(function (p) {
-             var keys = p.tags && Object.keys(p.tags);
-             var key = keys && keys.length && keys[0]; // pick the first tag
+       function getEarthRadius(options) {
+         if (isUndefinedOrNull(options)) {
+           return defaultEarthRadius;
+         } else if (isObjectNotArray(options)) {
+           var earthRadius = options.earthRadius;
+           return earthRadius === undefined ? defaultEarthRadius : earthRadius;
+         }
 
-             if (!key) return;
-             if (ignore.indexOf(key) !== -1) return;
+         return defaultEarthRadius;
+       }
 
-             if (p.geometry.indexOf('area') !== -1) {
-               // probably an area..
-               areaKeys[key] = areaKeys[key] || {};
-             }
-           }); // discardlist
+       function getDirection(options) {
+         if (isObjectNotArray(options) && options.rightHandRule) {
+           return -1;
+         }
 
-           presets.forEach(function (p) {
-             var key;
+         return 1;
+       }
 
-             for (key in p.addTags) {
-               // examine all addTags to get a better sense of what can be tagged on lines - #6800
-               var value = p.addTags[key];
+       function getBearing(options) {
+         if (isUndefinedOrNull(options)) {
+           return 0;
+         } else if (isObjectNotArray(options)) {
+           var bearing = options.bearing;
+           return bearing === undefined ? 0 : bearing;
+         }
 
-               if (key in areaKeys && // probably an area...
-               p.geometry.indexOf('line') !== -1 && // but sometimes a line
-               value !== '*') {
-                 areaKeys[key][value] = true;
-               }
-             }
-           });
-           return areaKeys;
-         };
+         return 0;
+       }
 
-         _this.pointTags = function () {
-           return _this.collection.reduce(function (pointTags, d) {
-             // ignore name-suggestion-index, deprecated, and generic presets
-             if (d.suggestion || d.replacement || d.searchable === false) return pointTags; // only care about the primary tag
+       function isObjectNotArray(argument) {
+         return argument !== null && _typeof(argument) === "object" && !Array.isArray(argument);
+       }
 
-             var keys = d.tags && Object.keys(d.tags);
-             var key = keys && keys.length && keys[0]; // pick the first tag
+       function isUndefinedOrNull(argument) {
+         return argument === null || argument === undefined;
+       }
 
-             if (!key) return pointTags; // if this can be a point
+       // `Number.EPSILON` constant
+       // https://tc39.es/ecma262/#sec-number.epsilon
+       _export({ target: 'Number', stat: true }, {
+         EPSILON: Math.pow(2, -52)
+       });
 
-             if (d.geometry.indexOf('point') !== -1) {
-               pointTags[key] = pointTags[key] || {};
-               pointTags[key][d.tags[key]] = true;
-             }
+       var quot = /"/g;
 
-             return pointTags;
-           }, {});
-         };
+       // `CreateHTML` abstract operation
+       // https://tc39.es/ecma262/#sec-createhtml
+       var createHtml = function (string, tag, attribute, value) {
+         var S = String(requireObjectCoercible(string));
+         var p1 = '<' + tag;
+         if (attribute !== '') p1 += ' ' + attribute + '="' + String(value).replace(quot, '&quot;') + '"';
+         return p1 + '>' + S + '</' + tag + '>';
+       };
 
-         _this.vertexTags = function () {
-           return _this.collection.reduce(function (vertexTags, d) {
-             // ignore name-suggestion-index, deprecated, and generic presets
-             if (d.suggestion || d.replacement || d.searchable === false) return vertexTags; // only care about the primary tag
+       // check the existence of a method, lowercase
+       // of a tag and escaping quotes in arguments
+       var stringHtmlForced = function (METHOD_NAME) {
+         return fails(function () {
+           var test = ''[METHOD_NAME]('"');
+           return test !== test.toLowerCase() || test.split('"').length > 3;
+         });
+       };
 
-             var keys = d.tags && Object.keys(d.tags);
-             var key = keys && keys.length && keys[0]; // pick the first tag
+       // `String.prototype.link` method
+       // https://tc39.es/ecma262/#sec-string.prototype.link
+       _export({ target: 'String', proto: true, forced: stringHtmlForced('link') }, {
+         link: function link(url) {
+           return createHtml(this, 'a', 'href', url);
+         }
+       });
 
-             if (!key) return vertexTags; // if this can be a vertex
+       /**
+        * splaytree v3.1.0
+        * Fast Splay tree for Node and browser
+        *
+        * @author Alexander Milevski <info@w8r.name>
+        * @license MIT
+        * @preserve
+        */
+       var Node$1 =
+       /** @class */
+       function () {
+         function Node(key, data) {
+           this.next = null;
+           this.key = key;
+           this.data = data;
+           this.left = null;
+           this.right = null;
+         }
 
-             if (d.geometry.indexOf('vertex') !== -1) {
-               vertexTags[key] = vertexTags[key] || {};
-               vertexTags[key][d.tags[key]] = true;
-             }
+         return Node;
+       }();
+       /* follows "An implementation of top-down splaying"
+        * by D. Sleator <sleator@cs.cmu.edu> March 1992
+        */
 
-             return vertexTags;
-           }, {});
-         };
 
-         _this.field = function (id) {
-           return _fields[id];
-         };
+       function DEFAULT_COMPARE(a, b) {
+         return a > b ? 1 : a < b ? -1 : 0;
+       }
+       /**
+        * Simple top down splay, not requiring i to be in the tree t.
+        */
 
-         _this.universal = function () {
-           return _universal;
-         };
 
-         _this.defaults = function (geometry, n, startWithRecents) {
-           var recents = [];
+       function splay(i, t, comparator) {
+         var N = new Node$1(null, null);
+         var l = N;
+         var r = N;
 
-           if (startWithRecents) {
-             recents = _this.recent().matchGeometry(geometry).collection.slice(0, 4);
-           }
+         while (true) {
+           var cmp = comparator(i, t.key); //if (i < t.key) {
 
-           var defaults;
+           if (cmp < 0) {
+             if (t.left === null) break; //if (i < t.left.key) {
 
-           if (_addablePresetIDs) {
-             defaults = Array.from(_addablePresetIDs).map(function (id) {
-               var preset = _this.item(id);
+             if (comparator(i, t.left.key) < 0) {
+               var y = t.left;
+               /* rotate right */
 
-               if (preset && preset.matchGeometry(geometry)) return preset;
-               return null;
-             }).filter(Boolean);
-           } else {
-             defaults = _defaults[geometry].collection.concat(_this.fallback(geometry));
-           }
+               t.left = y.right;
+               y.right = t;
+               t = y;
+               if (t.left === null) break;
+             }
 
-           return presetCollection(utilArrayUniq(recents.concat(defaults)).slice(0, n - 1));
-         }; // pass a Set of addable preset ids
+             r.left = t;
+             /* link right */
 
+             r = t;
+             t = t.left; //} else if (i > t.key) {
+           } else if (cmp > 0) {
+             if (t.right === null) break; //if (i > t.right.key) {
+
+             if (comparator(i, t.right.key) > 0) {
+               var y = t.right;
+               /* rotate left */
 
-         _this.addablePresetIDs = function (val) {
-           if (!arguments.length) return _addablePresetIDs; // accept and convert arrays
+               t.right = y.left;
+               y.left = t;
+               t = y;
+               if (t.right === null) break;
+             }
 
-           if (Array.isArray(val)) val = new Set(val);
-           _addablePresetIDs = val;
+             l.right = t;
+             /* link left */
 
-           if (_addablePresetIDs) {
-             // reset all presets
-             _this.collection.forEach(function (p) {
-               // categories aren't addable
-               if (p.addable) p.addable(_addablePresetIDs.has(p.id));
-             });
-           } else {
-             _this.collection.forEach(function (p) {
-               if (p.addable) p.addable(true);
-             });
-           }
+             l = t;
+             t = t.right;
+           } else break;
+         }
+         /* assemble */
 
-           return _this;
-         };
 
-         _this.recent = function () {
-           return presetCollection(utilArrayUniq(_this.getRecents().map(function (d) {
-             return d.preset;
-           })));
-         };
+         l.right = t.left;
+         r.left = t.right;
+         t.left = N.right;
+         t.right = N.left;
+         return t;
+       }
 
-         function RibbonItem(preset, source) {
-           var item = {};
-           item.preset = preset;
-           item.source = source;
+       function insert(i, data, t, comparator) {
+         var node = new Node$1(i, data);
 
-           item.isFavorite = function () {
-             return item.source === 'favorite';
-           };
+         if (t === null) {
+           node.left = node.right = null;
+           return node;
+         }
 
-           item.isRecent = function () {
-             return item.source === 'recent';
-           };
+         t = splay(i, t, comparator);
+         var cmp = comparator(i, t.key);
 
-           item.matches = function (preset) {
-             return item.preset.id === preset.id;
-           };
+         if (cmp < 0) {
+           node.left = t.left;
+           node.right = t;
+           t.left = null;
+         } else if (cmp >= 0) {
+           node.right = t.right;
+           node.left = t;
+           t.right = null;
+         }
 
-           item.minified = function () {
-             return {
-               pID: item.preset.id
-             };
-           };
+         return node;
+       }
 
-           return item;
-         }
+       function split$1(key, v, comparator) {
+         var left = null;
+         var right = null;
 
-         function ribbonItemForMinified(d, source) {
-           if (d && d.pID) {
-             var preset = _this.item(d.pID);
+         if (v) {
+           v = splay(key, v, comparator);
+           var cmp = comparator(v.key, key);
 
-             if (!preset) return null;
-             return RibbonItem(preset, source);
+           if (cmp === 0) {
+             left = v.left;
+             right = v.right;
+           } else if (cmp < 0) {
+             right = v.right;
+             v.right = null;
+             left = v;
+           } else {
+             left = v.left;
+             v.left = null;
+             right = v;
            }
-
-           return null;
          }
 
-         _this.getGenericRibbonItems = function () {
-           return ['point', 'line', 'area'].map(function (id) {
-             return RibbonItem(_this.item(id), 'generic');
-           });
+         return {
+           left: left,
+           right: right
          };
+       }
 
-         _this.getAddable = function () {
-           if (!_addablePresetIDs) return [];
-           return _addablePresetIDs.map(function (id) {
-             var preset = _this.item(id);
+       function merge$3(left, right, comparator) {
+         if (right === null) return left;
+         if (left === null) return right;
+         right = splay(left.key, right, comparator);
+         right.left = left;
+         return right;
+       }
+       /**
+        * Prints level of the tree
+        */
 
-             if (preset) return RibbonItem(preset, 'addable');
-             return null;
-           }).filter(Boolean);
-         };
 
-         function setRecents(items) {
-           _recents = items;
-           var minifiedItems = items.map(function (d) {
-             return d.minified();
-           });
-           corePreferences('preset_recents', JSON.stringify(minifiedItems));
-           dispatch$1.call('recentsChange');
+       function printRow(root, prefix, isTail, out, printNode) {
+         if (root) {
+           out("" + prefix + (isTail ? '└── ' : '├── ') + printNode(root) + "\n");
+           var indent = prefix + (isTail ? '    ' : '│   ');
+           if (root.left) printRow(root.left, indent, false, out, printNode);
+           if (root.right) printRow(root.right, indent, true, out, printNode);
          }
+       }
 
-         _this.getRecents = function () {
-           if (!_recents) {
-             // fetch from local storage
-             _recents = (JSON.parse(corePreferences('preset_recents')) || []).reduce(function (acc, d) {
-               var item = ribbonItemForMinified(d, 'recent');
-               if (item && item.preset.addable()) acc.push(item);
-               return acc;
-             }, []);
+       var Tree =
+       /** @class */
+       function () {
+         function Tree(comparator) {
+           if (comparator === void 0) {
+             comparator = DEFAULT_COMPARE;
            }
 
-           return _recents;
-         };
-
-         _this.addRecent = function (preset, besidePreset, after) {
-           var recents = _this.getRecents();
+           this._root = null;
+           this._size = 0;
+           this._comparator = comparator;
+         }
+         /**
+          * Inserts a key, allows duplicates
+          */
 
-           var beforeItem = _this.recentMatching(besidePreset);
 
-           var toIndex = recents.indexOf(beforeItem);
-           if (after) toIndex += 1;
-           var newItem = RibbonItem(preset, 'recent');
-           recents.splice(toIndex, 0, newItem);
-           setRecents(recents);
+         Tree.prototype.insert = function (key, data) {
+           this._size++;
+           return this._root = insert(key, data, this._root, this._comparator);
          };
+         /**
+          * Adds a key, if it is not present in the tree
+          */
 
-         _this.removeRecent = function (preset) {
-           var item = _this.recentMatching(preset);
 
-           if (item) {
-             var items = _this.getRecents();
+         Tree.prototype.add = function (key, data) {
+           var node = new Node$1(key, data);
 
-             items.splice(items.indexOf(item), 1);
-             setRecents(items);
+           if (this._root === null) {
+             node.left = node.right = null;
+             this._size++;
+             this._root = node;
            }
-         };
-
-         _this.recentMatching = function (preset) {
-           var items = _this.getRecents();
 
-           for (var i in items) {
-             if (items[i].matches(preset)) {
-               return items[i];
+           var comparator = this._comparator;
+           var t = splay(key, this._root, comparator);
+           var cmp = comparator(key, t.key);
+           if (cmp === 0) this._root = t;else {
+             if (cmp < 0) {
+               node.left = t.left;
+               node.right = t;
+               t.left = null;
+             } else if (cmp > 0) {
+               node.right = t.right;
+               node.left = t;
+               t.right = null;
              }
-           }
 
-           return null;
+             this._size++;
+             this._root = node;
+           }
+           return this._root;
          };
+         /**
+          * @param  {Key} key
+          * @return {Node|null}
+          */
 
-         _this.moveItem = function (items, fromIndex, toIndex) {
-           if (fromIndex === toIndex || fromIndex < 0 || toIndex < 0 || fromIndex >= items.length || toIndex >= items.length) return null;
-           items.splice(toIndex, 0, items.splice(fromIndex, 1)[0]);
-           return items;
+
+         Tree.prototype.remove = function (key) {
+           this._root = this._remove(key, this._root, this._comparator);
          };
+         /**
+          * Deletes i from the tree if it's there
+          */
 
-         _this.moveRecent = function (item, beforeItem) {
-           var recents = _this.getRecents();
 
-           var fromIndex = recents.indexOf(item);
-           var toIndex = recents.indexOf(beforeItem);
+         Tree.prototype._remove = function (i, t, comparator) {
+           var x;
+           if (t === null) return null;
+           t = splay(i, t, comparator);
+           var cmp = comparator(i, t.key);
 
-           var items = _this.moveItem(recents, fromIndex, toIndex);
+           if (cmp === 0) {
+             /* found it */
+             if (t.left === null) {
+               x = t.right;
+             } else {
+               x = splay(i, t.left, comparator);
+               x.right = t.right;
+             }
 
-           if (items) setRecents(items);
+             this._size--;
+             return x;
+           }
+
+           return t;
+           /* It wasn't there */
          };
+         /**
+          * Removes and returns the node with smallest key
+          */
 
-         _this.setMostRecent = function (preset) {
-           if (preset.searchable === false) return;
 
-           var items = _this.getRecents();
+         Tree.prototype.pop = function () {
+           var node = this._root;
 
-           var item = _this.recentMatching(preset);
+           if (node) {
+             while (node.left) {
+               node = node.left;
+             }
 
-           if (item) {
-             items.splice(items.indexOf(item), 1);
-           } else {
-             item = RibbonItem(preset, 'recent');
-           } // remove the last recent (first in, first out)
+             this._root = splay(node.key, this._root, this._comparator);
+             this._root = this._remove(node.key, this._root, this._comparator);
+             return {
+               key: node.key,
+               data: node.data
+             };
+           }
+
+           return null;
+         };
+         /**
+          * Find without splaying
+          */
 
 
-           while (items.length >= MAXRECENTS) {
-             items.pop();
-           } // prepend array
+         Tree.prototype.findStatic = function (key) {
+           var current = this._root;
+           var compare = this._comparator;
 
+           while (current) {
+             var cmp = compare(key, current.key);
+             if (cmp === 0) return current;else if (cmp < 0) current = current.left;else current = current.right;
+           }
 
-           items.unshift(item);
-           setRecents(items);
+           return null;
          };
 
-         function setFavorites(items) {
-           _favorites = items;
-           var minifiedItems = items.map(function (d) {
-             return d.minified();
-           });
-           corePreferences('preset_favorites', JSON.stringify(minifiedItems)); // call update
+         Tree.prototype.find = function (key) {
+           if (this._root) {
+             this._root = splay(key, this._root, this._comparator);
+             if (this._comparator(key, this._root.key) !== 0) return null;
+           }
 
-           dispatch$1.call('favoritePreset');
-         }
+           return this._root;
+         };
 
-         _this.addFavorite = function (preset, besidePreset, after) {
-           var favorites = _this.getFavorites();
+         Tree.prototype.contains = function (key) {
+           var current = this._root;
+           var compare = this._comparator;
 
-           var beforeItem = _this.favoriteMatching(besidePreset);
+           while (current) {
+             var cmp = compare(key, current.key);
+             if (cmp === 0) return true;else if (cmp < 0) current = current.left;else current = current.right;
+           }
 
-           var toIndex = favorites.indexOf(beforeItem);
-           if (after) toIndex += 1;
-           var newItem = RibbonItem(preset, 'favorite');
-           favorites.splice(toIndex, 0, newItem);
-           setFavorites(favorites);
+           return false;
          };
 
-         _this.toggleFavorite = function (preset) {
-           var favs = _this.getFavorites();
-
-           var favorite = _this.favoriteMatching(preset);
-
-           if (favorite) {
-             favs.splice(favs.indexOf(favorite), 1);
-           } else {
-             // only allow 10 favorites
-             if (favs.length === 10) {
-               // remove the last favorite (last in, first out)
-               favs.pop();
-             } // append array
+         Tree.prototype.forEach = function (visitor, ctx) {
+           var current = this._root;
+           var Q = [];
+           /* Initialize stack s */
 
+           var done = false;
 
-             favs.push(RibbonItem(preset, 'favorite'));
+           while (!done) {
+             if (current !== null) {
+               Q.push(current);
+               current = current.left;
+             } else {
+               if (Q.length !== 0) {
+                 current = Q.pop();
+                 visitor.call(ctx, current);
+                 current = current.right;
+               } else done = true;
+             }
            }
 
-           setFavorites(favs);
+           return this;
          };
+         /**
+          * Walk key range from `low` to `high`. Stops if `fn` returns a value.
+          */
 
-         _this.removeFavorite = function (preset) {
-           var item = _this.favoriteMatching(preset);
 
-           if (item) {
-             var items = _this.getFavorites();
+         Tree.prototype.range = function (low, high, fn, ctx) {
+           var Q = [];
+           var compare = this._comparator;
+           var node = this._root;
+           var cmp;
 
-             items.splice(items.indexOf(item), 1);
-             setFavorites(items);
-           }
-         };
+           while (Q.length !== 0 || node) {
+             if (node) {
+               Q.push(node);
+               node = node.left;
+             } else {
+               node = Q.pop();
+               cmp = compare(node.key, high);
 
-         _this.getFavorites = function () {
-           if (!_favorites) {
-             // fetch from local storage
-             var rawFavorites = JSON.parse(corePreferences('preset_favorites'));
+               if (cmp > 0) {
+                 break;
+               } else if (compare(node.key, low) >= 0) {
+                 if (fn.call(ctx, node)) return this; // stop if smth is returned
+               }
 
-             if (!rawFavorites) {
-               rawFavorites = [];
-               corePreferences('preset_favorites', JSON.stringify(rawFavorites));
+               node = node.right;
              }
-
-             _favorites = rawFavorites.reduce(function (output, d) {
-               var item = ribbonItemForMinified(d, 'favorite');
-               if (item && item.preset.addable()) output.push(item);
-               return output;
-             }, []);
            }
 
-           return _favorites;
+           return this;
          };
+         /**
+          * Returns array of keys
+          */
 
-         _this.favoriteMatching = function (preset) {
-           var favs = _this.getFavorites();
-
-           for (var index in favs) {
-             if (favs[index].matches(preset)) {
-               return favs[index];
-             }
-           }
 
-           return null;
+         Tree.prototype.keys = function () {
+           var keys = [];
+           this.forEach(function (_a) {
+             var key = _a.key;
+             return keys.push(key);
+           });
+           return keys;
          };
+         /**
+          * Returns array of all the data in the nodes
+          */
 
-         return utilRebind(_this, dispatch$1, 'on');
-       }
 
-       function utilTagText(entity) {
-         var obj = entity && entity.tags || {};
-         return Object.keys(obj).map(function (k) {
-           return k + '=' + obj[k];
-         }).join(', ');
-       }
-       function utilTotalExtent(array, graph) {
-         var extent = geoExtent();
-         var val, entity;
+         Tree.prototype.values = function () {
+           var values = [];
+           this.forEach(function (_a) {
+             var data = _a.data;
+             return values.push(data);
+           });
+           return values;
+         };
 
-         for (var i = 0; i < array.length; i++) {
-           val = array[i];
-           entity = typeof val === 'string' ? graph.hasEntity(val) : val;
+         Tree.prototype.min = function () {
+           if (this._root) return this.minNode(this._root).key;
+           return null;
+         };
 
-           if (entity) {
-             extent._extend(entity.extent(graph));
+         Tree.prototype.max = function () {
+           if (this._root) return this.maxNode(this._root).key;
+           return null;
+         };
+
+         Tree.prototype.minNode = function (t) {
+           if (t === void 0) {
+             t = this._root;
            }
-         }
 
-         return extent;
-       }
-       function utilTagDiff(oldTags, newTags) {
-         var tagDiff = [];
-         var keys = utilArrayUnion(Object.keys(oldTags), Object.keys(newTags)).sort();
-         keys.forEach(function (k) {
-           var oldVal = oldTags[k];
-           var newVal = newTags[k];
+           if (t) while (t.left) {
+             t = t.left;
+           }
+           return t;
+         };
 
-           if ((oldVal || oldVal === '') && (newVal === undefined || newVal !== oldVal)) {
-             tagDiff.push({
-               type: '-',
-               key: k,
-               oldVal: oldVal,
-               newVal: newVal,
-               display: '- ' + k + '=' + oldVal
-             });
+         Tree.prototype.maxNode = function (t) {
+           if (t === void 0) {
+             t = this._root;
            }
 
-           if ((newVal || newVal === '') && (oldVal === undefined || newVal !== oldVal)) {
-             tagDiff.push({
-               type: '+',
-               key: k,
-               oldVal: oldVal,
-               newVal: newVal,
-               display: '+ ' + k + '=' + newVal
-             });
+           if (t) while (t.right) {
+             t = t.right;
            }
-         });
-         return tagDiff;
-       }
-       function utilEntitySelector(ids) {
-         return ids.length ? '.' + ids.join(',.') : 'nothing';
-       } // returns an selector to select entity ids for:
-       //  - entityIDs passed in
-       //  - shallow descendant entityIDs for any of those entities that are relations
+           return t;
+         };
+         /**
+          * Returns node at given index
+          */
+
+
+         Tree.prototype.at = function (index) {
+           var current = this._root;
+           var done = false;
+           var i = 0;
+           var Q = [];
 
-       function utilEntityOrMemberSelector(ids, graph) {
-         var seen = new Set(ids);
-         ids.forEach(collectShallowDescendants);
-         return utilEntitySelector(Array.from(seen));
+           while (!done) {
+             if (current) {
+               Q.push(current);
+               current = current.left;
+             } else {
+               if (Q.length > 0) {
+                 current = Q.pop();
+                 if (i === index) return current;
+                 i++;
+                 current = current.right;
+               } else done = true;
+             }
+           }
 
-         function collectShallowDescendants(id) {
-           var entity = graph.hasEntity(id);
-           if (!entity || entity.type !== 'relation') return;
-           entity.members.map(function (member) {
-             return member.id;
-           }).forEach(function (id) {
-             seen.add(id);
-           });
-         }
-       } // returns an selector to select entity ids for:
-       //  - entityIDs passed in
-       //  - deep descendant entityIDs for any of those entities that are relations
+           return null;
+         };
 
-       function utilEntityOrDeepMemberSelector(ids, graph) {
-         return utilEntitySelector(utilEntityAndDeepMemberIDs(ids, graph));
-       } // returns an selector to select entity ids for:
-       //  - entityIDs passed in
-       //  - deep descendant entityIDs for any of those entities that are relations
+         Tree.prototype.next = function (d) {
+           var root = this._root;
+           var successor = null;
 
-       function utilEntityAndDeepMemberIDs(ids, graph) {
-         var seen = new Set();
-         ids.forEach(collectDeepDescendants);
-         return Array.from(seen);
+           if (d.right) {
+             successor = d.right;
 
-         function collectDeepDescendants(id) {
-           if (seen.has(id)) return;
-           seen.add(id);
-           var entity = graph.hasEntity(id);
-           if (!entity || entity.type !== 'relation') return;
-           entity.members.map(function (member) {
-             return member.id;
-           }).forEach(collectDeepDescendants); // recurse
-         }
-       } // returns an selector to select entity ids for:
-       //  - deep descendant entityIDs for any of those entities that are relations
+             while (successor.left) {
+               successor = successor.left;
+             }
 
-       function utilDeepMemberSelector(ids, graph, skipMultipolgonMembers) {
-         var idsSet = new Set(ids);
-         var seen = new Set();
-         var returners = new Set();
-         ids.forEach(collectDeepDescendants);
-         return utilEntitySelector(Array.from(returners));
+             return successor;
+           }
 
-         function collectDeepDescendants(id) {
-           if (seen.has(id)) return;
-           seen.add(id);
+           var comparator = this._comparator;
 
-           if (!idsSet.has(id)) {
-             returners.add(id);
+           while (root) {
+             var cmp = comparator(d.key, root.key);
+             if (cmp === 0) break;else if (cmp < 0) {
+               successor = root;
+               root = root.left;
+             } else root = root.right;
            }
 
-           var entity = graph.hasEntity(id);
-           if (!entity || entity.type !== 'relation') return;
-           if (skipMultipolgonMembers && entity.isMultipolygon()) return;
-           entity.members.map(function (member) {
-             return member.id;
-           }).forEach(collectDeepDescendants); // recurse
-         }
-       } // Adds or removes highlight styling for the specified entities
+           return successor;
+         };
 
-       function utilHighlightEntities(ids, highlighted, context) {
-         context.surface().selectAll(utilEntityOrDeepMemberSelector(ids, context.graph())).classed('highlighted', highlighted);
-       } // returns an Array that is the union of:
-       //  - nodes for any nodeIDs passed in
-       //  - child nodes of any wayIDs passed in
-       //  - descendant member and child nodes of relationIDs passed in
+         Tree.prototype.prev = function (d) {
+           var root = this._root;
+           var predecessor = null;
 
-       function utilGetAllNodes(ids, graph) {
-         var seen = new Set();
-         var nodes = new Set();
-         ids.forEach(collectNodes);
-         return Array.from(nodes);
+           if (d.left !== null) {
+             predecessor = d.left;
 
-         function collectNodes(id) {
-           if (seen.has(id)) return;
-           seen.add(id);
-           var entity = graph.hasEntity(id);
-           if (!entity) return;
+             while (predecessor.right) {
+               predecessor = predecessor.right;
+             }
 
-           if (entity.type === 'node') {
-             nodes.add(entity);
-           } else if (entity.type === 'way') {
-             entity.nodes.forEach(collectNodes);
-           } else {
-             entity.members.map(function (member) {
-               return member.id;
-             }).forEach(collectNodes); // recurse
+             return predecessor;
            }
-         }
-       }
-       function utilDisplayName(entity) {
-         var localizedNameKey = 'name:' + _mainLocalizer.languageCode().toLowerCase();
-         var name = entity.tags[localizedNameKey] || entity.tags.name || '';
-         var network = entity.tags.cycle_network || entity.tags.network;
 
-         if (!name && entity.tags.ref) {
-           name = entity.tags.ref;
+           var comparator = this._comparator;
 
-           if (network) {
-             name = network + ' ' + name;
+           while (root) {
+             var cmp = comparator(d.key, root.key);
+             if (cmp === 0) break;else if (cmp < 0) root = root.left;else {
+               predecessor = root;
+               root = root.right;
+             }
            }
-         }
 
-         return name;
-       }
-       function utilDisplayNameForPath(entity) {
-         var name = utilDisplayName(entity);
-         var isFirefox = utilDetect().browser.toLowerCase().indexOf('firefox') > -1;
+           return predecessor;
+         };
 
-         if (!isFirefox && name && rtlRegex.test(name)) {
-           name = fixRTLTextForSvg(name);
-         }
+         Tree.prototype.clear = function () {
+           this._root = null;
+           this._size = 0;
+           return this;
+         };
 
-         return name;
-       }
-       function utilDisplayType(id) {
-         return {
-           n: _t('inspector.node'),
-           w: _t('inspector.way'),
-           r: _t('inspector.relation')
-         }[id.charAt(0)];
-       }
-       function utilDisplayLabel(entity, graphOrGeometry) {
-         var displayName = utilDisplayName(entity);
+         Tree.prototype.toList = function () {
+           return toList(this._root);
+         };
+         /**
+          * Bulk-load items. Both array have to be same size
+          */
 
-         if (displayName) {
-           // use the display name if there is one
-           return displayName;
-         }
 
-         var preset = typeof graphOrGeometry === 'string' ? _mainPresetIndex.matchTags(entity.tags, graphOrGeometry) : _mainPresetIndex.match(entity, graphOrGeometry);
+         Tree.prototype.load = function (keys, values, presort) {
+           if (values === void 0) {
+             values = [];
+           }
 
-         if (preset && preset.name()) {
-           // use the preset name if there is a match
-           return preset.name();
-         } // fallback to the display type (node/way/relation)
+           if (presort === void 0) {
+             presort = false;
+           }
 
+           var size = keys.length;
+           var comparator = this._comparator; // sort if needed
 
-         return utilDisplayType(entity.id);
-       }
-       function utilEntityRoot(entityType) {
-         return {
-           node: 'n',
-           way: 'w',
-           relation: 'r'
-         }[entityType];
-       } // Returns a single object containing the tags of all the given entities.
-       // Example:
-       // {
-       //   highway: 'service',
-       //   service: 'parking_aisle'
-       // }
-       //           +
-       // {
-       //   highway: 'service',
-       //   service: 'driveway',
-       //   width: '3'
-       // }
-       //           =
-       // {
-       //   highway: 'service',
-       //   service: [ 'driveway', 'parking_aisle' ],
-       //   width: [ '3', undefined ]
-       // }
+           if (presort) sort(keys, values, 0, size - 1, comparator);
 
-       function utilCombinedTags(entityIDs, graph) {
-         var tags = {};
-         var tagCounts = {};
-         var allKeys = new Set();
-         var entities = entityIDs.map(function (entityID) {
-           return graph.hasEntity(entityID);
-         }).filter(Boolean); // gather the aggregate keys
+           if (this._root === null) {
+             // empty tree
+             this._root = loadRecursive(keys, values, 0, size);
+             this._size = size;
+           } else {
+             // that re-builds the whole tree from two in-order traversals
+             var mergedList = mergeLists(this.toList(), createList(keys, values), comparator);
+             size = this._size + size;
+             this._root = sortedListToBST({
+               head: mergedList
+             }, 0, size);
+           }
 
-         entities.forEach(function (entity) {
-           var keys = Object.keys(entity.tags).filter(Boolean);
-           keys.forEach(function (key) {
-             allKeys.add(key);
-           });
-         });
-         entities.forEach(function (entity) {
-           allKeys.forEach(function (key) {
-             var value = entity.tags[key]; // purposely allow `undefined`
+           return this;
+         };
 
-             if (!tags.hasOwnProperty(key)) {
-               // first value, set as raw
-               tags[key] = value;
-             } else {
-               if (!Array.isArray(tags[key])) {
-                 if (tags[key] !== value) {
-                   // first alternate value, replace single value with array
-                   tags[key] = [tags[key], value];
-                 }
-               } else {
-                 // type is array
-                 if (tags[key].indexOf(value) === -1) {
-                   // subsequent alternate value, add to array
-                   tags[key].push(value);
-                 }
-               }
-             }
+         Tree.prototype.isEmpty = function () {
+           return this._root === null;
+         };
 
-             var tagHash = key + '=' + value;
-             if (!tagCounts[tagHash]) tagCounts[tagHash] = 0;
-             tagCounts[tagHash] += 1;
-           });
+         Object.defineProperty(Tree.prototype, "size", {
+           get: function get() {
+             return this._size;
+           },
+           enumerable: true,
+           configurable: true
+         });
+         Object.defineProperty(Tree.prototype, "root", {
+           get: function get() {
+             return this._root;
+           },
+           enumerable: true,
+           configurable: true
          });
 
-         for (var key in tags) {
-           if (!Array.isArray(tags[key])) continue; // sort values by frequency then alphabetically
+         Tree.prototype.toString = function (printNode) {
+           if (printNode === void 0) {
+             printNode = function printNode(n) {
+               return String(n.key);
+             };
+           }
 
-           tags[key] = tags[key].sort(function (val1, val2) {
-             var key = key; // capture
+           var out = [];
+           printRow(this._root, '', true, function (v) {
+             return out.push(v);
+           }, printNode);
+           return out.join('');
+         };
 
-             var count2 = tagCounts[key + '=' + val2];
-             var count1 = tagCounts[key + '=' + val1];
+         Tree.prototype.update = function (key, newKey, newData) {
+           var comparator = this._comparator;
 
-             if (count2 !== count1) {
-               return count2 - count1;
-             }
+           var _a = split$1(key, this._root, comparator),
+               left = _a.left,
+               right = _a.right;
 
-             if (val2 && val1) {
-               return val1.localeCompare(val2);
-             }
+           if (comparator(key, newKey) < 0) {
+             right = insert(newKey, newData, right, comparator);
+           } else {
+             left = insert(newKey, newData, left, comparator);
+           }
 
-             return val1 ? 1 : -1;
-           });
-         }
+           this._root = merge$3(left, right, comparator);
+         };
 
-         return tags;
-       }
-       function utilStringQs(str) {
-         var i = 0; // advance past any leading '?' or '#' characters
+         Tree.prototype.split = function (key) {
+           return split$1(key, this._root, this._comparator);
+         };
 
-         while (i < str.length && (str[i] === '?' || str[i] === '#')) {
-           i++;
-         }
+         return Tree;
+       }();
 
-         str = str.slice(i);
-         return str.split('&').reduce(function (obj, pair) {
-           var parts = pair.split('=');
+       function loadRecursive(keys, values, start, end) {
+         var size = end - start;
 
-           if (parts.length === 2) {
-             obj[parts[0]] = null === parts[1] ? '' : decodeURIComponent(parts[1]);
-           }
+         if (size > 0) {
+           var middle = start + Math.floor(size / 2);
+           var key = keys[middle];
+           var data = values[middle];
+           var node = new Node$1(key, data);
+           node.left = loadRecursive(keys, values, start, middle);
+           node.right = loadRecursive(keys, values, middle + 1, end);
+           return node;
+         }
 
-           return obj;
-         }, {});
+         return null;
        }
-       function utilQsString(obj, noencode) {
-         // encode everything except special characters used in certain hash parameters:
-         // "/" in map states, ":", ",", {" and "}" in background
-         function softEncode(s) {
-           return encodeURIComponent(s).replace(/(%2F|%3A|%2C|%7B|%7D)/g, decodeURIComponent);
+
+       function createList(keys, values) {
+         var head = new Node$1(null, null);
+         var p = head;
+
+         for (var i = 0; i < keys.length; i++) {
+           p = p.next = new Node$1(keys[i], values[i]);
          }
 
-         return Object.keys(obj).sort().map(function (key) {
-           return encodeURIComponent(key) + '=' + (noencode ? softEncode(obj[key]) : encodeURIComponent(obj[key]));
-         }).join('&');
+         p.next = null;
+         return head.next;
        }
-       function utilPrefixDOMProperty(property) {
-         var prefixes = ['webkit', 'ms', 'moz', 'o'];
-         var i = -1;
-         var n = prefixes.length;
-         var s = document.body;
-         if (property in s) return property;
-         property = property.substr(0, 1).toUpperCase() + property.substr(1);
 
-         while (++i < n) {
-           if (prefixes[i] + property in s) {
-             return prefixes[i] + property;
+       function toList(root) {
+         var current = root;
+         var Q = [];
+         var done = false;
+         var head = new Node$1(null, null);
+         var p = head;
+
+         while (!done) {
+           if (current) {
+             Q.push(current);
+             current = current.left;
+           } else {
+             if (Q.length > 0) {
+               current = p = p.next = Q.pop();
+               current = current.right;
+             } else done = true;
            }
          }
 
-         return false;
+         p.next = null; // that'll work even if the tree was empty
+
+         return head.next;
        }
-       function utilPrefixCSSProperty(property) {
-         var prefixes = ['webkit', 'ms', 'Moz', 'O'];
-         var i = -1;
-         var n = prefixes.length;
-         var s = document.body.style;
 
-         if (property.toLowerCase() in s) {
-           return property.toLowerCase();
-         }
+       function sortedListToBST(list, start, end) {
+         var size = end - start;
 
-         while (++i < n) {
-           if (prefixes[i] + property in s) {
-             return '-' + prefixes[i].toLowerCase() + property.replace(/([A-Z])/g, '-$1').toLowerCase();
-           }
+         if (size > 0) {
+           var middle = start + Math.floor(size / 2);
+           var left = sortedListToBST(list, start, middle);
+           var root = list.head;
+           root.left = left;
+           list.head = list.head.next;
+           root.right = sortedListToBST(list, middle + 1, end);
+           return root;
          }
 
-         return false;
+         return null;
        }
-       var transformProperty;
-       function utilSetTransform(el, x, y, scale) {
-         var prop = transformProperty = transformProperty || utilPrefixCSSProperty('Transform');
-         var translate = utilDetect().opera ? 'translate(' + x + 'px,' + y + 'px)' : 'translate3d(' + x + 'px,' + y + 'px,0)';
-         return el.style(prop, translate + (scale ? ' scale(' + scale + ')' : ''));
-       } // Calculates Levenshtein distance between two strings
-       // see:  https://en.wikipedia.org/wiki/Levenshtein_distance
-       // first converts the strings to lowercase and replaces diacritic marks with ascii equivalents.
 
-       function utilEditDistance(a, b) {
-         a = remove$1(a.toLowerCase());
-         b = remove$1(b.toLowerCase());
-         if (a.length === 0) return b.length;
-         if (b.length === 0) return a.length;
-         var matrix = [];
-         var i, j;
+       function mergeLists(l1, l2, compare) {
+         var head = new Node$1(null, null); // dummy
 
-         for (i = 0; i <= b.length; i++) {
-           matrix[i] = [i];
+         var p = head;
+         var p1 = l1;
+         var p2 = l2;
+
+         while (p1 !== null && p2 !== null) {
+           if (compare(p1.key, p2.key) < 0) {
+             p.next = p1;
+             p1 = p1.next;
+           } else {
+             p.next = p2;
+             p2 = p2.next;
+           }
+
+           p = p.next;
          }
 
-         for (j = 0; j <= a.length; j++) {
-           matrix[0][j] = j;
+         if (p1 !== null) {
+           p.next = p1;
+         } else if (p2 !== null) {
+           p.next = p2;
          }
 
-         for (i = 1; i <= b.length; i++) {
-           for (j = 1; j <= a.length; j++) {
-             if (b.charAt(i - 1) === a.charAt(j - 1)) {
-               matrix[i][j] = matrix[i - 1][j - 1];
-             } else {
-               matrix[i][j] = Math.min(matrix[i - 1][j - 1] + 1, // substitution
-               Math.min(matrix[i][j - 1] + 1, // insertion
-               matrix[i - 1][j] + 1)); // deletion
-             }
-           }
+         return head.next;
+       }
+
+       function sort(keys, values, left, right, compare) {
+         if (left >= right) return;
+         var pivot = keys[left + right >> 1];
+         var i = left - 1;
+         var j = right + 1;
+
+         while (true) {
+           do {
+             i++;
+           } while (compare(keys[i], pivot) < 0);
+
+           do {
+             j--;
+           } while (compare(keys[j], pivot) > 0);
+
+           if (i >= j) break;
+           var tmp = keys[i];
+           keys[i] = keys[j];
+           keys[j] = tmp;
+           tmp = values[i];
+           values[i] = values[j];
+           values[j] = tmp;
          }
 
-         return matrix[b.length][a.length];
-       } // a d3.mouse-alike which
-       // 1. Only works on HTML elements, not SVG
-       // 2. Does not cause style recalculation
+         sort(keys, values, left, j, compare);
+         sort(keys, values, j + 1, right, compare);
+       }
 
-       function utilFastMouse(container) {
-         var rect = container.getBoundingClientRect();
-         var rectLeft = rect.left;
-         var rectTop = rect.top;
-         var clientLeft = +container.clientLeft;
-         var clientTop = +container.clientTop;
-         return function (e) {
-           return [e.clientX - rectLeft - clientLeft, e.clientY - rectTop - clientTop];
-         };
+       function _classCallCheck(instance, Constructor) {
+         if (!(instance instanceof Constructor)) {
+           throw new TypeError("Cannot call a class as a function");
+         }
        }
-       function utilAsyncMap(inputs, func, callback) {
-         var remaining = inputs.length;
-         var results = [];
-         var errors = [];
-         inputs.forEach(function (d, i) {
-           func(d, function done(err, data) {
-             errors[i] = err;
-             results[i] = data;
-             remaining--;
-             if (!remaining) callback(errors, results);
-           });
-         });
-       } // wraps an index to an interval [0..length-1]
 
-       function utilWrap(index, length) {
-         if (index < 0) {
-           index += Math.ceil(-index / length) * length;
+       function _defineProperties(target, props) {
+         for (var i = 0; i < props.length; i++) {
+           var descriptor = props[i];
+           descriptor.enumerable = descriptor.enumerable || false;
+           descriptor.configurable = true;
+           if ("value" in descriptor) descriptor.writable = true;
+           Object.defineProperty(target, descriptor.key, descriptor);
          }
+       }
 
-         return index % length;
+       function _createClass(Constructor, protoProps, staticProps) {
+         if (protoProps) _defineProperties(Constructor.prototype, protoProps);
+         if (staticProps) _defineProperties(Constructor, staticProps);
+         return Constructor;
        }
        /**
-        * a replacement for functor
+        * A bounding box has the format:
+        *
+        *  { ll: { x: xmin, y: ymin }, ur: { x: xmax, y: ymax } }
         *
-        * @param {*} value any value
-        * @returns {Function} a function that returns that value or the value if it's a function
         */
 
-       function utilFunctor(value) {
-         if (typeof value === 'function') return value;
-         return function () {
-           return value;
-         };
-       }
-       function utilNoAuto(selection) {
-         var isText = selection.size() && selection.node().tagName.toLowerCase() === 'textarea';
-         return selection // assign 'new-password' even for non-password fields to prevent browsers (Chrome) ignoring 'off'
-         .attr('autocomplete', 'new-password').attr('autocorrect', 'off').attr('autocapitalize', 'off').attr('spellcheck', isText ? 'true' : 'false');
-       } // https://stackoverflow.com/questions/194846/is-there-any-kind-of-hash-code-function-in-javascript
-       // https://werxltd.com/wp/2010/05/13/javascript-implementation-of-javas-string-hashcode-method/
 
-       function utilHashcode(str) {
-         var hash = 0;
+       var isInBbox = function isInBbox(bbox, point) {
+         return bbox.ll.x <= point.x && point.x <= bbox.ur.x && bbox.ll.y <= point.y && point.y <= bbox.ur.y;
+       };
+       /* Returns either null, or a bbox (aka an ordered pair of points)
+        * If there is only one point of overlap, a bbox with identical points
+        * will be returned */
 
-         if (str.length === 0) {
-           return hash;
-         }
 
-         for (var i = 0; i < str.length; i++) {
-           var _char = str.charCodeAt(i);
+       var getBboxOverlap = function getBboxOverlap(b1, b2) {
+         // check if the bboxes overlap at all
+         if (b2.ur.x < b1.ll.x || b1.ur.x < b2.ll.x || b2.ur.y < b1.ll.y || b1.ur.y < b2.ll.y) return null; // find the middle two X values
 
-           hash = (hash << 5) - hash + _char;
-           hash = hash & hash; // Convert to 32bit integer
-         }
+         var lowerX = b1.ll.x < b2.ll.x ? b2.ll.x : b1.ll.x;
+         var upperX = b1.ur.x < b2.ur.x ? b1.ur.x : b2.ur.x; // find the middle two Y values
 
-         return hash;
-       } // Returns version of `str` with all runs of special characters replaced by `_`;
-       // suitable for HTML ids, classes, selectors, etc.
+         var lowerY = b1.ll.y < b2.ll.y ? b2.ll.y : b1.ll.y;
+         var upperY = b1.ur.y < b2.ur.y ? b1.ur.y : b2.ur.y; // put those middle values together to get the overlap
 
-       function utilSafeClassName(str) {
-         return str.toLowerCase().replace(/[^a-z0-9]+/g, '_');
-       } // Returns string based on `val` that is highly unlikely to collide with an id
-       // used previously or that's present elsewhere in the document. Useful for preventing
-       // browser-provided autofills or when embedding iD on pages with unknown elements.
+         return {
+           ll: {
+             x: lowerX,
+             y: lowerY
+           },
+           ur: {
+             x: upperX,
+             y: upperY
+           }
+         };
+       };
+       /* Javascript doesn't do integer math. Everything is
+        * floating point with percision Number.EPSILON.
+        *
+        * https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/EPSILON
+        */
 
-       function utilUniqueDomId(val) {
-         return 'ideditor-' + utilSafeClassName(val.toString()) + '-' + new Date().getTime().toString();
-       } // Returns the length of `str` in unicode characters. This can be less than
-       // `String.length()` since a single unicode character can be composed of multiple
-       // JavaScript UTF-16 code units.
 
-       function utilUnicodeCharsCount(str) {
-         // Native ES2015 implementations of `Array.from` split strings into unicode characters
-         return Array.from(str).length;
-       } // Returns a new string representing `str` cut from its start to `limit` length
-       // in unicode characters. Note that this runs the risk of splitting graphemes.
+       var epsilon = Number.EPSILON; // IE Polyfill
 
-       function utilUnicodeCharsTruncated(str, limit) {
-         return Array.from(str).slice(0, limit).join('');
-       }
+       if (epsilon === undefined) epsilon = Math.pow(2, -52);
+       var EPSILON_SQ = epsilon * epsilon;
+       /* FLP comparator */
 
-       function osmEntity(attrs) {
-         // For prototypal inheritance.
-         if (this instanceof osmEntity) return; // Create the appropriate subtype.
+       var cmp = function cmp(a, b) {
+         // check if they're both 0
+         if (-epsilon < a && a < epsilon) {
+           if (-epsilon < b && b < epsilon) {
+             return 0;
+           }
+         } // check if they're flp equal
 
-         if (attrs && attrs.type) {
-           return osmEntity[attrs.type].apply(this, arguments);
-         } else if (attrs && attrs.id) {
-           return osmEntity[osmEntity.id.type(attrs.id)].apply(this, arguments);
-         } // Initialize a generic Entity (used only in tests).
 
+         var ab = a - b;
 
-         return new osmEntity().initialize(arguments);
-       }
+         if (ab * ab < EPSILON_SQ * a * b) {
+           return 0;
+         } // normal comparison
 
-       osmEntity.id = function (type) {
-         return osmEntity.id.fromOSM(type, osmEntity.id.next[type]--);
-       };
 
-       osmEntity.id.next = {
-         changeset: -1,
-         node: -1,
-         way: -1,
-         relation: -1
+         return a < b ? -1 : 1;
        };
+       /**
+        * This class rounds incoming values sufficiently so that
+        * floating points problems are, for the most part, avoided.
+        *
+        * Incoming points are have their x & y values tested against
+        * all previously seen x & y values. If either is 'too close'
+        * to a previously seen value, it's value is 'snapped' to the
+        * previously seen value.
+        *
+        * All points should be rounded by this class before being
+        * stored in any data structures in the rest of this algorithm.
+        */
 
-       osmEntity.id.fromOSM = function (type, id) {
-         return type[0] + id;
-       };
 
-       osmEntity.id.toOSM = function (id) {
-         return id.slice(1);
-       };
+       var PtRounder = /*#__PURE__*/function () {
+         function PtRounder() {
+           _classCallCheck(this, PtRounder);
 
-       osmEntity.id.type = function (id) {
-         return {
-           'c': 'changeset',
-           'n': 'node',
-           'w': 'way',
-           'r': 'relation'
-         }[id[0]];
-       }; // A function suitable for use as the second argument to d3.selection#data().
+           this.reset();
+         }
 
+         _createClass(PtRounder, [{
+           key: "reset",
+           value: function reset() {
+             this.xRounder = new CoordRounder();
+             this.yRounder = new CoordRounder();
+           }
+         }, {
+           key: "round",
+           value: function round(x, y) {
+             return {
+               x: this.xRounder.round(x),
+               y: this.yRounder.round(y)
+             };
+           }
+         }]);
 
-       osmEntity.key = function (entity) {
-         return entity.id + 'v' + (entity.v || 0);
-       };
+         return PtRounder;
+       }();
 
-       var _deprecatedTagValuesByKey;
+       var CoordRounder = /*#__PURE__*/function () {
+         function CoordRounder() {
+           _classCallCheck(this, CoordRounder);
 
-       osmEntity.deprecatedTagValuesByKey = function (dataDeprecated) {
-         if (!_deprecatedTagValuesByKey) {
-           _deprecatedTagValuesByKey = {};
-           dataDeprecated.forEach(function (d) {
-             var oldKeys = Object.keys(d.old);
+           this.tree = new Tree(); // preseed with 0 so we don't end up with values < Number.EPSILON
 
-             if (oldKeys.length === 1) {
-               var oldKey = oldKeys[0];
-               var oldValue = d.old[oldKey];
+           this.round(0);
+         } // Note: this can rounds input values backwards or forwards.
+         //       You might ask, why not restrict this to just rounding
+         //       forwards? Wouldn't that allow left endpoints to always
+         //       remain left endpoints during splitting (never change to
+         //       right). No - it wouldn't, because we snap intersections
+         //       to endpoints (to establish independence from the segment
+         //       angle for t-intersections).
 
-               if (oldValue !== '*') {
-                 if (!_deprecatedTagValuesByKey[oldKey]) {
-                   _deprecatedTagValuesByKey[oldKey] = [oldValue];
-                 } else {
-                   _deprecatedTagValuesByKey[oldKey].push(oldValue);
-                 }
-               }
-             }
-           });
-         }
 
-         return _deprecatedTagValuesByKey;
-       };
+         _createClass(CoordRounder, [{
+           key: "round",
+           value: function round(coord) {
+             var node = this.tree.add(coord);
+             var prevNode = this.tree.prev(node);
 
-       osmEntity.prototype = {
-         tags: {},
-         initialize: function initialize(sources) {
-           for (var i = 0; i < sources.length; ++i) {
-             var source = sources[i];
+             if (prevNode !== null && cmp(node.key, prevNode.key) === 0) {
+               this.tree.remove(coord);
+               return prevNode.key;
+             }
 
-             for (var prop in source) {
-               if (Object.prototype.hasOwnProperty.call(source, prop)) {
-                 if (source[prop] === undefined) {
-                   delete this[prop];
-                 } else {
-                   this[prop] = source[prop];
-                 }
-               }
+             var nextNode = this.tree.next(node);
+
+             if (nextNode !== null && cmp(node.key, nextNode.key) === 0) {
+               this.tree.remove(coord);
+               return nextNode.key;
              }
-           }
 
-           if (!this.id && this.type) {
-             this.id = osmEntity.id(this.type);
+             return coord;
            }
+         }]);
 
-           if (!this.hasOwnProperty('visible')) {
-             this.visible = true;
-           }
+         return CoordRounder;
+       }(); // singleton available by import
 
-           return this;
-         },
-         copy: function copy(resolver, copies) {
-           if (copies[this.id]) return copies[this.id];
-           var copy = osmEntity(this, {
-             id: undefined,
-             user: undefined,
-             version: undefined
-           });
-           copies[this.id] = copy;
-           return copy;
-         },
-         osmId: function osmId() {
-           return osmEntity.id.toOSM(this.id);
-         },
-         isNew: function isNew() {
-           return this.osmId() < 0;
-         },
-         update: function update(attrs) {
-           return osmEntity(this, attrs, {
-             v: 1 + (this.v || 0)
-           });
-         },
-         mergeTags: function mergeTags(tags) {
-           var merged = Object.assign({}, this.tags); // shallow copy
 
-           var changed = false;
+       var rounder = new PtRounder();
+       /* Cross Product of two vectors with first point at origin */
 
-           for (var k in tags) {
-             var t1 = merged[k];
-             var t2 = tags[k];
+       var crossProduct = function crossProduct(a, b) {
+         return a.x * b.y - a.y * b.x;
+       };
+       /* Dot Product of two vectors with first point at origin */
 
-             if (!t1) {
-               changed = true;
-               merged[k] = t2;
-             } else if (t1 !== t2) {
-               changed = true;
-               merged[k] = utilUnicodeCharsTruncated(utilArrayUnion(t1.split(/;\s*/), t2.split(/;\s*/)).join(';'), 255 // avoid exceeding character limit; see also services/osm.js -> maxCharsForTagValue()
-               );
-             }
-           }
 
-           return changed ? this.update({
-             tags: merged
-           }) : this;
-         },
-         intersects: function intersects(extent, resolver) {
-           return this.extent(resolver).intersects(extent);
-         },
-         hasNonGeometryTags: function hasNonGeometryTags() {
-           return Object.keys(this.tags).some(function (k) {
-             return k !== 'area';
-           });
-         },
-         hasParentRelations: function hasParentRelations(resolver) {
-           return resolver.parentRelations(this).length > 0;
-         },
-         hasInterestingTags: function hasInterestingTags() {
-           return Object.keys(this.tags).some(osmIsInterestingTag);
-         },
-         hasWikidata: function hasWikidata() {
-           return !!this.tags.wikidata || !!this.tags['brand:wikidata'];
-         },
-         isHighwayIntersection: function isHighwayIntersection() {
-           return false;
-         },
-         isDegenerate: function isDegenerate() {
-           return true;
-         },
-         deprecatedTags: function deprecatedTags(dataDeprecated) {
-           var tags = this.tags; // if there are no tags, none can be deprecated
+       var dotProduct = function dotProduct(a, b) {
+         return a.x * b.x + a.y * b.y;
+       };
+       /* Comparator for two vectors with same starting point */
 
-           if (Object.keys(tags).length === 0) return [];
-           var deprecated = [];
-           dataDeprecated.forEach(function (d) {
-             var oldKeys = Object.keys(d.old);
 
-             if (d.replace) {
-               var hasExistingValues = Object.keys(d.replace).some(function (replaceKey) {
-                 if (!tags[replaceKey] || d.old[replaceKey]) return false;
-                 var replaceValue = d.replace[replaceKey];
-                 if (replaceValue === '*') return false;
-                 if (replaceValue === tags[replaceKey]) return false;
-                 return true;
-               }); // don't flag deprecated tags if the upgrade path would overwrite existing data - #7843
+       var compareVectorAngles = function compareVectorAngles(basePt, endPt1, endPt2) {
+         var v1 = {
+           x: endPt1.x - basePt.x,
+           y: endPt1.y - basePt.y
+         };
+         var v2 = {
+           x: endPt2.x - basePt.x,
+           y: endPt2.y - basePt.y
+         };
+         var kross = crossProduct(v1, v2);
+         return cmp(kross, 0);
+       };
 
-               if (hasExistingValues) return;
-             }
+       var length = function length(v) {
+         return Math.sqrt(dotProduct(v, v));
+       };
+       /* Get the sine of the angle from pShared -> pAngle to pShaed -> pBase */
 
-             var matchesDeprecatedTags = oldKeys.every(function (oldKey) {
-               if (!tags[oldKey]) return false;
-               if (d.old[oldKey] === '*') return true;
-               if (d.old[oldKey] === tags[oldKey]) return true;
-               var vals = tags[oldKey].split(';').filter(Boolean);
 
-               if (vals.length === 0) {
-                 return false;
-               } else if (vals.length > 1) {
-                 return vals.indexOf(d.old[oldKey]) !== -1;
-               } else {
-                 if (tags[oldKey] === d.old[oldKey]) {
-                   if (d.replace && d.old[oldKey] === d.replace[oldKey]) {
-                     var replaceKeys = Object.keys(d.replace);
-                     return !replaceKeys.every(function (replaceKey) {
-                       return tags[replaceKey] === d.replace[replaceKey];
-                     });
-                   } else {
-                     return true;
-                   }
-                 }
-               }
+       var sineOfAngle = function sineOfAngle(pShared, pBase, pAngle) {
+         var vBase = {
+           x: pBase.x - pShared.x,
+           y: pBase.y - pShared.y
+         };
+         var vAngle = {
+           x: pAngle.x - pShared.x,
+           y: pAngle.y - pShared.y
+         };
+         return crossProduct(vAngle, vBase) / length(vAngle) / length(vBase);
+       };
+       /* Get the cosine of the angle from pShared -> pAngle to pShaed -> pBase */
 
-               return false;
-             });
 
-             if (matchesDeprecatedTags) {
-               deprecated.push(d);
-             }
-           });
-           return deprecated;
-         }
+       var cosineOfAngle = function cosineOfAngle(pShared, pBase, pAngle) {
+         var vBase = {
+           x: pBase.x - pShared.x,
+           y: pBase.y - pShared.y
+         };
+         var vAngle = {
+           x: pAngle.x - pShared.x,
+           y: pAngle.y - pShared.y
+         };
+         return dotProduct(vAngle, vBase) / length(vAngle) / length(vBase);
        };
+       /* Get the x coordinate where the given line (defined by a point and vector)
+        * crosses the horizontal line with the given y coordiante.
+        * In the case of parrallel lines (including overlapping ones) returns null. */
 
-       function osmLanes(entity) {
-         if (entity.type !== 'way') return null;
-         if (!entity.tags.highway) return null;
-         var tags = entity.tags;
-         var isOneWay = entity.isOneWay();
-         var laneCount = getLaneCount(tags, isOneWay);
-         var maxspeed = parseMaxspeed(tags);
-         var laneDirections = parseLaneDirections(tags, isOneWay, laneCount);
-         var forward = laneDirections.forward;
-         var backward = laneDirections.backward;
-         var bothways = laneDirections.bothways; // parse the piped string 'x|y|z' format
 
-         var turnLanes = {};
-         turnLanes.unspecified = parseTurnLanes(tags['turn:lanes']);
-         turnLanes.forward = parseTurnLanes(tags['turn:lanes:forward']);
-         turnLanes.backward = parseTurnLanes(tags['turn:lanes:backward']);
-         var maxspeedLanes = {};
-         maxspeedLanes.unspecified = parseMaxspeedLanes(tags['maxspeed:lanes'], maxspeed);
-         maxspeedLanes.forward = parseMaxspeedLanes(tags['maxspeed:lanes:forward'], maxspeed);
-         maxspeedLanes.backward = parseMaxspeedLanes(tags['maxspeed:lanes:backward'], maxspeed);
-         var psvLanes = {};
-         psvLanes.unspecified = parseMiscLanes(tags['psv:lanes']);
-         psvLanes.forward = parseMiscLanes(tags['psv:lanes:forward']);
-         psvLanes.backward = parseMiscLanes(tags['psv:lanes:backward']);
-         var busLanes = {};
-         busLanes.unspecified = parseMiscLanes(tags['bus:lanes']);
-         busLanes.forward = parseMiscLanes(tags['bus:lanes:forward']);
-         busLanes.backward = parseMiscLanes(tags['bus:lanes:backward']);
-         var taxiLanes = {};
-         taxiLanes.unspecified = parseMiscLanes(tags['taxi:lanes']);
-         taxiLanes.forward = parseMiscLanes(tags['taxi:lanes:forward']);
-         taxiLanes.backward = parseMiscLanes(tags['taxi:lanes:backward']);
-         var hovLanes = {};
-         hovLanes.unspecified = parseMiscLanes(tags['hov:lanes']);
-         hovLanes.forward = parseMiscLanes(tags['hov:lanes:forward']);
-         hovLanes.backward = parseMiscLanes(tags['hov:lanes:backward']);
-         var hgvLanes = {};
-         hgvLanes.unspecified = parseMiscLanes(tags['hgv:lanes']);
-         hgvLanes.forward = parseMiscLanes(tags['hgv:lanes:forward']);
-         hgvLanes.backward = parseMiscLanes(tags['hgv:lanes:backward']);
-         var bicyclewayLanes = {};
-         bicyclewayLanes.unspecified = parseBicycleWay(tags['bicycleway:lanes']);
-         bicyclewayLanes.forward = parseBicycleWay(tags['bicycleway:lanes:forward']);
-         bicyclewayLanes.backward = parseBicycleWay(tags['bicycleway:lanes:backward']);
-         var lanesObj = {
-           forward: [],
-           backward: [],
-           unspecified: []
-         }; // map forward/backward/unspecified of each lane type to lanesObj
+       var horizontalIntersection = function horizontalIntersection(pt, v, y) {
+         if (v.y === 0) return null;
+         return {
+           x: pt.x + v.x / v.y * (y - pt.y),
+           y: y
+         };
+       };
+       /* Get the y coordinate where the given line (defined by a point and vector)
+        * crosses the vertical line with the given x coordiante.
+        * In the case of parrallel lines (including overlapping ones) returns null. */
 
-         mapToLanesObj(lanesObj, turnLanes, 'turnLane');
-         mapToLanesObj(lanesObj, maxspeedLanes, 'maxspeed');
-         mapToLanesObj(lanesObj, psvLanes, 'psv');
-         mapToLanesObj(lanesObj, busLanes, 'bus');
-         mapToLanesObj(lanesObj, taxiLanes, 'taxi');
-         mapToLanesObj(lanesObj, hovLanes, 'hov');
-         mapToLanesObj(lanesObj, hgvLanes, 'hgv');
-         mapToLanesObj(lanesObj, bicyclewayLanes, 'bicycleway');
+
+       var verticalIntersection = function verticalIntersection(pt, v, x) {
+         if (v.x === 0) return null;
          return {
-           metadata: {
-             count: laneCount,
-             oneway: isOneWay,
-             forward: forward,
-             backward: backward,
-             bothways: bothways,
-             turnLanes: turnLanes,
-             maxspeed: maxspeed,
-             maxspeedLanes: maxspeedLanes,
-             psvLanes: psvLanes,
-             busLanes: busLanes,
-             taxiLanes: taxiLanes,
-             hovLanes: hovLanes,
-             hgvLanes: hgvLanes,
-             bicyclewayLanes: bicyclewayLanes
-           },
-           lanes: lanesObj
+           x: x,
+           y: pt.y + v.y / v.x * (x - pt.x)
+         };
+       };
+       /* Get the intersection of two lines, each defined by a base point and a vector.
+        * In the case of parrallel lines (including overlapping ones) returns null. */
+
+
+       var intersection = function intersection(pt1, v1, pt2, v2) {
+         // take some shortcuts for vertical and horizontal lines
+         // this also ensures we don't calculate an intersection and then discover
+         // it's actually outside the bounding box of the line
+         if (v1.x === 0) return verticalIntersection(pt2, v2, pt1.x);
+         if (v2.x === 0) return verticalIntersection(pt1, v1, pt2.x);
+         if (v1.y === 0) return horizontalIntersection(pt2, v2, pt1.y);
+         if (v2.y === 0) return horizontalIntersection(pt1, v1, pt2.y); // General case for non-overlapping segments.
+         // This algorithm is based on Schneider and Eberly.
+         // http://www.cimec.org.ar/~ncalvo/Schneider_Eberly.pdf - pg 244
+
+         var kross = crossProduct(v1, v2);
+         if (kross == 0) return null;
+         var ve = {
+           x: pt2.x - pt1.x,
+           y: pt2.y - pt1.y
+         };
+         var d1 = crossProduct(ve, v1) / kross;
+         var d2 = crossProduct(ve, v2) / kross; // take the average of the two calculations to minimize rounding error
+
+         var x1 = pt1.x + d2 * v1.x,
+             x2 = pt2.x + d1 * v2.x;
+         var y1 = pt1.y + d2 * v1.y,
+             y2 = pt2.y + d1 * v2.y;
+         var x = (x1 + x2) / 2;
+         var y = (y1 + y2) / 2;
+         return {
+           x: x,
+           y: y
          };
-       }
-
-       function getLaneCount(tags, isOneWay) {
-         var count;
-
-         if (tags.lanes) {
-           count = parseInt(tags.lanes, 10);
-
-           if (count > 0) {
-             return count;
-           }
-         }
+       };
 
-         switch (tags.highway) {
-           case 'trunk':
-           case 'motorway':
-             count = isOneWay ? 2 : 4;
-             break;
+       var SweepEvent = /*#__PURE__*/function () {
+         _createClass(SweepEvent, null, [{
+           key: "compare",
+           // for ordering sweep events in the sweep event queue
+           value: function compare(a, b) {
+             // favor event with a point that the sweep line hits first
+             var ptCmp = SweepEvent.comparePoints(a.point, b.point);
+             if (ptCmp !== 0) return ptCmp; // the points are the same, so link them if needed
 
-           default:
-             count = isOneWay ? 1 : 2;
-             break;
-         }
+             if (a.point !== b.point) a.link(b); // favor right events over left
 
-         return count;
-       }
+             if (a.isLeft !== b.isLeft) return a.isLeft ? 1 : -1; // we have two matching left or right endpoints
+             // ordering of this case is the same as for their segments
 
-       function parseMaxspeed(tags) {
-         var maxspeed = tags.maxspeed;
-         if (!maxspeed) return;
-         var maxspeedRegex = /^([0-9][\.0-9]+?)(?:[ ]?(?:km\/h|kmh|kph|mph|knots))?$/;
-         if (!maxspeedRegex.test(maxspeed)) return;
-         return parseInt(maxspeed, 10);
-       }
+             return Segment.compare(a.segment, b.segment);
+           } // for ordering points in sweep line order
 
-       function parseLaneDirections(tags, isOneWay, laneCount) {
-         var forward = parseInt(tags['lanes:forward'], 10);
-         var backward = parseInt(tags['lanes:backward'], 10);
-         var bothways = parseInt(tags['lanes:both_ways'], 10) > 0 ? 1 : 0;
+         }, {
+           key: "comparePoints",
+           value: function comparePoints(aPt, bPt) {
+             if (aPt.x < bPt.x) return -1;
+             if (aPt.x > bPt.x) return 1;
+             if (aPt.y < bPt.y) return -1;
+             if (aPt.y > bPt.y) return 1;
+             return 0;
+           } // Warning: 'point' input will be modified and re-used (for performance)
 
-         if (parseInt(tags.oneway, 10) === -1) {
-           forward = 0;
-           bothways = 0;
-           backward = laneCount;
-         } else if (isOneWay) {
-           forward = laneCount;
-           bothways = 0;
-           backward = 0;
-         } else if (isNaN(forward) && isNaN(backward)) {
-           backward = Math.floor((laneCount - bothways) / 2);
-           forward = laneCount - bothways - backward;
-         } else if (isNaN(forward)) {
-           if (backward > laneCount - bothways) {
-             backward = laneCount - bothways;
-           }
+         }]);
 
-           forward = laneCount - bothways - backward;
-         } else if (isNaN(backward)) {
-           if (forward > laneCount - bothways) {
-             forward = laneCount - bothways;
-           }
+         function SweepEvent(point, isLeft) {
+           _classCallCheck(this, SweepEvent);
 
-           backward = laneCount - bothways - forward;
+           if (point.events === undefined) point.events = [this];else point.events.push(this);
+           this.point = point;
+           this.isLeft = isLeft; // this.segment, this.otherSE set by factory
          }
 
-         return {
-           forward: forward,
-           backward: backward,
-           bothways: bothways
-         };
-       }
+         _createClass(SweepEvent, [{
+           key: "link",
+           value: function link(other) {
+             if (other.point === this.point) {
+               throw new Error('Tried to link already linked events');
+             }
 
-       function parseTurnLanes(tag) {
-         if (!tag) return;
-         var validValues = ['left', 'slight_left', 'sharp_left', 'through', 'right', 'slight_right', 'sharp_right', 'reverse', 'merge_to_left', 'merge_to_right', 'none'];
-         return tag.split('|').map(function (s) {
-           if (s === '') s = 'none';
-           return s.split(';').map(function (d) {
-             return validValues.indexOf(d) === -1 ? 'unknown' : d;
-           });
-         });
-       }
+             var otherEvents = other.point.events;
 
-       function parseMaxspeedLanes(tag, maxspeed) {
-         if (!tag) return;
-         return tag.split('|').map(function (s) {
-           if (s === 'none') return s;
-           var m = parseInt(s, 10);
-           if (s === '' || m === maxspeed) return null;
-           return isNaN(m) ? 'unknown' : m;
-         });
-       }
+             for (var i = 0, iMax = otherEvents.length; i < iMax; i++) {
+               var evt = otherEvents[i];
+               this.point.events.push(evt);
+               evt.point = this.point;
+             }
 
-       function parseMiscLanes(tag) {
-         if (!tag) return;
-         var validValues = ['yes', 'no', 'designated'];
-         return tag.split('|').map(function (s) {
-           if (s === '') s = 'no';
-           return validValues.indexOf(s) === -1 ? 'unknown' : s;
-         });
-       }
+             this.checkForConsuming();
+           }
+           /* Do a pass over our linked events and check to see if any pair
+            * of segments match, and should be consumed. */
 
-       function parseBicycleWay(tag) {
-         if (!tag) return;
-         var validValues = ['yes', 'no', 'designated', 'lane'];
-         return tag.split('|').map(function (s) {
-           if (s === '') s = 'no';
-           return validValues.indexOf(s) === -1 ? 'unknown' : s;
-         });
-       }
+         }, {
+           key: "checkForConsuming",
+           value: function checkForConsuming() {
+             // FIXME: The loops in this method run O(n^2) => no good.
+             //        Maintain little ordered sweep event trees?
+             //        Can we maintaining an ordering that avoids the need
+             //        for the re-sorting with getLeftmostComparator in geom-out?
+             // Compare each pair of events to see if other events also match
+             var numEvents = this.point.events.length;
 
-       function mapToLanesObj(lanesObj, data, key) {
-         if (data.forward) data.forward.forEach(function (l, i) {
-           if (!lanesObj.forward[i]) lanesObj.forward[i] = {};
-           lanesObj.forward[i][key] = l;
-         });
-         if (data.backward) data.backward.forEach(function (l, i) {
-           if (!lanesObj.backward[i]) lanesObj.backward[i] = {};
-           lanesObj.backward[i][key] = l;
-         });
-         if (data.unspecified) data.unspecified.forEach(function (l, i) {
-           if (!lanesObj.unspecified[i]) lanesObj.unspecified[i] = {};
-           lanesObj.unspecified[i][key] = l;
-         });
-       }
+             for (var i = 0; i < numEvents; i++) {
+               var evt1 = this.point.events[i];
+               if (evt1.segment.consumedBy !== undefined) continue;
 
-       function osmWay() {
-         if (!(this instanceof osmWay)) {
-           return new osmWay().initialize(arguments);
-         } else if (arguments.length) {
-           this.initialize(arguments);
-         }
-       }
-       osmEntity.way = osmWay;
-       osmWay.prototype = Object.create(osmEntity.prototype);
-       Object.assign(osmWay.prototype, {
-         type: 'way',
-         nodes: [],
-         copy: function copy(resolver, copies) {
-           if (copies[this.id]) return copies[this.id];
-           var copy = osmEntity.prototype.copy.call(this, resolver, copies);
-           var nodes = this.nodes.map(function (id) {
-             return resolver.entity(id).copy(resolver, copies).id;
-           });
-           copy = copy.update({
-             nodes: nodes
-           });
-           copies[this.id] = copy;
-           return copy;
-         },
-         extent: function extent(resolver) {
-           return resolver["transient"](this, 'extent', function () {
-             var extent = geoExtent();
+               for (var j = i + 1; j < numEvents; j++) {
+                 var evt2 = this.point.events[j];
+                 if (evt2.consumedBy !== undefined) continue;
+                 if (evt1.otherSE.point.events !== evt2.otherSE.point.events) continue;
+                 evt1.segment.consume(evt2.segment);
+               }
+             }
+           }
+         }, {
+           key: "getAvailableLinkedEvents",
+           value: function getAvailableLinkedEvents() {
+             // point.events is always of length 2 or greater
+             var events = [];
 
-             for (var i = 0; i < this.nodes.length; i++) {
-               var node = resolver.hasEntity(this.nodes[i]);
+             for (var i = 0, iMax = this.point.events.length; i < iMax; i++) {
+               var evt = this.point.events[i];
 
-               if (node) {
-                 extent._extend(node.extent());
+               if (evt !== this && !evt.segment.ringOut && evt.segment.isInResult()) {
+                 events.push(evt);
                }
              }
 
-             return extent;
-           });
-         },
-         first: function first() {
-           return this.nodes[0];
-         },
-         last: function last() {
-           return this.nodes[this.nodes.length - 1];
-         },
-         contains: function contains(node) {
-           return this.nodes.indexOf(node) >= 0;
-         },
-         affix: function affix(node) {
-           if (this.nodes[0] === node) return 'prefix';
-           if (this.nodes[this.nodes.length - 1] === node) return 'suffix';
-         },
-         layer: function layer() {
-           // explicit layer tag, clamp between -10, 10..
-           if (isFinite(this.tags.layer)) {
-             return Math.max(-10, Math.min(+this.tags.layer, 10));
-           } // implied layer tag..
+             return events;
+           }
+           /**
+            * Returns a comparator function for sorting linked events that will
+            * favor the event that will give us the smallest left-side angle.
+            * All ring construction starts as low as possible heading to the right,
+            * so by always turning left as sharp as possible we'll get polygons
+            * without uncessary loops & holes.
+            *
+            * The comparator function has a compute cache such that it avoids
+            * re-computing already-computed values.
+            */
 
+         }, {
+           key: "getLeftmostComparator",
+           value: function getLeftmostComparator(baseEvent) {
+             var _this = this;
 
-           if (this.tags.covered === 'yes') return -1;
-           if (this.tags.location === 'overground') return 1;
-           if (this.tags.location === 'underground') return -1;
-           if (this.tags.location === 'underwater') return -10;
-           if (this.tags.power === 'line') return 10;
-           if (this.tags.power === 'minor_line') return 10;
-           if (this.tags.aerialway) return 10;
-           if (this.tags.bridge) return 1;
-           if (this.tags.cutting) return -1;
-           if (this.tags.tunnel) return -1;
-           if (this.tags.waterway) return -1;
-           if (this.tags.man_made === 'pipeline') return -10;
-           if (this.tags.boundary) return -10;
-           return 0;
-         },
-         // the approximate width of the line based on its tags except its `width` tag
-         impliedLineWidthMeters: function impliedLineWidthMeters() {
-           var averageWidths = {
-             highway: {
-               // width is for single lane
-               motorway: 5,
-               motorway_link: 5,
-               trunk: 4.5,
-               trunk_link: 4.5,
-               primary: 4,
-               secondary: 4,
-               tertiary: 4,
-               primary_link: 4,
-               secondary_link: 4,
-               tertiary_link: 4,
-               unclassified: 4,
-               road: 4,
-               living_street: 4,
-               bus_guideway: 4,
-               pedestrian: 4,
-               residential: 3.5,
-               service: 3.5,
-               track: 3,
-               cycleway: 2.5,
-               bridleway: 2,
-               corridor: 2,
-               steps: 2,
-               path: 1.5,
-               footway: 1.5
-             },
-             railway: {
-               // width includes ties and rail bed, not just track gauge
-               rail: 2.5,
-               light_rail: 2.5,
-               tram: 2.5,
-               subway: 2.5,
-               monorail: 2.5,
-               funicular: 2.5,
-               disused: 2.5,
-               preserved: 2.5,
-               miniature: 1.5,
-               narrow_gauge: 1.5
-             },
-             waterway: {
-               river: 50,
-               canal: 25,
-               stream: 5,
-               tidal_channel: 5,
-               fish_pass: 2.5,
-               drain: 2.5,
-               ditch: 1.5
-             }
-           };
+             var cache = new Map();
 
-           for (var key in averageWidths) {
-             if (this.tags[key] && averageWidths[key][this.tags[key]]) {
-               var width = averageWidths[key][this.tags[key]];
+             var fillCache = function fillCache(linkedEvent) {
+               var nextEvent = linkedEvent.otherSE;
+               cache.set(linkedEvent, {
+                 sine: sineOfAngle(_this.point, baseEvent.point, nextEvent.point),
+                 cosine: cosineOfAngle(_this.point, baseEvent.point, nextEvent.point)
+               });
+             };
 
-               if (key === 'highway') {
-                 var laneCount = this.tags.lanes && parseInt(this.tags.lanes, 10);
-                 if (!laneCount) laneCount = this.isOneWay() ? 1 : 2;
-                 return width * laneCount;
-               }
+             return function (a, b) {
+               if (!cache.has(a)) fillCache(a);
+               if (!cache.has(b)) fillCache(b);
 
-               return width;
-             }
-           }
+               var _cache$get = cache.get(a),
+                   asine = _cache$get.sine,
+                   acosine = _cache$get.cosine;
 
-           return null;
-         },
-         isOneWay: function isOneWay() {
-           // explicit oneway tag..
-           var values = {
-             'yes': true,
-             '1': true,
-             '-1': true,
-             'reversible': true,
-             'alternating': true,
-             'no': false,
-             '0': false
-           };
+               var _cache$get2 = cache.get(b),
+                   bsine = _cache$get2.sine,
+                   bcosine = _cache$get2.cosine; // both on or above x-axis
 
-           if (values[this.tags.oneway] !== undefined) {
-             return values[this.tags.oneway];
-           } // implied oneway tag..
 
+               if (asine >= 0 && bsine >= 0) {
+                 if (acosine < bcosine) return 1;
+                 if (acosine > bcosine) return -1;
+                 return 0;
+               } // both below x-axis
 
-           for (var key in this.tags) {
-             if (key in osmOneWayTags && this.tags[key] in osmOneWayTags[key]) return true;
-           }
 
-           return false;
-         },
-         // Some identifier for tag that implies that this way is "sided",
-         // i.e. the right side is the 'inside' (e.g. the right side of a
-         // natural=cliff is lower).
-         sidednessIdentifier: function sidednessIdentifier() {
-           for (var key in this.tags) {
-             var value = this.tags[key];
+               if (asine < 0 && bsine < 0) {
+                 if (acosine < bcosine) return -1;
+                 if (acosine > bcosine) return 1;
+                 return 0;
+               } // one above x-axis, one below
 
-             if (key in osmRightSideIsInsideTags && value in osmRightSideIsInsideTags[key]) {
-               if (osmRightSideIsInsideTags[key][value] === true) {
-                 return key;
-               } else {
-                 // if the map's value is something other than a
-                 // literal true, we should use it so we can
-                 // special case some keys (e.g. natural=coastline
-                 // is handled differently to other naturals).
-                 return osmRightSideIsInsideTags[key][value];
-               }
-             }
-           }
 
-           return null;
-         },
-         isSided: function isSided() {
-           if (this.tags.two_sided === 'yes') {
-             return false;
+               if (bsine < asine) return -1;
+               if (bsine > asine) return 1;
+               return 0;
+             };
            }
+         }]);
 
-           return this.sidednessIdentifier() !== null;
-         },
-         lanes: function lanes() {
-           return osmLanes(this);
-         },
-         isClosed: function isClosed() {
-           return this.nodes.length > 1 && this.first() === this.last();
-         },
-         isConvex: function isConvex(resolver) {
-           if (!this.isClosed() || this.isDegenerate()) return null;
-           var nodes = utilArrayUniq(resolver.childNodes(this));
-           var coords = nodes.map(function (n) {
-             return n.loc;
-           });
-           var curr = 0;
-           var prev = 0;
+         return SweepEvent;
+       }(); // segments and sweep events when all else is identical
 
-           for (var i = 0; i < coords.length; i++) {
-             var o = coords[(i + 1) % coords.length];
-             var a = coords[i];
-             var b = coords[(i + 2) % coords.length];
-             var res = geoVecCross(a, b, o);
-             curr = res > 0 ? 1 : res < 0 ? -1 : 0;
 
-             if (curr === 0) {
-               continue;
-             } else if (prev && curr !== prev) {
-               return false;
-             }
+       var segmentId = 0;
 
-             prev = curr;
-           }
+       var Segment = /*#__PURE__*/function () {
+         _createClass(Segment, null, [{
+           key: "compare",
 
-           return true;
-         },
-         // returns an object with the tag that implies this is an area, if any
-         tagSuggestingArea: function tagSuggestingArea() {
-           return osmTagSuggestingArea(this.tags);
-         },
-         isArea: function isArea() {
-           if (this.tags.area === 'yes') return true;
-           if (!this.isClosed() || this.tags.area === 'no') return false;
-           return this.tagSuggestingArea() !== null;
-         },
-         isDegenerate: function isDegenerate() {
-           return new Set(this.nodes).size < (this.isArea() ? 3 : 2);
-         },
-         areAdjacent: function areAdjacent(n1, n2) {
-           for (var i = 0; i < this.nodes.length; i++) {
-             if (this.nodes[i] === n1) {
-               if (this.nodes[i - 1] === n2) return true;
-               if (this.nodes[i + 1] === n2) return true;
-             }
-           }
+           /* This compare() function is for ordering segments in the sweep
+            * line tree, and does so according to the following criteria:
+            *
+            * Consider the vertical line that lies an infinestimal step to the
+            * right of the right-more of the two left endpoints of the input
+            * segments. Imagine slowly moving a point up from negative infinity
+            * in the increasing y direction. Which of the two segments will that
+            * point intersect first? That segment comes 'before' the other one.
+            *
+            * If neither segment would be intersected by such a line, (if one
+            * or more of the segments are vertical) then the line to be considered
+            * is directly on the right-more of the two left inputs.
+            */
+           value: function compare(a, b) {
+             var alx = a.leftSE.point.x;
+             var blx = b.leftSE.point.x;
+             var arx = a.rightSE.point.x;
+             var brx = b.rightSE.point.x; // check if they're even in the same vertical plane
 
-           return false;
-         },
-         geometry: function geometry(graph) {
-           return graph["transient"](this, 'geometry', function () {
-             return this.isArea() ? 'area' : 'line';
-           });
-         },
-         // returns an array of objects representing the segments between the nodes in this way
-         segments: function segments(graph) {
-           function segmentExtent(graph) {
-             var n1 = graph.hasEntity(this.nodes[0]);
-             var n2 = graph.hasEntity(this.nodes[1]);
-             return n1 && n2 && geoExtent([[Math.min(n1.loc[0], n2.loc[0]), Math.min(n1.loc[1], n2.loc[1])], [Math.max(n1.loc[0], n2.loc[0]), Math.max(n1.loc[1], n2.loc[1])]]);
-           }
+             if (brx < alx) return 1;
+             if (arx < blx) return -1;
+             var aly = a.leftSE.point.y;
+             var bly = b.leftSE.point.y;
+             var ary = a.rightSE.point.y;
+             var bry = b.rightSE.point.y; // is left endpoint of segment B the right-more?
 
-           return graph["transient"](this, 'segments', function () {
-             var segments = [];
+             if (alx < blx) {
+               // are the two segments in the same horizontal plane?
+               if (bly < aly && bly < ary) return 1;
+               if (bly > aly && bly > ary) return -1; // is the B left endpoint colinear to segment A?
 
-             for (var i = 0; i < this.nodes.length - 1; i++) {
-               segments.push({
-                 id: this.id + '-' + i,
-                 wayId: this.id,
-                 index: i,
-                 nodes: [this.nodes[i], this.nodes[i + 1]],
-                 extent: segmentExtent
-               });
-             }
+               var aCmpBLeft = a.comparePoint(b.leftSE.point);
+               if (aCmpBLeft < 0) return 1;
+               if (aCmpBLeft > 0) return -1; // is the A right endpoint colinear to segment B ?
+
+               var bCmpARight = b.comparePoint(a.rightSE.point);
+               if (bCmpARight !== 0) return bCmpARight; // colinear segments, consider the one with left-more
+               // left endpoint to be first (arbitrary?)
+
+               return -1;
+             } // is left endpoint of segment A the right-more?
 
-             return segments;
-           });
-         },
-         // If this way is not closed, append the beginning node to the end of the nodelist to close it.
-         close: function close() {
-           if (this.isClosed() || !this.nodes.length) return this;
-           var nodes = this.nodes.slice();
-           nodes = nodes.filter(noRepeatNodes);
-           nodes.push(nodes[0]);
-           return this.update({
-             nodes: nodes
-           });
-         },
-         // If this way is closed, remove any connector nodes from the end of the nodelist to unclose it.
-         unclose: function unclose() {
-           if (!this.isClosed()) return this;
-           var nodes = this.nodes.slice();
-           var connector = this.first();
-           var i = nodes.length - 1; // remove trailing connectors..
 
-           while (i > 0 && nodes.length > 1 && nodes[i] === connector) {
-             nodes.splice(i, 1);
-             i = nodes.length - 1;
-           }
+             if (alx > blx) {
+               if (aly < bly && aly < bry) return -1;
+               if (aly > bly && aly > bry) return 1; // is the A left endpoint colinear to segment B?
 
-           nodes = nodes.filter(noRepeatNodes);
-           return this.update({
-             nodes: nodes
-           });
-         },
-         // Adds a node (id) in front of the node which is currently at position index.
-         // If index is undefined, the node will be added to the end of the way for linear ways,
-         //   or just before the final connecting node for circular ways.
-         // Consecutive duplicates are eliminated including existing ones.
-         // Circularity is always preserved when adding a node.
-         addNode: function addNode(id, index) {
-           var nodes = this.nodes.slice();
-           var isClosed = this.isClosed();
-           var max = isClosed ? nodes.length - 1 : nodes.length;
+               var bCmpALeft = b.comparePoint(a.leftSE.point);
+               if (bCmpALeft !== 0) return bCmpALeft; // is the B right endpoint colinear to segment A?
 
-           if (index === undefined) {
-             index = max;
-           }
+               var aCmpBRight = a.comparePoint(b.rightSE.point);
+               if (aCmpBRight < 0) return 1;
+               if (aCmpBRight > 0) return -1; // colinear segments, consider the one with left-more
+               // left endpoint to be first (arbitrary?)
 
-           if (index < 0 || index > max) {
-             throw new RangeError('index ' + index + ' out of range 0..' + max);
-           } // If this is a closed way, remove all connector nodes except the first one
-           // (there may be duplicates) and adjust index if necessary..
+               return 1;
+             } // if we get here, the two left endpoints are in the same
+             // vertical plane, ie alx === blx
+             // consider the lower left-endpoint to come first
 
 
-           if (isClosed) {
-             var connector = this.first(); // leading connectors..
+             if (aly < bly) return -1;
+             if (aly > bly) return 1; // left endpoints are identical
+             // check for colinearity by using the left-more right endpoint
+             // is the A right endpoint more left-more?
 
-             var i = 1;
+             if (arx < brx) {
+               var _bCmpARight = b.comparePoint(a.rightSE.point);
 
-             while (i < nodes.length && nodes.length > 2 && nodes[i] === connector) {
-               nodes.splice(i, 1);
-               if (index > i) index--;
-             } // trailing connectors..
+               if (_bCmpARight !== 0) return _bCmpARight;
+             } // is the B right endpoint more left-more?
 
 
-             i = nodes.length - 1;
+             if (arx > brx) {
+               var _aCmpBRight = a.comparePoint(b.rightSE.point);
 
-             while (i > 0 && nodes.length > 1 && nodes[i] === connector) {
-               nodes.splice(i, 1);
-               if (index > i) index--;
-               i = nodes.length - 1;
+               if (_aCmpBRight < 0) return 1;
+               if (_aCmpBRight > 0) return -1;
              }
-           }
 
-           nodes.splice(index, 0, id);
-           nodes = nodes.filter(noRepeatNodes); // If the way was closed before, append a connector node to keep it closed..
+             if (arx !== brx) {
+               // are these two [almost] vertical segments with opposite orientation?
+               // if so, the one with the lower right endpoint comes first
+               var ay = ary - aly;
+               var ax = arx - alx;
+               var by = bry - bly;
+               var bx = brx - blx;
+               if (ay > ax && by < bx) return 1;
+               if (ay < ax && by > bx) return -1;
+             } // we have colinear segments with matching orientation
+             // consider the one with more left-more right endpoint to be first
 
-           if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
-             nodes.push(nodes[0]);
-           }
 
-           return this.update({
-             nodes: nodes
-           });
-         },
-         // Replaces the node which is currently at position index with the given node (id).
-         // Consecutive duplicates are eliminated including existing ones.
-         // Circularity is preserved when updating a node.
-         updateNode: function updateNode(id, index) {
-           var nodes = this.nodes.slice();
-           var isClosed = this.isClosed();
-           var max = nodes.length - 1;
+             if (arx > brx) return 1;
+             if (arx < brx) return -1; // if we get here, two two right endpoints are in the same
+             // vertical plane, ie arx === brx
+             // consider the lower right-endpoint to come first
 
-           if (index === undefined || index < 0 || index > max) {
-             throw new RangeError('index ' + index + ' out of range 0..' + max);
-           } // If this is a closed way, remove all connector nodes except the first one
-           // (there may be duplicates) and adjust index if necessary..
+             if (ary < bry) return -1;
+             if (ary > bry) return 1; // right endpoints identical as well, so the segments are idential
+             // fall back on creation order as consistent tie-breaker
 
+             if (a.id < b.id) return -1;
+             if (a.id > b.id) return 1; // identical segment, ie a === b
 
-           if (isClosed) {
-             var connector = this.first(); // leading connectors..
+             return 0;
+           }
+           /* Warning: a reference to ringWindings input will be stored,
+            *  and possibly will be later modified */
 
-             var i = 1;
+         }]);
 
-             while (i < nodes.length && nodes.length > 2 && nodes[i] === connector) {
-               nodes.splice(i, 1);
-               if (index > i) index--;
-             } // trailing connectors..
+         function Segment(leftSE, rightSE, rings, windings) {
+           _classCallCheck(this, Segment);
 
+           this.id = ++segmentId;
+           this.leftSE = leftSE;
+           leftSE.segment = this;
+           leftSE.otherSE = rightSE;
+           this.rightSE = rightSE;
+           rightSE.segment = this;
+           rightSE.otherSE = leftSE;
+           this.rings = rings;
+           this.windings = windings; // left unset for performance, set later in algorithm
+           // this.ringOut, this.consumedBy, this.prev
+         }
 
-             i = nodes.length - 1;
+         _createClass(Segment, [{
+           key: "replaceRightSE",
 
-             while (i > 0 && nodes.length > 1 && nodes[i] === connector) {
-               nodes.splice(i, 1);
-               if (index === i) index = 0; // update leading connector instead
+           /* When a segment is split, the rightSE is replaced with a new sweep event */
+           value: function replaceRightSE(newRightSE) {
+             this.rightSE = newRightSE;
+             this.rightSE.segment = this;
+             this.rightSE.otherSE = this.leftSE;
+             this.leftSE.otherSE = this.rightSE;
+           }
+         }, {
+           key: "bbox",
+           value: function bbox() {
+             var y1 = this.leftSE.point.y;
+             var y2 = this.rightSE.point.y;
+             return {
+               ll: {
+                 x: this.leftSE.point.x,
+                 y: y1 < y2 ? y1 : y2
+               },
+               ur: {
+                 x: this.rightSE.point.x,
+                 y: y1 > y2 ? y1 : y2
+               }
+             };
+           }
+           /* A vector from the left point to the right */
 
-               i = nodes.length - 1;
-             }
+         }, {
+           key: "vector",
+           value: function vector() {
+             return {
+               x: this.rightSE.point.x - this.leftSE.point.x,
+               y: this.rightSE.point.y - this.leftSE.point.y
+             };
+           }
+         }, {
+           key: "isAnEndpoint",
+           value: function isAnEndpoint(pt) {
+             return pt.x === this.leftSE.point.x && pt.y === this.leftSE.point.y || pt.x === this.rightSE.point.x && pt.y === this.rightSE.point.y;
            }
+           /* Compare this segment with a point.
+            *
+            * A point P is considered to be colinear to a segment if there
+            * exists a distance D such that if we travel along the segment
+            * from one * endpoint towards the other a distance D, we find
+            * ourselves at point P.
+            *
+            * Return value indicates:
+            *
+            *   1: point lies above the segment (to the left of vertical)
+            *   0: point is colinear to segment
+            *  -1: point lies below the segment (to the right of vertical)
+            */
 
-           nodes.splice(index, 1, id);
-           nodes = nodes.filter(noRepeatNodes); // If the way was closed before, append a connector node to keep it closed..
+         }, {
+           key: "comparePoint",
+           value: function comparePoint(point) {
+             if (this.isAnEndpoint(point)) return 0;
+             var lPt = this.leftSE.point;
+             var rPt = this.rightSE.point;
+             var v = this.vector(); // Exactly vertical segments.
 
-           if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
-             nodes.push(nodes[0]);
-           }
+             if (lPt.x === rPt.x) {
+               if (point.x === lPt.x) return 0;
+               return point.x < lPt.x ? 1 : -1;
+             } // Nearly vertical segments with an intersection.
+             // Check to see where a point on the line with matching Y coordinate is.
 
-           return this.update({
-             nodes: nodes
-           });
-         },
-         // Replaces each occurrence of node id needle with replacement.
-         // Consecutive duplicates are eliminated including existing ones.
-         // Circularity is preserved.
-         replaceNode: function replaceNode(needleID, replacementID) {
-           var nodes = this.nodes.slice();
-           var isClosed = this.isClosed();
 
-           for (var i = 0; i < nodes.length; i++) {
-             if (nodes[i] === needleID) {
-               nodes[i] = replacementID;
-             }
+             var yDist = (point.y - lPt.y) / v.y;
+             var xFromYDist = lPt.x + yDist * v.x;
+             if (point.x === xFromYDist) return 0; // General case.
+             // Check to see where a point on the line with matching X coordinate is.
+
+             var xDist = (point.x - lPt.x) / v.x;
+             var yFromXDist = lPt.y + xDist * v.y;
+             if (point.y === yFromXDist) return 0;
+             return point.y < yFromXDist ? -1 : 1;
            }
+           /**
+            * Given another segment, returns the first non-trivial intersection
+            * between the two segments (in terms of sweep line ordering), if it exists.
+            *
+            * A 'non-trivial' intersection is one that will cause one or both of the
+            * segments to be split(). As such, 'trivial' vs. 'non-trivial' intersection:
+            *
+            *   * endpoint of segA with endpoint of segB --> trivial
+            *   * endpoint of segA with point along segB --> non-trivial
+            *   * endpoint of segB with point along segA --> non-trivial
+            *   * point along segA with point along segB --> non-trivial
+            *
+            * If no non-trivial intersection exists, return null
+            * Else, return null.
+            */
 
-           nodes = nodes.filter(noRepeatNodes); // If the way was closed before, append a connector node to keep it closed..
+         }, {
+           key: "getIntersection",
+           value: function getIntersection(other) {
+             // If bboxes don't overlap, there can't be any intersections
+             var tBbox = this.bbox();
+             var oBbox = other.bbox();
+             var bboxOverlap = getBboxOverlap(tBbox, oBbox);
+             if (bboxOverlap === null) return null; // We first check to see if the endpoints can be considered intersections.
+             // This will 'snap' intersections to endpoints if possible, and will
+             // handle cases of colinearity.
 
-           if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
-             nodes.push(nodes[0]);
-           }
+             var tlp = this.leftSE.point;
+             var trp = this.rightSE.point;
+             var olp = other.leftSE.point;
+             var orp = other.rightSE.point; // does each endpoint touch the other segment?
+             // note that we restrict the 'touching' definition to only allow segments
+             // to touch endpoints that lie forward from where we are in the sweep line pass
 
-           return this.update({
-             nodes: nodes
-           });
-         },
-         // Removes each occurrence of node id.
-         // Consecutive duplicates are eliminated including existing ones.
-         // Circularity is preserved.
-         removeNode: function removeNode(id) {
-           var nodes = this.nodes.slice();
-           var isClosed = this.isClosed();
-           nodes = nodes.filter(function (node) {
-             return node !== id;
-           }).filter(noRepeatNodes); // If the way was closed before, append a connector node to keep it closed..
+             var touchesOtherLSE = isInBbox(tBbox, olp) && this.comparePoint(olp) === 0;
+             var touchesThisLSE = isInBbox(oBbox, tlp) && other.comparePoint(tlp) === 0;
+             var touchesOtherRSE = isInBbox(tBbox, orp) && this.comparePoint(orp) === 0;
+             var touchesThisRSE = isInBbox(oBbox, trp) && other.comparePoint(trp) === 0; // do left endpoints match?
 
-           if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
-             nodes.push(nodes[0]);
-           }
+             if (touchesThisLSE && touchesOtherLSE) {
+               // these two cases are for colinear segments with matching left
+               // endpoints, and one segment being longer than the other
+               if (touchesThisRSE && !touchesOtherRSE) return trp;
+               if (!touchesThisRSE && touchesOtherRSE) return orp; // either the two segments match exactly (two trival intersections)
+               // or just on their left endpoint (one trivial intersection
 
-           return this.update({
-             nodes: nodes
-           });
-         },
-         asJXON: function asJXON(changeset_id) {
-           var r = {
-             way: {
-               '@id': this.osmId(),
-               '@version': this.version || 0,
-               nd: this.nodes.map(function (id) {
-                 return {
-                   keyAttributes: {
-                     ref: osmEntity.id.toOSM(id)
-                   }
-                 };
-               }, this),
-               tag: Object.keys(this.tags).map(function (k) {
-                 return {
-                   keyAttributes: {
-                     k: k,
-                     v: this.tags[k]
-                   }
-                 };
-               }, this)
-             }
-           };
+               return null;
+             } // does this left endpoint matches (other doesn't)
 
-           if (changeset_id) {
-             r.way['@changeset'] = changeset_id;
-           }
 
-           return r;
-         },
-         asGeoJSON: function asGeoJSON(resolver) {
-           return resolver["transient"](this, 'GeoJSON', function () {
-             var coordinates = resolver.childNodes(this).map(function (n) {
-               return n.loc;
-             });
+             if (touchesThisLSE) {
+               // check for segments that just intersect on opposing endpoints
+               if (touchesOtherRSE) {
+                 if (tlp.x === orp.x && tlp.y === orp.y) return null;
+               } // t-intersection on left endpoint
 
-             if (this.isArea() && this.isClosed()) {
-               return {
-                 type: 'Polygon',
-                 coordinates: [coordinates]
-               };
-             } else {
-               return {
-                 type: 'LineString',
-                 coordinates: coordinates
-               };
-             }
-           });
-         },
-         area: function area(resolver) {
-           return resolver["transient"](this, 'area', function () {
-             var nodes = resolver.childNodes(this);
-             var json = {
-               type: 'Polygon',
-               coordinates: [nodes.map(function (n) {
-                 return n.loc;
-               })]
-             };
 
-             if (!this.isClosed() && nodes.length) {
-               json.coordinates[0].push(nodes[0].loc);
-             }
+               return tlp;
+             } // does other left endpoint matches (this doesn't)
 
-             var area = d3_geoArea(json); // Heuristic for detecting counterclockwise winding order. Assumes
-             // that OpenStreetMap polygons are not hemisphere-spanning.
 
-             if (area > 2 * Math.PI) {
-               json.coordinates[0] = json.coordinates[0].reverse();
-               area = d3_geoArea(json);
-             }
+             if (touchesOtherLSE) {
+               // check for segments that just intersect on opposing endpoints
+               if (touchesThisRSE) {
+                 if (trp.x === olp.x && trp.y === olp.y) return null;
+               } // t-intersection on left endpoint
 
-             return isNaN(area) ? 0 : area;
-           });
-         }
-       }); // Filter function to eliminate consecutive duplicates.
 
-       function noRepeatNodes(node, i, arr) {
-         return i === 0 || node !== arr[i - 1];
-       }
+               return olp;
+             } // trivial intersection on right endpoints
 
-       //
-       // 1. Relation tagged with `type=multipolygon` and no interesting tags.
-       // 2. One and only one member with the `outer` role. Must be a way with interesting tags.
-       // 3. No members without a role.
-       //
-       // Old multipolygons are no longer recommended but are still rendered as areas by iD.
 
-       function osmOldMultipolygonOuterMemberOfRelation(entity, graph) {
-         if (entity.type !== 'relation' || !entity.isMultipolygon() || Object.keys(entity.tags).filter(osmIsInterestingTag).length > 1) {
-           return false;
-         }
+             if (touchesThisRSE && touchesOtherRSE) return null; // t-intersections on just one right endpoint
 
-         var outerMember;
+             if (touchesThisRSE) return trp;
+             if (touchesOtherRSE) return orp; // None of our endpoints intersect. Look for a general intersection between
+             // infinite lines laid over the segments
 
-         for (var memberIndex in entity.members) {
-           var member = entity.members[memberIndex];
+             var pt = intersection(tlp, this.vector(), olp, other.vector()); // are the segments parrallel? Note that if they were colinear with overlap,
+             // they would have an endpoint intersection and that case was already handled above
 
-           if (!member.role || member.role === 'outer') {
-             if (outerMember) return false;
-             if (member.type !== 'way') return false;
-             if (!graph.hasEntity(member.id)) return false;
-             outerMember = graph.entity(member.id);
+             if (pt === null) return null; // is the intersection found between the lines not on the segments?
 
-             if (Object.keys(outerMember.tags).filter(osmIsInterestingTag).length === 0) {
-               return false;
-             }
+             if (!isInBbox(bboxOverlap, pt)) return null; // round the the computed point if needed
+
+             return rounder.round(pt.x, pt.y);
            }
-         }
+           /**
+            * Split the given segment into multiple segments on the given points.
+            *  * Each existing segment will retain its leftSE and a new rightSE will be
+            *    generated for it.
+            *  * A new segment will be generated which will adopt the original segment's
+            *    rightSE, and a new leftSE will be generated for it.
+            *  * If there are more than two points given to split on, new segments
+            *    in the middle will be generated with new leftSE and rightSE's.
+            *  * An array of the newly generated SweepEvents will be returned.
+            *
+            * Warning: input array of points is modified
+            */
 
-         return outerMember;
-       } // For fixing up rendering of multipolygons with tags on the outer member.
-       // https://github.com/openstreetmap/iD/issues/613
+         }, {
+           key: "split",
+           value: function split(point) {
+             var newEvents = [];
+             var alreadyLinked = point.events !== undefined;
+             var newLeftSE = new SweepEvent(point, true);
+             var newRightSE = new SweepEvent(point, false);
+             var oldRightSE = this.rightSE;
+             this.replaceRightSE(newRightSE);
+             newEvents.push(newRightSE);
+             newEvents.push(newLeftSE);
+             var newSeg = new Segment(newLeftSE, oldRightSE, this.rings.slice(), this.windings.slice()); // when splitting a nearly vertical downward-facing segment,
+             // sometimes one of the resulting new segments is vertical, in which
+             // case its left and right events may need to be swapped
 
-       function osmIsOldMultipolygonOuterMember(entity, graph) {
-         if (entity.type !== 'way' || Object.keys(entity.tags).filter(osmIsInterestingTag).length === 0) return false;
-         var parents = graph.parentRelations(entity);
-         if (parents.length !== 1) return false;
-         var parent = parents[0];
-         if (!parent.isMultipolygon() || Object.keys(parent.tags).filter(osmIsInterestingTag).length > 1) return false;
-         var members = parent.members,
-             member;
+             if (SweepEvent.comparePoints(newSeg.leftSE.point, newSeg.rightSE.point) > 0) {
+               newSeg.swapEvents();
+             }
 
-         for (var i = 0; i < members.length; i++) {
-           member = members[i];
-           if (member.id === entity.id && member.role && member.role !== 'outer') return false; // Not outer member
+             if (SweepEvent.comparePoints(this.leftSE.point, this.rightSE.point) > 0) {
+               this.swapEvents();
+             } // in the point we just used to create new sweep events with was already
+             // linked to other events, we need to check if either of the affected
+             // segments should be consumed
 
-           if (member.id !== entity.id && (!member.role || member.role === 'outer')) return false; // Not a simple multipolygon
-         }
 
-         return parent;
-       }
-       function osmOldMultipolygonOuterMember(entity, graph) {
-         if (entity.type !== 'way') return false;
-         var parents = graph.parentRelations(entity);
-         if (parents.length !== 1) return false;
-         var parent = parents[0];
-         if (!parent.isMultipolygon() || Object.keys(parent.tags).filter(osmIsInterestingTag).length > 1) return false;
-         var members = parent.members,
-             member,
-             outerMember;
+             if (alreadyLinked) {
+               newLeftSE.checkForConsuming();
+               newRightSE.checkForConsuming();
+             }
 
-         for (var i = 0; i < members.length; i++) {
-           member = members[i];
+             return newEvents;
+           }
+           /* Swap which event is left and right */
+
+         }, {
+           key: "swapEvents",
+           value: function swapEvents() {
+             var tmpEvt = this.rightSE;
+             this.rightSE = this.leftSE;
+             this.leftSE = tmpEvt;
+             this.leftSE.isLeft = true;
+             this.rightSE.isLeft = false;
+
+             for (var i = 0, iMax = this.windings.length; i < iMax; i++) {
+               this.windings[i] *= -1;
+             }
+           }
+           /* Consume another segment. We take their rings under our wing
+            * and mark them as consumed. Use for perfectly overlapping segments */
+
+         }, {
+           key: "consume",
+           value: function consume(other) {
+             var consumer = this;
+             var consumee = other;
+
+             while (consumer.consumedBy) {
+               consumer = consumer.consumedBy;
+             }
 
-           if (!member.role || member.role === 'outer') {
-             if (outerMember) return false; // Not a simple multipolygon
+             while (consumee.consumedBy) {
+               consumee = consumee.consumedBy;
+             }
 
-             outerMember = member;
-           }
-         }
+             var cmp = Segment.compare(consumer, consumee);
+             if (cmp === 0) return; // already consumed
+             // the winner of the consumption is the earlier segment
+             // according to sweep line ordering
 
-         if (!outerMember) return false;
-         var outerEntity = graph.hasEntity(outerMember.id);
-         if (!outerEntity || !Object.keys(outerEntity.tags).filter(osmIsInterestingTag).length) return false;
-         return outerEntity;
-       } // Join `toJoin` array into sequences of connecting ways.
-       // Segments which share identical start/end nodes will, as much as possible,
-       // be connected with each other.
-       //
-       // The return value is a nested array. Each constituent array contains elements
-       // of `toJoin` which have been determined to connect.
-       //
-       // Each consitituent array also has a `nodes` property whose value is an
-       // ordered array of member nodes, with appropriate order reversal and
-       // start/end coordinate de-duplication.
-       //
-       // Members of `toJoin` must have, at minimum, `type` and `id` properties.
-       // Thus either an array of `osmWay`s or a relation member array may be used.
-       //
-       // If an member is an `osmWay`, its tags and childnodes may be reversed via
-       // `actionReverse` in the output.
-       //
-       // The returned sequences array also has an `actions` array property, containing
-       // any reversal actions that should be applied to the graph, should the calling
-       // code attempt to actually join the given ways.
-       //
-       // Incomplete members (those for which `graph.hasEntity(element.id)` returns
-       // false) and non-way members are ignored.
-       //
+             if (cmp > 0) {
+               var tmp = consumer;
+               consumer = consumee;
+               consumee = tmp;
+             } // make sure a segment doesn't consume it's prev
 
-       function osmJoinWays(toJoin, graph) {
-         function resolve(member) {
-           return graph.childNodes(graph.entity(member.id));
-         }
 
-         function reverse(item) {
-           var action = actionReverse(item.id, {
-             reverseOneway: true
-           });
-           sequences.actions.push(action);
-           return item instanceof osmWay ? action(graph).entity(item.id) : item;
-         } // make a copy containing only the items to join
+             if (consumer.prev === consumee) {
+               var _tmp = consumer;
+               consumer = consumee;
+               consumee = _tmp;
+             }
 
+             for (var i = 0, iMax = consumee.rings.length; i < iMax; i++) {
+               var ring = consumee.rings[i];
+               var winding = consumee.windings[i];
+               var index = consumer.rings.indexOf(ring);
 
-         toJoin = toJoin.filter(function (member) {
-           return member.type === 'way' && graph.hasEntity(member.id);
-         }); // Are the things we are joining relation members or `osmWays`?
-         // If `osmWays`, skip the "prefer a forward path" code below (see #4872)
+               if (index === -1) {
+                 consumer.rings.push(ring);
+                 consumer.windings.push(winding);
+               } else consumer.windings[index] += winding;
+             }
 
-         var i;
-         var joinAsMembers = true;
+             consumee.rings = null;
+             consumee.windings = null;
+             consumee.consumedBy = consumer; // mark sweep events consumed as to maintain ordering in sweep event queue
 
-         for (i = 0; i < toJoin.length; i++) {
-           if (toJoin[i] instanceof osmWay) {
-             joinAsMembers = false;
-             break;
+             consumee.leftSE.consumedBy = consumer.leftSE;
+             consumee.rightSE.consumedBy = consumer.rightSE;
            }
-         }
-
-         var sequences = [];
-         sequences.actions = [];
+           /* The first segment previous segment chain that is in the result */
 
-         while (toJoin.length) {
-           // start a new sequence
-           var item = toJoin.shift();
-           var currWays = [item];
-           var currNodes = resolve(item).slice(); // add to it
+         }, {
+           key: "prevInResult",
+           value: function prevInResult() {
+             if (this._prevInResult !== undefined) return this._prevInResult;
+             if (!this.prev) this._prevInResult = null;else if (this.prev.isInResult()) this._prevInResult = this.prev;else this._prevInResult = this.prev.prevInResult();
+             return this._prevInResult;
+           }
+         }, {
+           key: "beforeState",
+           value: function beforeState() {
+             if (this._beforeState !== undefined) return this._beforeState;
+             if (!this.prev) this._beforeState = {
+               rings: [],
+               windings: [],
+               multiPolys: []
+             };else {
+               var seg = this.prev.consumedBy || this.prev;
+               this._beforeState = seg.afterState();
+             }
+             return this._beforeState;
+           }
+         }, {
+           key: "afterState",
+           value: function afterState() {
+             if (this._afterState !== undefined) return this._afterState;
+             var beforeState = this.beforeState();
+             this._afterState = {
+               rings: beforeState.rings.slice(0),
+               windings: beforeState.windings.slice(0),
+               multiPolys: []
+             };
+             var ringsAfter = this._afterState.rings;
+             var windingsAfter = this._afterState.windings;
+             var mpsAfter = this._afterState.multiPolys; // calculate ringsAfter, windingsAfter
 
-           while (toJoin.length) {
-             var start = currNodes[0];
-             var end = currNodes[currNodes.length - 1];
-             var fn = null;
-             var nodes = null; // Find the next way/member to join.
+             for (var i = 0, iMax = this.rings.length; i < iMax; i++) {
+               var ring = this.rings[i];
+               var winding = this.windings[i];
+               var index = ringsAfter.indexOf(ring);
 
-             for (i = 0; i < toJoin.length; i++) {
-               item = toJoin[i];
-               nodes = resolve(item); // (for member ordering only, not way ordering - see #4872)
-               // Strongly prefer to generate a forward path that preserves the order
-               // of the members array. For multipolygons and most relations, member
-               // order does not matter - but for routes, it does. (see #4589)
-               // If we started this sequence backwards (i.e. next member way attaches to
-               // the start node and not the end node), reverse the initial way before continuing.
+               if (index === -1) {
+                 ringsAfter.push(ring);
+                 windingsAfter.push(winding);
+               } else windingsAfter[index] += winding;
+             } // calcualte polysAfter
 
-               if (joinAsMembers && currWays.length === 1 && nodes[0] !== end && nodes[nodes.length - 1] !== end && (nodes[nodes.length - 1] === start || nodes[0] === start)) {
-                 currWays[0] = reverse(currWays[0]);
-                 currNodes.reverse();
-                 start = currNodes[0];
-                 end = currNodes[currNodes.length - 1];
-               }
 
-               if (nodes[0] === end) {
-                 fn = currNodes.push; // join to end
+             var polysAfter = [];
+             var polysExclude = [];
 
-                 nodes = nodes.slice(1);
-                 break;
-               } else if (nodes[nodes.length - 1] === end) {
-                 fn = currNodes.push; // join to end
+             for (var _i = 0, _iMax = ringsAfter.length; _i < _iMax; _i++) {
+               if (windingsAfter[_i] === 0) continue; // non-zero rule
 
-                 nodes = nodes.slice(0, -1).reverse();
-                 item = reverse(item);
-                 break;
-               } else if (nodes[nodes.length - 1] === start) {
-                 fn = currNodes.unshift; // join to beginning
+               var _ring = ringsAfter[_i];
+               var poly = _ring.poly;
+               if (polysExclude.indexOf(poly) !== -1) continue;
+               if (_ring.isExterior) polysAfter.push(poly);else {
+                 if (polysExclude.indexOf(poly) === -1) polysExclude.push(poly);
 
-                 nodes = nodes.slice(0, -1);
-                 break;
-               } else if (nodes[0] === start) {
-                 fn = currNodes.unshift; // join to beginning
+                 var _index = polysAfter.indexOf(_ring.poly);
 
-                 nodes = nodes.slice(1).reverse();
-                 item = reverse(item);
-                 break;
-               } else {
-                 fn = nodes = null;
+                 if (_index !== -1) polysAfter.splice(_index, 1);
                }
-             }
+             } // calculate multiPolysAfter
 
-             if (!nodes) {
-               // couldn't find a joinable way/member
-               break;
+
+             for (var _i2 = 0, _iMax2 = polysAfter.length; _i2 < _iMax2; _i2++) {
+               var mp = polysAfter[_i2].multiPoly;
+               if (mpsAfter.indexOf(mp) === -1) mpsAfter.push(mp);
              }
 
-             fn.apply(currWays, [item]);
-             fn.apply(currNodes, nodes);
-             toJoin.splice(i, 1);
+             return this._afterState;
            }
+           /* Is this segment part of the final result? */
 
-           currWays.nodes = currNodes;
-           sequences.push(currWays);
-         }
+         }, {
+           key: "isInResult",
+           value: function isInResult() {
+             // if we've been consumed, we're not in the result
+             if (this.consumedBy) return false;
+             if (this._isInResult !== undefined) return this._isInResult;
+             var mpsBefore = this.beforeState().multiPolys;
+             var mpsAfter = this.afterState().multiPolys;
 
-         return sequences;
-       }
+             switch (operation.type) {
+               case 'union':
+                 {
+                   // UNION - included iff:
+                   //  * On one side of us there is 0 poly interiors AND
+                   //  * On the other side there is 1 or more.
+                   var noBefores = mpsBefore.length === 0;
+                   var noAfters = mpsAfter.length === 0;
+                   this._isInResult = noBefores !== noAfters;
+                   break;
+                 }
 
-       function actionAddMember(relationId, member, memberIndex, insertPair) {
-         return function action(graph) {
-           var relation = graph.entity(relationId); // There are some special rules for Public Transport v2 routes.
+               case 'intersection':
+                 {
+                   // INTERSECTION - included iff:
+                   //  * on one side of us all multipolys are rep. with poly interiors AND
+                   //  * on the other side of us, not all multipolys are repsented
+                   //    with poly interiors
+                   var least;
+                   var most;
 
-           var isPTv2 = /stop|platform/.test(member.role);
+                   if (mpsBefore.length < mpsAfter.length) {
+                     least = mpsBefore.length;
+                     most = mpsAfter.length;
+                   } else {
+                     least = mpsAfter.length;
+                     most = mpsBefore.length;
+                   }
 
-           if ((isNaN(memberIndex) || insertPair) && member.type === 'way' && !isPTv2) {
-             // Try to perform sensible inserts based on how the ways join together
-             graph = addWayMember(relation, graph);
-           } else {
-             // see https://wiki.openstreetmap.org/wiki/Public_transport#Service_routes
-             // Stops and Platforms for PTv2 should be ordered first.
-             // hack: We do not currently have the ability to place them in the exactly correct order.
-             if (isPTv2 && isNaN(memberIndex)) {
-               memberIndex = 0;
+                   this._isInResult = most === operation.numMultiPolys && least < most;
+                   break;
+                 }
+
+               case 'xor':
+                 {
+                   // XOR - included iff:
+                   //  * the difference between the number of multipolys represented
+                   //    with poly interiors on our two sides is an odd number
+                   var diff = Math.abs(mpsBefore.length - mpsAfter.length);
+                   this._isInResult = diff % 2 === 1;
+                   break;
+                 }
+
+               case 'difference':
+                 {
+                   // DIFFERENCE included iff:
+                   //  * on exactly one side, we have just the subject
+                   var isJustSubject = function isJustSubject(mps) {
+                     return mps.length === 1 && mps[0].isSubject;
+                   };
+
+                   this._isInResult = isJustSubject(mpsBefore) !== isJustSubject(mpsAfter);
+                   break;
+                 }
+
+               default:
+                 throw new Error("Unrecognized operation type found ".concat(operation.type));
              }
 
-             graph = graph.replace(relation.addMember(member, memberIndex));
+             return this._isInResult;
            }
+         }], [{
+           key: "fromRing",
+           value: function fromRing(pt1, pt2, ring) {
+             var leftPt, rightPt, winding; // ordering the two points according to sweep line ordering
 
-           return graph;
-         }; // Add a way member into the relation "wherever it makes sense".
-         // In this situation we were not supplied a memberIndex.
+             var cmpPts = SweepEvent.comparePoints(pt1, pt2);
 
-         function addWayMember(relation, graph) {
-           var groups, tempWay, item, i, j, k; // remove PTv2 stops and platforms before doing anything.
+             if (cmpPts < 0) {
+               leftPt = pt1;
+               rightPt = pt2;
+               winding = 1;
+             } else if (cmpPts > 0) {
+               leftPt = pt2;
+               rightPt = pt1;
+               winding = -1;
+             } else throw new Error("Tried to create degenerate segment at [".concat(pt1.x, ", ").concat(pt1.y, "]"));
 
-           var PTv2members = [];
-           var members = [];
+             var leftSE = new SweepEvent(leftPt, true);
+             var rightSE = new SweepEvent(rightPt, false);
+             return new Segment(leftSE, rightSE, [ring], [winding]);
+           }
+         }]);
 
-           for (i = 0; i < relation.members.length; i++) {
-             var m = relation.members[i];
+         return Segment;
+       }();
 
-             if (/stop|platform/.test(m.role)) {
-               PTv2members.push(m);
-             } else {
-               members.push(m);
-             }
+       var RingIn = /*#__PURE__*/function () {
+         function RingIn(geomRing, poly, isExterior) {
+           _classCallCheck(this, RingIn);
+
+           if (!Array.isArray(geomRing) || geomRing.length === 0) {
+             throw new Error('Input geometry is not a valid Polygon or MultiPolygon');
            }
 
-           relation = relation.update({
-             members: members
-           });
+           this.poly = poly;
+           this.isExterior = isExterior;
+           this.segments = [];
 
-           if (insertPair) {
-             // We're adding a member that must stay paired with an existing member.
-             // (This feature is used by `actionSplit`)
-             //
-             // This is tricky because the members may exist multiple times in the
-             // member list, and with different A-B/B-A ordering and different roles.
-             // (e.g. a bus route that loops out and back - #4589).
-             //
-             // Replace the existing member with a temporary way,
-             // so that `osmJoinWays` can treat the pair like a single way.
-             tempWay = osmWay({
-               id: 'wTemp',
-               nodes: insertPair.nodes
-             });
-             graph = graph.replace(tempWay);
-             var tempMember = {
-               id: tempWay.id,
-               type: 'way',
-               role: member.role
-             };
-             var tempRelation = relation.replaceMember({
-               id: insertPair.originalID
-             }, tempMember, true);
-             groups = utilArrayGroupBy(tempRelation.members, 'type');
-             groups.way = groups.way || [];
-           } else {
-             // Add the member anywhere, one time. Just push and let `osmJoinWays` decide where to put it.
-             groups = utilArrayGroupBy(relation.members, 'type');
-             groups.way = groups.way || [];
-             groups.way.push(member);
+           if (typeof geomRing[0][0] !== 'number' || typeof geomRing[0][1] !== 'number') {
+             throw new Error('Input geometry is not a valid Polygon or MultiPolygon');
            }
 
-           members = withIndex(groups.way);
-           var joined = osmJoinWays(members, graph); // `joined` might not contain all of the way members,
-           // But will contain only the completed (downloaded) members
-
-           for (i = 0; i < joined.length; i++) {
-             var segment = joined[i];
-             var nodes = segment.nodes.slice();
-             var startIndex = segment[0].index; // j = array index in `members` where this segment starts
+           var firstPoint = rounder.round(geomRing[0][0], geomRing[0][1]);
+           this.bbox = {
+             ll: {
+               x: firstPoint.x,
+               y: firstPoint.y
+             },
+             ur: {
+               x: firstPoint.x,
+               y: firstPoint.y
+             }
+           };
+           var prevPoint = firstPoint;
 
-             for (j = 0; j < members.length; j++) {
-               if (members[j].index === startIndex) {
-                 break;
-               }
-             } // k = each member in segment
+           for (var i = 1, iMax = geomRing.length; i < iMax; i++) {
+             if (typeof geomRing[i][0] !== 'number' || typeof geomRing[i][1] !== 'number') {
+               throw new Error('Input geometry is not a valid Polygon or MultiPolygon');
+             }
 
+             var point = rounder.round(geomRing[i][0], geomRing[i][1]); // skip repeated points
 
-             for (k = 0; k < segment.length; k++) {
-               item = segment[k];
-               var way = graph.entity(item.id); // If this is a paired item, generate members in correct order and role
+             if (point.x === prevPoint.x && point.y === prevPoint.y) continue;
+             this.segments.push(Segment.fromRing(prevPoint, point, this));
+             if (point.x < this.bbox.ll.x) this.bbox.ll.x = point.x;
+             if (point.y < this.bbox.ll.y) this.bbox.ll.y = point.y;
+             if (point.x > this.bbox.ur.x) this.bbox.ur.x = point.x;
+             if (point.y > this.bbox.ur.y) this.bbox.ur.y = point.y;
+             prevPoint = point;
+           } // add segment from last to first if last is not the same as first
 
-               if (tempWay && item.id === tempWay.id) {
-                 if (nodes[0].id === insertPair.nodes[0]) {
-                   item.pair = [{
-                     id: insertPair.originalID,
-                     type: 'way',
-                     role: item.role
-                   }, {
-                     id: insertPair.insertedID,
-                     type: 'way',
-                     role: item.role
-                   }];
-                 } else {
-                   item.pair = [{
-                     id: insertPair.insertedID,
-                     type: 'way',
-                     role: item.role
-                   }, {
-                     id: insertPair.originalID,
-                     type: 'way',
-                     role: item.role
-                   }];
-                 }
-               } // reorder `members` if necessary
 
+           if (firstPoint.x !== prevPoint.x || firstPoint.y !== prevPoint.y) {
+             this.segments.push(Segment.fromRing(prevPoint, firstPoint, this));
+           }
+         }
 
-               if (k > 0) {
-                 if (j + k >= members.length || item.index !== members[j + k].index) {
-                   moveMember(members, item.index, j + k);
-                 }
-               }
+         _createClass(RingIn, [{
+           key: "getSweepEvents",
+           value: function getSweepEvents() {
+             var sweepEvents = [];
 
-               nodes.splice(0, way.nodes.length - 1);
+             for (var i = 0, iMax = this.segments.length; i < iMax; i++) {
+               var segment = this.segments[i];
+               sweepEvents.push(segment.leftSE);
+               sweepEvents.push(segment.rightSE);
              }
+
+             return sweepEvents;
            }
+         }]);
 
-           if (tempWay) {
-             graph = graph.remove(tempWay);
-           } // Final pass: skip dead items, split pairs, remove index properties
+         return RingIn;
+       }();
 
+       var PolyIn = /*#__PURE__*/function () {
+         function PolyIn(geomPoly, multiPoly) {
+           _classCallCheck(this, PolyIn);
 
-           var wayMembers = [];
+           if (!Array.isArray(geomPoly)) {
+             throw new Error('Input geometry is not a valid Polygon or MultiPolygon');
+           }
 
-           for (i = 0; i < members.length; i++) {
-             item = members[i];
-             if (item.index === -1) continue;
+           this.exteriorRing = new RingIn(geomPoly[0], this, true); // copy by value
 
-             if (item.pair) {
-               wayMembers.push(item.pair[0]);
-               wayMembers.push(item.pair[1]);
-             } else {
-               wayMembers.push(utilObjectOmit(item, ['index']));
+           this.bbox = {
+             ll: {
+               x: this.exteriorRing.bbox.ll.x,
+               y: this.exteriorRing.bbox.ll.y
+             },
+             ur: {
+               x: this.exteriorRing.bbox.ur.x,
+               y: this.exteriorRing.bbox.ur.y
              }
-           } // Put stops and platforms first, then nodes, ways, relations
-           // This is recommended for Public Transport v2 routes:
-           // see https://wiki.openstreetmap.org/wiki/Public_transport#Service_routes
+           };
+           this.interiorRings = [];
+
+           for (var i = 1, iMax = geomPoly.length; i < iMax; i++) {
+             var ring = new RingIn(geomPoly[i], this, false);
+             if (ring.bbox.ll.x < this.bbox.ll.x) this.bbox.ll.x = ring.bbox.ll.x;
+             if (ring.bbox.ll.y < this.bbox.ll.y) this.bbox.ll.y = ring.bbox.ll.y;
+             if (ring.bbox.ur.x > this.bbox.ur.x) this.bbox.ur.x = ring.bbox.ur.x;
+             if (ring.bbox.ur.y > this.bbox.ur.y) this.bbox.ur.y = ring.bbox.ur.y;
+             this.interiorRings.push(ring);
+           }
 
+           this.multiPoly = multiPoly;
+         }
 
-           var newMembers = PTv2members.concat(groups.node || [], wayMembers, groups.relation || []);
-           return graph.replace(relation.update({
-             members: newMembers
-           })); // `moveMember()` changes the `members` array in place by splicing
-           // the item with `.index = findIndex` to where it belongs,
-           // and marking the old position as "dead" with `.index = -1`
-           //
-           // j=5, k=0                jk
-           // segment                 5 4 7 6
-           // members       0 1 2 3 4 5 6 7 8 9        keep 5 in j+k
-           //
-           // j=5, k=1                j k
-           // segment                 5 4 7 6
-           // members       0 1 2 3 4 5 6 7 8 9        move 4 to j+k
-           // members       0 1 2 3 x 5 4 6 7 8 9      moved
-           //
-           // j=5, k=2                j   k
-           // segment                 5 4 7 6
-           // members       0 1 2 3 x 5 4 6 7 8 9      move 7 to j+k
-           // members       0 1 2 3 x 5 4 7 6 x 8 9    moved
-           //
-           // j=5, k=3                j     k
-           // segment                 5 4 7 6
-           // members       0 1 2 3 x 5 4 7 6 x 8 9    keep 6 in j+k
-           //
+         _createClass(PolyIn, [{
+           key: "getSweepEvents",
+           value: function getSweepEvents() {
+             var sweepEvents = this.exteriorRing.getSweepEvents();
 
-           function moveMember(arr, findIndex, toIndex) {
-             var i;
+             for (var i = 0, iMax = this.interiorRings.length; i < iMax; i++) {
+               var ringSweepEvents = this.interiorRings[i].getSweepEvents();
 
-             for (i = 0; i < arr.length; i++) {
-               if (arr[i].index === findIndex) {
-                 break;
+               for (var j = 0, jMax = ringSweepEvents.length; j < jMax; j++) {
+                 sweepEvents.push(ringSweepEvents[j]);
                }
              }
 
-             var item = Object.assign({}, arr[i]); // shallow copy
-
-             arr[i].index = -1; // mark as dead
+             return sweepEvents;
+           }
+         }]);
 
-             item.index = toIndex;
-             arr.splice(toIndex, 0, item);
-           } // This is the same as `Relation.indexedMembers`,
-           // Except we don't want to index all the members, only the ways
+         return PolyIn;
+       }();
 
+       var MultiPolyIn = /*#__PURE__*/function () {
+         function MultiPolyIn(geom, isSubject) {
+           _classCallCheck(this, MultiPolyIn);
 
-           function withIndex(arr) {
-             var result = new Array(arr.length);
+           if (!Array.isArray(geom)) {
+             throw new Error('Input geometry is not a valid Polygon or MultiPolygon');
+           }
 
-             for (var i = 0; i < arr.length; i++) {
-               result[i] = Object.assign({}, arr[i]); // shallow copy
+           try {
+             // if the input looks like a polygon, convert it to a multipolygon
+             if (typeof geom[0][0][0] === 'number') geom = [geom];
+           } catch (ex) {// The input is either malformed or has empty arrays.
+             // In either case, it will be handled later on.
+           }
 
-               result[i].index = i;
+           this.polys = [];
+           this.bbox = {
+             ll: {
+               x: Number.POSITIVE_INFINITY,
+               y: Number.POSITIVE_INFINITY
+             },
+             ur: {
+               x: Number.NEGATIVE_INFINITY,
+               y: Number.NEGATIVE_INFINITY
              }
+           };
 
-             return result;
+           for (var i = 0, iMax = geom.length; i < iMax; i++) {
+             var poly = new PolyIn(geom[i], this);
+             if (poly.bbox.ll.x < this.bbox.ll.x) this.bbox.ll.x = poly.bbox.ll.x;
+             if (poly.bbox.ll.y < this.bbox.ll.y) this.bbox.ll.y = poly.bbox.ll.y;
+             if (poly.bbox.ur.x > this.bbox.ur.x) this.bbox.ur.x = poly.bbox.ur.x;
+             if (poly.bbox.ur.y > this.bbox.ur.y) this.bbox.ur.y = poly.bbox.ur.y;
+             this.polys.push(poly);
            }
+
+           this.isSubject = isSubject;
          }
-       }
 
-       function actionAddMidpoint(midpoint, node) {
-         return function (graph) {
-           graph = graph.replace(node.move(midpoint.loc));
-           var parents = utilArrayIntersection(graph.parentWays(graph.entity(midpoint.edge[0])), graph.parentWays(graph.entity(midpoint.edge[1])));
-           parents.forEach(function (way) {
-             for (var i = 0; i < way.nodes.length - 1; i++) {
-               if (geoEdgeEqual([way.nodes[i], way.nodes[i + 1]], midpoint.edge)) {
-                 graph = graph.replace(graph.entity(way.id).addNode(node.id, i + 1)); // Add only one midpoint on doubled-back segments,
-                 // turning them into self-intersections.
+         _createClass(MultiPolyIn, [{
+           key: "getSweepEvents",
+           value: function getSweepEvents() {
+             var sweepEvents = [];
 
-                 return;
+             for (var i = 0, iMax = this.polys.length; i < iMax; i++) {
+               var polySweepEvents = this.polys[i].getSweepEvents();
+
+               for (var j = 0, jMax = polySweepEvents.length; j < jMax; j++) {
+                 sweepEvents.push(polySweepEvents[j]);
                }
              }
-           });
-           return graph;
-         };
-       }
 
-       // https://github.com/openstreetmap/potlatch2/blob/master/net/systemeD/halcyon/connection/actions/AddNodeToWayAction.as
-       function actionAddVertex(wayId, nodeId, index) {
-         return function (graph) {
-           return graph.replace(graph.entity(wayId).addNode(nodeId, index));
-         };
-       }
+             return sweepEvents;
+           }
+         }]);
 
-       function actionChangeMember(relationId, member, memberIndex) {
-         return function (graph) {
-           return graph.replace(graph.entity(relationId).updateMember(member, memberIndex));
-         };
-       }
+         return MultiPolyIn;
+       }();
 
-       function actionChangePreset(entityID, oldPreset, newPreset, skipFieldDefaults) {
-         return function action(graph) {
-           var entity = graph.entity(entityID);
-           var geometry = entity.geometry(graph);
-           var tags = entity.tags;
-           if (oldPreset) tags = oldPreset.unsetTags(tags, geometry);
-           if (newPreset) tags = newPreset.setTags(tags, geometry, skipFieldDefaults);
-           return graph.replace(entity.update({
-             tags: tags
-           }));
-         };
-       }
+       var RingOut = /*#__PURE__*/function () {
+         _createClass(RingOut, null, [{
+           key: "factory",
 
-       function actionChangeTags(entityId, tags) {
-         return function (graph) {
-           var entity = graph.entity(entityId);
-           return graph.replace(entity.update({
-             tags: tags
-           }));
-         };
-       }
+           /* Given the segments from the sweep line pass, compute & return a series
+            * of closed rings from all the segments marked to be part of the result */
+           value: function factory(allSegments) {
+             var ringsOut = [];
 
-       function osmNode() {
-         if (!(this instanceof osmNode)) {
-           return new osmNode().initialize(arguments);
-         } else if (arguments.length) {
-           this.initialize(arguments);
-         }
-       }
-       osmEntity.node = osmNode;
-       osmNode.prototype = Object.create(osmEntity.prototype);
-       Object.assign(osmNode.prototype, {
-         type: 'node',
-         loc: [9999, 9999],
-         extent: function extent() {
-           return new geoExtent(this.loc);
-         },
-         geometry: function geometry(graph) {
-           return graph["transient"](this, 'geometry', function () {
-             return graph.isPoi(this) ? 'point' : 'vertex';
-           });
-         },
-         move: function move(loc) {
-           return this.update({
-             loc: loc
-           });
-         },
-         isDegenerate: function isDegenerate() {
-           return !(Array.isArray(this.loc) && this.loc.length === 2 && this.loc[0] >= -180 && this.loc[0] <= 180 && this.loc[1] >= -90 && this.loc[1] <= 90);
-         },
-         // Inspect tags and geometry to determine which direction(s) this node/vertex points
-         directions: function directions(resolver, projection) {
-           var val;
-           var i; // which tag to use?
+             for (var i = 0, iMax = allSegments.length; i < iMax; i++) {
+               var segment = allSegments[i];
+               if (!segment.isInResult() || segment.ringOut) continue;
+               var prevEvent = null;
+               var event = segment.leftSE;
+               var nextEvent = segment.rightSE;
+               var events = [event];
+               var startingPoint = event.point;
+               var intersectionLEs = [];
+               /* Walk the chain of linked events to form a closed ring */
 
-           if (this.isHighwayIntersection(resolver) && (this.tags.stop || '').toLowerCase() === 'all') {
-             // all-way stop tag on a highway intersection
-             val = 'all';
-           } else {
-             // generic direction tag
-             val = (this.tags.direction || '').toLowerCase(); // better suffix-style direction tag
+               while (true) {
+                 prevEvent = event;
+                 event = nextEvent;
+                 events.push(event);
+                 /* Is the ring complete? */
 
-             var re = /:direction$/i;
-             var keys = Object.keys(this.tags);
+                 if (event.point === startingPoint) break;
 
-             for (i = 0; i < keys.length; i++) {
-               if (re.test(keys[i])) {
-                 val = this.tags[keys[i]].toLowerCase();
-                 break;
-               }
-             }
-           }
+                 while (true) {
+                   var availableLEs = event.getAvailableLinkedEvents();
+                   /* Did we hit a dead end? This shouldn't happen. Indicates some earlier
+                    * part of the algorithm malfunctioned... please file a bug report. */
 
-           if (val === '') return [];
-           var cardinal = {
-             north: 0,
-             n: 0,
-             northnortheast: 22,
-             nne: 22,
-             northeast: 45,
-             ne: 45,
-             eastnortheast: 67,
-             ene: 67,
-             east: 90,
-             e: 90,
-             eastsoutheast: 112,
-             ese: 112,
-             southeast: 135,
-             se: 135,
-             southsoutheast: 157,
-             sse: 157,
-             south: 180,
-             s: 180,
-             southsouthwest: 202,
-             ssw: 202,
-             southwest: 225,
-             sw: 225,
-             westsouthwest: 247,
-             wsw: 247,
-             west: 270,
-             w: 270,
-             westnorthwest: 292,
-             wnw: 292,
-             northwest: 315,
-             nw: 315,
-             northnorthwest: 337,
-             nnw: 337
-           };
-           var values = val.split(';');
-           var results = [];
-           values.forEach(function (v) {
-             // swap cardinal for numeric directions
-             if (cardinal[v] !== undefined) {
-               v = cardinal[v];
-             } // numeric direction - just add to results
+                   if (availableLEs.length === 0) {
+                     var firstPt = events[0].point;
+                     var lastPt = events[events.length - 1].point;
+                     throw new Error("Unable to complete output ring starting at [".concat(firstPt.x, ",") + " ".concat(firstPt.y, "]. Last matching segment found ends at") + " [".concat(lastPt.x, ", ").concat(lastPt.y, "]."));
+                   }
+                   /* Only one way to go, so cotinue on the path */
 
 
-             if (v !== '' && !isNaN(+v)) {
-               results.push(+v);
-               return;
-             } // string direction - inspect parent ways
+                   if (availableLEs.length === 1) {
+                     nextEvent = availableLEs[0].otherSE;
+                     break;
+                   }
+                   /* We must have an intersection. Check for a completed loop */
 
 
-             var lookBackward = this.tags['traffic_sign:backward'] || v === 'backward' || v === 'both' || v === 'all';
-             var lookForward = this.tags['traffic_sign:forward'] || v === 'forward' || v === 'both' || v === 'all';
-             if (!lookForward && !lookBackward) return;
-             var nodeIds = {};
-             resolver.parentWays(this).forEach(function (parent) {
-               var nodes = parent.nodes;
+                   var indexLE = null;
 
-               for (i = 0; i < nodes.length; i++) {
-                 if (nodes[i] === this.id) {
-                   // match current entity
-                   if (lookForward && i > 0) {
-                     nodeIds[nodes[i - 1]] = true; // look back to prev node
+                   for (var j = 0, jMax = intersectionLEs.length; j < jMax; j++) {
+                     if (intersectionLEs[j].point === event.point) {
+                       indexLE = j;
+                       break;
+                     }
                    }
+                   /* Found a completed loop. Cut that off and make a ring */
 
-                   if (lookBackward && i < nodes.length - 1) {
-                     nodeIds[nodes[i + 1]] = true; // look ahead to next node
+
+                   if (indexLE !== null) {
+                     var intersectionLE = intersectionLEs.splice(indexLE)[0];
+                     var ringEvents = events.splice(intersectionLE.index);
+                     ringEvents.unshift(ringEvents[0].otherSE);
+                     ringsOut.push(new RingOut(ringEvents.reverse()));
+                     continue;
                    }
+                   /* register the intersection */
+
+
+                   intersectionLEs.push({
+                     index: events.length,
+                     point: event.point
+                   });
+                   /* Choose the left-most option to continue the walk */
+
+                   var comparator = event.getLeftmostComparator(prevEvent);
+                   nextEvent = availableLEs.sort(comparator)[0].otherSE;
+                   break;
                  }
                }
-             }, this);
-             Object.keys(nodeIds).forEach(function (nodeId) {
-               // +90 because geoAngle returns angle from X axis, not Y (north)
-               results.push(geoAngle(this, resolver.entity(nodeId), projection) * (180 / Math.PI) + 90);
-             }, this);
-           }, this);
-           return utilArrayUniq(results);
-         },
-         isEndpoint: function isEndpoint(resolver) {
-           return resolver["transient"](this, 'isEndpoint', function () {
-             var id = this.id;
-             return resolver.parentWays(this).filter(function (parent) {
-               return !parent.isClosed() && !!parent.affix(id);
-             }).length > 0;
-           });
-         },
-         isConnected: function isConnected(resolver) {
-           return resolver["transient"](this, 'isConnected', function () {
-             var parents = resolver.parentWays(this);
 
-             if (parents.length > 1) {
-               // vertex is connected to multiple parent ways
-               for (var i in parents) {
-                 if (parents[i].geometry(resolver) === 'line' && parents[i].hasInterestingTags()) return true;
-               }
-             } else if (parents.length === 1) {
-               var way = parents[0];
-               var nodes = way.nodes.slice();
+               ringsOut.push(new RingOut(events));
+             }
 
-               if (way.isClosed()) {
-                 nodes.pop();
-               } // ignore connecting node if closed
-               // return true if vertex appears multiple times (way is self intersecting)
+             return ringsOut;
+           }
+         }]);
 
+         function RingOut(events) {
+           _classCallCheck(this, RingOut);
 
-               return nodes.indexOf(this.id) !== nodes.lastIndexOf(this.id);
-             }
+           this.events = events;
 
-             return false;
-           });
-         },
-         parentIntersectionWays: function parentIntersectionWays(resolver) {
-           return resolver["transient"](this, 'parentIntersectionWays', function () {
-             return resolver.parentWays(this).filter(function (parent) {
-               return (parent.tags.highway || parent.tags.waterway || parent.tags.railway || parent.tags.aeroway) && parent.geometry(resolver) === 'line';
-             });
-           });
-         },
-         isIntersection: function isIntersection(resolver) {
-           return this.parentIntersectionWays(resolver).length > 1;
-         },
-         isHighwayIntersection: function isHighwayIntersection(resolver) {
-           return resolver["transient"](this, 'isHighwayIntersection', function () {
-             return resolver.parentWays(this).filter(function (parent) {
-               return parent.tags.highway && parent.geometry(resolver) === 'line';
-             }).length > 1;
-           });
-         },
-         isOnAddressLine: function isOnAddressLine(resolver) {
-           return resolver["transient"](this, 'isOnAddressLine', function () {
-             return resolver.parentWays(this).filter(function (parent) {
-               return parent.tags.hasOwnProperty('addr:interpolation') && parent.geometry(resolver) === 'line';
-             }).length > 0;
-           });
-         },
-         asJXON: function asJXON(changeset_id) {
-           var r = {
-             node: {
-               '@id': this.osmId(),
-               '@lon': this.loc[0],
-               '@lat': this.loc[1],
-               '@version': this.version || 0,
-               tag: Object.keys(this.tags).map(function (k) {
-                 return {
-                   keyAttributes: {
-                     k: k,
-                     v: this.tags[k]
-                   }
-                 };
-               }, this)
-             }
-           };
-           if (changeset_id) r.node['@changeset'] = changeset_id;
-           return r;
-         },
-         asGeoJSON: function asGeoJSON() {
-           return {
-             type: 'Point',
-             coordinates: this.loc
-           };
+           for (var i = 0, iMax = events.length; i < iMax; i++) {
+             events[i].segment.ringOut = this;
+           }
+
+           this.poly = null;
          }
-       });
 
-       function actionCircularize(wayId, projection, maxAngle) {
-         maxAngle = (maxAngle || 20) * Math.PI / 180;
+         _createClass(RingOut, [{
+           key: "getGeom",
+           value: function getGeom() {
+             // Remove superfluous points (ie extra points along a straight line),
+             var prevPt = this.events[0].point;
+             var points = [prevPt];
 
-         var action = function action(graph, t) {
-           if (t === null || !isFinite(t)) t = 1;
-           t = Math.min(Math.max(+t, 0), 1);
-           var way = graph.entity(wayId);
-           var origNodes = {};
-           graph.childNodes(way).forEach(function (node) {
-             if (!origNodes[node.id]) origNodes[node.id] = node;
-           });
+             for (var i = 1, iMax = this.events.length - 1; i < iMax; i++) {
+               var _pt = this.events[i].point;
+               var _nextPt = this.events[i + 1].point;
+               if (compareVectorAngles(_pt, prevPt, _nextPt) === 0) continue;
+               points.push(_pt);
+               prevPt = _pt;
+             } // ring was all (within rounding error of angle calc) colinear points
 
-           if (!way.isConvex(graph)) {
-             graph = action.makeConvex(graph);
-           }
 
-           var nodes = utilArrayUniq(graph.childNodes(way));
-           var keyNodes = nodes.filter(function (n) {
-             return graph.parentWays(n).length !== 1;
-           });
-           var points = nodes.map(function (n) {
-             return projection(n.loc);
-           });
-           var keyPoints = keyNodes.map(function (n) {
-             return projection(n.loc);
-           });
-           var centroid = points.length === 2 ? geoVecInterp(points[0], points[1], 0.5) : d3_polygonCentroid(points);
-           var radius = d3_median(points, function (p) {
-             return geoVecLength(centroid, p);
-           });
-           var sign = d3_polygonArea(points) > 0 ? 1 : -1;
-           var ids, i, j, k; // we need at least two key nodes for the algorithm to work
+             if (points.length === 1) return null; // check if the starting point is necessary
 
-           if (!keyNodes.length) {
-             keyNodes = [nodes[0]];
-             keyPoints = [points[0]];
-           }
+             var pt = points[0];
+             var nextPt = points[1];
+             if (compareVectorAngles(pt, prevPt, nextPt) === 0) points.shift();
+             points.push(points[0]);
+             var step = this.isExteriorRing() ? 1 : -1;
+             var iStart = this.isExteriorRing() ? 0 : points.length - 1;
+             var iEnd = this.isExteriorRing() ? points.length : -1;
+             var orderedPoints = [];
 
-           if (keyNodes.length === 1) {
-             var index = nodes.indexOf(keyNodes[0]);
-             var oppositeIndex = Math.floor((index + nodes.length / 2) % nodes.length);
-             keyNodes.push(nodes[oppositeIndex]);
-             keyPoints.push(points[oppositeIndex]);
-           } // key points and nodes are those connected to the ways,
-           // 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 (var _i = iStart; _i != iEnd; _i += step) {
+               orderedPoints.push([points[_i].x, points[_i].y]);
+             }
 
+             return orderedPoints;
+           }
+         }, {
+           key: "isExteriorRing",
+           value: function isExteriorRing() {
+             if (this._isExteriorRing === undefined) {
+               var enclosing = this.enclosingRing();
+               this._isExteriorRing = enclosing ? !enclosing.isExteriorRing() : true;
+             }
 
-           for (i = 0; i < keyPoints.length; i++) {
-             var nextKeyNodeIndex = (i + 1) % keyNodes.length;
-             var startNode = keyNodes[i];
-             var endNode = keyNodes[nextKeyNodeIndex];
-             var startNodeIndex = nodes.indexOf(startNode);
-             var endNodeIndex = nodes.indexOf(endNode);
-             var numberNewPoints = -1;
-             var indexRange = endNodeIndex - startNodeIndex;
-             var nearNodes = {};
-             var inBetweenNodes = [];
-             var startAngle, endAngle, totalAngle, eachAngle;
-             var angle, loc, node, origNode;
+             return this._isExteriorRing;
+           }
+         }, {
+           key: "enclosingRing",
+           value: function enclosingRing() {
+             if (this._enclosingRing === undefined) {
+               this._enclosingRing = this._calcEnclosingRing();
+             }
 
-             if (indexRange < 0) {
-               indexRange += nodes.length;
-             } // position this key node
+             return this._enclosingRing;
+           }
+           /* Returns the ring that encloses this one, if any */
 
+         }, {
+           key: "_calcEnclosingRing",
+           value: function _calcEnclosingRing() {
+             // start with the ealier sweep line event so that the prevSeg
+             // chain doesn't lead us inside of a loop of ours
+             var leftMostEvt = this.events[0];
 
-             var distance = geoVecLength(centroid, keyPoints[i]) || 1e-4;
-             keyPoints[i] = [centroid[0] + (keyPoints[i][0] - centroid[0]) / distance * radius, centroid[1] + (keyPoints[i][1] - centroid[1]) / distance * radius];
-             loc = projection.invert(keyPoints[i]);
-             node = keyNodes[i];
-             origNode = origNodes[node.id];
-             node = node.move(geoVecInterp(origNode.loc, loc, t));
-             graph = graph.replace(node); // figure out the between delta angle we want to match to
+             for (var i = 1, iMax = this.events.length; i < iMax; i++) {
+               var evt = this.events[i];
+               if (SweepEvent.compare(leftMostEvt, evt) > 0) leftMostEvt = evt;
+             }
 
-             startAngle = Math.atan2(keyPoints[i][1] - centroid[1], keyPoints[i][0] - centroid[0]);
-             endAngle = Math.atan2(keyPoints[nextKeyNodeIndex][1] - centroid[1], keyPoints[nextKeyNodeIndex][0] - centroid[0]);
-             totalAngle = endAngle - startAngle; // detects looping around -pi/pi
+             var prevSeg = leftMostEvt.segment.prevInResult();
+             var prevPrevSeg = prevSeg ? prevSeg.prevInResult() : null;
+
+             while (true) {
+               // no segment found, thus no ring can enclose us
+               if (!prevSeg) return null; // no segments below prev segment found, thus the ring of the prev
+               // segment must loop back around and enclose us
 
-             if (totalAngle * sign > 0) {
-               totalAngle = -sign * (2 * Math.PI - Math.abs(totalAngle));
-             }
+               if (!prevPrevSeg) return prevSeg.ringOut; // if the two segments are of different rings, the ring of the prev
+               // segment must either loop around us or the ring of the prev prev
+               // seg, which would make us and the ring of the prev peers
 
-             do {
-               numberNewPoints++;
-               eachAngle = totalAngle / (indexRange + numberNewPoints);
-             } while (Math.abs(eachAngle) > maxAngle); // move existing nodes
+               if (prevPrevSeg.ringOut !== prevSeg.ringOut) {
+                 if (prevPrevSeg.ringOut.enclosingRing() !== prevSeg.ringOut) {
+                   return prevSeg.ringOut;
+                 } else return prevSeg.ringOut.enclosingRing();
+               } // two segments are from the same ring, so this was a penisula
+               // of that ring. iterate downward, keep searching
 
 
-             for (j = 1; j < indexRange; j++) {
-               angle = startAngle + j * eachAngle;
-               loc = projection.invert([centroid[0] + Math.cos(angle) * radius, centroid[1] + Math.sin(angle) * radius]);
-               node = nodes[(j + startNodeIndex) % nodes.length];
-               origNode = origNodes[node.id];
-               nearNodes[node.id] = angle;
-               node = node.move(geoVecInterp(origNode.loc, loc, t));
-               graph = graph.replace(node);
-             } // add new in between nodes if necessary
+               prevSeg = prevPrevSeg.prevInResult();
+               prevPrevSeg = prevSeg ? prevSeg.prevInResult() : null;
+             }
+           }
+         }]);
 
+         return RingOut;
+       }();
 
-             for (j = 0; j < numberNewPoints; j++) {
-               angle = startAngle + (indexRange + j) * eachAngle;
-               loc = projection.invert([centroid[0] + Math.cos(angle) * radius, centroid[1] + Math.sin(angle) * radius]); // choose a nearnode to use as the original
+       var PolyOut = /*#__PURE__*/function () {
+         function PolyOut(exteriorRing) {
+           _classCallCheck(this, PolyOut);
 
-               var min = Infinity;
+           this.exteriorRing = exteriorRing;
+           exteriorRing.poly = this;
+           this.interiorRings = [];
+         }
 
-               for (var nodeId in nearNodes) {
-                 var nearAngle = nearNodes[nodeId];
-                 var dist = Math.abs(nearAngle - angle);
+         _createClass(PolyOut, [{
+           key: "addInterior",
+           value: function addInterior(ring) {
+             this.interiorRings.push(ring);
+             ring.poly = this;
+           }
+         }, {
+           key: "getGeom",
+           value: function getGeom() {
+             var geom = [this.exteriorRing.getGeom()]; // exterior ring was all (within rounding error of angle calc) colinear points
 
-                 if (dist < min) {
-                   min = dist;
-                   origNode = origNodes[nodeId];
-                 }
-               }
+             if (geom[0] === null) return null;
 
-               node = osmNode({
-                 loc: geoVecInterp(origNode.loc, loc, t)
-               });
-               graph = graph.replace(node);
-               nodes.splice(endNodeIndex + j, 0, node);
-               inBetweenNodes.push(node.id);
-             } // Check for other ways that share these keyNodes..
-             // If keyNodes are adjacent in both ways,
-             // we can add inBetweenNodes to that shared way too..
+             for (var i = 0, iMax = this.interiorRings.length; i < iMax; i++) {
+               var ringGeom = this.interiorRings[i].getGeom(); // interior ring was all (within rounding error of angle calc) colinear points
 
+               if (ringGeom === null) continue;
+               geom.push(ringGeom);
+             }
 
-             if (indexRange === 1 && inBetweenNodes.length) {
-               var startIndex1 = way.nodes.lastIndexOf(startNode.id);
-               var endIndex1 = way.nodes.lastIndexOf(endNode.id);
-               var wayDirection1 = endIndex1 - startIndex1;
+             return geom;
+           }
+         }]);
 
-               if (wayDirection1 < -1) {
-                 wayDirection1 = 1;
-               }
+         return PolyOut;
+       }();
 
-               var parentWays = graph.parentWays(keyNodes[i]);
+       var MultiPolyOut = /*#__PURE__*/function () {
+         function MultiPolyOut(rings) {
+           _classCallCheck(this, MultiPolyOut);
 
-               for (j = 0; j < parentWays.length; j++) {
-                 var sharedWay = parentWays[j];
-                 if (sharedWay === way) continue;
+           this.rings = rings;
+           this.polys = this._composePolys(rings);
+         }
 
-                 if (sharedWay.areAdjacent(startNode.id, endNode.id)) {
-                   var startIndex2 = sharedWay.nodes.lastIndexOf(startNode.id);
-                   var endIndex2 = sharedWay.nodes.lastIndexOf(endNode.id);
-                   var wayDirection2 = endIndex2 - startIndex2;
-                   var insertAt = endIndex2;
+         _createClass(MultiPolyOut, [{
+           key: "getGeom",
+           value: function getGeom() {
+             var geom = [];
 
-                   if (wayDirection2 < -1) {
-                     wayDirection2 = 1;
-                   }
+             for (var i = 0, iMax = this.polys.length; i < iMax; i++) {
+               var polyGeom = this.polys[i].getGeom(); // exterior ring was all (within rounding error of angle calc) colinear points
 
-                   if (wayDirection1 !== wayDirection2) {
-                     inBetweenNodes.reverse();
-                     insertAt = startIndex2;
-                   }
+               if (polyGeom === null) continue;
+               geom.push(polyGeom);
+             }
 
-                   for (k = 0; k < inBetweenNodes.length; k++) {
-                     sharedWay = sharedWay.addNode(inBetweenNodes[k], insertAt + k);
-                   }
+             return geom;
+           }
+         }, {
+           key: "_composePolys",
+           value: function _composePolys(rings) {
+             var polys = [];
 
-                   graph = graph.replace(sharedWay);
-                 }
+             for (var i = 0, iMax = rings.length; i < iMax; i++) {
+               var ring = rings[i];
+               if (ring.poly) continue;
+               if (ring.isExteriorRing()) polys.push(new PolyOut(ring));else {
+                 var enclosingRing = ring.enclosingRing();
+                 if (!enclosingRing.poly) polys.push(new PolyOut(enclosingRing));
+                 enclosingRing.poly.addInterior(ring);
                }
              }
-           } // update the way to have all the new nodes
 
+             return polys;
+           }
+         }]);
 
-           ids = nodes.map(function (n) {
-             return n.id;
-           });
-           ids.push(ids[0]);
-           way = way.update({
-             nodes: ids
-           });
-           graph = graph.replace(way);
-           return graph;
-         };
+         return MultiPolyOut;
+       }();
+       /**
+        * NOTE:  We must be careful not to change any segments while
+        *        they are in the SplayTree. AFAIK, there's no way to tell
+        *        the tree to rebalance itself - thus before splitting
+        *        a segment that's in the tree, we remove it from the tree,
+        *        do the split, then re-insert it. (Even though splitting a
+        *        segment *shouldn't* change its correct position in the
+        *        sweep line tree, the reality is because of rounding errors,
+        *        it sometimes does.)
+        */
 
-         action.makeConvex = function (graph) {
-           var way = graph.entity(wayId);
-           var nodes = utilArrayUniq(graph.childNodes(way));
-           var points = nodes.map(function (n) {
-             return projection(n.loc);
-           });
-           var sign = d3_polygonArea(points) > 0 ? 1 : -1;
-           var hull = d3_polygonHull(points);
-           var i, j; // D3 convex hulls go counterclockwise..
 
-           if (sign === -1) {
-             nodes.reverse();
-             points.reverse();
-           }
+       var SweepLine = /*#__PURE__*/function () {
+         function SweepLine(queue) {
+           var comparator = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : Segment.compare;
 
-           for (i = 0; i < hull.length - 1; i++) {
-             var startIndex = points.indexOf(hull[i]);
-             var endIndex = points.indexOf(hull[i + 1]);
-             var indexRange = endIndex - startIndex;
+           _classCallCheck(this, SweepLine);
 
-             if (indexRange < 0) {
-               indexRange += nodes.length;
-             } // move interior nodes to the surface of the convex hull..
+           this.queue = queue;
+           this.tree = new Tree(comparator);
+           this.segments = [];
+         }
 
+         _createClass(SweepLine, [{
+           key: "process",
+           value: function process(event) {
+             var segment = event.segment;
+             var newEvents = []; // if we've already been consumed by another segment,
+             // clean up our body parts and get out
 
-             for (j = 1; j < indexRange; j++) {
-               var point = geoVecInterp(hull[i], hull[i + 1], j / indexRange);
-               var node = nodes[(j + startIndex) % nodes.length].move(projection.invert(point));
-               graph = graph.replace(node);
+             if (event.consumedBy) {
+               if (event.isLeft) this.queue.remove(event.otherSE);else this.tree.remove(segment);
+               return newEvents;
              }
-           }
 
-           return graph;
-         };
+             var node = event.isLeft ? this.tree.insert(segment) : this.tree.find(segment);
+             if (!node) throw new Error("Unable to find segment #".concat(segment.id, " ") + "[".concat(segment.leftSE.point.x, ", ").concat(segment.leftSE.point.y, "] -> ") + "[".concat(segment.rightSE.point.x, ", ").concat(segment.rightSE.point.y, "] ") + 'in SweepLine tree. Please submit a bug report.');
+             var prevNode = node;
+             var nextNode = node;
+             var prevSeg = undefined;
+             var nextSeg = undefined; // skip consumed segments still in tree
 
-         action.disabled = function (graph) {
-           if (!graph.entity(wayId).isClosed()) {
-             return 'not_closed';
-           } //disable when already circular
+             while (prevSeg === undefined) {
+               prevNode = this.tree.prev(prevNode);
+               if (prevNode === null) prevSeg = null;else if (prevNode.key.consumedBy === undefined) prevSeg = prevNode.key;
+             } // skip consumed segments still in tree
 
 
-           var way = graph.entity(wayId);
-           var nodes = utilArrayUniq(graph.childNodes(way));
-           var points = nodes.map(function (n) {
-             return projection(n.loc);
-           });
-           var hull = d3_polygonHull(points);
-           var epsilonAngle = Math.PI / 180;
+             while (nextSeg === undefined) {
+               nextNode = this.tree.next(nextNode);
+               if (nextNode === null) nextSeg = null;else if (nextNode.key.consumedBy === undefined) nextSeg = nextNode.key;
+             }
 
-           if (hull.length !== points.length || hull.length < 3) {
-             return false;
-           }
+             if (event.isLeft) {
+               // Check for intersections against the previous segment in the sweep line
+               var prevMySplitter = null;
 
-           var centroid = d3_polygonCentroid(points);
-           var radius = geoVecLengthSquare(centroid, points[0]);
-           var i, actualPoint; // compare distances between centroid and points
+               if (prevSeg) {
+                 var prevInter = prevSeg.getIntersection(segment);
 
-           for (i = 0; i < hull.length; i++) {
-             actualPoint = hull[i];
-             var actualDist = geoVecLengthSquare(actualPoint, centroid);
-             var diff = Math.abs(actualDist - radius); //compare distances with epsilon-error (5%)
+                 if (prevInter !== null) {
+                   if (!segment.isAnEndpoint(prevInter)) prevMySplitter = prevInter;
 
-             if (diff > 0.05 * radius) {
-               return false;
-             }
-           } //check if central angles are smaller than maxAngle
+                   if (!prevSeg.isAnEndpoint(prevInter)) {
+                     var newEventsFromSplit = this._splitSafely(prevSeg, prevInter);
 
+                     for (var i = 0, iMax = newEventsFromSplit.length; i < iMax; i++) {
+                       newEvents.push(newEventsFromSplit[i]);
+                     }
+                   }
+                 }
+               } // Check for intersections against the next segment in the sweep line
 
-           for (i = 0; i < hull.length; i++) {
-             actualPoint = hull[i];
-             var nextPoint = hull[(i + 1) % hull.length];
-             var startAngle = Math.atan2(actualPoint[1] - centroid[1], actualPoint[0] - centroid[0]);
-             var endAngle = Math.atan2(nextPoint[1] - centroid[1], nextPoint[0] - centroid[0]);
-             var angle = endAngle - startAngle;
 
-             if (angle < 0) {
-               angle = -angle;
-             }
+               var nextMySplitter = null;
 
-             if (angle > Math.PI) {
-               angle = 2 * Math.PI - angle;
-             }
+               if (nextSeg) {
+                 var nextInter = nextSeg.getIntersection(segment);
 
-             if (angle > maxAngle + epsilonAngle) {
-               return false;
-             }
-           }
+                 if (nextInter !== null) {
+                   if (!segment.isAnEndpoint(nextInter)) nextMySplitter = nextInter;
 
-           return 'already_circular';
-         };
+                   if (!nextSeg.isAnEndpoint(nextInter)) {
+                     var _newEventsFromSplit = this._splitSafely(nextSeg, nextInter);
 
-         action.transitionable = true;
-         return action;
-       }
+                     for (var _i = 0, _iMax = _newEventsFromSplit.length; _i < _iMax; _i++) {
+                       newEvents.push(_newEventsFromSplit[_i]);
+                     }
+                   }
+                 }
+               } // For simplicity, even if we find more than one intersection we only
+               // spilt on the 'earliest' (sweep-line style) of the intersections.
+               // The other intersection will be handled in a future process().
 
-       function actionDeleteWay(wayID) {
-         function canDeleteNode(node, graph) {
-           // don't delete nodes still attached to ways or relations
-           if (graph.parentWays(node).length || graph.parentRelations(node).length) return false;
-           var geometries = osmNodeGeometriesForTags(node.tags); // don't delete if this node can be a standalone point
 
-           if (geometries.point) return false; // delete if this node only be a vertex
+               if (prevMySplitter !== null || nextMySplitter !== null) {
+                 var mySplitter = null;
+                 if (prevMySplitter === null) mySplitter = nextMySplitter;else if (nextMySplitter === null) mySplitter = prevMySplitter;else {
+                   var cmpSplitters = SweepEvent.comparePoints(prevMySplitter, nextMySplitter);
+                   mySplitter = cmpSplitters <= 0 ? prevMySplitter : nextMySplitter;
+                 } // Rounding errors can cause changes in ordering,
+                 // so remove afected segments and right sweep events before splitting
 
-           if (geometries.vertex) return true; // iD doesn't know if this should be a point or vertex,
-           // so only delete if there are no interesting tags
+                 this.queue.remove(segment.rightSE);
+                 newEvents.push(segment.rightSE);
 
-           return !node.hasInterestingTags();
-         }
+                 var _newEventsFromSplit2 = segment.split(mySplitter);
 
-         var action = function action(graph) {
-           var way = graph.entity(wayID);
-           graph.parentRelations(way).forEach(function (parent) {
-             parent = parent.removeMembersWithID(wayID);
-             graph = graph.replace(parent);
+                 for (var _i2 = 0, _iMax2 = _newEventsFromSplit2.length; _i2 < _iMax2; _i2++) {
+                   newEvents.push(_newEventsFromSplit2[_i2]);
+                 }
+               }
 
-             if (parent.isDegenerate()) {
-               graph = actionDeleteRelation(parent.id)(graph);
-             }
-           });
-           new Set(way.nodes).forEach(function (nodeID) {
-             graph = graph.replace(way.removeNode(nodeID));
-             var node = graph.entity(nodeID);
+               if (newEvents.length > 0) {
+                 // We found some intersections, so re-do the current event to
+                 // make sure sweep line ordering is totally consistent for later
+                 // use with the segment 'prev' pointers
+                 this.tree.remove(segment);
+                 newEvents.push(event);
+               } else {
+                 // done with left event
+                 this.segments.push(segment);
+                 segment.prev = prevSeg;
+               }
+             } else {
+               // event.isRight
+               // since we're about to be removed from the sweep line, check for
+               // intersections between our previous and next segments
+               if (prevSeg && nextSeg) {
+                 var inter = prevSeg.getIntersection(nextSeg);
 
-             if (canDeleteNode(node, graph)) {
-               graph = graph.remove(node);
-             }
-           });
-           return graph.remove(way);
-         };
+                 if (inter !== null) {
+                   if (!prevSeg.isAnEndpoint(inter)) {
+                     var _newEventsFromSplit3 = this._splitSafely(prevSeg, inter);
 
-         return action;
-       }
+                     for (var _i3 = 0, _iMax3 = _newEventsFromSplit3.length; _i3 < _iMax3; _i3++) {
+                       newEvents.push(_newEventsFromSplit3[_i3]);
+                     }
+                   }
 
-       function actionDeleteMultiple(ids) {
-         var actions = {
-           way: actionDeleteWay,
-           node: actionDeleteNode,
-           relation: actionDeleteRelation
-         };
+                   if (!nextSeg.isAnEndpoint(inter)) {
+                     var _newEventsFromSplit4 = this._splitSafely(nextSeg, inter);
 
-         var action = function action(graph) {
-           ids.forEach(function (id) {
-             if (graph.hasEntity(id)) {
-               // It may have been deleted already.
-               graph = actions[graph.entity(id).type](id)(graph);
+                     for (var _i4 = 0, _iMax4 = _newEventsFromSplit4.length; _i4 < _iMax4; _i4++) {
+                       newEvents.push(_newEventsFromSplit4[_i4]);
+                     }
+                   }
+                 }
+               }
+
+               this.tree.remove(segment);
              }
-           });
-           return graph;
-         };
 
-         return action;
-       }
+             return newEvents;
+           }
+           /* Safely split a segment that is currently in the datastructures
+            * IE - a segment other than the one that is currently being processed. */
 
-       function actionDeleteRelation(relationID, allowUntaggedMembers) {
-         function canDeleteEntity(entity, graph) {
-           return !graph.parentWays(entity).length && !graph.parentRelations(entity).length && !entity.hasInterestingTags() && !allowUntaggedMembers;
+         }, {
+           key: "_splitSafely",
+           value: function _splitSafely(seg, pt) {
+             // Rounding errors can cause changes in ordering,
+             // so remove afected segments and right sweep events before splitting
+             // removeNode() doesn't work, so have re-find the seg
+             // https://github.com/w8r/splay-tree/pull/5
+             this.tree.remove(seg);
+             var rightSE = seg.rightSE;
+             this.queue.remove(rightSE);
+             var newEvents = seg.split(pt);
+             newEvents.push(rightSE); // splitting can trigger consumption
+
+             if (seg.consumedBy === undefined) this.tree.insert(seg);
+             return newEvents;
+           }
+         }]);
+
+         return SweepLine;
+       }();
+
+       var POLYGON_CLIPPING_MAX_QUEUE_SIZE = typeof process !== 'undefined' && process.env.POLYGON_CLIPPING_MAX_QUEUE_SIZE || 1000000;
+       var POLYGON_CLIPPING_MAX_SWEEPLINE_SEGMENTS = typeof process !== 'undefined' && process.env.POLYGON_CLIPPING_MAX_SWEEPLINE_SEGMENTS || 1000000;
+
+       var Operation = /*#__PURE__*/function () {
+         function Operation() {
+           _classCallCheck(this, Operation);
          }
 
-         var action = function action(graph) {
-           var relation = graph.entity(relationID);
-           graph.parentRelations(relation).forEach(function (parent) {
-             parent = parent.removeMembersWithID(relationID);
-             graph = graph.replace(parent);
+         _createClass(Operation, [{
+           key: "run",
+           value: function run(type, geom, moreGeoms) {
+             operation.type = type;
+             rounder.reset();
+             /* Convert inputs to MultiPoly objects */
 
-             if (parent.isDegenerate()) {
-               graph = actionDeleteRelation(parent.id)(graph);
-             }
-           });
-           var memberIDs = utilArrayUniq(relation.members.map(function (m) {
-             return m.id;
-           }));
-           memberIDs.forEach(function (memberID) {
-             graph = graph.replace(relation.removeMembersWithID(memberID));
-             var entity = graph.entity(memberID);
+             var multipolys = [new MultiPolyIn(geom, true)];
 
-             if (canDeleteEntity(entity, graph)) {
-               graph = actionDeleteMultiple([memberID])(graph);
+             for (var i = 0, iMax = moreGeoms.length; i < iMax; i++) {
+               multipolys.push(new MultiPolyIn(moreGeoms[i], false));
              }
-           });
-           return graph.remove(relation);
-         };
 
-         return action;
-       }
+             operation.numMultiPolys = multipolys.length;
+             /* BBox optimization for difference operation
+              * If the bbox of a multipolygon that's part of the clipping doesn't
+              * intersect the bbox of the subject at all, we can just drop that
+              * multiploygon. */
 
-       function actionDeleteNode(nodeId) {
-         var action = function action(graph) {
-           var node = graph.entity(nodeId);
-           graph.parentWays(node).forEach(function (parent) {
-             parent = parent.removeNode(nodeId);
-             graph = graph.replace(parent);
+             if (operation.type === 'difference') {
+               // in place removal
+               var subject = multipolys[0];
+               var _i = 1;
 
-             if (parent.isDegenerate()) {
-               graph = actionDeleteWay(parent.id)(graph);
+               while (_i < multipolys.length) {
+                 if (getBboxOverlap(multipolys[_i].bbox, subject.bbox) !== null) _i++;else multipolys.splice(_i, 1);
+               }
              }
-           });
-           graph.parentRelations(node).forEach(function (parent) {
-             parent = parent.removeMembersWithID(nodeId);
-             graph = graph.replace(parent);
+             /* BBox optimization for intersection operation
+              * If we can find any pair of multipolygons whose bbox does not overlap,
+              * then the result will be empty. */
 
-             if (parent.isDegenerate()) {
-               graph = actionDeleteRelation(parent.id)(graph);
-             }
-           });
-           return graph.remove(node);
-         };
 
-         return action;
-       }
+             if (operation.type === 'intersection') {
+               // TODO: this is O(n^2) in number of polygons. By sorting the bboxes,
+               //       it could be optimized to O(n * ln(n))
+               for (var _i2 = 0, _iMax = multipolys.length; _i2 < _iMax; _i2++) {
+                 var mpA = multipolys[_i2];
 
-       //
-       // First choose a node to be the survivor, with preference given
-       // to an existing (not new) node.
-       //
-       // Tags and relation memberships of of non-surviving nodes are merged
-       // to the survivor.
-       //
-       // This is the inverse of `iD.actionDisconnect`.
-       //
-       // Reference:
-       //   https://github.com/openstreetmap/potlatch2/blob/master/net/systemeD/halcyon/connection/actions/MergeNodesAction.as
-       //   https://github.com/openstreetmap/josm/blob/mirror/src/org/openstreetmap/josm/actions/MergeNodesAction.java
-       //
+                 for (var j = _i2 + 1, jMax = multipolys.length; j < jMax; j++) {
+                   if (getBboxOverlap(mpA.bbox, multipolys[j].bbox) === null) return [];
+                 }
+               }
+             }
+             /* Put segment endpoints in a priority queue */
 
-       function actionConnect(nodeIDs) {
-         var action = function action(graph) {
-           var survivor;
-           var node;
-           var parents;
-           var i, j; // Choose a survivor node, prefer an existing (not new) node - #4974
 
-           for (i = 0; i < nodeIDs.length; i++) {
-             survivor = graph.entity(nodeIDs[i]);
-             if (survivor.version) break; // found one
-           } // Replace all non-surviving nodes with the survivor and merge tags.
+             var queue = new Tree(SweepEvent.compare);
 
+             for (var _i3 = 0, _iMax2 = multipolys.length; _i3 < _iMax2; _i3++) {
+               var sweepEvents = multipolys[_i3].getSweepEvents();
 
-           for (i = 0; i < nodeIDs.length; i++) {
-             node = graph.entity(nodeIDs[i]);
-             if (node.id === survivor.id) continue;
-             parents = graph.parentWays(node);
+               for (var _j = 0, _jMax = sweepEvents.length; _j < _jMax; _j++) {
+                 queue.insert(sweepEvents[_j]);
 
-             for (j = 0; j < parents.length; j++) {
-               graph = graph.replace(parents[j].replaceNode(node.id, survivor.id));
+                 if (queue.size > POLYGON_CLIPPING_MAX_QUEUE_SIZE) {
+                   // prevents an infinite loop, an otherwise common manifestation of bugs
+                   throw new Error('Infinite loop when putting segment endpoints in a priority queue ' + '(queue size too big). Please file a bug report.');
+                 }
+               }
              }
+             /* Pass the sweep line over those endpoints */
 
-             parents = graph.parentRelations(node);
 
-             for (j = 0; j < parents.length; j++) {
-               graph = graph.replace(parents[j].replaceMember(node, survivor));
-             }
+             var sweepLine = new SweepLine(queue);
+             var prevQueueSize = queue.size;
+             var node = queue.pop();
 
-             survivor = survivor.mergeTags(node.tags);
-             graph = actionDeleteNode(node.id)(graph);
-           }
+             while (node) {
+               var evt = node.key;
 
-           graph = graph.replace(survivor); // find and delete any degenerate ways created by connecting adjacent vertices
+               if (queue.size === prevQueueSize) {
+                 // prevents an infinite loop, an otherwise common manifestation of bugs
+                 var seg = evt.segment;
+                 throw new Error("Unable to pop() ".concat(evt.isLeft ? 'left' : 'right', " SweepEvent ") + "[".concat(evt.point.x, ", ").concat(evt.point.y, "] from segment #").concat(seg.id, " ") + "[".concat(seg.leftSE.point.x, ", ").concat(seg.leftSE.point.y, "] -> ") + "[".concat(seg.rightSE.point.x, ", ").concat(seg.rightSE.point.y, "] from queue. ") + 'Please file a bug report.');
+               }
 
-           parents = graph.parentWays(survivor);
+               if (queue.size > POLYGON_CLIPPING_MAX_QUEUE_SIZE) {
+                 // prevents an infinite loop, an otherwise common manifestation of bugs
+                 throw new Error('Infinite loop when passing sweep line over endpoints ' + '(queue size too big). Please file a bug report.');
+               }
 
-           for (i = 0; i < parents.length; i++) {
-             if (parents[i].isDegenerate()) {
-               graph = actionDeleteWay(parents[i].id)(graph);
-             }
-           }
+               if (sweepLine.segments.length > POLYGON_CLIPPING_MAX_SWEEPLINE_SEGMENTS) {
+                 // prevents an infinite loop, an otherwise common manifestation of bugs
+                 throw new Error('Infinite loop when passing sweep line over endpoints ' + '(too many sweep line segments). Please file a bug report.');
+               }
 
-           return graph;
-         };
+               var newEvents = sweepLine.process(evt);
 
-         action.disabled = function (graph) {
-           var seen = {};
-           var restrictionIDs = [];
-           var survivor;
-           var node, way;
-           var relations, relation, role;
-           var i, j, k; // Choose a survivor node, prefer an existing (not new) node - #4974
+               for (var _i4 = 0, _iMax3 = newEvents.length; _i4 < _iMax3; _i4++) {
+                 var _evt = newEvents[_i4];
+                 if (_evt.consumedBy === undefined) queue.insert(_evt);
+               }
 
-           for (i = 0; i < nodeIDs.length; i++) {
-             survivor = graph.entity(nodeIDs[i]);
-             if (survivor.version) break; // found one
-           } // 1. disable if the nodes being connected have conflicting relation roles
+               prevQueueSize = queue.size;
+               node = queue.pop();
+             } // free some memory we don't need anymore
 
 
-           for (i = 0; i < nodeIDs.length; i++) {
-             node = graph.entity(nodeIDs[i]);
-             relations = graph.parentRelations(node);
+             rounder.reset();
+             /* Collect and compile segments we're keeping into a multipolygon */
 
-             for (j = 0; j < relations.length; j++) {
-               relation = relations[j];
-               role = relation.memberById(node.id).role || ''; // if this node is a via node in a restriction, remember for later
+             var ringsOut = RingOut.factory(sweepLine.segments);
+             var result = new MultiPolyOut(ringsOut);
+             return result.getGeom();
+           }
+         }]);
 
-               if (relation.hasFromViaTo()) {
-                 restrictionIDs.push(relation.id);
-               }
+         return Operation;
+       }(); // singleton available by import
 
-               if (seen[relation.id] !== undefined && seen[relation.id] !== role) {
-                 return 'relation';
-               } else {
-                 seen[relation.id] = role;
-               }
-             }
-           } // gather restrictions for parent ways
 
+       var operation = new Operation();
 
-           for (i = 0; i < nodeIDs.length; i++) {
-             node = graph.entity(nodeIDs[i]);
-             var parents = graph.parentWays(node);
+       var union = function union(geom) {
+         for (var _len = arguments.length, moreGeoms = new Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) {
+           moreGeoms[_key - 1] = arguments[_key];
+         }
 
-             for (j = 0; j < parents.length; j++) {
-               var parent = parents[j];
-               relations = graph.parentRelations(parent);
+         return operation.run('union', geom, moreGeoms);
+       };
 
-               for (k = 0; k < relations.length; k++) {
-                 relation = relations[k];
+       var intersection$1 = function intersection(geom) {
+         for (var _len2 = arguments.length, moreGeoms = new Array(_len2 > 1 ? _len2 - 1 : 0), _key2 = 1; _key2 < _len2; _key2++) {
+           moreGeoms[_key2 - 1] = arguments[_key2];
+         }
 
-                 if (relation.hasFromViaTo()) {
-                   restrictionIDs.push(relation.id);
-                 }
-               }
-             }
-           } // test restrictions
+         return operation.run('intersection', geom, moreGeoms);
+       };
 
+       var xor = function xor(geom) {
+         for (var _len3 = arguments.length, moreGeoms = new Array(_len3 > 1 ? _len3 - 1 : 0), _key3 = 1; _key3 < _len3; _key3++) {
+           moreGeoms[_key3 - 1] = arguments[_key3];
+         }
 
-           restrictionIDs = utilArrayUniq(restrictionIDs);
+         return operation.run('xor', geom, moreGeoms);
+       };
 
-           for (i = 0; i < restrictionIDs.length; i++) {
-             relation = graph.entity(restrictionIDs[i]);
-             if (!relation.isComplete(graph)) continue;
-             var memberWays = relation.members.filter(function (m) {
-               return m.type === 'way';
-             }).map(function (m) {
-               return graph.entity(m.id);
-             });
-             memberWays = utilArrayUniq(memberWays);
-             var f = relation.memberByRole('from');
-             var t = relation.memberByRole('to');
-             var isUturn = f.id === t.id; // 2a. disable if connection would damage a restriction
-             // (a key node is a node at the junction of ways)
+       var difference = function difference(subjectGeom) {
+         for (var _len4 = arguments.length, clippingGeoms = new Array(_len4 > 1 ? _len4 - 1 : 0), _key4 = 1; _key4 < _len4; _key4++) {
+           clippingGeoms[_key4 - 1] = arguments[_key4];
+         }
 
-             var nodes = {
-               from: [],
-               via: [],
-               to: [],
-               keyfrom: [],
-               keyto: []
-             };
+         return operation.run('difference', subjectGeom, clippingGeoms);
+       };
 
-             for (j = 0; j < relation.members.length; j++) {
-               collectNodes(relation.members[j], nodes);
+       var index = {
+         union: union,
+         intersection: intersection$1,
+         xor: xor,
+         difference: difference
+       };
+
+       var geojsonPrecision = createCommonjsModule(function (module) {
+         (function () {
+           function parse(t, coordinatePrecision, extrasPrecision) {
+             function point(p) {
+               return p.map(function (e, index) {
+                 if (index < 2) {
+                   return 1 * e.toFixed(coordinatePrecision);
+                 } else {
+                   return 1 * e.toFixed(extrasPrecision);
+                 }
+               });
              }
 
-             nodes.keyfrom = utilArrayUniq(nodes.keyfrom.filter(hasDuplicates));
-             nodes.keyto = utilArrayUniq(nodes.keyto.filter(hasDuplicates));
-             var filter = keyNodeFilter(nodes.keyfrom, nodes.keyto);
-             nodes.from = nodes.from.filter(filter);
-             nodes.via = nodes.via.filter(filter);
-             nodes.to = nodes.to.filter(filter);
-             var connectFrom = false;
-             var connectVia = false;
-             var connectTo = false;
-             var connectKeyFrom = false;
-             var connectKeyTo = false;
+             function multi(l) {
+               return l.map(point);
+             }
 
-             for (j = 0; j < nodeIDs.length; j++) {
-               var n = nodeIDs[j];
+             function poly(p) {
+               return p.map(multi);
+             }
 
-               if (nodes.from.indexOf(n) !== -1) {
-                 connectFrom = true;
-               }
+             function multiPoly(m) {
+               return m.map(poly);
+             }
 
-               if (nodes.via.indexOf(n) !== -1) {
-                 connectVia = true;
+             function geometry(obj) {
+               if (!obj) {
+                 return {};
                }
 
-               if (nodes.to.indexOf(n) !== -1) {
-                 connectTo = true;
-               }
+               switch (obj.type) {
+                 case "Point":
+                   obj.coordinates = point(obj.coordinates);
+                   return obj;
 
-               if (nodes.keyfrom.indexOf(n) !== -1) {
-                 connectKeyFrom = true;
-               }
+                 case "LineString":
+                 case "MultiPoint":
+                   obj.coordinates = multi(obj.coordinates);
+                   return obj;
 
-               if (nodes.keyto.indexOf(n) !== -1) {
-                 connectKeyTo = true;
+                 case "Polygon":
+                 case "MultiLineString":
+                   obj.coordinates = poly(obj.coordinates);
+                   return obj;
+
+                 case "MultiPolygon":
+                   obj.coordinates = multiPoly(obj.coordinates);
+                   return obj;
+
+                 case "GeometryCollection":
+                   obj.geometries = obj.geometries.map(geometry);
+                   return obj;
+
+                 default:
+                   return {};
                }
              }
 
-             if (connectFrom && connectTo && !isUturn) {
-               return 'restriction';
+             function feature(obj) {
+               obj.geometry = geometry(obj.geometry);
+               return obj;
              }
 
-             if (connectFrom && connectVia) {
-               return 'restriction';
+             function featureCollection(f) {
+               f.features = f.features.map(feature);
+               return f;
              }
 
-             if (connectTo && connectVia) {
-               return 'restriction';
-             } // connecting to a key node -
-             // if both nodes are on a member way (i.e. part of the turn restriction),
-             // the connecting node must be adjacent to the key node.
+             function geometryCollection(g) {
+               g.geometries = g.geometries.map(geometry);
+               return g;
+             }
 
+             if (!t) {
+               return t;
+             }
 
-             if (connectKeyFrom || connectKeyTo) {
-               if (nodeIDs.length !== 2) {
-                 return 'restriction';
-               }
+             switch (t.type) {
+               case "Feature":
+                 return feature(t);
 
-               var n0 = null;
-               var n1 = null;
+               case "GeometryCollection":
+                 return geometryCollection(t);
 
-               for (j = 0; j < memberWays.length; j++) {
-                 way = memberWays[j];
+               case "FeatureCollection":
+                 return featureCollection(t);
 
-                 if (way.contains(nodeIDs[0])) {
-                   n0 = nodeIDs[0];
-                 }
+               case "Point":
+               case "LineString":
+               case "Polygon":
+               case "MultiPoint":
+               case "MultiPolygon":
+               case "MultiLineString":
+                 return geometry(t);
 
-                 if (way.contains(nodeIDs[1])) {
-                   n1 = nodeIDs[1];
-                 }
-               }
+               default:
+                 return t;
+             }
+           }
 
-               if (n0 && n1) {
-                 // both nodes are part of the restriction
-                 var ok = false;
+           module.exports = parse;
+           module.exports.parse = parse;
+         })();
+       });
 
-                 for (j = 0; j < memberWays.length; j++) {
-                   way = memberWays[j];
+       var FORCED$3 = fails(function () {
+         return new Date(NaN).toJSON() !== null
+           || Date.prototype.toJSON.call({ toISOString: function () { return 1; } }) !== 1;
+       });
 
-                   if (way.areAdjacent(n0, n1)) {
-                     ok = true;
-                     break;
-                   }
-                 }
+       // `Date.prototype.toJSON` method
+       // https://tc39.es/ecma262/#sec-date.prototype.tojson
+       _export({ target: 'Date', proto: true, forced: FORCED$3 }, {
+         // eslint-disable-next-line no-unused-vars -- required for `.length`
+         toJSON: function toJSON(key) {
+           var O = toObject(this);
+           var pv = toPrimitive(O);
+           return typeof pv == 'number' && !isFinite(pv) ? null : O.toISOString();
+         }
+       });
 
-                 if (!ok) {
-                   return 'restriction';
-                 }
-               }
-             } // 2b. disable if nodes being connected will destroy a member way in a restriction
-             // (to test, make a copy and try actually connecting the nodes)
+       // `URL.prototype.toJSON` method
+       // https://url.spec.whatwg.org/#dom-url-tojson
+       _export({ target: 'URL', proto: true, enumerable: true }, {
+         toJSON: function toJSON() {
+           return URL.prototype.toString.call(this);
+         }
+       });
 
+       function isObject$3(obj) {
+         return _typeof(obj) === 'object' && obj !== null;
+       }
 
-             for (j = 0; j < memberWays.length; j++) {
-               way = memberWays[j].update({}); // make copy
+       function forEach(obj, cb) {
+         if (Array.isArray(obj)) {
+           obj.forEach(cb);
+         } else if (isObject$3(obj)) {
+           Object.keys(obj).forEach(function (key) {
+             var val = obj[key];
+             cb(val, key);
+           });
+         }
+       }
 
-               for (k = 0; k < nodeIDs.length; k++) {
-                 if (nodeIDs[k] === survivor.id) continue;
+       function getTreeDepth(obj) {
+         var depth = 0;
 
-                 if (way.areAdjacent(nodeIDs[k], survivor.id)) {
-                   way = way.removeNode(nodeIDs[k]);
-                 } else {
-                   way = way.replaceNode(nodeIDs[k], survivor.id);
-                 }
-               }
+         if (Array.isArray(obj) || isObject$3(obj)) {
+           forEach(obj, function (val) {
+             if (Array.isArray(val) || isObject$3(val)) {
+               var tmpDepth = getTreeDepth(val);
 
-               if (way.isDegenerate()) {
-                 return 'restriction';
+               if (tmpDepth > depth) {
+                 depth = tmpDepth;
                }
              }
-           }
+           });
+           return depth + 1;
+         }
 
-           return false; // if a key node appears multiple times (indexOf !== lastIndexOf) it's a FROM-VIA or TO-VIA junction
+         return depth;
+       }
 
-           function hasDuplicates(n, i, arr) {
-             return arr.indexOf(n) !== arr.lastIndexOf(n);
+       function stringify(obj, options) {
+         options = options || {};
+         var indent = JSON.stringify([1], null, get(options, 'indent', 2)).slice(2, -3);
+         var addMargin = get(options, 'margins', false);
+         var addArrayMargin = get(options, 'arrayMargins', false);
+         var addObjectMargin = get(options, 'objectMargins', false);
+         var maxLength = indent === '' ? Infinity : get(options, 'maxLength', 80);
+         var maxNesting = get(options, 'maxNesting', Infinity);
+         return function _stringify(obj, currentIndent, reserved) {
+           if (obj && typeof obj.toJSON === 'function') {
+             obj = obj.toJSON();
            }
 
-           function keyNodeFilter(froms, tos) {
-             return function (n) {
-               return froms.indexOf(n) === -1 && tos.indexOf(n) === -1;
-             };
+           var string = JSON.stringify(obj);
+
+           if (string === undefined) {
+             return string;
            }
 
-           function collectNodes(member, collection) {
-             var entity = graph.hasEntity(member.id);
-             if (!entity) return;
-             var role = member.role || '';
+           var length = maxLength - currentIndent.length - reserved;
+           var treeDepth = getTreeDepth(obj);
 
-             if (!collection[role]) {
-               collection[role] = [];
+           if (treeDepth <= maxNesting && string.length <= length) {
+             var prettified = prettify(string, {
+               addMargin: addMargin,
+               addArrayMargin: addArrayMargin,
+               addObjectMargin: addObjectMargin
+             });
+
+             if (prettified.length <= length) {
+               return prettified;
              }
+           }
 
-             if (member.type === 'node') {
-               collection[role].push(member.id);
+           if (isObject$3(obj)) {
+             var nextIndent = currentIndent + indent;
+             var items = [];
+             var delimiters;
 
-               if (role === 'via') {
-                 collection.keyfrom.push(member.id);
-                 collection.keyto.push(member.id);
-               }
-             } else if (member.type === 'way') {
-               collection[role].push.apply(collection[role], entity.nodes);
+             var comma = function comma(array, index) {
+               return index === array.length - 1 ? 0 : 1;
+             };
 
-               if (role === 'from' || role === 'via') {
-                 collection.keyfrom.push(entity.first());
-                 collection.keyfrom.push(entity.last());
+             if (Array.isArray(obj)) {
+               for (var index = 0; index < obj.length; index++) {
+                 items.push(_stringify(obj[index], nextIndent, comma(obj, index)) || 'null');
                }
 
-               if (role === 'to' || role === 'via') {
-                 collection.keyto.push(entity.first());
-                 collection.keyto.push(entity.last());
-               }
+               delimiters = '[]';
+             } else {
+               Object.keys(obj).forEach(function (key, index, array) {
+                 var keyPart = JSON.stringify(key) + ': ';
+
+                 var value = _stringify(obj[key], nextIndent, keyPart.length + comma(array, index));
+
+                 if (value !== undefined) {
+                   items.push(keyPart + value);
+                 }
+               });
+               delimiters = '{}';
+             }
+
+             if (items.length > 0) {
+               return [delimiters[0], indent + items.join(',\n' + nextIndent), delimiters[1]].join('\n' + currentIndent);
              }
            }
+
+           return string;
+         }(obj, '', 0);
+       } // Note: This regex matches even invalid JSON strings, but since we’re
+       // working on the output of `JSON.stringify` we know that only valid strings
+       // are present (unless the user supplied a weird `options.indent` but in
+       // that case we don’t care since the output would be invalid anyway).
+
+
+       var stringOrChar = /("(?:[^\\"]|\\.)*")|[:,\][}{]/g;
+
+       function prettify(string, options) {
+         options = options || {};
+         var tokens = {
+           '{': '{',
+           '}': '}',
+           '[': '[',
+           ']': ']',
+           ',': ', ',
+           ':': ': '
          };
 
-         return action;
+         if (options.addMargin || options.addObjectMargin) {
+           tokens['{'] = '{ ';
+           tokens['}'] = ' }';
+         }
+
+         if (options.addMargin || options.addArrayMargin) {
+           tokens['['] = '[ ';
+           tokens[']'] = ' ]';
+         }
+
+         return string.replace(stringOrChar, function (match, string) {
+           return string ? match : tokens[match];
+         });
        }
 
-       function actionCopyEntities(ids, fromGraph) {
-         var _copies = {};
+       function get(options, name, defaultValue) {
+         return name in options ? options[name] : defaultValue;
+       }
+
+       var jsonStringifyPrettyCompact = stringify;
+
+       var _default = /*#__PURE__*/function () {
+         // constructor
+         //
+         // `fc`  Optional FeatureCollection of known features
+         //
+         // Optionally pass a GeoJSON FeatureCollection of known features which we can refer to later.
+         // Each feature must have a filename-like `id`, for example: `something.geojson`
+         //
+         // {
+         //   "type": "FeatureCollection"
+         //   "features": [
+         //     {
+         //       "type": "Feature",
+         //       "id": "philly_metro.geojson",
+         //       "properties": { … },
+         //       "geometry": { … }
+         //     }
+         //   ]
+         // }
+         function _default(fc) {
+           var _this = this;
+
+           _classCallCheck$1(this, _default);
+
+           // The _cache retains resolved features, so if you ask for the same thing multiple times
+           // we don't repeat the expensive resolving/clipping operations.
+           //
+           // Each feature has a stable identifier that is used as the cache key.
+           // The identifiers look like:
+           // - for point locations, the stringified point:          e.g. '[8.67039,49.41882]'
+           // - for geojson locations, the geojson id:               e.g. 'de-hamburg.geojson'
+           // - for countrycoder locations, feature.id property:     e.g. 'Q2'  (countrycoder uses Wikidata identifiers)
+           // - for aggregated locationSets, +[include]-[exclude]:   e.g '+[Q2]-[Q18,Q27611]'
+           this._cache = {}; // When strict mode = true, throw on invalid locations or locationSets.
+           // When strict mode = false, return `null` for invalid locations or locationSets.
+
+           this._strict = true; // process input FeatureCollection
+
+           if (fc && fc.type === 'FeatureCollection' && Array.isArray(fc.features)) {
+             fc.features.forEach(function (feature) {
+               feature.properties = feature.properties || {};
+               var props = feature.properties; // Get `id` from either `id` or `properties`
 
-         var action = function action(graph) {
-           ids.forEach(function (id) {
-             fromGraph.entity(id).copy(fromGraph, _copies);
-           });
+               var id = feature.id || props.id;
+               if (!id || !/^\S+\.geojson$/i.test(id)) return; // Ensure `id` exists and is lowercase
 
-           for (var id in _copies) {
-             graph = graph.replace(_copies[id]);
-           }
+               id = id.toLowerCase();
+               feature.id = id;
+               props.id = id; // Ensure `area` property exists
 
-           return graph;
-         };
+               if (!props.area) {
+                 var area = geojsonArea.geometry(feature.geometry) / 1e6; // m² to km²
 
-         action.copies = function () {
-           return _copies;
-         };
+                 props.area = Number(area.toFixed(2));
+               }
 
-         return action;
-       }
+               _this._cache[id] = feature;
+             });
+           } // Replace CountryCoder world geometry to be a polygon covering the world.
 
-       function actionDeleteMember(relationId, memberIndex) {
-         return function (graph) {
-           var relation = graph.entity(relationId).removeMember(memberIndex);
-           graph = graph.replace(relation);
-           if (relation.isDegenerate()) graph = actionDeleteRelation(relation.id)(graph);
-           return graph;
-         };
-       }
 
-       function actionDiscardTags(difference, discardTags) {
-         discardTags = discardTags || {};
-         return function (graph) {
-           difference.modified().forEach(checkTags);
-           difference.created().forEach(checkTags);
-           return graph;
+           var world = _cloneDeep(feature$1('Q2'));
 
-           function checkTags(entity) {
-             var keys = Object.keys(entity.tags);
-             var didDiscard = false;
-             var tags = {};
+           world.geometry = {
+             type: 'Polygon',
+             coordinates: [[[-180, -90], [180, -90], [180, 90], [-180, 90], [-180, -90]]]
+           };
+           world.id = 'Q2';
+           world.properties.id = 'Q2';
+           world.properties.area = geojsonArea.geometry(world.geometry) / 1e6; // m² to km²
 
-             for (var i = 0; i < keys.length; i++) {
-               var k = keys[i];
+           this._cache.Q2 = world;
+         } // validateLocation
+         // `location`  The location to validate
+         //
+         // Pass a `location` value to validate
+         //
+         // Returns a result like:
+         //   {
+         //     type:     'point', 'geojson', or 'countrycoder'
+         //     location:  the queried location
+         //     id:        the stable identifier for the feature
+         //   }
+         // or `null` if the location is invalid
+         //
 
-               if (discardTags[k] || !entity.tags[k]) {
-                 didDiscard = true;
-               } else {
-                 tags[k] = entity.tags[k];
-               }
-             }
 
-             if (didDiscard) {
-               graph = graph.replace(entity.update({
-                 tags: tags
-               }));
-             }
-           }
-         };
-       }
+         _createClass$1(_default, [{
+           key: "validateLocation",
+           value: function validateLocation(location) {
+             if (Array.isArray(location) && (location.length === 2 || location.length === 3)) {
+               // [lon, lat] or [lon, lat, radius] point?
+               var lon = location[0];
+               var lat = location[1];
+               var radius = location[2];
 
-       //
-       // Optionally, disconnect only the given ways.
-       //
-       // For testing convenience, accepts an ID to assign to the (first) new node.
-       // Normally, this will be undefined and the way will automatically
-       // be assigned a new ID.
-       //
-       // This is the inverse of `iD.actionConnect`.
-       //
-       // Reference:
-       //   https://github.com/openstreetmap/potlatch2/blob/master/net/systemeD/halcyon/connection/actions/UnjoinNodeAction.as
-       //   https://github.com/openstreetmap/josm/blob/mirror/src/org/openstreetmap/josm/actions/UnGlueAction.java
-       //
+               if (Number.isFinite(lon) && lon >= -180 && lon <= 180 && Number.isFinite(lat) && lat >= -90 && lat <= 90 && (location.length === 2 || Number.isFinite(radius) && radius > 0)) {
+                 var id = '[' + location.toString() + ']';
+                 return {
+                   type: 'point',
+                   location: location,
+                   id: id
+                 };
+               }
+             } else if (typeof location === 'string' && /^\S+\.geojson$/i.test(location)) {
+               // a .geojson filename?
+               var _id = location.toLowerCase();
 
-       function actionDisconnect(nodeId, newNodeId) {
-         var wayIds;
+               if (this._cache[_id]) {
+                 return {
+                   type: 'geojson',
+                   location: location,
+                   id: _id
+                 };
+               }
+             } else if (typeof location === 'string' || typeof location === 'number') {
+               // a country-coder value?
+               var feature = feature$1(location);
 
-         var action = function action(graph) {
-           var node = graph.entity(nodeId);
-           var connections = action.connections(graph);
-           connections.forEach(function (connection) {
-             var way = graph.entity(connection.wayID);
-             var newNode = osmNode({
-               id: newNodeId,
-               loc: node.loc,
-               tags: node.tags
-             });
-             graph = graph.replace(newNode);
+               if (feature) {
+                 // Use wikidata QID as the identifier, since that seems to be the one
+                 // property that everything in CountryCoder is guaranteed to have.
+                 var _id2 = feature.properties.wikidata;
+                 return {
+                   type: 'countrycoder',
+                   location: location,
+                   id: _id2
+                 };
+               }
+             }
 
-             if (connection.index === 0 && way.isArea()) {
-               // replace shared node with shared node..
-               graph = graph.replace(way.replaceNode(way.nodes[0], newNode.id));
-             } else if (way.isClosed() && connection.index === way.nodes.length - 1) {
-               // replace closing node with new new node..
-               graph = graph.replace(way.unclose().addNode(newNode.id));
+             if (this._strict) {
+               throw new Error("validateLocation:  Invalid location: \"".concat(location, "\"."));
              } else {
-               // replace shared node with multiple new nodes..
-               graph = graph.replace(way.updateNode(newNode.id, connection.index));
+               return null;
              }
-           });
-           return graph;
-         };
-
-         action.connections = function (graph) {
-           var candidates = [];
-           var keeping = false;
-           var parentWays = graph.parentWays(graph.entity(nodeId));
-           var way, waynode;
-
-           for (var i = 0; i < parentWays.length; i++) {
-             way = parentWays[i];
+           } // resolveLocation
+           // `location`  The location to resolve
+           //
+           // Pass a `location` value to resolve
+           //
+           // Returns a result like:
+           //   {
+           //     type:      'point', 'geojson', or 'countrycoder'
+           //     location:  the queried location
+           //     id:        a stable identifier for the feature
+           //     feature:   the resolved GeoJSON feature
+           //   }
+           //  or `null` if the location is invalid
+           //
 
-             if (wayIds && wayIds.indexOf(way.id) === -1) {
-               keeping = true;
-               continue;
-             }
+         }, {
+           key: "resolveLocation",
+           value: function resolveLocation(location) {
+             var valid = this.validateLocation(location);
+             if (!valid) return null;
+             var id = valid.id; // Return a result from cache if we can
 
-             if (way.isArea() && way.nodes[0] === nodeId) {
-               candidates.push({
-                 wayID: way.id,
-                 index: 0
+             if (this._cache[id]) {
+               return Object.assign(valid, {
+                 feature: this._cache[id]
                });
-             } else {
-               for (var j = 0; j < way.nodes.length; j++) {
-                 waynode = way.nodes[j];
+             } // A [lon,lat] coordinate pair?
 
-                 if (waynode === nodeId) {
-                   if (way.isClosed() && parentWays.length > 1 && wayIds && wayIds.indexOf(way.id) !== -1 && j === way.nodes.length - 1) {
-                     continue;
-                   }
 
-                   candidates.push({
-                     wayID: way.id,
-                     index: j
-                   });
-                 }
-               }
-             }
-           }
+             if (valid.type === 'point') {
+               var lon = location[0];
+               var lat = location[1];
+               var radius = location[2] || 25; // km
 
-           return keeping ? candidates : candidates.slice(1);
-         };
+               var EDGES = 10;
+               var PRECISION = 3;
+               var area = Math.PI * radius * radius;
+               var feature = this._cache[id] = geojsonPrecision({
+                 type: 'Feature',
+                 id: id,
+                 properties: {
+                   id: id,
+                   area: Number(area.toFixed(2))
+                 },
+                 geometry: circleToPolygon([lon, lat], radius * 1000, EDGES) // km to m
 
-         action.disabled = function (graph) {
-           var connections = action.connections(graph);
-           if (connections.length === 0) return 'not_connected';
-           var parentWays = graph.parentWays(graph.entity(nodeId));
-           var seenRelationIds = {};
-           var sharedRelation;
-           parentWays.forEach(function (way) {
-             var relations = graph.parentRelations(way);
-             relations.forEach(function (relation) {
-               if (relation.id in seenRelationIds) {
-                 if (wayIds) {
-                   if (wayIds.indexOf(way.id) !== -1 || wayIds.indexOf(seenRelationIds[relation.id]) !== -1) {
-                     sharedRelation = relation;
-                   }
-                 } else {
-                   sharedRelation = relation;
-                 }
-               } else {
-                 seenRelationIds[relation.id] = way.id;
-               }
-             });
-           });
-           if (sharedRelation) return 'relation';
-         };
+               }, PRECISION);
+               return Object.assign(valid, {
+                 feature: feature
+               }); // A .geojson filename?
+             } else if (valid.type === 'geojson') ; else if (valid.type === 'countrycoder') {
+               var _feature = _cloneDeep(feature$1(id));
 
-         action.limitWays = function (val) {
-           if (!arguments.length) return wayIds;
-           wayIds = val;
-           return action;
-         };
+               var props = _feature.properties; // -> This block of code is weird and requires some explanation. <-
+               // CountryCoder includes higher level features which are made up of members.
+               // These features don't have their own geometry, but CountryCoder provides an
+               //   `aggregateFeature` method to combine these members into a MultiPolygon.
+               // In the past, Turf/JSTS/martinez could not handle the aggregated features,
+               //   so we'd iteratively union them all together.  (this was slow)
+               // But now mfogel/polygon-clipping handles these MultiPolygons like a boss.
+               // This approach also has the benefit of removing all the internal boaders and
+               //   simplifying the regional polygons a lot.
 
-         return action;
-       }
+               if (Array.isArray(props.members)) {
+                 var aggregate = aggregateFeature(id);
+                 aggregate.geometry.coordinates = _clip([aggregate], 'UNION').geometry.coordinates;
+                 _feature.geometry = aggregate.geometry;
+               } // Ensure `area` property exists
 
-       var geojsonRewind = rewind;
 
-       function rewind(gj, outer) {
-         var type = gj && gj.type,
-             i;
+               if (!props.area) {
+                 var _area = geojsonArea.geometry(_feature.geometry) / 1e6; // m² to km²
 
-         if (type === 'FeatureCollection') {
-           for (i = 0; i < gj.features.length; i++) {
-             rewind(gj.features[i], outer);
-           }
-         } else if (type === 'GeometryCollection') {
-           for (i = 0; i < gj.geometries.length; i++) {
-             rewind(gj.geometries[i], outer);
-           }
-         } else if (type === 'Feature') {
-           rewind(gj.geometry, outer);
-         } else if (type === 'Polygon') {
-           rewindRings(gj.coordinates, outer);
-         } else if (type === 'MultiPolygon') {
-           for (i = 0; i < gj.coordinates.length; i++) {
-             rewindRings(gj.coordinates[i], outer);
-           }
-         }
 
-         return gj;
-       }
+                 props.area = Number(_area.toFixed(2));
+               } // Ensure `id` property exists
 
-       function rewindRings(rings, outer) {
-         if (rings.length === 0) return;
-         rewindRing(rings[0], outer);
 
-         for (var i = 1; i < rings.length; i++) {
-           rewindRing(rings[i], !outer);
-         }
-       }
+               _feature.id = id;
+               props.id = id;
+               this._cache[id] = _feature;
+               return Object.assign(valid, {
+                 feature: _feature
+               });
+             }
 
-       function rewindRing(ring, dir) {
-         var area = 0;
+             if (this._strict) {
+               throw new Error("resolveLocation:  Couldn't resolve location \"".concat(location, "\"."));
+             } else {
+               return null;
+             }
+           } // validateLocationSet
+           // `locationSet`  the locationSet to validate
+           //
+           // Pass a locationSet Object to validate like:
+           //   {
+           //     include: [ Array of locations ],
+           //     exclude: [ Array of locations ]
+           //   }
+           //
+           // Returns a result like:
+           //   {
+           //     type:         'locationset'
+           //     locationSet:  the queried locationSet
+           //     id:           the stable identifier for the feature
+           //   }
+           // or `null` if the locationSet is invalid
+           //
 
-         for (var i = 0, len = ring.length, j = len - 1; i < len; j = i++) {
-           area += (ring[i][0] - ring[j][0]) * (ring[j][1] + ring[i][1]);
-         }
+         }, {
+           key: "validateLocationSet",
+           value: function validateLocationSet(locationSet) {
+             locationSet = locationSet || {};
+             var validator = this.validateLocation.bind(this);
+             var include = (locationSet.include || []).map(validator).filter(Boolean);
+             var exclude = (locationSet.exclude || []).map(validator).filter(Boolean);
 
-         if (area >= 0 !== !!dir) ring.reverse();
-       }
+             if (!include.length) {
+               if (this._strict) {
+                 throw new Error("validateLocationSet:  LocationSet includes nothing.");
+               } else {
+                 // non-strict mode, replace an empty locationSet with one that includes "the world"
+                 locationSet.include = ['Q2'];
+                 include = [{
+                   type: 'countrycoder',
+                   location: 'Q2',
+                   id: 'Q2'
+                 }];
+               }
+             } // Generate stable identifier
 
-       function actionExtract(entityID) {
-         var extractedNodeID;
 
-         var action = function action(graph) {
-           var entity = graph.entity(entityID);
+             include.sort(_sortLocations);
+             var id = '+[' + include.map(function (d) {
+               return d.id;
+             }).join(',') + ']';
 
-           if (entity.type === 'node') {
-             return extractFromNode(entity, graph);
-           }
+             if (exclude.length) {
+               exclude.sort(_sortLocations);
+               id += '-[' + exclude.map(function (d) {
+                 return d.id;
+               }).join(',') + ']';
+             }
 
-           return extractFromWayOrRelation(entity, graph);
-         };
+             return {
+               type: 'locationset',
+               locationSet: locationSet,
+               id: id
+             };
+           } // resolveLocationSet
+           // `locationSet`  the locationSet to resolve
+           //
+           // Pass a locationSet Object to validate like:
+           //   {
+           //     include: [ Array of locations ],
+           //     exclude: [ Array of locations ]
+           //   }
+           //
+           // Returns a result like:
+           //   {
+           //     type:         'locationset'
+           //     locationSet:  the queried locationSet
+           //     id:           the stable identifier for the feature
+           //     feature:      the resolved GeoJSON feature
+           //   }
+           // or `null` if the locationSet is invalid
+           //
 
-         function extractFromNode(node, graph) {
-           extractedNodeID = node.id; // Create a new node to replace the one we will detach
+         }, {
+           key: "resolveLocationSet",
+           value: function resolveLocationSet(locationSet) {
+             locationSet = locationSet || {};
+             var valid = this.validateLocationSet(locationSet);
+             if (!valid) return null;
+             var id = valid.id; // Return a result from cache if we can
 
-           var replacement = osmNode({
-             loc: node.loc
-           });
-           graph = graph.replace(replacement); // Process each way in turn, updating the graph as we go
+             if (this._cache[id]) {
+               return Object.assign(valid, {
+                 feature: this._cache[id]
+               });
+             }
 
-           graph = graph.parentWays(node).reduce(function (accGraph, parentWay) {
-             return accGraph.replace(parentWay.replaceNode(entityID, replacement.id));
-           }, graph); // Process any relations too
+             var resolver = this.resolveLocation.bind(this);
+             var includes = (locationSet.include || []).map(resolver).filter(Boolean);
+             var excludes = (locationSet.exclude || []).map(resolver).filter(Boolean); // Return quickly if it's a single included location..
 
-           return graph.parentRelations(node).reduce(function (accGraph, parentRel) {
-             return accGraph.replace(parentRel.replaceMember(node, replacement));
-           }, graph);
-         }
+             if (includes.length === 1 && excludes.length === 0) {
+               return Object.assign(valid, {
+                 feature: includes[0].feature
+               });
+             } // Calculate unions
 
-         function extractFromWayOrRelation(entity, graph) {
-           var fromGeometry = entity.geometry(graph);
-           var keysToCopyAndRetain = ['source', 'wheelchair'];
-           var keysToRetain = ['area'];
-           var buildingKeysToRetain = ['architect', 'building', 'height', 'layer']; // d3_geoCentroid is wrong for counterclockwise-wound polygons, so wind them clockwise
 
-           var extractedLoc = d3_geoCentroid(geojsonRewind(Object.assign({}, entity.asGeoJSON(graph)), true));
+             var includeGeoJSON = _clip(includes.map(function (d) {
+               return d.feature;
+             }), 'UNION');
 
-           if (!extractedLoc || !isFinite(extractedLoc[0]) || !isFinite(extractedLoc[1])) {
-             extractedLoc = entity.extent(graph).center();
-           }
+             var excludeGeoJSON = _clip(excludes.map(function (d) {
+               return d.feature;
+             }), 'UNION'); // Calculate difference, update `area` and return result
 
-           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 = {};
+             var resultGeoJSON = excludeGeoJSON ? _clip([includeGeoJSON, excludeGeoJSON], 'DIFFERENCE') : includeGeoJSON;
+             var area = geojsonArea.geometry(resultGeoJSON.geometry) / 1e6; // m² to km²
 
-           for (var key in entityTags) {
-             if (entity.type === 'relation' && key === 'type') {
-               continue;
-             }
+             resultGeoJSON.id = id;
+             resultGeoJSON.properties = {
+               id: id,
+               area: Number(area.toFixed(2))
+             };
+             this._cache[id] = resultGeoJSON;
+             return Object.assign(valid, {
+               feature: resultGeoJSON
+             });
+           } // strict
+           //
 
-             if (keysToRetain.indexOf(key) !== -1) {
-               continue;
+         }, {
+           key: "strict",
+           value: function strict(val) {
+             if (val === undefined) {
+               // get
+               return this._strict;
+             } else {
+               // set
+               this._strict = val;
+               return this;
              }
+           } // cache
+           // convenience method to access the internal cache
 
-             if (isBuilding) {
-               // don't transfer building-related tags
-               if (buildingKeysToRetain.indexOf(key) !== -1 || key.match(/^building:.{1,}/) || key.match(/^roof:.{1,}/)) continue;
-             } // leave `indoor` tag on the area
+         }, {
+           key: "cache",
+           value: function cache() {
+             return this._cache;
+           } // stringify
+           // convenience method to prettyStringify the given object
 
+         }, {
+           key: "stringify",
+           value: function stringify(obj, options) {
+             return jsonStringifyPrettyCompact(obj, options);
+           }
+         }]);
 
-             if (isIndoorArea && key === 'indoor') {
-               continue;
-             } // copy the tag from the entity to the point
+         return _default;
+       }(); // Wrap the mfogel/polygon-clipping library and return a GeoJSON feature.
 
+       function _clip(features, which) {
+         if (!Array.isArray(features) || !features.length) return null;
+         var fn = {
+           UNION: index.union,
+           DIFFERENCE: index.difference
+         }[which];
+         var args = features.map(function (feature) {
+           return feature.geometry.coordinates;
+         });
+         var coords = fn.apply(null, args);
+         return {
+           type: 'Feature',
+           properties: {},
+           geometry: {
+             type: whichType(coords),
+             coordinates: coords
+           }
+         }; // is this a Polygon or a MultiPolygon?
 
-             pointTags[key] = entityTags[key]; // leave addresses and some other tags so they're on both features
+         function whichType(coords) {
+           var a = Array.isArray(coords);
+           var b = a && Array.isArray(coords[0]);
+           var c = b && Array.isArray(coords[0][0]);
+           var d = c && Array.isArray(coords[0][0][0]);
+           return d ? 'MultiPolygon' : 'Polygon';
+         }
+       }
 
-             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
+       function _cloneDeep(obj) {
+         return JSON.parse(JSON.stringify(obj));
+       } // Sorting the location lists is ok because they end up unioned together.
+       // This sorting makes it possible to generate a deterministic id.
+
+
+       function _sortLocations(a, b) {
+         var rank = {
+           countrycoder: 1,
+           geojson: 2,
+           point: 3
+         };
+         var aRank = rank[a.type];
+         var bRank = rank[b.type];
+         return aRank > bRank ? 1 : aRank < bRank ? -1 : a.id.localeCompare(b.id);
+       }
+
+       // `Number.MAX_SAFE_INTEGER` constant
+       // https://tc39.es/ecma262/#sec-number.max_safe_integer
+       _export({ target: 'Number', stat: true }, {
+         MAX_SAFE_INTEGER: 0x1FFFFFFFFFFFFF
+       });
 
+       var aesJs = createCommonjsModule(function (module, exports) {
+         (function (root) {
 
-             delete entityTags[key];
+           function checkInt(value) {
+             return parseInt(value) === value;
            }
 
-           if (!isBuilding && !isIndoorArea && fromGeometry === 'area') {
-             // ensure that areas keep area geometry
-             entityTags.area = 'yes';
-           }
+           function checkInts(arrayish) {
+             if (!checkInt(arrayish.length)) {
+               return false;
+             }
 
-           var replacement = osmNode({
-             loc: extractedLoc,
-             tags: pointTags
-           });
-           graph = graph.replace(replacement);
-           extractedNodeID = replacement.id;
-           return graph.replace(entity.update({
-             tags: entityTags
-           }));
-         }
+             for (var i = 0; i < arrayish.length; i++) {
+               if (!checkInt(arrayish[i]) || arrayish[i] < 0 || arrayish[i] > 255) {
+                 return false;
+               }
+             }
 
-         action.getExtractedNodeID = function () {
-           return extractedNodeID;
-         };
+             return true;
+           }
 
-         return action;
-       }
+           function coerceArray(arg, copy) {
+             // ArrayBuffer view
+             if (arg.buffer && arg.name === 'Uint8Array') {
+               if (copy) {
+                 if (arg.slice) {
+                   arg = arg.slice();
+                 } else {
+                   arg = Array.prototype.slice.call(arg);
+                 }
+               }
 
-       //
-       // This is the inverse of `iD.actionSplit`.
-       //
-       // 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
-       //
+               return arg;
+             } // It's an array; check it is a valid representation of a byte
 
-       function actionJoin(ids) {
-         function groupEntitiesByGeometry(graph) {
-           var entities = ids.map(function (id) {
-             return graph.entity(id);
-           });
-           return Object.assign({
-             line: []
-           }, utilArrayGroupBy(entities, function (entity) {
-             return entity.geometry(graph);
-           }));
-         }
 
-         var action = function action(graph) {
-           var ways = ids.map(graph.entity, graph);
-           var survivorID = ways[0].id; // if any of the ways are sided (e.g. coastline, cliff, kerb)
-           // sort them first so they establish the overall order - #6033
+             if (Array.isArray(arg)) {
+               if (!checkInts(arg)) {
+                 throw new Error('Array contains invalid value: ' + arg);
+               }
 
-           ways.sort(function (a, b) {
-             var aSided = a.isSided();
-             var bSided = b.isSided();
-             return aSided && !bSided ? -1 : bSided && !aSided ? 1 : 0;
-           }); // Prefer to keep an existing way.
+               return new Uint8Array(arg);
+             } // Something else, but behaves like an array (maybe a Buffer? Arguments?)
 
-           for (var i = 0; i < ways.length; i++) {
-             if (!ways[i].isNew()) {
-               survivorID = ways[i].id;
-               break;
+
+             if (checkInt(arg.length) && checkInts(arg)) {
+               return new Uint8Array(arg);
              }
+
+             throw new Error('unsupported array-like object');
            }
 
-           var sequences = osmJoinWays(ways, graph);
-           var joined = sequences[0]; // We might need to reverse some of these ways before joining them.  #4688
-           // `joined.actions` property will contain any actions we need to apply.
+           function createArray(length) {
+             return new Uint8Array(length);
+           }
 
-           graph = sequences.actions.reduce(function (g, action) {
-             return action(g);
-           }, graph);
-           var survivor = graph.entity(survivorID);
-           survivor = survivor.update({
-             nodes: joined.nodes.map(function (n) {
-               return n.id;
-             })
-           });
-           graph = graph.replace(survivor);
-           joined.forEach(function (way) {
-             if (way.id === survivorID) return;
-             graph.parentRelations(way).forEach(function (parent) {
-               graph = graph.replace(parent.replaceMember(way, survivor));
-             });
-             survivor = survivor.mergeTags(way.tags);
-             graph = graph.replace(survivor);
-             graph = actionDeleteWay(way.id)(graph);
-           }); // Finds if the join created a single-member multipolygon,
-           // and if so turns it into a basic area instead
+           function copyArray(sourceArray, targetArray, targetStart, sourceStart, sourceEnd) {
+             if (sourceStart != null || sourceEnd != null) {
+               if (sourceArray.slice) {
+                 sourceArray = sourceArray.slice(sourceStart, sourceEnd);
+               } else {
+                 sourceArray = Array.prototype.slice.call(sourceArray, sourceStart, sourceEnd);
+               }
+             }
 
-           function checkForSimpleMultipolygon() {
-             if (!survivor.isClosed()) return;
-             var multipolygons = graph.parentMultipolygons(survivor).filter(function (multipolygon) {
-               // find multipolygons where the survivor is the only member
-               return multipolygon.members.length === 1;
-             }); // skip if this is the single member of multiple multipolygons
+             targetArray.set(sourceArray, targetStart);
+           }
 
-             if (multipolygons.length !== 1) return;
-             var multipolygon = multipolygons[0];
+           var convertUtf8 = function () {
+             function toBytes(text) {
+               var result = [],
+                   i = 0;
+               text = encodeURI(text);
 
-             for (var key in survivor.tags) {
-               if (multipolygon.tags[key] && // don't collapse if tags cannot be cleanly merged
-               multipolygon.tags[key] !== survivor.tags[key]) return;
-             }
+               while (i < text.length) {
+                 var c = text.charCodeAt(i++); // if it is a % sign, encode the following 2 bytes as a hex value
 
-             survivor = survivor.mergeTags(multipolygon.tags);
-             graph = graph.replace(survivor);
-             graph = actionDeleteRelation(multipolygon.id, true
-             /* allow untagged members */
-             )(graph);
-             var tags = Object.assign({}, survivor.tags);
+                 if (c === 37) {
+                   result.push(parseInt(text.substr(i, 2), 16));
+                   i += 2; // otherwise, just the actual byte
+                 } else {
+                   result.push(c);
+                 }
+               }
 
-             if (survivor.geometry(graph) !== 'area') {
-               // ensure the feature persists as an area
-               tags.area = 'yes';
+               return coerceArray(result);
              }
 
-             delete tags.type; // remove type=multipolygon
-
-             survivor = survivor.update({
-               tags: tags
-             });
-             graph = graph.replace(survivor);
-           }
+             function fromBytes(bytes) {
+               var result = [],
+                   i = 0;
 
-           checkForSimpleMultipolygon();
-           return graph;
-         }; // Returns the number of nodes the resultant way is expected to have
+               while (i < bytes.length) {
+                 var c = bytes[i];
 
+                 if (c < 128) {
+                   result.push(String.fromCharCode(c));
+                   i++;
+                 } else if (c > 191 && c < 224) {
+                   result.push(String.fromCharCode((c & 0x1f) << 6 | bytes[i + 1] & 0x3f));
+                   i += 2;
+                 } else {
+                   result.push(String.fromCharCode((c & 0x0f) << 12 | (bytes[i + 1] & 0x3f) << 6 | bytes[i + 2] & 0x3f));
+                   i += 3;
+                 }
+               }
 
-         action.resultingWayNodesLength = function (graph) {
-           return ids.reduce(function (count, id) {
-             return count + graph.entity(id).nodes.length;
-           }, 0) - ids.length - 1;
-         };
+               return result.join('');
+             }
 
-         action.disabled = function (graph) {
-           var geometries = groupEntitiesByGeometry(graph);
+             return {
+               toBytes: toBytes,
+               fromBytes: fromBytes
+             };
+           }();
 
-           if (ids.length < 2 || ids.length !== geometries.line.length) {
-             return 'not_eligible';
-           }
+           var convertHex = function () {
+             function toBytes(text) {
+               var result = [];
 
-           var joined = osmJoinWays(ids.map(graph.entity, graph), graph);
+               for (var i = 0; i < text.length; i += 2) {
+                 result.push(parseInt(text.substr(i, 2), 16));
+               }
 
-           if (joined.length > 1) {
-             return 'not_adjacent';
-           } // Loop through all combinations of path-pairs
-           // to check potential intersections between all pairs
+               return result;
+             } // http://ixti.net/development/javascript/2011/11/11/base64-encodedecode-of-utf8-in-browser-with-js.html
 
 
-           for (var i = 0; i < ids.length - 1; i++) {
-             for (var j = i + 1; j < ids.length; j++) {
-               var path1 = graph.childNodes(graph.entity(ids[i])).map(function (e) {
-                 return e.loc;
-               });
-               var path2 = graph.childNodes(graph.entity(ids[j])).map(function (e) {
-                 return e.loc;
-               });
-               var intersections = geoPathIntersections(path1, path2); // Check if intersections are just nodes lying on top of
-               // each other/the line, as opposed to crossing it
+             var Hex = '0123456789abcdef';
 
-               var common = utilArrayIntersection(joined[0].nodes.map(function (n) {
-                 return n.loc.toString();
-               }), intersections.map(function (n) {
-                 return n.toString();
-               }));
+             function fromBytes(bytes) {
+               var result = [];
 
-               if (common.length !== intersections.length) {
-                 return 'paths_intersect';
+               for (var i = 0; i < bytes.length; i++) {
+                 var v = bytes[i];
+                 result.push(Hex[(v & 0xf0) >> 4] + Hex[v & 0x0f]);
                }
+
+               return result.join('');
              }
-           }
 
-           var nodeIds = joined[0].nodes.map(function (n) {
-             return n.id;
-           }).slice(1, -1);
-           var relation;
-           var tags = {};
-           var 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;
-               }
-             });
+             return {
+               toBytes: toBytes,
+               fromBytes: fromBytes
+             };
+           }(); // Number of rounds by keysize
 
-             for (var k in way.tags) {
-               if (!(k in tags)) {
-                 tags[k] = way.tags[k];
-               } else if (tags[k] && osmIsInterestingTag(k) && tags[k] !== way.tags[k]) {
-                 conflicting = true;
-               }
-             }
-           });
 
-           if (relation) {
-             return 'restriction';
-           }
+           var numberOfRounds = {
+             16: 10,
+             24: 12,
+             32: 14
+           }; // Round constant words
 
-           if (conflicting) {
-             return 'conflicting_tags';
-           }
-         };
+           var rcon = [0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36, 0x6c, 0xd8, 0xab, 0x4d, 0x9a, 0x2f, 0x5e, 0xbc, 0x63, 0xc6, 0x97, 0x35, 0x6a, 0xd4, 0xb3, 0x7d, 0xfa, 0xef, 0xc5, 0x91]; // S-box and Inverse S-box (S is for Substitution)
 
-         return action;
-       }
+           var S = [0x63, 0x7c, 0x77, 0x7b, 0xf2, 0x6b, 0x6f, 0xc5, 0x30, 0x01, 0x67, 0x2b, 0xfe, 0xd7, 0xab, 0x76, 0xca, 0x82, 0xc9, 0x7d, 0xfa, 0x59, 0x47, 0xf0, 0xad, 0xd4, 0xa2, 0xaf, 0x9c, 0xa4, 0x72, 0xc0, 0xb7, 0xfd, 0x93, 0x26, 0x36, 0x3f, 0xf7, 0xcc, 0x34, 0xa5, 0xe5, 0xf1, 0x71, 0xd8, 0x31, 0x15, 0x04, 0xc7, 0x23, 0xc3, 0x18, 0x96, 0x05, 0x9a, 0x07, 0x12, 0x80, 0xe2, 0xeb, 0x27, 0xb2, 0x75, 0x09, 0x83, 0x2c, 0x1a, 0x1b, 0x6e, 0x5a, 0xa0, 0x52, 0x3b, 0xd6, 0xb3, 0x29, 0xe3, 0x2f, 0x84, 0x53, 0xd1, 0x00, 0xed, 0x20, 0xfc, 0xb1, 0x5b, 0x6a, 0xcb, 0xbe, 0x39, 0x4a, 0x4c, 0x58, 0xcf, 0xd0, 0xef, 0xaa, 0xfb, 0x43, 0x4d, 0x33, 0x85, 0x45, 0xf9, 0x02, 0x7f, 0x50, 0x3c, 0x9f, 0xa8, 0x51, 0xa3, 0x40, 0x8f, 0x92, 0x9d, 0x38, 0xf5, 0xbc, 0xb6, 0xda, 0x21, 0x10, 0xff, 0xf3, 0xd2, 0xcd, 0x0c, 0x13, 0xec, 0x5f, 0x97, 0x44, 0x17, 0xc4, 0xa7, 0x7e, 0x3d, 0x64, 0x5d, 0x19, 0x73, 0x60, 0x81, 0x4f, 0xdc, 0x22, 0x2a, 0x90, 0x88, 0x46, 0xee, 0xb8, 0x14, 0xde, 0x5e, 0x0b, 0xdb, 0xe0, 0x32, 0x3a, 0x0a, 0x49, 0x06, 0x24, 0x5c, 0xc2, 0xd3, 0xac, 0x62, 0x91, 0x95, 0xe4, 0x79, 0xe7, 0xc8, 0x37, 0x6d, 0x8d, 0xd5, 0x4e, 0xa9, 0x6c, 0x56, 0xf4, 0xea, 0x65, 0x7a, 0xae, 0x08, 0xba, 0x78, 0x25, 0x2e, 0x1c, 0xa6, 0xb4, 0xc6, 0xe8, 0xdd, 0x74, 0x1f, 0x4b, 0xbd, 0x8b, 0x8a, 0x70, 0x3e, 0xb5, 0x66, 0x48, 0x03, 0xf6, 0x0e, 0x61, 0x35, 0x57, 0xb9, 0x86, 0xc1, 0x1d, 0x9e, 0xe1, 0xf8, 0x98, 0x11, 0x69, 0xd9, 0x8e, 0x94, 0x9b, 0x1e, 0x87, 0xe9, 0xce, 0x55, 0x28, 0xdf, 0x8c, 0xa1, 0x89, 0x0d, 0xbf, 0xe6, 0x42, 0x68, 0x41, 0x99, 0x2d, 0x0f, 0xb0, 0x54, 0xbb, 0x16];
+           var Si = [0x52, 0x09, 0x6a, 0xd5, 0x30, 0x36, 0xa5, 0x38, 0xbf, 0x40, 0xa3, 0x9e, 0x81, 0xf3, 0xd7, 0xfb, 0x7c, 0xe3, 0x39, 0x82, 0x9b, 0x2f, 0xff, 0x87, 0x34, 0x8e, 0x43, 0x44, 0xc4, 0xde, 0xe9, 0xcb, 0x54, 0x7b, 0x94, 0x32, 0xa6, 0xc2, 0x23, 0x3d, 0xee, 0x4c, 0x95, 0x0b, 0x42, 0xfa, 0xc3, 0x4e, 0x08, 0x2e, 0xa1, 0x66, 0x28, 0xd9, 0x24, 0xb2, 0x76, 0x5b, 0xa2, 0x49, 0x6d, 0x8b, 0xd1, 0x25, 0x72, 0xf8, 0xf6, 0x64, 0x86, 0x68, 0x98, 0x16, 0xd4, 0xa4, 0x5c, 0xcc, 0x5d, 0x65, 0xb6, 0x92, 0x6c, 0x70, 0x48, 0x50, 0xfd, 0xed, 0xb9, 0xda, 0x5e, 0x15, 0x46, 0x57, 0xa7, 0x8d, 0x9d, 0x84, 0x90, 0xd8, 0xab, 0x00, 0x8c, 0xbc, 0xd3, 0x0a, 0xf7, 0xe4, 0x58, 0x05, 0xb8, 0xb3, 0x45, 0x06, 0xd0, 0x2c, 0x1e, 0x8f, 0xca, 0x3f, 0x0f, 0x02, 0xc1, 0xaf, 0xbd, 0x03, 0x01, 0x13, 0x8a, 0x6b, 0x3a, 0x91, 0x11, 0x41, 0x4f, 0x67, 0xdc, 0xea, 0x97, 0xf2, 0xcf, 0xce, 0xf0, 0xb4, 0xe6, 0x73, 0x96, 0xac, 0x74, 0x22, 0xe7, 0xad, 0x35, 0x85, 0xe2, 0xf9, 0x37, 0xe8, 0x1c, 0x75, 0xdf, 0x6e, 0x47, 0xf1, 0x1a, 0x71, 0x1d, 0x29, 0xc5, 0x89, 0x6f, 0xb7, 0x62, 0x0e, 0xaa, 0x18, 0xbe, 0x1b, 0xfc, 0x56, 0x3e, 0x4b, 0xc6, 0xd2, 0x79, 0x20, 0x9a, 0xdb, 0xc0, 0xfe, 0x78, 0xcd, 0x5a, 0xf4, 0x1f, 0xdd, 0xa8, 0x33, 0x88, 0x07, 0xc7, 0x31, 0xb1, 0x12, 0x10, 0x59, 0x27, 0x80, 0xec, 0x5f, 0x60, 0x51, 0x7f, 0xa9, 0x19, 0xb5, 0x4a, 0x0d, 0x2d, 0xe5, 0x7a, 0x9f, 0x93, 0xc9, 0x9c, 0xef, 0xa0, 0xe0, 0x3b, 0x4d, 0xae, 0x2a, 0xf5, 0xb0, 0xc8, 0xeb, 0xbb, 0x3c, 0x83, 0x53, 0x99, 0x61, 0x17, 0x2b, 0x04, 0x7e, 0xba, 0x77, 0xd6, 0x26, 0xe1, 0x69, 0x14, 0x63, 0x55, 0x21, 0x0c, 0x7d]; // Transformations for encryption
 
-       function actionMerge(ids) {
-         function groupEntitiesByGeometry(graph) {
-           var entities = ids.map(function (id) {
-             return graph.entity(id);
-           });
-           return Object.assign({
-             point: [],
-             area: [],
-             line: [],
-             relation: []
-           }, utilArrayGroupBy(entities, function (entity) {
-             return entity.geometry(graph);
-           }));
-         }
+           var T1 = [0xc66363a5, 0xf87c7c84, 0xee777799, 0xf67b7b8d, 0xfff2f20d, 0xd66b6bbd, 0xde6f6fb1, 0x91c5c554, 0x60303050, 0x02010103, 0xce6767a9, 0x562b2b7d, 0xe7fefe19, 0xb5d7d762, 0x4dababe6, 0xec76769a, 0x8fcaca45, 0x1f82829d, 0x89c9c940, 0xfa7d7d87, 0xeffafa15, 0xb25959eb, 0x8e4747c9, 0xfbf0f00b, 0x41adadec, 0xb3d4d467, 0x5fa2a2fd, 0x45afafea, 0x239c9cbf, 0x53a4a4f7, 0xe4727296, 0x9bc0c05b, 0x75b7b7c2, 0xe1fdfd1c, 0x3d9393ae, 0x4c26266a, 0x6c36365a, 0x7e3f3f41, 0xf5f7f702, 0x83cccc4f, 0x6834345c, 0x51a5a5f4, 0xd1e5e534, 0xf9f1f108, 0xe2717193, 0xabd8d873, 0x62313153, 0x2a15153f, 0x0804040c, 0x95c7c752, 0x46232365, 0x9dc3c35e, 0x30181828, 0x379696a1, 0x0a05050f, 0x2f9a9ab5, 0x0e070709, 0x24121236, 0x1b80809b, 0xdfe2e23d, 0xcdebeb26, 0x4e272769, 0x7fb2b2cd, 0xea75759f, 0x1209091b, 0x1d83839e, 0x582c2c74, 0x341a1a2e, 0x361b1b2d, 0xdc6e6eb2, 0xb45a5aee, 0x5ba0a0fb, 0xa45252f6, 0x763b3b4d, 0xb7d6d661, 0x7db3b3ce, 0x5229297b, 0xdde3e33e, 0x5e2f2f71, 0x13848497, 0xa65353f5, 0xb9d1d168, 0x00000000, 0xc1eded2c, 0x40202060, 0xe3fcfc1f, 0x79b1b1c8, 0xb65b5bed, 0xd46a6abe, 0x8dcbcb46, 0x67bebed9, 0x7239394b, 0x944a4ade, 0x984c4cd4, 0xb05858e8, 0x85cfcf4a, 0xbbd0d06b, 0xc5efef2a, 0x4faaaae5, 0xedfbfb16, 0x864343c5, 0x9a4d4dd7, 0x66333355, 0x11858594, 0x8a4545cf, 0xe9f9f910, 0x04020206, 0xfe7f7f81, 0xa05050f0, 0x783c3c44, 0x259f9fba, 0x4ba8a8e3, 0xa25151f3, 0x5da3a3fe, 0x804040c0, 0x058f8f8a, 0x3f9292ad, 0x219d9dbc, 0x70383848, 0xf1f5f504, 0x63bcbcdf, 0x77b6b6c1, 0xafdada75, 0x42212163, 0x20101030, 0xe5ffff1a, 0xfdf3f30e, 0xbfd2d26d, 0x81cdcd4c, 0x180c0c14, 0x26131335, 0xc3ecec2f, 0xbe5f5fe1, 0x359797a2, 0x884444cc, 0x2e171739, 0x93c4c457, 0x55a7a7f2, 0xfc7e7e82, 0x7a3d3d47, 0xc86464ac, 0xba5d5de7, 0x3219192b, 0xe6737395, 0xc06060a0, 0x19818198, 0x9e4f4fd1, 0xa3dcdc7f, 0x44222266, 0x542a2a7e, 0x3b9090ab, 0x0b888883, 0x8c4646ca, 0xc7eeee29, 0x6bb8b8d3, 0x2814143c, 0xa7dede79, 0xbc5e5ee2, 0x160b0b1d, 0xaddbdb76, 0xdbe0e03b, 0x64323256, 0x743a3a4e, 0x140a0a1e, 0x924949db, 0x0c06060a, 0x4824246c, 0xb85c5ce4, 0x9fc2c25d, 0xbdd3d36e, 0x43acacef, 0xc46262a6, 0x399191a8, 0x319595a4, 0xd3e4e437, 0xf279798b, 0xd5e7e732, 0x8bc8c843, 0x6e373759, 0xda6d6db7, 0x018d8d8c, 0xb1d5d564, 0x9c4e4ed2, 0x49a9a9e0, 0xd86c6cb4, 0xac5656fa, 0xf3f4f407, 0xcfeaea25, 0xca6565af, 0xf47a7a8e, 0x47aeaee9, 0x10080818, 0x6fbabad5, 0xf0787888, 0x4a25256f, 0x5c2e2e72, 0x381c1c24, 0x57a6a6f1, 0x73b4b4c7, 0x97c6c651, 0xcbe8e823, 0xa1dddd7c, 0xe874749c, 0x3e1f1f21, 0x964b4bdd, 0x61bdbddc, 0x0d8b8b86, 0x0f8a8a85, 0xe0707090, 0x7c3e3e42, 0x71b5b5c4, 0xcc6666aa, 0x904848d8, 0x06030305, 0xf7f6f601, 0x1c0e0e12, 0xc26161a3, 0x6a35355f, 0xae5757f9, 0x69b9b9d0, 0x17868691, 0x99c1c158, 0x3a1d1d27, 0x279e9eb9, 0xd9e1e138, 0xebf8f813, 0x2b9898b3, 0x22111133, 0xd26969bb, 0xa9d9d970, 0x078e8e89, 0x339494a7, 0x2d9b9bb6, 0x3c1e1e22, 0x15878792, 0xc9e9e920, 0x87cece49, 0xaa5555ff, 0x50282878, 0xa5dfdf7a, 0x038c8c8f, 0x59a1a1f8, 0x09898980, 0x1a0d0d17, 0x65bfbfda, 0xd7e6e631, 0x844242c6, 0xd06868b8, 0x824141c3, 0x299999b0, 0x5a2d2d77, 0x1e0f0f11, 0x7bb0b0cb, 0xa85454fc, 0x6dbbbbd6, 0x2c16163a];
+           var T2 = [0xa5c66363, 0x84f87c7c, 0x99ee7777, 0x8df67b7b, 0x0dfff2f2, 0xbdd66b6b, 0xb1de6f6f, 0x5491c5c5, 0x50603030, 0x03020101, 0xa9ce6767, 0x7d562b2b, 0x19e7fefe, 0x62b5d7d7, 0xe64dabab, 0x9aec7676, 0x458fcaca, 0x9d1f8282, 0x4089c9c9, 0x87fa7d7d, 0x15effafa, 0xebb25959, 0xc98e4747, 0x0bfbf0f0, 0xec41adad, 0x67b3d4d4, 0xfd5fa2a2, 0xea45afaf, 0xbf239c9c, 0xf753a4a4, 0x96e47272, 0x5b9bc0c0, 0xc275b7b7, 0x1ce1fdfd, 0xae3d9393, 0x6a4c2626, 0x5a6c3636, 0x417e3f3f, 0x02f5f7f7, 0x4f83cccc, 0x5c683434, 0xf451a5a5, 0x34d1e5e5, 0x08f9f1f1, 0x93e27171, 0x73abd8d8, 0x53623131, 0x3f2a1515, 0x0c080404, 0x5295c7c7, 0x65462323, 0x5e9dc3c3, 0x28301818, 0xa1379696, 0x0f0a0505, 0xb52f9a9a, 0x090e0707, 0x36241212, 0x9b1b8080, 0x3ddfe2e2, 0x26cdebeb, 0x694e2727, 0xcd7fb2b2, 0x9fea7575, 0x1b120909, 0x9e1d8383, 0x74582c2c, 0x2e341a1a, 0x2d361b1b, 0xb2dc6e6e, 0xeeb45a5a, 0xfb5ba0a0, 0xf6a45252, 0x4d763b3b, 0x61b7d6d6, 0xce7db3b3, 0x7b522929, 0x3edde3e3, 0x715e2f2f, 0x97138484, 0xf5a65353, 0x68b9d1d1, 0x00000000, 0x2cc1eded, 0x60402020, 0x1fe3fcfc, 0xc879b1b1, 0xedb65b5b, 0xbed46a6a, 0x468dcbcb, 0xd967bebe, 0x4b723939, 0xde944a4a, 0xd4984c4c, 0xe8b05858, 0x4a85cfcf, 0x6bbbd0d0, 0x2ac5efef, 0xe54faaaa, 0x16edfbfb, 0xc5864343, 0xd79a4d4d, 0x55663333, 0x94118585, 0xcf8a4545, 0x10e9f9f9, 0x06040202, 0x81fe7f7f, 0xf0a05050, 0x44783c3c, 0xba259f9f, 0xe34ba8a8, 0xf3a25151, 0xfe5da3a3, 0xc0804040, 0x8a058f8f, 0xad3f9292, 0xbc219d9d, 0x48703838, 0x04f1f5f5, 0xdf63bcbc, 0xc177b6b6, 0x75afdada, 0x63422121, 0x30201010, 0x1ae5ffff, 0x0efdf3f3, 0x6dbfd2d2, 0x4c81cdcd, 0x14180c0c, 0x35261313, 0x2fc3ecec, 0xe1be5f5f, 0xa2359797, 0xcc884444, 0x392e1717, 0x5793c4c4, 0xf255a7a7, 0x82fc7e7e, 0x477a3d3d, 0xacc86464, 0xe7ba5d5d, 0x2b321919, 0x95e67373, 0xa0c06060, 0x98198181, 0xd19e4f4f, 0x7fa3dcdc, 0x66442222, 0x7e542a2a, 0xab3b9090, 0x830b8888, 0xca8c4646, 0x29c7eeee, 0xd36bb8b8, 0x3c281414, 0x79a7dede, 0xe2bc5e5e, 0x1d160b0b, 0x76addbdb, 0x3bdbe0e0, 0x56643232, 0x4e743a3a, 0x1e140a0a, 0xdb924949, 0x0a0c0606, 0x6c482424, 0xe4b85c5c, 0x5d9fc2c2, 0x6ebdd3d3, 0xef43acac, 0xa6c46262, 0xa8399191, 0xa4319595, 0x37d3e4e4, 0x8bf27979, 0x32d5e7e7, 0x438bc8c8, 0x596e3737, 0xb7da6d6d, 0x8c018d8d, 0x64b1d5d5, 0xd29c4e4e, 0xe049a9a9, 0xb4d86c6c, 0xfaac5656, 0x07f3f4f4, 0x25cfeaea, 0xafca6565, 0x8ef47a7a, 0xe947aeae, 0x18100808, 0xd56fbaba, 0x88f07878, 0x6f4a2525, 0x725c2e2e, 0x24381c1c, 0xf157a6a6, 0xc773b4b4, 0x5197c6c6, 0x23cbe8e8, 0x7ca1dddd, 0x9ce87474, 0x213e1f1f, 0xdd964b4b, 0xdc61bdbd, 0x860d8b8b, 0x850f8a8a, 0x90e07070, 0x427c3e3e, 0xc471b5b5, 0xaacc6666, 0xd8904848, 0x05060303, 0x01f7f6f6, 0x121c0e0e, 0xa3c26161, 0x5f6a3535, 0xf9ae5757, 0xd069b9b9, 0x91178686, 0x5899c1c1, 0x273a1d1d, 0xb9279e9e, 0x38d9e1e1, 0x13ebf8f8, 0xb32b9898, 0x33221111, 0xbbd26969, 0x70a9d9d9, 0x89078e8e, 0xa7339494, 0xb62d9b9b, 0x223c1e1e, 0x92158787, 0x20c9e9e9, 0x4987cece, 0xffaa5555, 0x78502828, 0x7aa5dfdf, 0x8f038c8c, 0xf859a1a1, 0x80098989, 0x171a0d0d, 0xda65bfbf, 0x31d7e6e6, 0xc6844242, 0xb8d06868, 0xc3824141, 0xb0299999, 0x775a2d2d, 0x111e0f0f, 0xcb7bb0b0, 0xfca85454, 0xd66dbbbb, 0x3a2c1616];
+           var T3 = [0x63a5c663, 0x7c84f87c, 0x7799ee77, 0x7b8df67b, 0xf20dfff2, 0x6bbdd66b, 0x6fb1de6f, 0xc55491c5, 0x30506030, 0x01030201, 0x67a9ce67, 0x2b7d562b, 0xfe19e7fe, 0xd762b5d7, 0xabe64dab, 0x769aec76, 0xca458fca, 0x829d1f82, 0xc94089c9, 0x7d87fa7d, 0xfa15effa, 0x59ebb259, 0x47c98e47, 0xf00bfbf0, 0xadec41ad, 0xd467b3d4, 0xa2fd5fa2, 0xafea45af, 0x9cbf239c, 0xa4f753a4, 0x7296e472, 0xc05b9bc0, 0xb7c275b7, 0xfd1ce1fd, 0x93ae3d93, 0x266a4c26, 0x365a6c36, 0x3f417e3f, 0xf702f5f7, 0xcc4f83cc, 0x345c6834, 0xa5f451a5, 0xe534d1e5, 0xf108f9f1, 0x7193e271, 0xd873abd8, 0x31536231, 0x153f2a15, 0x040c0804, 0xc75295c7, 0x23654623, 0xc35e9dc3, 0x18283018, 0x96a13796, 0x050f0a05, 0x9ab52f9a, 0x07090e07, 0x12362412, 0x809b1b80, 0xe23ddfe2, 0xeb26cdeb, 0x27694e27, 0xb2cd7fb2, 0x759fea75, 0x091b1209, 0x839e1d83, 0x2c74582c, 0x1a2e341a, 0x1b2d361b, 0x6eb2dc6e, 0x5aeeb45a, 0xa0fb5ba0, 0x52f6a452, 0x3b4d763b, 0xd661b7d6, 0xb3ce7db3, 0x297b5229, 0xe33edde3, 0x2f715e2f, 0x84971384, 0x53f5a653, 0xd168b9d1, 0x00000000, 0xed2cc1ed, 0x20604020, 0xfc1fe3fc, 0xb1c879b1, 0x5bedb65b, 0x6abed46a, 0xcb468dcb, 0xbed967be, 0x394b7239, 0x4ade944a, 0x4cd4984c, 0x58e8b058, 0xcf4a85cf, 0xd06bbbd0, 0xef2ac5ef, 0xaae54faa, 0xfb16edfb, 0x43c58643, 0x4dd79a4d, 0x33556633, 0x85941185, 0x45cf8a45, 0xf910e9f9, 0x02060402, 0x7f81fe7f, 0x50f0a050, 0x3c44783c, 0x9fba259f, 0xa8e34ba8, 0x51f3a251, 0xa3fe5da3, 0x40c08040, 0x8f8a058f, 0x92ad3f92, 0x9dbc219d, 0x38487038, 0xf504f1f5, 0xbcdf63bc, 0xb6c177b6, 0xda75afda, 0x21634221, 0x10302010, 0xff1ae5ff, 0xf30efdf3, 0xd26dbfd2, 0xcd4c81cd, 0x0c14180c, 0x13352613, 0xec2fc3ec, 0x5fe1be5f, 0x97a23597, 0x44cc8844, 0x17392e17, 0xc45793c4, 0xa7f255a7, 0x7e82fc7e, 0x3d477a3d, 0x64acc864, 0x5de7ba5d, 0x192b3219, 0x7395e673, 0x60a0c060, 0x81981981, 0x4fd19e4f, 0xdc7fa3dc, 0x22664422, 0x2a7e542a, 0x90ab3b90, 0x88830b88, 0x46ca8c46, 0xee29c7ee, 0xb8d36bb8, 0x143c2814, 0xde79a7de, 0x5ee2bc5e, 0x0b1d160b, 0xdb76addb, 0xe03bdbe0, 0x32566432, 0x3a4e743a, 0x0a1e140a, 0x49db9249, 0x060a0c06, 0x246c4824, 0x5ce4b85c, 0xc25d9fc2, 0xd36ebdd3, 0xacef43ac, 0x62a6c462, 0x91a83991, 0x95a43195, 0xe437d3e4, 0x798bf279, 0xe732d5e7, 0xc8438bc8, 0x37596e37, 0x6db7da6d, 0x8d8c018d, 0xd564b1d5, 0x4ed29c4e, 0xa9e049a9, 0x6cb4d86c, 0x56faac56, 0xf407f3f4, 0xea25cfea, 0x65afca65, 0x7a8ef47a, 0xaee947ae, 0x08181008, 0xbad56fba, 0x7888f078, 0x256f4a25, 0x2e725c2e, 0x1c24381c, 0xa6f157a6, 0xb4c773b4, 0xc65197c6, 0xe823cbe8, 0xdd7ca1dd, 0x749ce874, 0x1f213e1f, 0x4bdd964b, 0xbddc61bd, 0x8b860d8b, 0x8a850f8a, 0x7090e070, 0x3e427c3e, 0xb5c471b5, 0x66aacc66, 0x48d89048, 0x03050603, 0xf601f7f6, 0x0e121c0e, 0x61a3c261, 0x355f6a35, 0x57f9ae57, 0xb9d069b9, 0x86911786, 0xc15899c1, 0x1d273a1d, 0x9eb9279e, 0xe138d9e1, 0xf813ebf8, 0x98b32b98, 0x11332211, 0x69bbd269, 0xd970a9d9, 0x8e89078e, 0x94a73394, 0x9bb62d9b, 0x1e223c1e, 0x87921587, 0xe920c9e9, 0xce4987ce, 0x55ffaa55, 0x28785028, 0xdf7aa5df, 0x8c8f038c, 0xa1f859a1, 0x89800989, 0x0d171a0d, 0xbfda65bf, 0xe631d7e6, 0x42c68442, 0x68b8d068, 0x41c38241, 0x99b02999, 0x2d775a2d, 0x0f111e0f, 0xb0cb7bb0, 0x54fca854, 0xbbd66dbb, 0x163a2c16];
+           var T4 = [0x6363a5c6, 0x7c7c84f8, 0x777799ee, 0x7b7b8df6, 0xf2f20dff, 0x6b6bbdd6, 0x6f6fb1de, 0xc5c55491, 0x30305060, 0x01010302, 0x6767a9ce, 0x2b2b7d56, 0xfefe19e7, 0xd7d762b5, 0xababe64d, 0x76769aec, 0xcaca458f, 0x82829d1f, 0xc9c94089, 0x7d7d87fa, 0xfafa15ef, 0x5959ebb2, 0x4747c98e, 0xf0f00bfb, 0xadadec41, 0xd4d467b3, 0xa2a2fd5f, 0xafafea45, 0x9c9cbf23, 0xa4a4f753, 0x727296e4, 0xc0c05b9b, 0xb7b7c275, 0xfdfd1ce1, 0x9393ae3d, 0x26266a4c, 0x36365a6c, 0x3f3f417e, 0xf7f702f5, 0xcccc4f83, 0x34345c68, 0xa5a5f451, 0xe5e534d1, 0xf1f108f9, 0x717193e2, 0xd8d873ab, 0x31315362, 0x15153f2a, 0x04040c08, 0xc7c75295, 0x23236546, 0xc3c35e9d, 0x18182830, 0x9696a137, 0x05050f0a, 0x9a9ab52f, 0x0707090e, 0x12123624, 0x80809b1b, 0xe2e23ddf, 0xebeb26cd, 0x2727694e, 0xb2b2cd7f, 0x75759fea, 0x09091b12, 0x83839e1d, 0x2c2c7458, 0x1a1a2e34, 0x1b1b2d36, 0x6e6eb2dc, 0x5a5aeeb4, 0xa0a0fb5b, 0x5252f6a4, 0x3b3b4d76, 0xd6d661b7, 0xb3b3ce7d, 0x29297b52, 0xe3e33edd, 0x2f2f715e, 0x84849713, 0x5353f5a6, 0xd1d168b9, 0x00000000, 0xeded2cc1, 0x20206040, 0xfcfc1fe3, 0xb1b1c879, 0x5b5bedb6, 0x6a6abed4, 0xcbcb468d, 0xbebed967, 0x39394b72, 0x4a4ade94, 0x4c4cd498, 0x5858e8b0, 0xcfcf4a85, 0xd0d06bbb, 0xefef2ac5, 0xaaaae54f, 0xfbfb16ed, 0x4343c586, 0x4d4dd79a, 0x33335566, 0x85859411, 0x4545cf8a, 0xf9f910e9, 0x02020604, 0x7f7f81fe, 0x5050f0a0, 0x3c3c4478, 0x9f9fba25, 0xa8a8e34b, 0x5151f3a2, 0xa3a3fe5d, 0x4040c080, 0x8f8f8a05, 0x9292ad3f, 0x9d9dbc21, 0x38384870, 0xf5f504f1, 0xbcbcdf63, 0xb6b6c177, 0xdada75af, 0x21216342, 0x10103020, 0xffff1ae5, 0xf3f30efd, 0xd2d26dbf, 0xcdcd4c81, 0x0c0c1418, 0x13133526, 0xecec2fc3, 0x5f5fe1be, 0x9797a235, 0x4444cc88, 0x1717392e, 0xc4c45793, 0xa7a7f255, 0x7e7e82fc, 0x3d3d477a, 0x6464acc8, 0x5d5de7ba, 0x19192b32, 0x737395e6, 0x6060a0c0, 0x81819819, 0x4f4fd19e, 0xdcdc7fa3, 0x22226644, 0x2a2a7e54, 0x9090ab3b, 0x8888830b, 0x4646ca8c, 0xeeee29c7, 0xb8b8d36b, 0x14143c28, 0xdede79a7, 0x5e5ee2bc, 0x0b0b1d16, 0xdbdb76ad, 0xe0e03bdb, 0x32325664, 0x3a3a4e74, 0x0a0a1e14, 0x4949db92, 0x06060a0c, 0x24246c48, 0x5c5ce4b8, 0xc2c25d9f, 0xd3d36ebd, 0xacacef43, 0x6262a6c4, 0x9191a839, 0x9595a431, 0xe4e437d3, 0x79798bf2, 0xe7e732d5, 0xc8c8438b, 0x3737596e, 0x6d6db7da, 0x8d8d8c01, 0xd5d564b1, 0x4e4ed29c, 0xa9a9e049, 0x6c6cb4d8, 0x5656faac, 0xf4f407f3, 0xeaea25cf, 0x6565afca, 0x7a7a8ef4, 0xaeaee947, 0x08081810, 0xbabad56f, 0x787888f0, 0x25256f4a, 0x2e2e725c, 0x1c1c2438, 0xa6a6f157, 0xb4b4c773, 0xc6c65197, 0xe8e823cb, 0xdddd7ca1, 0x74749ce8, 0x1f1f213e, 0x4b4bdd96, 0xbdbddc61, 0x8b8b860d, 0x8a8a850f, 0x707090e0, 0x3e3e427c, 0xb5b5c471, 0x6666aacc, 0x4848d890, 0x03030506, 0xf6f601f7, 0x0e0e121c, 0x6161a3c2, 0x35355f6a, 0x5757f9ae, 0xb9b9d069, 0x86869117, 0xc1c15899, 0x1d1d273a, 0x9e9eb927, 0xe1e138d9, 0xf8f813eb, 0x9898b32b, 0x11113322, 0x6969bbd2, 0xd9d970a9, 0x8e8e8907, 0x9494a733, 0x9b9bb62d, 0x1e1e223c, 0x87879215, 0xe9e920c9, 0xcece4987, 0x5555ffaa, 0x28287850, 0xdfdf7aa5, 0x8c8c8f03, 0xa1a1f859, 0x89898009, 0x0d0d171a, 0xbfbfda65, 0xe6e631d7, 0x4242c684, 0x6868b8d0, 0x4141c382, 0x9999b029, 0x2d2d775a, 0x0f0f111e, 0xb0b0cb7b, 0x5454fca8, 0xbbbbd66d, 0x16163a2c]; // Transformations for decryption
 
-         var action = function action(graph) {
-           var geometries = groupEntitiesByGeometry(graph);
-           var target = geometries.area[0] || geometries.line[0];
-           var points = geometries.point;
-           points.forEach(function (point) {
-             target = target.mergeTags(point.tags);
-             graph = graph.replace(target);
-             graph.parentRelations(point).forEach(function (parent) {
-               graph = graph.replace(parent.replaceMember(point, target));
-             });
-             var nodes = utilArrayUniq(graph.childNodes(target));
-             var removeNode = point;
+           var T5 = [0x51f4a750, 0x7e416553, 0x1a17a4c3, 0x3a275e96, 0x3bab6bcb, 0x1f9d45f1, 0xacfa58ab, 0x4be30393, 0x2030fa55, 0xad766df6, 0x88cc7691, 0xf5024c25, 0x4fe5d7fc, 0xc52acbd7, 0x26354480, 0xb562a38f, 0xdeb15a49, 0x25ba1b67, 0x45ea0e98, 0x5dfec0e1, 0xc32f7502, 0x814cf012, 0x8d4697a3, 0x6bd3f9c6, 0x038f5fe7, 0x15929c95, 0xbf6d7aeb, 0x955259da, 0xd4be832d, 0x587421d3, 0x49e06929, 0x8ec9c844, 0x75c2896a, 0xf48e7978, 0x99583e6b, 0x27b971dd, 0xbee14fb6, 0xf088ad17, 0xc920ac66, 0x7dce3ab4, 0x63df4a18, 0xe51a3182, 0x97513360, 0x62537f45, 0xb16477e0, 0xbb6bae84, 0xfe81a01c, 0xf9082b94, 0x70486858, 0x8f45fd19, 0x94de6c87, 0x527bf8b7, 0xab73d323, 0x724b02e2, 0xe31f8f57, 0x6655ab2a, 0xb2eb2807, 0x2fb5c203, 0x86c57b9a, 0xd33708a5, 0x302887f2, 0x23bfa5b2, 0x02036aba, 0xed16825c, 0x8acf1c2b, 0xa779b492, 0xf307f2f0, 0x4e69e2a1, 0x65daf4cd, 0x0605bed5, 0xd134621f, 0xc4a6fe8a, 0x342e539d, 0xa2f355a0, 0x058ae132, 0xa4f6eb75, 0x0b83ec39, 0x4060efaa, 0x5e719f06, 0xbd6e1051, 0x3e218af9, 0x96dd063d, 0xdd3e05ae, 0x4de6bd46, 0x91548db5, 0x71c45d05, 0x0406d46f, 0x605015ff, 0x1998fb24, 0xd6bde997, 0x894043cc, 0x67d99e77, 0xb0e842bd, 0x07898b88, 0xe7195b38, 0x79c8eedb, 0xa17c0a47, 0x7c420fe9, 0xf8841ec9, 0x00000000, 0x09808683, 0x322bed48, 0x1e1170ac, 0x6c5a724e, 0xfd0efffb, 0x0f853856, 0x3daed51e, 0x362d3927, 0x0a0fd964, 0x685ca621, 0x9b5b54d1, 0x24362e3a, 0x0c0a67b1, 0x9357e70f, 0xb4ee96d2, 0x1b9b919e, 0x80c0c54f, 0x61dc20a2, 0x5a774b69, 0x1c121a16, 0xe293ba0a, 0xc0a02ae5, 0x3c22e043, 0x121b171d, 0x0e090d0b, 0xf28bc7ad, 0x2db6a8b9, 0x141ea9c8, 0x57f11985, 0xaf75074c, 0xee99ddbb, 0xa37f60fd, 0xf701269f, 0x5c72f5bc, 0x44663bc5, 0x5bfb7e34, 0x8b432976, 0xcb23c6dc, 0xb6edfc68, 0xb8e4f163, 0xd731dcca, 0x42638510, 0x13972240, 0x84c61120, 0x854a247d, 0xd2bb3df8, 0xaef93211, 0xc729a16d, 0x1d9e2f4b, 0xdcb230f3, 0x0d8652ec, 0x77c1e3d0, 0x2bb3166c, 0xa970b999, 0x119448fa, 0x47e96422, 0xa8fc8cc4, 0xa0f03f1a, 0x567d2cd8, 0x223390ef, 0x87494ec7, 0xd938d1c1, 0x8ccaa2fe, 0x98d40b36, 0xa6f581cf, 0xa57ade28, 0xdab78e26, 0x3fadbfa4, 0x2c3a9de4, 0x5078920d, 0x6a5fcc9b, 0x547e4662, 0xf68d13c2, 0x90d8b8e8, 0x2e39f75e, 0x82c3aff5, 0x9f5d80be, 0x69d0937c, 0x6fd52da9, 0xcf2512b3, 0xc8ac993b, 0x10187da7, 0xe89c636e, 0xdb3bbb7b, 0xcd267809, 0x6e5918f4, 0xec9ab701, 0x834f9aa8, 0xe6956e65, 0xaaffe67e, 0x21bccf08, 0xef15e8e6, 0xbae79bd9, 0x4a6f36ce, 0xea9f09d4, 0x29b07cd6, 0x31a4b2af, 0x2a3f2331, 0xc6a59430, 0x35a266c0, 0x744ebc37, 0xfc82caa6, 0xe090d0b0, 0x33a7d815, 0xf104984a, 0x41ecdaf7, 0x7fcd500e, 0x1791f62f, 0x764dd68d, 0x43efb04d, 0xccaa4d54, 0xe49604df, 0x9ed1b5e3, 0x4c6a881b, 0xc12c1fb8, 0x4665517f, 0x9d5eea04, 0x018c355d, 0xfa877473, 0xfb0b412e, 0xb3671d5a, 0x92dbd252, 0xe9105633, 0x6dd64713, 0x9ad7618c, 0x37a10c7a, 0x59f8148e, 0xeb133c89, 0xcea927ee, 0xb761c935, 0xe11ce5ed, 0x7a47b13c, 0x9cd2df59, 0x55f2733f, 0x1814ce79, 0x73c737bf, 0x53f7cdea, 0x5ffdaa5b, 0xdf3d6f14, 0x7844db86, 0xcaaff381, 0xb968c43e, 0x3824342c, 0xc2a3405f, 0x161dc372, 0xbce2250c, 0x283c498b, 0xff0d9541, 0x39a80171, 0x080cb3de, 0xd8b4e49c, 0x6456c190, 0x7bcb8461, 0xd532b670, 0x486c5c74, 0xd0b85742];
+           var T6 = [0x5051f4a7, 0x537e4165, 0xc31a17a4, 0x963a275e, 0xcb3bab6b, 0xf11f9d45, 0xabacfa58, 0x934be303, 0x552030fa, 0xf6ad766d, 0x9188cc76, 0x25f5024c, 0xfc4fe5d7, 0xd7c52acb, 0x80263544, 0x8fb562a3, 0x49deb15a, 0x6725ba1b, 0x9845ea0e, 0xe15dfec0, 0x02c32f75, 0x12814cf0, 0xa38d4697, 0xc66bd3f9, 0xe7038f5f, 0x9515929c, 0xebbf6d7a, 0xda955259, 0x2dd4be83, 0xd3587421, 0x2949e069, 0x448ec9c8, 0x6a75c289, 0x78f48e79, 0x6b99583e, 0xdd27b971, 0xb6bee14f, 0x17f088ad, 0x66c920ac, 0xb47dce3a, 0x1863df4a, 0x82e51a31, 0x60975133, 0x4562537f, 0xe0b16477, 0x84bb6bae, 0x1cfe81a0, 0x94f9082b, 0x58704868, 0x198f45fd, 0x8794de6c, 0xb7527bf8, 0x23ab73d3, 0xe2724b02, 0x57e31f8f, 0x2a6655ab, 0x07b2eb28, 0x032fb5c2, 0x9a86c57b, 0xa5d33708, 0xf2302887, 0xb223bfa5, 0xba02036a, 0x5ced1682, 0x2b8acf1c, 0x92a779b4, 0xf0f307f2, 0xa14e69e2, 0xcd65daf4, 0xd50605be, 0x1fd13462, 0x8ac4a6fe, 0x9d342e53, 0xa0a2f355, 0x32058ae1, 0x75a4f6eb, 0x390b83ec, 0xaa4060ef, 0x065e719f, 0x51bd6e10, 0xf93e218a, 0x3d96dd06, 0xaedd3e05, 0x464de6bd, 0xb591548d, 0x0571c45d, 0x6f0406d4, 0xff605015, 0x241998fb, 0x97d6bde9, 0xcc894043, 0x7767d99e, 0xbdb0e842, 0x8807898b, 0x38e7195b, 0xdb79c8ee, 0x47a17c0a, 0xe97c420f, 0xc9f8841e, 0x00000000, 0x83098086, 0x48322bed, 0xac1e1170, 0x4e6c5a72, 0xfbfd0eff, 0x560f8538, 0x1e3daed5, 0x27362d39, 0x640a0fd9, 0x21685ca6, 0xd19b5b54, 0x3a24362e, 0xb10c0a67, 0x0f9357e7, 0xd2b4ee96, 0x9e1b9b91, 0x4f80c0c5, 0xa261dc20, 0x695a774b, 0x161c121a, 0x0ae293ba, 0xe5c0a02a, 0x433c22e0, 0x1d121b17, 0x0b0e090d, 0xadf28bc7, 0xb92db6a8, 0xc8141ea9, 0x8557f119, 0x4caf7507, 0xbbee99dd, 0xfda37f60, 0x9ff70126, 0xbc5c72f5, 0xc544663b, 0x345bfb7e, 0x768b4329, 0xdccb23c6, 0x68b6edfc, 0x63b8e4f1, 0xcad731dc, 0x10426385, 0x40139722, 0x2084c611, 0x7d854a24, 0xf8d2bb3d, 0x11aef932, 0x6dc729a1, 0x4b1d9e2f, 0xf3dcb230, 0xec0d8652, 0xd077c1e3, 0x6c2bb316, 0x99a970b9, 0xfa119448, 0x2247e964, 0xc4a8fc8c, 0x1aa0f03f, 0xd8567d2c, 0xef223390, 0xc787494e, 0xc1d938d1, 0xfe8ccaa2, 0x3698d40b, 0xcfa6f581, 0x28a57ade, 0x26dab78e, 0xa43fadbf, 0xe42c3a9d, 0x0d507892, 0x9b6a5fcc, 0x62547e46, 0xc2f68d13, 0xe890d8b8, 0x5e2e39f7, 0xf582c3af, 0xbe9f5d80, 0x7c69d093, 0xa96fd52d, 0xb3cf2512, 0x3bc8ac99, 0xa710187d, 0x6ee89c63, 0x7bdb3bbb, 0x09cd2678, 0xf46e5918, 0x01ec9ab7, 0xa8834f9a, 0x65e6956e, 0x7eaaffe6, 0x0821bccf, 0xe6ef15e8, 0xd9bae79b, 0xce4a6f36, 0xd4ea9f09, 0xd629b07c, 0xaf31a4b2, 0x312a3f23, 0x30c6a594, 0xc035a266, 0x37744ebc, 0xa6fc82ca, 0xb0e090d0, 0x1533a7d8, 0x4af10498, 0xf741ecda, 0x0e7fcd50, 0x2f1791f6, 0x8d764dd6, 0x4d43efb0, 0x54ccaa4d, 0xdfe49604, 0xe39ed1b5, 0x1b4c6a88, 0xb8c12c1f, 0x7f466551, 0x049d5eea, 0x5d018c35, 0x73fa8774, 0x2efb0b41, 0x5ab3671d, 0x5292dbd2, 0x33e91056, 0x136dd647, 0x8c9ad761, 0x7a37a10c, 0x8e59f814, 0x89eb133c, 0xeecea927, 0x35b761c9, 0xede11ce5, 0x3c7a47b1, 0x599cd2df, 0x3f55f273, 0x791814ce, 0xbf73c737, 0xea53f7cd, 0x5b5ffdaa, 0x14df3d6f, 0x867844db, 0x81caaff3, 0x3eb968c4, 0x2c382434, 0x5fc2a340, 0x72161dc3, 0x0cbce225, 0x8b283c49, 0x41ff0d95, 0x7139a801, 0xde080cb3, 0x9cd8b4e4, 0x906456c1, 0x617bcb84, 0x70d532b6, 0x74486c5c, 0x42d0b857];
+           var T7 = [0xa75051f4, 0x65537e41, 0xa4c31a17, 0x5e963a27, 0x6bcb3bab, 0x45f11f9d, 0x58abacfa, 0x03934be3, 0xfa552030, 0x6df6ad76, 0x769188cc, 0x4c25f502, 0xd7fc4fe5, 0xcbd7c52a, 0x44802635, 0xa38fb562, 0x5a49deb1, 0x1b6725ba, 0x0e9845ea, 0xc0e15dfe, 0x7502c32f, 0xf012814c, 0x97a38d46, 0xf9c66bd3, 0x5fe7038f, 0x9c951592, 0x7aebbf6d, 0x59da9552, 0x832dd4be, 0x21d35874, 0x692949e0, 0xc8448ec9, 0x896a75c2, 0x7978f48e, 0x3e6b9958, 0x71dd27b9, 0x4fb6bee1, 0xad17f088, 0xac66c920, 0x3ab47dce, 0x4a1863df, 0x3182e51a, 0x33609751, 0x7f456253, 0x77e0b164, 0xae84bb6b, 0xa01cfe81, 0x2b94f908, 0x68587048, 0xfd198f45, 0x6c8794de, 0xf8b7527b, 0xd323ab73, 0x02e2724b, 0x8f57e31f, 0xab2a6655, 0x2807b2eb, 0xc2032fb5, 0x7b9a86c5, 0x08a5d337, 0x87f23028, 0xa5b223bf, 0x6aba0203, 0x825ced16, 0x1c2b8acf, 0xb492a779, 0xf2f0f307, 0xe2a14e69, 0xf4cd65da, 0xbed50605, 0x621fd134, 0xfe8ac4a6, 0x539d342e, 0x55a0a2f3, 0xe132058a, 0xeb75a4f6, 0xec390b83, 0xefaa4060, 0x9f065e71, 0x1051bd6e, 0x8af93e21, 0x063d96dd, 0x05aedd3e, 0xbd464de6, 0x8db59154, 0x5d0571c4, 0xd46f0406, 0x15ff6050, 0xfb241998, 0xe997d6bd, 0x43cc8940, 0x9e7767d9, 0x42bdb0e8, 0x8b880789, 0x5b38e719, 0xeedb79c8, 0x0a47a17c, 0x0fe97c42, 0x1ec9f884, 0x00000000, 0x86830980, 0xed48322b, 0x70ac1e11, 0x724e6c5a, 0xfffbfd0e, 0x38560f85, 0xd51e3dae, 0x3927362d, 0xd9640a0f, 0xa621685c, 0x54d19b5b, 0x2e3a2436, 0x67b10c0a, 0xe70f9357, 0x96d2b4ee, 0x919e1b9b, 0xc54f80c0, 0x20a261dc, 0x4b695a77, 0x1a161c12, 0xba0ae293, 0x2ae5c0a0, 0xe0433c22, 0x171d121b, 0x0d0b0e09, 0xc7adf28b, 0xa8b92db6, 0xa9c8141e, 0x198557f1, 0x074caf75, 0xddbbee99, 0x60fda37f, 0x269ff701, 0xf5bc5c72, 0x3bc54466, 0x7e345bfb, 0x29768b43, 0xc6dccb23, 0xfc68b6ed, 0xf163b8e4, 0xdccad731, 0x85104263, 0x22401397, 0x112084c6, 0x247d854a, 0x3df8d2bb, 0x3211aef9, 0xa16dc729, 0x2f4b1d9e, 0x30f3dcb2, 0x52ec0d86, 0xe3d077c1, 0x166c2bb3, 0xb999a970, 0x48fa1194, 0x642247e9, 0x8cc4a8fc, 0x3f1aa0f0, 0x2cd8567d, 0x90ef2233, 0x4ec78749, 0xd1c1d938, 0xa2fe8cca, 0x0b3698d4, 0x81cfa6f5, 0xde28a57a, 0x8e26dab7, 0xbfa43fad, 0x9de42c3a, 0x920d5078, 0xcc9b6a5f, 0x4662547e, 0x13c2f68d, 0xb8e890d8, 0xf75e2e39, 0xaff582c3, 0x80be9f5d, 0x937c69d0, 0x2da96fd5, 0x12b3cf25, 0x993bc8ac, 0x7da71018, 0x636ee89c, 0xbb7bdb3b, 0x7809cd26, 0x18f46e59, 0xb701ec9a, 0x9aa8834f, 0x6e65e695, 0xe67eaaff, 0xcf0821bc, 0xe8e6ef15, 0x9bd9bae7, 0x36ce4a6f, 0x09d4ea9f, 0x7cd629b0, 0xb2af31a4, 0x23312a3f, 0x9430c6a5, 0x66c035a2, 0xbc37744e, 0xcaa6fc82, 0xd0b0e090, 0xd81533a7, 0x984af104, 0xdaf741ec, 0x500e7fcd, 0xf62f1791, 0xd68d764d, 0xb04d43ef, 0x4d54ccaa, 0x04dfe496, 0xb5e39ed1, 0x881b4c6a, 0x1fb8c12c, 0x517f4665, 0xea049d5e, 0x355d018c, 0x7473fa87, 0x412efb0b, 0x1d5ab367, 0xd25292db, 0x5633e910, 0x47136dd6, 0x618c9ad7, 0x0c7a37a1, 0x148e59f8, 0x3c89eb13, 0x27eecea9, 0xc935b761, 0xe5ede11c, 0xb13c7a47, 0xdf599cd2, 0x733f55f2, 0xce791814, 0x37bf73c7, 0xcdea53f7, 0xaa5b5ffd, 0x6f14df3d, 0xdb867844, 0xf381caaf, 0xc43eb968, 0x342c3824, 0x405fc2a3, 0xc372161d, 0x250cbce2, 0x498b283c, 0x9541ff0d, 0x017139a8, 0xb3de080c, 0xe49cd8b4, 0xc1906456, 0x84617bcb, 0xb670d532, 0x5c74486c, 0x5742d0b8];
+           var T8 = [0xf4a75051, 0x4165537e, 0x17a4c31a, 0x275e963a, 0xab6bcb3b, 0x9d45f11f, 0xfa58abac, 0xe303934b, 0x30fa5520, 0x766df6ad, 0xcc769188, 0x024c25f5, 0xe5d7fc4f, 0x2acbd7c5, 0x35448026, 0x62a38fb5, 0xb15a49de, 0xba1b6725, 0xea0e9845, 0xfec0e15d, 0x2f7502c3, 0x4cf01281, 0x4697a38d, 0xd3f9c66b, 0x8f5fe703, 0x929c9515, 0x6d7aebbf, 0x5259da95, 0xbe832dd4, 0x7421d358, 0xe0692949, 0xc9c8448e, 0xc2896a75, 0x8e7978f4, 0x583e6b99, 0xb971dd27, 0xe14fb6be, 0x88ad17f0, 0x20ac66c9, 0xce3ab47d, 0xdf4a1863, 0x1a3182e5, 0x51336097, 0x537f4562, 0x6477e0b1, 0x6bae84bb, 0x81a01cfe, 0x082b94f9, 0x48685870, 0x45fd198f, 0xde6c8794, 0x7bf8b752, 0x73d323ab, 0x4b02e272, 0x1f8f57e3, 0x55ab2a66, 0xeb2807b2, 0xb5c2032f, 0xc57b9a86, 0x3708a5d3, 0x2887f230, 0xbfa5b223, 0x036aba02, 0x16825ced, 0xcf1c2b8a, 0x79b492a7, 0x07f2f0f3, 0x69e2a14e, 0xdaf4cd65, 0x05bed506, 0x34621fd1, 0xa6fe8ac4, 0x2e539d34, 0xf355a0a2, 0x8ae13205, 0xf6eb75a4, 0x83ec390b, 0x60efaa40, 0x719f065e, 0x6e1051bd, 0x218af93e, 0xdd063d96, 0x3e05aedd, 0xe6bd464d, 0x548db591, 0xc45d0571, 0x06d46f04, 0x5015ff60, 0x98fb2419, 0xbde997d6, 0x4043cc89, 0xd99e7767, 0xe842bdb0, 0x898b8807, 0x195b38e7, 0xc8eedb79, 0x7c0a47a1, 0x420fe97c, 0x841ec9f8, 0x00000000, 0x80868309, 0x2bed4832, 0x1170ac1e, 0x5a724e6c, 0x0efffbfd, 0x8538560f, 0xaed51e3d, 0x2d392736, 0x0fd9640a, 0x5ca62168, 0x5b54d19b, 0x362e3a24, 0x0a67b10c, 0x57e70f93, 0xee96d2b4, 0x9b919e1b, 0xc0c54f80, 0xdc20a261, 0x774b695a, 0x121a161c, 0x93ba0ae2, 0xa02ae5c0, 0x22e0433c, 0x1b171d12, 0x090d0b0e, 0x8bc7adf2, 0xb6a8b92d, 0x1ea9c814, 0xf1198557, 0x75074caf, 0x99ddbbee, 0x7f60fda3, 0x01269ff7, 0x72f5bc5c, 0x663bc544, 0xfb7e345b, 0x4329768b, 0x23c6dccb, 0xedfc68b6, 0xe4f163b8, 0x31dccad7, 0x63851042, 0x97224013, 0xc6112084, 0x4a247d85, 0xbb3df8d2, 0xf93211ae, 0x29a16dc7, 0x9e2f4b1d, 0xb230f3dc, 0x8652ec0d, 0xc1e3d077, 0xb3166c2b, 0x70b999a9, 0x9448fa11, 0xe9642247, 0xfc8cc4a8, 0xf03f1aa0, 0x7d2cd856, 0x3390ef22, 0x494ec787, 0x38d1c1d9, 0xcaa2fe8c, 0xd40b3698, 0xf581cfa6, 0x7ade28a5, 0xb78e26da, 0xadbfa43f, 0x3a9de42c, 0x78920d50, 0x5fcc9b6a, 0x7e466254, 0x8d13c2f6, 0xd8b8e890, 0x39f75e2e, 0xc3aff582, 0x5d80be9f, 0xd0937c69, 0xd52da96f, 0x2512b3cf, 0xac993bc8, 0x187da710, 0x9c636ee8, 0x3bbb7bdb, 0x267809cd, 0x5918f46e, 0x9ab701ec, 0x4f9aa883, 0x956e65e6, 0xffe67eaa, 0xbccf0821, 0x15e8e6ef, 0xe79bd9ba, 0x6f36ce4a, 0x9f09d4ea, 0xb07cd629, 0xa4b2af31, 0x3f23312a, 0xa59430c6, 0xa266c035, 0x4ebc3774, 0x82caa6fc, 0x90d0b0e0, 0xa7d81533, 0x04984af1, 0xecdaf741, 0xcd500e7f, 0x91f62f17, 0x4dd68d76, 0xefb04d43, 0xaa4d54cc, 0x9604dfe4, 0xd1b5e39e, 0x6a881b4c, 0x2c1fb8c1, 0x65517f46, 0x5eea049d, 0x8c355d01, 0x877473fa, 0x0b412efb, 0x671d5ab3, 0xdbd25292, 0x105633e9, 0xd647136d, 0xd7618c9a, 0xa10c7a37, 0xf8148e59, 0x133c89eb, 0xa927eece, 0x61c935b7, 0x1ce5ede1, 0x47b13c7a, 0xd2df599c, 0xf2733f55, 0x14ce7918, 0xc737bf73, 0xf7cdea53, 0xfdaa5b5f, 0x3d6f14df, 0x44db8678, 0xaff381ca, 0x68c43eb9, 0x24342c38, 0xa3405fc2, 0x1dc37216, 0xe2250cbc, 0x3c498b28, 0x0d9541ff, 0xa8017139, 0x0cb3de08, 0xb4e49cd8, 0x56c19064, 0xcb84617b, 0x32b670d5, 0x6c5c7448, 0xb85742d0]; // Transformations for decryption key expansion
 
-             for (var i = 0; i < nodes.length; i++) {
-               var node = nodes[i];
+           var U1 = [0x00000000, 0x0e090d0b, 0x1c121a16, 0x121b171d, 0x3824342c, 0x362d3927, 0x24362e3a, 0x2a3f2331, 0x70486858, 0x7e416553, 0x6c5a724e, 0x62537f45, 0x486c5c74, 0x4665517f, 0x547e4662, 0x5a774b69, 0xe090d0b0, 0xee99ddbb, 0xfc82caa6, 0xf28bc7ad, 0xd8b4e49c, 0xd6bde997, 0xc4a6fe8a, 0xcaaff381, 0x90d8b8e8, 0x9ed1b5e3, 0x8ccaa2fe, 0x82c3aff5, 0xa8fc8cc4, 0xa6f581cf, 0xb4ee96d2, 0xbae79bd9, 0xdb3bbb7b, 0xd532b670, 0xc729a16d, 0xc920ac66, 0xe31f8f57, 0xed16825c, 0xff0d9541, 0xf104984a, 0xab73d323, 0xa57ade28, 0xb761c935, 0xb968c43e, 0x9357e70f, 0x9d5eea04, 0x8f45fd19, 0x814cf012, 0x3bab6bcb, 0x35a266c0, 0x27b971dd, 0x29b07cd6, 0x038f5fe7, 0x0d8652ec, 0x1f9d45f1, 0x119448fa, 0x4be30393, 0x45ea0e98, 0x57f11985, 0x59f8148e, 0x73c737bf, 0x7dce3ab4, 0x6fd52da9, 0x61dc20a2, 0xad766df6, 0xa37f60fd, 0xb16477e0, 0xbf6d7aeb, 0x955259da, 0x9b5b54d1, 0x894043cc, 0x87494ec7, 0xdd3e05ae, 0xd33708a5, 0xc12c1fb8, 0xcf2512b3, 0xe51a3182, 0xeb133c89, 0xf9082b94, 0xf701269f, 0x4de6bd46, 0x43efb04d, 0x51f4a750, 0x5ffdaa5b, 0x75c2896a, 0x7bcb8461, 0x69d0937c, 0x67d99e77, 0x3daed51e, 0x33a7d815, 0x21bccf08, 0x2fb5c203, 0x058ae132, 0x0b83ec39, 0x1998fb24, 0x1791f62f, 0x764dd68d, 0x7844db86, 0x6a5fcc9b, 0x6456c190, 0x4e69e2a1, 0x4060efaa, 0x527bf8b7, 0x5c72f5bc, 0x0605bed5, 0x080cb3de, 0x1a17a4c3, 0x141ea9c8, 0x3e218af9, 0x302887f2, 0x223390ef, 0x2c3a9de4, 0x96dd063d, 0x98d40b36, 0x8acf1c2b, 0x84c61120, 0xaef93211, 0xa0f03f1a, 0xb2eb2807, 0xbce2250c, 0xe6956e65, 0xe89c636e, 0xfa877473, 0xf48e7978, 0xdeb15a49, 0xd0b85742, 0xc2a3405f, 0xccaa4d54, 0x41ecdaf7, 0x4fe5d7fc, 0x5dfec0e1, 0x53f7cdea, 0x79c8eedb, 0x77c1e3d0, 0x65daf4cd, 0x6bd3f9c6, 0x31a4b2af, 0x3fadbfa4, 0x2db6a8b9, 0x23bfa5b2, 0x09808683, 0x07898b88, 0x15929c95, 0x1b9b919e, 0xa17c0a47, 0xaf75074c, 0xbd6e1051, 0xb3671d5a, 0x99583e6b, 0x97513360, 0x854a247d, 0x8b432976, 0xd134621f, 0xdf3d6f14, 0xcd267809, 0xc32f7502, 0xe9105633, 0xe7195b38, 0xf5024c25, 0xfb0b412e, 0x9ad7618c, 0x94de6c87, 0x86c57b9a, 0x88cc7691, 0xa2f355a0, 0xacfa58ab, 0xbee14fb6, 0xb0e842bd, 0xea9f09d4, 0xe49604df, 0xf68d13c2, 0xf8841ec9, 0xd2bb3df8, 0xdcb230f3, 0xcea927ee, 0xc0a02ae5, 0x7a47b13c, 0x744ebc37, 0x6655ab2a, 0x685ca621, 0x42638510, 0x4c6a881b, 0x5e719f06, 0x5078920d, 0x0a0fd964, 0x0406d46f, 0x161dc372, 0x1814ce79, 0x322bed48, 0x3c22e043, 0x2e39f75e, 0x2030fa55, 0xec9ab701, 0xe293ba0a, 0xf088ad17, 0xfe81a01c, 0xd4be832d, 0xdab78e26, 0xc8ac993b, 0xc6a59430, 0x9cd2df59, 0x92dbd252, 0x80c0c54f, 0x8ec9c844, 0xa4f6eb75, 0xaaffe67e, 0xb8e4f163, 0xb6edfc68, 0x0c0a67b1, 0x02036aba, 0x10187da7, 0x1e1170ac, 0x342e539d, 0x3a275e96, 0x283c498b, 0x26354480, 0x7c420fe9, 0x724b02e2, 0x605015ff, 0x6e5918f4, 0x44663bc5, 0x4a6f36ce, 0x587421d3, 0x567d2cd8, 0x37a10c7a, 0x39a80171, 0x2bb3166c, 0x25ba1b67, 0x0f853856, 0x018c355d, 0x13972240, 0x1d9e2f4b, 0x47e96422, 0x49e06929, 0x5bfb7e34, 0x55f2733f, 0x7fcd500e, 0x71c45d05, 0x63df4a18, 0x6dd64713, 0xd731dcca, 0xd938d1c1, 0xcb23c6dc, 0xc52acbd7, 0xef15e8e6, 0xe11ce5ed, 0xf307f2f0, 0xfd0efffb, 0xa779b492, 0xa970b999, 0xbb6bae84, 0xb562a38f, 0x9f5d80be, 0x91548db5, 0x834f9aa8, 0x8d4697a3];
+           var U2 = [0x00000000, 0x0b0e090d, 0x161c121a, 0x1d121b17, 0x2c382434, 0x27362d39, 0x3a24362e, 0x312a3f23, 0x58704868, 0x537e4165, 0x4e6c5a72, 0x4562537f, 0x74486c5c, 0x7f466551, 0x62547e46, 0x695a774b, 0xb0e090d0, 0xbbee99dd, 0xa6fc82ca, 0xadf28bc7, 0x9cd8b4e4, 0x97d6bde9, 0x8ac4a6fe, 0x81caaff3, 0xe890d8b8, 0xe39ed1b5, 0xfe8ccaa2, 0xf582c3af, 0xc4a8fc8c, 0xcfa6f581, 0xd2b4ee96, 0xd9bae79b, 0x7bdb3bbb, 0x70d532b6, 0x6dc729a1, 0x66c920ac, 0x57e31f8f, 0x5ced1682, 0x41ff0d95, 0x4af10498, 0x23ab73d3, 0x28a57ade, 0x35b761c9, 0x3eb968c4, 0x0f9357e7, 0x049d5eea, 0x198f45fd, 0x12814cf0, 0xcb3bab6b, 0xc035a266, 0xdd27b971, 0xd629b07c, 0xe7038f5f, 0xec0d8652, 0xf11f9d45, 0xfa119448, 0x934be303, 0x9845ea0e, 0x8557f119, 0x8e59f814, 0xbf73c737, 0xb47dce3a, 0xa96fd52d, 0xa261dc20, 0xf6ad766d, 0xfda37f60, 0xe0b16477, 0xebbf6d7a, 0xda955259, 0xd19b5b54, 0xcc894043, 0xc787494e, 0xaedd3e05, 0xa5d33708, 0xb8c12c1f, 0xb3cf2512, 0x82e51a31, 0x89eb133c, 0x94f9082b, 0x9ff70126, 0x464de6bd, 0x4d43efb0, 0x5051f4a7, 0x5b5ffdaa, 0x6a75c289, 0x617bcb84, 0x7c69d093, 0x7767d99e, 0x1e3daed5, 0x1533a7d8, 0x0821bccf, 0x032fb5c2, 0x32058ae1, 0x390b83ec, 0x241998fb, 0x2f1791f6, 0x8d764dd6, 0x867844db, 0x9b6a5fcc, 0x906456c1, 0xa14e69e2, 0xaa4060ef, 0xb7527bf8, 0xbc5c72f5, 0xd50605be, 0xde080cb3, 0xc31a17a4, 0xc8141ea9, 0xf93e218a, 0xf2302887, 0xef223390, 0xe42c3a9d, 0x3d96dd06, 0x3698d40b, 0x2b8acf1c, 0x2084c611, 0x11aef932, 0x1aa0f03f, 0x07b2eb28, 0x0cbce225, 0x65e6956e, 0x6ee89c63, 0x73fa8774, 0x78f48e79, 0x49deb15a, 0x42d0b857, 0x5fc2a340, 0x54ccaa4d, 0xf741ecda, 0xfc4fe5d7, 0xe15dfec0, 0xea53f7cd, 0xdb79c8ee, 0xd077c1e3, 0xcd65daf4, 0xc66bd3f9, 0xaf31a4b2, 0xa43fadbf, 0xb92db6a8, 0xb223bfa5, 0x83098086, 0x8807898b, 0x9515929c, 0x9e1b9b91, 0x47a17c0a, 0x4caf7507, 0x51bd6e10, 0x5ab3671d, 0x6b99583e, 0x60975133, 0x7d854a24, 0x768b4329, 0x1fd13462, 0x14df3d6f, 0x09cd2678, 0x02c32f75, 0x33e91056, 0x38e7195b, 0x25f5024c, 0x2efb0b41, 0x8c9ad761, 0x8794de6c, 0x9a86c57b, 0x9188cc76, 0xa0a2f355, 0xabacfa58, 0xb6bee14f, 0xbdb0e842, 0xd4ea9f09, 0xdfe49604, 0xc2f68d13, 0xc9f8841e, 0xf8d2bb3d, 0xf3dcb230, 0xeecea927, 0xe5c0a02a, 0x3c7a47b1, 0x37744ebc, 0x2a6655ab, 0x21685ca6, 0x10426385, 0x1b4c6a88, 0x065e719f, 0x0d507892, 0x640a0fd9, 0x6f0406d4, 0x72161dc3, 0x791814ce, 0x48322bed, 0x433c22e0, 0x5e2e39f7, 0x552030fa, 0x01ec9ab7, 0x0ae293ba, 0x17f088ad, 0x1cfe81a0, 0x2dd4be83, 0x26dab78e, 0x3bc8ac99, 0x30c6a594, 0x599cd2df, 0x5292dbd2, 0x4f80c0c5, 0x448ec9c8, 0x75a4f6eb, 0x7eaaffe6, 0x63b8e4f1, 0x68b6edfc, 0xb10c0a67, 0xba02036a, 0xa710187d, 0xac1e1170, 0x9d342e53, 0x963a275e, 0x8b283c49, 0x80263544, 0xe97c420f, 0xe2724b02, 0xff605015, 0xf46e5918, 0xc544663b, 0xce4a6f36, 0xd3587421, 0xd8567d2c, 0x7a37a10c, 0x7139a801, 0x6c2bb316, 0x6725ba1b, 0x560f8538, 0x5d018c35, 0x40139722, 0x4b1d9e2f, 0x2247e964, 0x2949e069, 0x345bfb7e, 0x3f55f273, 0x0e7fcd50, 0x0571c45d, 0x1863df4a, 0x136dd647, 0xcad731dc, 0xc1d938d1, 0xdccb23c6, 0xd7c52acb, 0xe6ef15e8, 0xede11ce5, 0xf0f307f2, 0xfbfd0eff, 0x92a779b4, 0x99a970b9, 0x84bb6bae, 0x8fb562a3, 0xbe9f5d80, 0xb591548d, 0xa8834f9a, 0xa38d4697];
+           var U3 = [0x00000000, 0x0d0b0e09, 0x1a161c12, 0x171d121b, 0x342c3824, 0x3927362d, 0x2e3a2436, 0x23312a3f, 0x68587048, 0x65537e41, 0x724e6c5a, 0x7f456253, 0x5c74486c, 0x517f4665, 0x4662547e, 0x4b695a77, 0xd0b0e090, 0xddbbee99, 0xcaa6fc82, 0xc7adf28b, 0xe49cd8b4, 0xe997d6bd, 0xfe8ac4a6, 0xf381caaf, 0xb8e890d8, 0xb5e39ed1, 0xa2fe8cca, 0xaff582c3, 0x8cc4a8fc, 0x81cfa6f5, 0x96d2b4ee, 0x9bd9bae7, 0xbb7bdb3b, 0xb670d532, 0xa16dc729, 0xac66c920, 0x8f57e31f, 0x825ced16, 0x9541ff0d, 0x984af104, 0xd323ab73, 0xde28a57a, 0xc935b761, 0xc43eb968, 0xe70f9357, 0xea049d5e, 0xfd198f45, 0xf012814c, 0x6bcb3bab, 0x66c035a2, 0x71dd27b9, 0x7cd629b0, 0x5fe7038f, 0x52ec0d86, 0x45f11f9d, 0x48fa1194, 0x03934be3, 0x0e9845ea, 0x198557f1, 0x148e59f8, 0x37bf73c7, 0x3ab47dce, 0x2da96fd5, 0x20a261dc, 0x6df6ad76, 0x60fda37f, 0x77e0b164, 0x7aebbf6d, 0x59da9552, 0x54d19b5b, 0x43cc8940, 0x4ec78749, 0x05aedd3e, 0x08a5d337, 0x1fb8c12c, 0x12b3cf25, 0x3182e51a, 0x3c89eb13, 0x2b94f908, 0x269ff701, 0xbd464de6, 0xb04d43ef, 0xa75051f4, 0xaa5b5ffd, 0x896a75c2, 0x84617bcb, 0x937c69d0, 0x9e7767d9, 0xd51e3dae, 0xd81533a7, 0xcf0821bc, 0xc2032fb5, 0xe132058a, 0xec390b83, 0xfb241998, 0xf62f1791, 0xd68d764d, 0xdb867844, 0xcc9b6a5f, 0xc1906456, 0xe2a14e69, 0xefaa4060, 0xf8b7527b, 0xf5bc5c72, 0xbed50605, 0xb3de080c, 0xa4c31a17, 0xa9c8141e, 0x8af93e21, 0x87f23028, 0x90ef2233, 0x9de42c3a, 0x063d96dd, 0x0b3698d4, 0x1c2b8acf, 0x112084c6, 0x3211aef9, 0x3f1aa0f0, 0x2807b2eb, 0x250cbce2, 0x6e65e695, 0x636ee89c, 0x7473fa87, 0x7978f48e, 0x5a49deb1, 0x5742d0b8, 0x405fc2a3, 0x4d54ccaa, 0xdaf741ec, 0xd7fc4fe5, 0xc0e15dfe, 0xcdea53f7, 0xeedb79c8, 0xe3d077c1, 0xf4cd65da, 0xf9c66bd3, 0xb2af31a4, 0xbfa43fad, 0xa8b92db6, 0xa5b223bf, 0x86830980, 0x8b880789, 0x9c951592, 0x919e1b9b, 0x0a47a17c, 0x074caf75, 0x1051bd6e, 0x1d5ab367, 0x3e6b9958, 0x33609751, 0x247d854a, 0x29768b43, 0x621fd134, 0x6f14df3d, 0x7809cd26, 0x7502c32f, 0x5633e910, 0x5b38e719, 0x4c25f502, 0x412efb0b, 0x618c9ad7, 0x6c8794de, 0x7b9a86c5, 0x769188cc, 0x55a0a2f3, 0x58abacfa, 0x4fb6bee1, 0x42bdb0e8, 0x09d4ea9f, 0x04dfe496, 0x13c2f68d, 0x1ec9f884, 0x3df8d2bb, 0x30f3dcb2, 0x27eecea9, 0x2ae5c0a0, 0xb13c7a47, 0xbc37744e, 0xab2a6655, 0xa621685c, 0x85104263, 0x881b4c6a, 0x9f065e71, 0x920d5078, 0xd9640a0f, 0xd46f0406, 0xc372161d, 0xce791814, 0xed48322b, 0xe0433c22, 0xf75e2e39, 0xfa552030, 0xb701ec9a, 0xba0ae293, 0xad17f088, 0xa01cfe81, 0x832dd4be, 0x8e26dab7, 0x993bc8ac, 0x9430c6a5, 0xdf599cd2, 0xd25292db, 0xc54f80c0, 0xc8448ec9, 0xeb75a4f6, 0xe67eaaff, 0xf163b8e4, 0xfc68b6ed, 0x67b10c0a, 0x6aba0203, 0x7da71018, 0x70ac1e11, 0x539d342e, 0x5e963a27, 0x498b283c, 0x44802635, 0x0fe97c42, 0x02e2724b, 0x15ff6050, 0x18f46e59, 0x3bc54466, 0x36ce4a6f, 0x21d35874, 0x2cd8567d, 0x0c7a37a1, 0x017139a8, 0x166c2bb3, 0x1b6725ba, 0x38560f85, 0x355d018c, 0x22401397, 0x2f4b1d9e, 0x642247e9, 0x692949e0, 0x7e345bfb, 0x733f55f2, 0x500e7fcd, 0x5d0571c4, 0x4a1863df, 0x47136dd6, 0xdccad731, 0xd1c1d938, 0xc6dccb23, 0xcbd7c52a, 0xe8e6ef15, 0xe5ede11c, 0xf2f0f307, 0xfffbfd0e, 0xb492a779, 0xb999a970, 0xae84bb6b, 0xa38fb562, 0x80be9f5d, 0x8db59154, 0x9aa8834f, 0x97a38d46];
+           var U4 = [0x00000000, 0x090d0b0e, 0x121a161c, 0x1b171d12, 0x24342c38, 0x2d392736, 0x362e3a24, 0x3f23312a, 0x48685870, 0x4165537e, 0x5a724e6c, 0x537f4562, 0x6c5c7448, 0x65517f46, 0x7e466254, 0x774b695a, 0x90d0b0e0, 0x99ddbbee, 0x82caa6fc, 0x8bc7adf2, 0xb4e49cd8, 0xbde997d6, 0xa6fe8ac4, 0xaff381ca, 0xd8b8e890, 0xd1b5e39e, 0xcaa2fe8c, 0xc3aff582, 0xfc8cc4a8, 0xf581cfa6, 0xee96d2b4, 0xe79bd9ba, 0x3bbb7bdb, 0x32b670d5, 0x29a16dc7, 0x20ac66c9, 0x1f8f57e3, 0x16825ced, 0x0d9541ff, 0x04984af1, 0x73d323ab, 0x7ade28a5, 0x61c935b7, 0x68c43eb9, 0x57e70f93, 0x5eea049d, 0x45fd198f, 0x4cf01281, 0xab6bcb3b, 0xa266c035, 0xb971dd27, 0xb07cd629, 0x8f5fe703, 0x8652ec0d, 0x9d45f11f, 0x9448fa11, 0xe303934b, 0xea0e9845, 0xf1198557, 0xf8148e59, 0xc737bf73, 0xce3ab47d, 0xd52da96f, 0xdc20a261, 0x766df6ad, 0x7f60fda3, 0x6477e0b1, 0x6d7aebbf, 0x5259da95, 0x5b54d19b, 0x4043cc89, 0x494ec787, 0x3e05aedd, 0x3708a5d3, 0x2c1fb8c1, 0x2512b3cf, 0x1a3182e5, 0x133c89eb, 0x082b94f9, 0x01269ff7, 0xe6bd464d, 0xefb04d43, 0xf4a75051, 0xfdaa5b5f, 0xc2896a75, 0xcb84617b, 0xd0937c69, 0xd99e7767, 0xaed51e3d, 0xa7d81533, 0xbccf0821, 0xb5c2032f, 0x8ae13205, 0x83ec390b, 0x98fb2419, 0x91f62f17, 0x4dd68d76, 0x44db8678, 0x5fcc9b6a, 0x56c19064, 0x69e2a14e, 0x60efaa40, 0x7bf8b752, 0x72f5bc5c, 0x05bed506, 0x0cb3de08, 0x17a4c31a, 0x1ea9c814, 0x218af93e, 0x2887f230, 0x3390ef22, 0x3a9de42c, 0xdd063d96, 0xd40b3698, 0xcf1c2b8a, 0xc6112084, 0xf93211ae, 0xf03f1aa0, 0xeb2807b2, 0xe2250cbc, 0x956e65e6, 0x9c636ee8, 0x877473fa, 0x8e7978f4, 0xb15a49de, 0xb85742d0, 0xa3405fc2, 0xaa4d54cc, 0xecdaf741, 0xe5d7fc4f, 0xfec0e15d, 0xf7cdea53, 0xc8eedb79, 0xc1e3d077, 0xdaf4cd65, 0xd3f9c66b, 0xa4b2af31, 0xadbfa43f, 0xb6a8b92d, 0xbfa5b223, 0x80868309, 0x898b8807, 0x929c9515, 0x9b919e1b, 0x7c0a47a1, 0x75074caf, 0x6e1051bd, 0x671d5ab3, 0x583e6b99, 0x51336097, 0x4a247d85, 0x4329768b, 0x34621fd1, 0x3d6f14df, 0x267809cd, 0x2f7502c3, 0x105633e9, 0x195b38e7, 0x024c25f5, 0x0b412efb, 0xd7618c9a, 0xde6c8794, 0xc57b9a86, 0xcc769188, 0xf355a0a2, 0xfa58abac, 0xe14fb6be, 0xe842bdb0, 0x9f09d4ea, 0x9604dfe4, 0x8d13c2f6, 0x841ec9f8, 0xbb3df8d2, 0xb230f3dc, 0xa927eece, 0xa02ae5c0, 0x47b13c7a, 0x4ebc3774, 0x55ab2a66, 0x5ca62168, 0x63851042, 0x6a881b4c, 0x719f065e, 0x78920d50, 0x0fd9640a, 0x06d46f04, 0x1dc37216, 0x14ce7918, 0x2bed4832, 0x22e0433c, 0x39f75e2e, 0x30fa5520, 0x9ab701ec, 0x93ba0ae2, 0x88ad17f0, 0x81a01cfe, 0xbe832dd4, 0xb78e26da, 0xac993bc8, 0xa59430c6, 0xd2df599c, 0xdbd25292, 0xc0c54f80, 0xc9c8448e, 0xf6eb75a4, 0xffe67eaa, 0xe4f163b8, 0xedfc68b6, 0x0a67b10c, 0x036aba02, 0x187da710, 0x1170ac1e, 0x2e539d34, 0x275e963a, 0x3c498b28, 0x35448026, 0x420fe97c, 0x4b02e272, 0x5015ff60, 0x5918f46e, 0x663bc544, 0x6f36ce4a, 0x7421d358, 0x7d2cd856, 0xa10c7a37, 0xa8017139, 0xb3166c2b, 0xba1b6725, 0x8538560f, 0x8c355d01, 0x97224013, 0x9e2f4b1d, 0xe9642247, 0xe0692949, 0xfb7e345b, 0xf2733f55, 0xcd500e7f, 0xc45d0571, 0xdf4a1863, 0xd647136d, 0x31dccad7, 0x38d1c1d9, 0x23c6dccb, 0x2acbd7c5, 0x15e8e6ef, 0x1ce5ede1, 0x07f2f0f3, 0x0efffbfd, 0x79b492a7, 0x70b999a9, 0x6bae84bb, 0x62a38fb5, 0x5d80be9f, 0x548db591, 0x4f9aa883, 0x4697a38d];
 
-               if (graph.parentWays(node).length > 1 || graph.parentRelations(node).length || node.hasInterestingTags()) {
-                 continue;
-               } // Found an uninteresting child node on the target way.
-               // Move orig point into its place to preserve point's history. #3683
+           function convertToInt32(bytes) {
+             var result = [];
 
+             for (var i = 0; i < bytes.length; i += 4) {
+               result.push(bytes[i] << 24 | bytes[i + 1] << 16 | bytes[i + 2] << 8 | bytes[i + 3]);
+             }
 
-               graph = graph.replace(point.update({
-                 tags: {},
-                 loc: node.loc
-               }));
-               target = target.replaceNode(node.id, point.id);
-               graph = graph.replace(target);
-               removeNode = node;
-               break;
+             return result;
+           }
+
+           var AES = function AES(key) {
+             if (!(this instanceof AES)) {
+               throw Error('AES must be instanitated with `new`');
              }
 
-             graph = graph.remove(removeNode);
-           });
+             Object.defineProperty(this, 'key', {
+               value: coerceArray(key, true)
+             });
 
-           if (target.tags.area === 'yes') {
-             var tags = Object.assign({}, target.tags); // shallow copy
+             this._prepare();
+           };
 
-             delete tags.area;
+           AES.prototype._prepare = function () {
+             var rounds = numberOfRounds[this.key.length];
 
-             if (osmTagSuggestingArea(tags)) {
-               // remove the `area` tag if area geometry is now implied - #3851
-               target = target.update({
-                 tags: tags
-               });
-               graph = graph.replace(target);
-             }
-           }
+             if (rounds == null) {
+               throw new Error('invalid key size (must be 16, 24 or 32 bytes)');
+             } // encryption round keys
 
-           return graph;
-         };
 
-         action.disabled = function (graph) {
-           var geometries = groupEntitiesByGeometry(graph);
+             this._Ke = []; // decryption round keys
 
-           if (geometries.point.length === 0 || geometries.area.length + geometries.line.length !== 1 || geometries.relation.length !== 0) {
-             return 'not_eligible';
-           }
-         };
+             this._Kd = [];
 
-         return action;
-       }
+             for (var i = 0; i <= rounds; i++) {
+               this._Ke.push([0, 0, 0, 0]);
 
-       //
-       // 1. move all the nodes to a common location
-       // 2. `actionConnect` them
+               this._Kd.push([0, 0, 0, 0]);
+             }
 
-       function actionMergeNodes(nodeIDs, loc) {
-         // If there is a single "interesting" node, use that as the location.
-         // Otherwise return the average location of all the nodes.
-         function chooseLoc(graph) {
-           if (!nodeIDs.length) return null;
-           var sum = [0, 0];
-           var interestingCount = 0;
-           var interestingLoc;
+             var roundKeyCount = (rounds + 1) * 4;
+             var KC = this.key.length / 4; // convert the key into ints
 
-           for (var i = 0; i < nodeIDs.length; i++) {
-             var node = graph.entity(nodeIDs[i]);
+             var tk = convertToInt32(this.key); // copy values into round key arrays
 
-             if (node.hasInterestingTags()) {
-               interestingLoc = ++interestingCount === 1 ? node.loc : null;
-             }
+             var index;
 
-             sum = geoVecAdd(sum, node.loc);
-           }
+             for (var i = 0; i < KC; i++) {
+               index = i >> 2;
+               this._Ke[index][i % 4] = tk[i];
+               this._Kd[rounds - index][i % 4] = tk[i];
+             } // key expansion (fips-197 section 5.2)
 
-           return interestingLoc || geoVecScale(sum, 1 / nodeIDs.length);
-         }
 
-         var action = function action(graph) {
-           if (nodeIDs.length < 2) return graph;
-           var toLoc = loc;
+             var rconpointer = 0;
+             var t = KC,
+                 tt;
 
-           if (!toLoc) {
-             toLoc = chooseLoc(graph);
-           }
+             while (t < roundKeyCount) {
+               tt = tk[KC - 1];
+               tk[0] ^= S[tt >> 16 & 0xFF] << 24 ^ S[tt >> 8 & 0xFF] << 16 ^ S[tt & 0xFF] << 8 ^ S[tt >> 24 & 0xFF] ^ rcon[rconpointer] << 24;
+               rconpointer += 1; // key expansion (for non-256 bit)
 
-           for (var i = 0; i < nodeIDs.length; i++) {
-             var node = graph.entity(nodeIDs[i]);
+               if (KC != 8) {
+                 for (var i = 1; i < KC; i++) {
+                   tk[i] ^= tk[i - 1];
+                 } // key expansion for 256-bit keys is "slightly different" (fips-197)
 
-             if (node.loc !== toLoc) {
-               graph = graph.replace(node.move(toLoc));
-             }
-           }
+               } else {
+                 for (var i = 1; i < KC / 2; i++) {
+                   tk[i] ^= tk[i - 1];
+                 }
 
-           return actionConnect(nodeIDs)(graph);
-         };
+                 tt = tk[KC / 2 - 1];
+                 tk[KC / 2] ^= S[tt & 0xFF] ^ S[tt >> 8 & 0xFF] << 8 ^ S[tt >> 16 & 0xFF] << 16 ^ S[tt >> 24 & 0xFF] << 24;
 
-         action.disabled = function (graph) {
-           if (nodeIDs.length < 2) return 'not_eligible';
+                 for (var i = KC / 2 + 1; i < KC; i++) {
+                   tk[i] ^= tk[i - 1];
+                 }
+               } // copy values into round key arrays
 
-           for (var i = 0; i < nodeIDs.length; i++) {
-             var entity = graph.entity(nodeIDs[i]);
-             if (entity.type !== 'node') return 'not_eligible';
-           }
 
-           return actionConnect(nodeIDs).disabled(graph);
-         };
+               var i = 0,
+                   r,
+                   c;
 
-         return action;
-       }
+               while (i < KC && t < roundKeyCount) {
+                 r = t >> 2;
+                 c = t % 4;
+                 this._Ke[r][c] = tk[i];
+                 this._Kd[rounds - r][c] = tk[i++];
+                 t++;
+               }
+             } // inverse-cipher-ify the decryption round key (fips-197 section 5.3)
 
-       function osmChangeset() {
-         if (!(this instanceof osmChangeset)) {
-           return new osmChangeset().initialize(arguments);
-         } else if (arguments.length) {
-           this.initialize(arguments);
-         }
-       }
-       osmEntity.changeset = osmChangeset;
-       osmChangeset.prototype = Object.create(osmEntity.prototype);
-       Object.assign(osmChangeset.prototype, {
-         type: 'changeset',
-         extent: function extent() {
-           return new geoExtent();
-         },
-         geometry: function geometry() {
-           return 'changeset';
-         },
-         asJXON: function asJXON() {
-           return {
-             osm: {
-               changeset: {
-                 tag: Object.keys(this.tags).map(function (k) {
-                   return {
-                     '@k': k,
-                     '@v': this.tags[k]
-                   };
-                 }, this),
-                 '@version': 0.6,
-                 '@generator': 'iD'
+
+             for (var r = 1; r < rounds; r++) {
+               for (var c = 0; c < 4; c++) {
+                 tt = this._Kd[r][c];
+                 this._Kd[r][c] = U1[tt >> 24 & 0xFF] ^ U2[tt >> 16 & 0xFF] ^ U3[tt >> 8 & 0xFF] ^ U4[tt & 0xFF];
                }
              }
            };
-         },
-         // Generate [osmChange](http://wiki.openstreetmap.org/wiki/OsmChange)
-         // XML. Returns a string.
-         osmChangeJXON: function osmChangeJXON(changes) {
-           var changeset_id = this.id;
-
-           function nest(x, order) {
-             var groups = {};
 
-             for (var i = 0; i < x.length; i++) {
-               var tagName = Object.keys(x[i])[0];
-               if (!groups[tagName]) groups[tagName] = [];
-               groups[tagName].push(x[i][tagName]);
+           AES.prototype.encrypt = function (plaintext) {
+             if (plaintext.length != 16) {
+               throw new Error('invalid plaintext size (must be 16 bytes)');
              }
 
-             var ordered = {};
-             order.forEach(function (o) {
-               if (groups[o]) ordered[o] = groups[o];
-             });
-             return ordered;
-           } // sort relations in a changeset by dependencies
-
+             var rounds = this._Ke.length - 1;
+             var a = [0, 0, 0, 0]; // convert plaintext to (ints ^ key)
 
-           function sort(changes) {
-             // find a referenced relation in the current changeset
-             function resolve(item) {
-               return relations.find(function (relation) {
-                 return item.keyAttributes.type === 'relation' && item.keyAttributes.ref === relation['@id'];
-               });
-             } // a new item is an item that has not been already processed
+             var t = convertToInt32(plaintext);
 
+             for (var i = 0; i < 4; i++) {
+               t[i] ^= this._Ke[0][i];
+             } // apply round transforms
 
-             function isNew(item) {
-               return !sorted[item['@id']] && !processing.find(function (proc) {
-                 return proc['@id'] === item['@id'];
-               });
-             }
 
-             var processing = [];
-             var sorted = {};
-             var relations = changes.relation;
-             if (!relations) return changes;
+             for (var r = 1; r < rounds; r++) {
+               for (var i = 0; i < 4; i++) {
+                 a[i] = T1[t[i] >> 24 & 0xff] ^ T2[t[(i + 1) % 4] >> 16 & 0xff] ^ T3[t[(i + 2) % 4] >> 8 & 0xff] ^ T4[t[(i + 3) % 4] & 0xff] ^ this._Ke[r][i];
+               }
 
-             for (var i = 0; i < relations.length; i++) {
-               var relation = relations[i]; // skip relation if already sorted
+               t = a.slice();
+             } // the last round is special
 
-               if (!sorted[relation['@id']]) {
-                 processing.push(relation);
-               }
 
-               while (processing.length > 0) {
-                 var next = processing[0],
-                     deps = next.member.map(resolve).filter(Boolean).filter(isNew);
+             var result = createArray(16),
+                 tt;
 
-                 if (deps.length === 0) {
-                   sorted[next['@id']] = next;
-                   processing.shift();
-                 } else {
-                   processing = deps.concat(processing);
-                 }
-               }
+             for (var i = 0; i < 4; i++) {
+               tt = this._Ke[rounds][i];
+               result[4 * i] = (S[t[i] >> 24 & 0xff] ^ tt >> 24) & 0xff;
+               result[4 * i + 1] = (S[t[(i + 1) % 4] >> 16 & 0xff] ^ tt >> 16) & 0xff;
+               result[4 * i + 2] = (S[t[(i + 2) % 4] >> 8 & 0xff] ^ tt >> 8) & 0xff;
+               result[4 * i + 3] = (S[t[(i + 3) % 4] & 0xff] ^ tt) & 0xff;
              }
 
-             changes.relation = Object.values(sorted);
-             return changes;
-           }
-
-           function rep(entity) {
-             return entity.asJXON(changeset_id);
-           }
+             return result;
+           };
 
-           return {
-             osmChange: {
-               '@version': 0.6,
-               '@generator': 'iD',
-               'create': sort(nest(changes.created.map(rep), ['node', 'way', 'relation'])),
-               'modify': nest(changes.modified.map(rep), ['node', 'way', 'relation']),
-               'delete': Object.assign(nest(changes.deleted.map(rep), ['relation', 'way', 'node']), {
-                 '@if-unused': true
-               })
+           AES.prototype.decrypt = function (ciphertext) {
+             if (ciphertext.length != 16) {
+               throw new Error('invalid ciphertext size (must be 16 bytes)');
              }
-           };
-         },
-         asGeoJSON: function asGeoJSON() {
-           return {};
-         }
-       });
 
-       function osmNote() {
-         if (!(this instanceof osmNote)) {
-           return new osmNote().initialize(arguments);
-         } else if (arguments.length) {
-           this.initialize(arguments);
-         }
-       }
+             var rounds = this._Kd.length - 1;
+             var a = [0, 0, 0, 0]; // convert plaintext to (ints ^ key)
 
-       osmNote.id = function () {
-         return osmNote.id.next--;
-       };
+             var t = convertToInt32(ciphertext);
 
-       osmNote.id.next = -1;
-       Object.assign(osmNote.prototype, {
-         type: 'note',
-         initialize: function initialize(sources) {
-           for (var i = 0; i < sources.length; ++i) {
-             var source = sources[i];
+             for (var i = 0; i < 4; i++) {
+               t[i] ^= this._Kd[0][i];
+             } // apply round transforms
 
-             for (var prop in source) {
-               if (Object.prototype.hasOwnProperty.call(source, prop)) {
-                 if (source[prop] === undefined) {
-                   delete this[prop];
-                 } else {
-                   this[prop] = source[prop];
-                 }
+
+             for (var r = 1; r < rounds; r++) {
+               for (var i = 0; i < 4; i++) {
+                 a[i] = T5[t[i] >> 24 & 0xff] ^ T6[t[(i + 3) % 4] >> 16 & 0xff] ^ T7[t[(i + 2) % 4] >> 8 & 0xff] ^ T8[t[(i + 1) % 4] & 0xff] ^ this._Kd[r][i];
                }
-             }
-           }
 
-           if (!this.id) {
-             this.id = osmNote.id().toString();
-           }
+               t = a.slice();
+             } // the last round is special
 
-           return this;
-         },
-         extent: function extent() {
-           return new geoExtent(this.loc);
-         },
-         update: function update(attrs) {
-           return osmNote(this, attrs); // {v: 1 + (this.v || 0)}
-         },
-         isNew: function isNew() {
-           return this.id < 0;
-         },
-         move: function move(loc) {
-           return this.update({
-             loc: loc
-           });
-         }
-       });
 
-       function osmRelation() {
-         if (!(this instanceof osmRelation)) {
-           return new osmRelation().initialize(arguments);
-         } else if (arguments.length) {
-           this.initialize(arguments);
-         }
-       }
-       osmEntity.relation = osmRelation;
-       osmRelation.prototype = Object.create(osmEntity.prototype);
+             var result = createArray(16),
+                 tt;
 
-       osmRelation.creationOrder = function (a, b) {
-         var aId = parseInt(osmEntity.id.toOSM(a.id), 10);
-         var bId = parseInt(osmEntity.id.toOSM(b.id), 10);
-         if (aId < 0 || bId < 0) return aId - bId;
-         return bId - aId;
-       };
+             for (var i = 0; i < 4; i++) {
+               tt = this._Kd[rounds][i];
+               result[4 * i] = (Si[t[i] >> 24 & 0xff] ^ tt >> 24) & 0xff;
+               result[4 * i + 1] = (Si[t[(i + 3) % 4] >> 16 & 0xff] ^ tt >> 16) & 0xff;
+               result[4 * i + 2] = (Si[t[(i + 2) % 4] >> 8 & 0xff] ^ tt >> 8) & 0xff;
+               result[4 * i + 3] = (Si[t[(i + 1) % 4] & 0xff] ^ tt) & 0xff;
+             }
 
-       Object.assign(osmRelation.prototype, {
-         type: 'relation',
-         members: [],
-         copy: function copy(resolver, copies) {
-           if (copies[this.id]) return copies[this.id];
-           var copy = osmEntity.prototype.copy.call(this, resolver, copies);
-           var members = this.members.map(function (member) {
-             return Object.assign({}, member, {
-               id: resolver.entity(member.id).copy(resolver, copies).id
-             });
-           });
-           copy = copy.update({
-             members: members
-           });
-           copies[this.id] = copy;
-           return copy;
-         },
-         extent: function extent(resolver, memo) {
-           return resolver["transient"](this, 'extent', function () {
-             if (memo && memo[this.id]) return geoExtent();
-             memo = memo || {};
-             memo[this.id] = true;
-             var extent = geoExtent();
+             return result;
+           };
+           /**
+            *  Mode Of Operation - Electonic Codebook (ECB)
+            */
 
-             for (var i = 0; i < this.members.length; i++) {
-               var member = resolver.hasEntity(this.members[i].id);
 
-               if (member) {
-                 extent._extend(member.extent(resolver, memo));
-               }
+           var ModeOfOperationECB = function ModeOfOperationECB(key) {
+             if (!(this instanceof ModeOfOperationECB)) {
+               throw Error('AES must be instanitated with `new`');
              }
 
-             return extent;
-           });
-         },
-         geometry: function geometry(graph) {
-           return graph["transient"](this, 'geometry', function () {
-             return this.isMultipolygon() ? 'area' : 'relation';
-           });
-         },
-         isDegenerate: function isDegenerate() {
-           return this.members.length === 0;
-         },
-         // Return an array of members, each extended with an 'index' property whose value
-         // is the member index.
-         indexedMembers: function indexedMembers() {
-           var result = new Array(this.members.length);
+             this.description = "Electronic Code Block";
+             this.name = "ecb";
+             this._aes = new AES(key);
+           };
 
-           for (var i = 0; i < this.members.length; i++) {
-             result[i] = Object.assign({}, this.members[i], {
-               index: i
-             });
-           }
+           ModeOfOperationECB.prototype.encrypt = function (plaintext) {
+             plaintext = coerceArray(plaintext);
 
-           return result;
-         },
-         // Return the first member with the given role. A copy of the member object
-         // is returned, extended with an 'index' property whose value is the member index.
-         memberByRole: function memberByRole(role) {
-           for (var i = 0; i < this.members.length; i++) {
-             if (this.members[i].role === role) {
-               return Object.assign({}, this.members[i], {
-                 index: i
-               });
+             if (plaintext.length % 16 !== 0) {
+               throw new Error('invalid plaintext size (must be multiple of 16 bytes)');
              }
-           }
-         },
-         // Same as memberByRole, but returns all members with the given role
-         membersByRole: function membersByRole(role) {
-           var result = [];
 
-           for (var i = 0; i < this.members.length; i++) {
-             if (this.members[i].role === role) {
-               result.push(Object.assign({}, this.members[i], {
-                 index: i
-               }));
+             var ciphertext = createArray(plaintext.length);
+             var block = createArray(16);
+
+             for (var i = 0; i < plaintext.length; i += 16) {
+               copyArray(plaintext, block, 0, i, i + 16);
+               block = this._aes.encrypt(block);
+               copyArray(block, ciphertext, i);
              }
-           }
 
-           return result;
-         },
-         // Return the first member with the given id. A copy of the member object
-         // is returned, extended with an 'index' property whose value is the member index.
-         memberById: function memberById(id) {
-           for (var i = 0; i < this.members.length; i++) {
-             if (this.members[i].id === id) {
-               return Object.assign({}, this.members[i], {
-                 index: i
-               });
+             return ciphertext;
+           };
+
+           ModeOfOperationECB.prototype.decrypt = function (ciphertext) {
+             ciphertext = coerceArray(ciphertext);
+
+             if (ciphertext.length % 16 !== 0) {
+               throw new Error('invalid ciphertext size (must be multiple of 16 bytes)');
              }
-           }
-         },
-         // Return the first member with the given id and role. A copy of the member object
-         // is returned, extended with an 'index' property whose value is the member index.
-         memberByIdAndRole: function memberByIdAndRole(id, role) {
-           for (var i = 0; i < this.members.length; i++) {
-             if (this.members[i].id === id && this.members[i].role === role) {
-               return Object.assign({}, this.members[i], {
-                 index: i
-               });
+
+             var plaintext = createArray(ciphertext.length);
+             var block = createArray(16);
+
+             for (var i = 0; i < ciphertext.length; i += 16) {
+               copyArray(ciphertext, block, 0, i, i + 16);
+               block = this._aes.decrypt(block);
+               copyArray(block, plaintext, i);
              }
-           }
-         },
-         addMember: function addMember(member, index) {
-           var members = this.members.slice();
-           members.splice(index === undefined ? members.length : index, 0, member);
-           return this.update({
-             members: members
-           });
-         },
-         updateMember: function updateMember(member, index) {
-           var members = this.members.slice();
-           members.splice(index, 1, Object.assign({}, members[index], member));
-           return this.update({
-             members: members
-           });
-         },
-         removeMember: function removeMember(index) {
-           var members = this.members.slice();
-           members.splice(index, 1);
-           return this.update({
-             members: members
-           });
-         },
-         removeMembersWithID: function removeMembersWithID(id) {
-           var members = this.members.filter(function (m) {
-             return m.id !== id;
-           });
-           return this.update({
-             members: members
-           });
-         },
-         moveMember: function moveMember(fromIndex, toIndex) {
-           var members = this.members.slice();
-           members.splice(toIndex, 0, members.splice(fromIndex, 1)[0]);
-           return this.update({
-             members: members
-           });
-         },
-         // Wherever a member appears with id `needle.id`, replace it with a member
-         // with id `replacement.id`, type `replacement.type`, and the original role,
-         // By default, adding a duplicate member (by id and role) is prevented.
-         // Return an updated relation.
-         replaceMember: function replaceMember(needle, replacement, keepDuplicates) {
-           if (!this.memberById(needle.id)) return this;
-           var members = [];
 
-           for (var i = 0; i < this.members.length; i++) {
-             var member = this.members[i];
+             return plaintext;
+           };
+           /**
+            *  Mode Of Operation - Cipher Block Chaining (CBC)
+            */
 
-             if (member.id !== needle.id) {
-               members.push(member);
-             } else if (keepDuplicates || !this.memberByIdAndRole(replacement.id, member.role)) {
-               members.push({
-                 id: replacement.id,
-                 type: replacement.type,
-                 role: member.role
-               });
+
+           var ModeOfOperationCBC = function ModeOfOperationCBC(key, iv) {
+             if (!(this instanceof ModeOfOperationCBC)) {
+               throw Error('AES must be instanitated with `new`');
              }
-           }
 
-           return this.update({
-             members: members
-           });
-         },
-         asJXON: function asJXON(changeset_id) {
-           var r = {
-             relation: {
-               '@id': this.osmId(),
-               '@version': this.version || 0,
-               member: this.members.map(function (member) {
-                 return {
-                   keyAttributes: {
-                     type: member.type,
-                     role: member.role,
-                     ref: osmEntity.id.toOSM(member.id)
-                   }
-                 };
-               }, this),
-               tag: Object.keys(this.tags).map(function (k) {
-                 return {
-                   keyAttributes: {
-                     k: k,
-                     v: this.tags[k]
-                   }
-                 };
-               }, this)
+             this.description = "Cipher Block Chaining";
+             this.name = "cbc";
+
+             if (!iv) {
+               iv = createArray(16);
+             } else if (iv.length != 16) {
+               throw new Error('invalid initialation vector size (must be 16 bytes)');
              }
+
+             this._lastCipherblock = coerceArray(iv, true);
+             this._aes = new AES(key);
            };
 
-           if (changeset_id) {
-             r.relation['@changeset'] = changeset_id;
-           }
+           ModeOfOperationCBC.prototype.encrypt = function (plaintext) {
+             plaintext = coerceArray(plaintext);
 
-           return r;
-         },
-         asGeoJSON: function asGeoJSON(resolver) {
-           return resolver["transient"](this, 'GeoJSON', function () {
-             if (this.isMultipolygon()) {
-               return {
-                 type: 'MultiPolygon',
-                 coordinates: this.multipolygon(resolver)
-               };
-             } else {
-               return {
-                 type: 'FeatureCollection',
-                 properties: this.tags,
-                 features: this.members.map(function (member) {
-                   return Object.assign({
-                     role: member.role
-                   }, resolver.entity(member.id).asGeoJSON(resolver));
-                 })
-               };
-             }
-           });
-         },
-         area: function area(resolver) {
-           return resolver["transient"](this, 'area', function () {
-             return d3_geoArea(this.asGeoJSON(resolver));
-           });
-         },
-         isMultipolygon: function isMultipolygon() {
-           return this.tags.type === 'multipolygon';
-         },
-         isComplete: function isComplete(resolver) {
-           for (var i = 0; i < this.members.length; i++) {
-             if (!resolver.hasEntity(this.members[i].id)) {
-               return false;
+             if (plaintext.length % 16 !== 0) {
+               throw new Error('invalid plaintext size (must be multiple of 16 bytes)');
              }
-           }
 
-           return true;
-         },
-         hasFromViaTo: function hasFromViaTo() {
-           return this.members.some(function (m) {
-             return m.role === 'from';
-           }) && this.members.some(function (m) {
-             return m.role === 'via';
-           }) && this.members.some(function (m) {
-             return m.role === 'to';
-           });
-         },
-         isRestriction: function isRestriction() {
-           return !!(this.tags.type && this.tags.type.match(/^restriction:?/));
-         },
-         isValidRestriction: function isValidRestriction() {
-           if (!this.isRestriction()) return false;
-           var froms = this.members.filter(function (m) {
-             return m.role === 'from';
-           });
-           var vias = this.members.filter(function (m) {
-             return m.role === 'via';
-           });
-           var tos = this.members.filter(function (m) {
-             return m.role === 'to';
-           });
-           if (froms.length !== 1 && this.tags.restriction !== 'no_entry') return false;
-           if (froms.some(function (m) {
-             return m.type !== 'way';
-           })) return false;
-           if (tos.length !== 1 && this.tags.restriction !== 'no_exit') return false;
-           if (tos.some(function (m) {
-             return m.type !== 'way';
-           })) return false;
-           if (vias.length === 0) return false;
-           if (vias.length > 1 && vias.some(function (m) {
-             return m.type !== 'way';
-           })) return false;
-           return true;
-         },
-         // Returns an array [A0, ... An], each Ai being an array of node arrays [Nds0, ... Ndsm],
-         // where Nds0 is an outer ring and subsequent Ndsi's (if any i > 0) being inner rings.
-         //
-         // This corresponds to the structure needed for rendering a multipolygon path using a
-         // `evenodd` fill rule, as well as the structure of a GeoJSON MultiPolygon geometry.
-         //
-         // In the case of invalid geometries, this function will still return a result which
-         // includes the nodes of all way members, but some Nds may be unclosed and some inner
-         // rings not matched with the intended outer ring.
-         //
-         multipolygon: function multipolygon(resolver) {
-           var outers = this.members.filter(function (m) {
-             return 'outer' === (m.role || 'outer');
-           });
-           var inners = this.members.filter(function (m) {
-             return 'inner' === m.role;
-           });
-           outers = osmJoinWays(outers, resolver);
-           inners = osmJoinWays(inners, resolver);
+             var ciphertext = createArray(plaintext.length);
+             var block = createArray(16);
 
-           var sequenceToLineString = function sequenceToLineString(sequence) {
-             if (sequence.nodes.length > 2 && sequence.nodes[0] !== sequence.nodes[sequence.nodes.length - 1]) {
-               // close unclosed parts to ensure correct area rendering - #2945
-               sequence.nodes.push(sequence.nodes[0]);
+             for (var i = 0; i < plaintext.length; i += 16) {
+               copyArray(plaintext, block, 0, i, i + 16);
+
+               for (var j = 0; j < 16; j++) {
+                 block[j] ^= this._lastCipherblock[j];
+               }
+
+               this._lastCipherblock = this._aes.encrypt(block);
+               copyArray(this._lastCipherblock, ciphertext, i);
              }
 
-             return sequence.nodes.map(function (node) {
-               return node.loc;
-             });
+             return ciphertext;
            };
 
-           outers = outers.map(sequenceToLineString);
-           inners = inners.map(sequenceToLineString);
-           var result = outers.map(function (o) {
-             // Heuristic for detecting counterclockwise winding order. Assumes
-             // that OpenStreetMap polygons are not hemisphere-spanning.
-             return [d3_geoArea({
-               type: 'Polygon',
-               coordinates: [o]
-             }) > 2 * Math.PI ? o.reverse() : o];
-           });
+           ModeOfOperationCBC.prototype.decrypt = function (ciphertext) {
+             ciphertext = coerceArray(ciphertext);
 
-           function findOuter(inner) {
-             var o, outer;
+             if (ciphertext.length % 16 !== 0) {
+               throw new Error('invalid ciphertext size (must be multiple of 16 bytes)');
+             }
 
-             for (o = 0; o < outers.length; o++) {
-               outer = outers[o];
-               if (geoPolygonContainsPolygon(outer, inner)) return o;
+             var plaintext = createArray(ciphertext.length);
+             var block = createArray(16);
+
+             for (var i = 0; i < ciphertext.length; i += 16) {
+               copyArray(ciphertext, block, 0, i, i + 16);
+               block = this._aes.decrypt(block);
+
+               for (var j = 0; j < 16; j++) {
+                 plaintext[i + j] = block[j] ^ this._lastCipherblock[j];
+               }
+
+               copyArray(ciphertext, this._lastCipherblock, 0, i, i + 16);
              }
 
-             for (o = 0; o < outers.length; o++) {
-               outer = outers[o];
-               if (geoPolygonIntersectsPolygon(outer, inner, false)) return o;
+             return plaintext;
+           };
+           /**
+            *  Mode Of Operation - Cipher Feedback (CFB)
+            */
+
+
+           var ModeOfOperationCFB = function ModeOfOperationCFB(key, iv, segmentSize) {
+             if (!(this instanceof ModeOfOperationCFB)) {
+               throw Error('AES must be instanitated with `new`');
              }
-           }
 
-           for (var i = 0; i < inners.length; i++) {
-             var inner = inners[i];
+             this.description = "Cipher Feedback";
+             this.name = "cfb";
 
-             if (d3_geoArea({
-               type: 'Polygon',
-               coordinates: [inner]
-             }) < 2 * Math.PI) {
-               inner = inner.reverse();
+             if (!iv) {
+               iv = createArray(16);
+             } else if (iv.length != 16) {
+               throw new Error('invalid initialation vector size (must be 16 size)');
              }
 
-             var o = findOuter(inners[i]);
+             if (!segmentSize) {
+               segmentSize = 1;
+             }
 
-             if (o !== undefined) {
-               result[o].push(inners[i]);
-             } else {
-               result.push([inners[i]]); // Invalid geometry
+             this.segmentSize = segmentSize;
+             this._shiftRegister = coerceArray(iv, true);
+             this._aes = new AES(key);
+           };
+
+           ModeOfOperationCFB.prototype.encrypt = function (plaintext) {
+             if (plaintext.length % this.segmentSize != 0) {
+               throw new Error('invalid plaintext size (must be segmentSize bytes)');
              }
-           }
 
-           return result;
-         }
-       });
+             var encrypted = coerceArray(plaintext, true);
+             var xorSegment;
 
-       var QAItem = /*#__PURE__*/function () {
-         function QAItem(loc, service, itemType, id, props) {
-           _classCallCheck(this, QAItem);
+             for (var i = 0; i < encrypted.length; i += this.segmentSize) {
+               xorSegment = this._aes.encrypt(this._shiftRegister);
 
-           // Store required properties
-           this.loc = loc;
-           this.service = service.title;
-           this.itemType = itemType; // All issues must have an ID for selection, use generic if none specified
+               for (var j = 0; j < this.segmentSize; j++) {
+                 encrypted[i + j] ^= xorSegment[j];
+               } // Shift the register
 
-           this.id = id ? id : "".concat(QAItem.id());
-           this.update(props); // Some QA services have marker icons to differentiate issues
 
-           if (service && typeof service.getIcon === 'function') {
-             this.icon = service.getIcon(itemType);
-           }
-         }
+               copyArray(this._shiftRegister, this._shiftRegister, 0, this.segmentSize);
+               copyArray(encrypted, this._shiftRegister, 16 - this.segmentSize, i, i + this.segmentSize);
+             }
 
-         _createClass(QAItem, [{
-           key: "update",
-           value: function update(props) {
-             var _this = this;
+             return encrypted;
+           };
 
-             // You can't override this initial information
-             var loc = this.loc,
-                 service = this.service,
-                 itemType = this.itemType,
-                 id = this.id;
-             Object.keys(props).forEach(function (prop) {
-               return _this[prop] = props[prop];
-             });
-             this.loc = loc;
-             this.service = service;
-             this.itemType = itemType;
-             this.id = id;
-             return this;
-           } // Generic handling for newly created QAItems
+           ModeOfOperationCFB.prototype.decrypt = function (ciphertext) {
+             if (ciphertext.length % this.segmentSize != 0) {
+               throw new Error('invalid ciphertext size (must be segmentSize bytes)');
+             }
 
-         }], [{
-           key: "id",
-           value: function id() {
-             return this.nextId--;
-           }
-         }]);
+             var plaintext = coerceArray(ciphertext, true);
+             var xorSegment;
 
-         return QAItem;
-       }();
-       QAItem.nextId = -1;
+             for (var i = 0; i < plaintext.length; i += this.segmentSize) {
+               xorSegment = this._aes.encrypt(this._shiftRegister);
 
-       //
-       // Optionally, split only the given ways, if multiple ways share
-       // the given node.
-       //
-       // This is the inverse of `iD.actionJoin`.
-       //
-       // For testing convenience, accepts an ID to assign to the new way.
-       // Normally, this will be undefined and the way will automatically
-       // be assigned a new ID.
-       //
-       // Reference:
-       //   https://github.com/systemed/potlatch2/blob/master/net/systemeD/halcyon/connection/actions/SplitWayAction.as
-       //
+               for (var j = 0; j < this.segmentSize; j++) {
+                 plaintext[i + j] ^= xorSegment[j];
+               } // Shift the register
 
-       function actionSplit(nodeIds, newWayIds) {
-         // accept single ID for backwards-compatiblity
-         if (typeof nodeIds === 'string') nodeIds = [nodeIds];
 
-         var _wayIDs; // the strategy for picking which way will have a new version and which way is newly created
+               copyArray(this._shiftRegister, this._shiftRegister, 0, this.segmentSize);
+               copyArray(ciphertext, this._shiftRegister, 16 - this.segmentSize, i, i + this.segmentSize);
+             }
 
+             return plaintext;
+           };
+           /**
+            *  Mode Of Operation - Output Feedback (OFB)
+            */
 
-         var _keepHistoryOn = 'longest'; // 'longest', 'first'
-         // The IDs of the ways actually created by running this action
 
-         var _createdWayIDs = [];
+           var ModeOfOperationOFB = function ModeOfOperationOFB(key, iv) {
+             if (!(this instanceof ModeOfOperationOFB)) {
+               throw Error('AES must be instanitated with `new`');
+             }
 
-         function dist(graph, nA, nB) {
-           var locA = graph.entity(nA).loc;
-           var locB = graph.entity(nB).loc;
-           var epsilon = 1e-6;
-           return locA && locB ? geoSphericalDistance(locA, locB) : epsilon;
-         } // If the way is closed, we need to search for a partner node
-         // to split the way at.
-         //
-         // The following looks for a node that is both far away from
-         // the initial node in terms of way segment length and nearby
-         // in terms of beeline-distance. This assures that areas get
-         // split on the most "natural" points (independent of the number
-         // of nodes).
-         // For example: bone-shaped areas get split across their waist
-         // line, circles across the diameter.
+             this.description = "Output Feedback";
+             this.name = "ofb";
 
+             if (!iv) {
+               iv = createArray(16);
+             } else if (iv.length != 16) {
+               throw new Error('invalid initialation vector size (must be 16 bytes)');
+             }
 
-         function splitArea(nodes, idxA, graph) {
-           var lengths = new Array(nodes.length);
-           var length;
-           var i;
-           var best = 0;
-           var idxB;
+             this._lastPrecipher = coerceArray(iv, true);
+             this._lastPrecipherIndex = 16;
+             this._aes = new AES(key);
+           };
 
-           function wrap(index) {
-             return utilWrap(index, nodes.length);
-           } // calculate lengths
+           ModeOfOperationOFB.prototype.encrypt = function (plaintext) {
+             var encrypted = coerceArray(plaintext, true);
 
+             for (var i = 0; i < encrypted.length; i++) {
+               if (this._lastPrecipherIndex === 16) {
+                 this._lastPrecipher = this._aes.encrypt(this._lastPrecipher);
+                 this._lastPrecipherIndex = 0;
+               }
 
-           length = 0;
+               encrypted[i] ^= this._lastPrecipher[this._lastPrecipherIndex++];
+             }
 
-           for (i = wrap(idxA + 1); i !== idxA; i = wrap(i + 1)) {
-             length += dist(graph, nodes[i], nodes[wrap(i - 1)]);
-             lengths[i] = length;
-           }
+             return encrypted;
+           }; // Decryption is symetric
 
-           length = 0;
 
-           for (i = wrap(idxA - 1); i !== idxA; i = wrap(i - 1)) {
-             length += dist(graph, nodes[i], nodes[wrap(i + 1)]);
+           ModeOfOperationOFB.prototype.decrypt = ModeOfOperationOFB.prototype.encrypt;
+           /**
+            *  Counter object for CTR common mode of operation
+            */
 
-             if (length < lengths[i]) {
-               lengths[i] = length;
-             }
-           } // determine best opposite node to split
+           var Counter = function Counter(initialValue) {
+             if (!(this instanceof Counter)) {
+               throw Error('Counter must be instanitated with `new`');
+             } // We allow 0, but anything false-ish uses the default 1
 
 
-           for (i = 0; i < nodes.length; i++) {
-             var cost = lengths[i] / dist(graph, nodes[idxA], nodes[i]);
+             if (initialValue !== 0 && !initialValue) {
+               initialValue = 1;
+             }
 
-             if (cost > best) {
-               idxB = i;
-               best = cost;
+             if (typeof initialValue === 'number') {
+               this._counter = createArray(16);
+               this.setValue(initialValue);
+             } else {
+               this.setBytes(initialValue);
              }
-           }
+           };
 
-           return idxB;
-         }
+           Counter.prototype.setValue = function (value) {
+             if (typeof value !== 'number' || parseInt(value) != value) {
+               throw new Error('invalid counter value (must be an integer)');
+             } // We cannot safely handle numbers beyond the safe range for integers
 
-         function totalLengthBetweenNodes(graph, nodes) {
-           var totalLength = 0;
 
-           for (var i = 0; i < nodes.length - 1; i++) {
-             totalLength += dist(graph, nodes[i], nodes[i + 1]);
-           }
+             if (value > Number.MAX_SAFE_INTEGER) {
+               throw new Error('integer value out of safe range');
+             }
 
-           return totalLength;
-         }
+             for (var index = 15; index >= 0; --index) {
+               this._counter[index] = value % 256;
+               value = parseInt(value / 256);
+             }
+           };
 
-         function split(graph, nodeId, wayA, newWayId) {
-           var wayB = osmWay({
-             id: newWayId,
-             tags: wayA.tags
-           }); // `wayB` is the NEW way
+           Counter.prototype.setBytes = function (bytes) {
+             bytes = coerceArray(bytes, true);
 
-           var origNodes = wayA.nodes.slice();
-           var nodesA;
-           var nodesB;
-           var isArea = wayA.isArea();
-           var isOuter = osmIsOldMultipolygonOuterMember(wayA, graph);
+             if (bytes.length != 16) {
+               throw new Error('invalid counter bytes size (must be 16 bytes)');
+             }
 
-           if (wayA.isClosed()) {
-             var nodes = wayA.nodes.slice(0, -1);
-             var idxA = nodes.indexOf(nodeId);
-             var idxB = splitArea(nodes, idxA, graph);
+             this._counter = bytes;
+           };
 
-             if (idxB < idxA) {
-               nodesA = nodes.slice(idxA).concat(nodes.slice(0, idxB + 1));
-               nodesB = nodes.slice(idxB, idxA + 1);
-             } else {
-               nodesA = nodes.slice(idxA, idxB + 1);
-               nodesB = nodes.slice(idxB).concat(nodes.slice(0, idxA + 1));
+           Counter.prototype.increment = function () {
+             for (var i = 15; i >= 0; i--) {
+               if (this._counter[i] === 255) {
+                 this._counter[i] = 0;
+               } else {
+                 this._counter[i]++;
+                 break;
+               }
              }
-           } else {
-             var idx = wayA.nodes.indexOf(nodeId, 1);
-             nodesA = wayA.nodes.slice(0, idx + 1);
-             nodesB = wayA.nodes.slice(idx);
-           }
+           };
+           /**
+            *  Mode Of Operation - Counter (CTR)
+            */
 
-           var lengthA = totalLengthBetweenNodes(graph, nodesA);
-           var lengthB = totalLengthBetweenNodes(graph, nodesB);
 
-           if (_keepHistoryOn === 'longest' && lengthB > lengthA) {
-             // keep the history on the longer way, regardless of the node count
-             wayA = wayA.update({
-               nodes: nodesB
-             });
-             wayB = wayB.update({
-               nodes: nodesA
-             });
-             var temp = lengthA;
-             lengthA = lengthB;
-             lengthB = temp;
-           } else {
-             wayA = wayA.update({
-               nodes: nodesA
-             });
-             wayB = wayB.update({
-               nodes: nodesB
-             });
-           }
+           var ModeOfOperationCTR = function ModeOfOperationCTR(key, counter) {
+             if (!(this instanceof ModeOfOperationCTR)) {
+               throw Error('AES must be instanitated with `new`');
+             }
 
-           if (wayA.tags.step_count) {
-             // divide up the the step count proportionally between the two ways
-             var stepCount = parseFloat(wayA.tags.step_count);
+             this.description = "Counter";
+             this.name = "ctr";
 
-             if (stepCount && // ensure a number
-             isFinite(stepCount) && // ensure positive
-             stepCount > 0 && // ensure integer
-             Math.round(stepCount) === stepCount) {
-               var tagsA = Object.assign({}, wayA.tags);
-               var tagsB = Object.assign({}, wayB.tags);
-               var ratioA = lengthA / (lengthA + lengthB);
-               var countA = Math.round(stepCount * ratioA);
-               tagsA.step_count = countA.toString();
-               tagsB.step_count = (stepCount - countA).toString();
-               wayA = wayA.update({
-                 tags: tagsA
-               });
-               wayB = wayB.update({
-                 tags: tagsB
-               });
+             if (!(counter instanceof Counter)) {
+               counter = new Counter(counter);
              }
-           }
 
-           graph = graph.replace(wayA);
-           graph = graph.replace(wayB);
-           graph.parentRelations(wayA).forEach(function (relation) {
-             var member; // Turn restrictions - make sure:
-             // 1. Splitting a FROM/TO way - only `wayA` OR `wayB` remains in relation
-             //    (whichever one is connected to the VIA node/ways)
-             // 2. Splitting a VIA way - `wayB` remains in relation as a VIA way
+             this._counter = counter;
+             this._remainingCounter = null;
+             this._remainingCounterIndex = 16;
+             this._aes = new AES(key);
+           };
 
-             if (relation.hasFromViaTo()) {
-               var f = relation.memberByRole('from');
-               var v = relation.membersByRole('via');
-               var t = relation.memberByRole('to');
-               var i; // 1. split a FROM/TO
+           ModeOfOperationCTR.prototype.encrypt = function (plaintext) {
+             var encrypted = coerceArray(plaintext, true);
 
-               if (f.id === wayA.id || t.id === wayA.id) {
-                 var keepB = false;
+             for (var i = 0; i < encrypted.length; i++) {
+               if (this._remainingCounterIndex === 16) {
+                 this._remainingCounter = this._aes.encrypt(this._counter._counter);
+                 this._remainingCounterIndex = 0;
 
-                 if (v.length === 1 && v[0].type === 'node') {
-                   // check via node
-                   keepB = wayB.contains(v[0].id);
-                 } else {
-                   // check via way(s)
-                   for (i = 0; i < v.length; i++) {
-                     if (v[i].type === 'way') {
-                       var wayVia = graph.hasEntity(v[i].id);
+                 this._counter.increment();
+               }
 
-                       if (wayVia && utilArrayIntersection(wayB.nodes, wayVia.nodes).length) {
-                         keepB = true;
-                         break;
-                       }
-                     }
-                   }
-                 }
+               encrypted[i] ^= this._remainingCounter[this._remainingCounterIndex++];
+             }
 
-                 if (keepB) {
-                   relation = relation.replaceMember(wayA, wayB);
-                   graph = graph.replace(relation);
-                 } // 2. split a VIA
+             return encrypted;
+           }; // Decryption is symetric
 
-               } else {
-                 for (i = 0; i < v.length; i++) {
-                   if (v[i].type === 'way' && v[i].id === wayA.id) {
-                     member = {
-                       id: wayB.id,
-                       type: 'way',
-                       role: 'via'
-                     };
-                     graph = actionAddMember(relation.id, member, v[i].index + 1)(graph);
-                     break;
-                   }
-                 }
-               } // All other relations (Routes, Multipolygons, etc):
-               // 1. Both `wayA` and `wayB` remain in the relation
-               // 2. But must be inserted as a pair (see `actionAddMember` for details)
 
-             } else {
-               if (relation === isOuter) {
-                 graph = graph.replace(relation.mergeTags(wayA.tags));
-                 graph = graph.replace(wayA.update({
-                   tags: {}
-                 }));
-                 graph = graph.replace(wayB.update({
-                   tags: {}
-                 }));
-               }
+           ModeOfOperationCTR.prototype.decrypt = ModeOfOperationCTR.prototype.encrypt; ///////////////////////
+           // Padding
+           // See:https://tools.ietf.org/html/rfc2315
 
-               member = {
-                 id: wayB.id,
-                 type: 'way',
-                 role: relation.memberById(wayA.id).role
-               };
-               var insertPair = {
-                 originalID: wayA.id,
-                 insertedID: wayB.id,
-                 nodes: origNodes
-               };
-               graph = actionAddMember(relation.id, member, undefined, insertPair)(graph);
+           function pkcs7pad(data) {
+             data = coerceArray(data, true);
+             var padder = 16 - data.length % 16;
+             var result = createArray(data.length + padder);
+             copyArray(data, result);
+
+             for (var i = data.length; i < result.length; i++) {
+               result[i] = padder;
              }
-           });
 
-           if (!isOuter && isArea) {
-             var multipolygon = osmRelation({
-               tags: Object.assign({}, wayA.tags, {
-                 type: 'multipolygon'
-               }),
-               members: [{
-                 id: wayA.id,
-                 role: 'outer',
-                 type: 'way'
-               }, {
-                 id: wayB.id,
-                 role: 'outer',
-                 type: 'way'
-               }]
-             });
-             graph = graph.replace(multipolygon);
-             graph = graph.replace(wayA.update({
-               tags: {}
-             }));
-             graph = graph.replace(wayB.update({
-               tags: {}
-             }));
+             return result;
            }
 
-           _createdWayIDs.push(wayB.id);
-
-           return graph;
-         }
+           function pkcs7strip(data) {
+             data = coerceArray(data, true);
 
-         var action = function action(graph) {
-           _createdWayIDs = [];
-           var newWayIndex = 0;
+             if (data.length < 16) {
+               throw new Error('PKCS#7 invalid length');
+             }
 
-           for (var i = 0; i < nodeIds.length; i++) {
-             var nodeId = nodeIds[i];
-             var candidates = action.waysForNode(nodeId, graph);
+             var padder = data[data.length - 1];
 
-             for (var j = 0; j < candidates.length; j++) {
-               graph = split(graph, nodeId, candidates[j], newWayIds && newWayIds[newWayIndex]);
-               newWayIndex += 1;
+             if (padder > 16) {
+               throw new Error('PKCS#7 padding byte out of range');
              }
-           }
 
-           return graph;
-         };
+             var length = data.length - padder;
 
-         action.getCreatedWayIDs = function () {
-           return _createdWayIDs;
-         };
+             for (var i = 0; i < padder; i++) {
+               if (data[length + i] !== padder) {
+                 throw new Error('PKCS#7 invalid padding byte');
+               }
+             }
 
-         action.waysForNode = function (nodeId, graph) {
-           var node = graph.entity(nodeId);
-           var splittableParents = graph.parentWays(node).filter(isSplittable);
+             var result = createArray(length);
+             copyArray(data, result, 0, 0, length);
+             return result;
+           } ///////////////////////
+           // Exporting
+           // The block cipher
 
-           if (!_wayIDs) {
-             // If the ways to split aren't specified, only split the lines.
-             // If there are no lines to split, split the areas.
-             var hasLine = splittableParents.some(function (parent) {
-               return parent.geometry(graph) === 'line';
-             });
 
-             if (hasLine) {
-               return splittableParents.filter(function (parent) {
-                 return parent.geometry(graph) === 'line';
-               });
+           var aesjs = {
+             AES: AES,
+             Counter: Counter,
+             ModeOfOperation: {
+               ecb: ModeOfOperationECB,
+               cbc: ModeOfOperationCBC,
+               cfb: ModeOfOperationCFB,
+               ofb: ModeOfOperationOFB,
+               ctr: ModeOfOperationCTR
+             },
+             utils: {
+               hex: convertHex,
+               utf8: convertUtf8
+             },
+             padding: {
+               pkcs7: {
+                 pad: pkcs7pad,
+                 strip: pkcs7strip
+               }
+             },
+             _arrayTest: {
+               coerceArray: coerceArray,
+               createArray: createArray,
+               copyArray: copyArray
              }
+           }; // node.js
+
+           {
+             module.exports = aesjs; // RequireJS/AMD
+             // http://www.requirejs.org/docs/api.html
+             // https://github.com/amdjs/amdjs-api/wiki/AMD
            }
+         })();
+       });
 
-           return splittableParents;
+       // We can use keys that are 128 bits (16 bytes), 192 bits (24 bytes) or 256 bits (32 bytes).
+       // To generate a random key:  window.crypto.getRandomValues(new Uint8Array(16));
+       // This default signing key is built into iD and can be used to mask/unmask sensitive values.
 
-           function isSplittable(parent) {
-             // If the ways to split are specified, ignore everything else.
-             if (_wayIDs && _wayIDs.indexOf(parent.id) === -1) return false; // We can fake splitting closed ways at their endpoints...
+       var DEFAULT_128 = [250, 157, 60, 79, 142, 134, 229, 129, 138, 126, 210, 129, 29, 71, 160, 208];
+       function utilAesEncrypt(text, key) {
+         key = key || DEFAULT_128;
+         var textBytes = aesJs.utils.utf8.toBytes(text);
+         var aesCtr = new aesJs.ModeOfOperation.ctr(key);
+         var encryptedBytes = aesCtr.encrypt(textBytes);
+         var encryptedHex = aesJs.utils.hex.fromBytes(encryptedBytes);
+         return encryptedHex;
+       }
+       function utilAesDecrypt(encryptedHex, key) {
+         key = key || DEFAULT_128;
+         var encryptedBytes = aesJs.utils.hex.toBytes(encryptedHex);
+         var aesCtr = new aesJs.ModeOfOperation.ctr(key);
+         var decryptedBytes = aesCtr.decrypt(encryptedBytes);
+         var text = aesJs.utils.utf8.fromBytes(decryptedBytes);
+         return text;
+       }
 
-             if (parent.isClosed()) return true; // otherwise, we can't split nodes at their endpoints.
+       function utilCleanTags(tags) {
+         var out = {};
 
-             for (var i = 1; i < parent.nodes.length - 1; i++) {
-               if (parent.nodes[i] === nodeId) return true;
-             }
+         for (var k in tags) {
+           if (!k) continue;
+           var v = tags[k];
 
-             return false;
+           if (v !== undefined) {
+             out[k] = cleanValue(k, v);
            }
-         };
+         }
 
-         action.ways = function (graph) {
-           return utilArrayUniq([].concat.apply([], nodeIds.map(function (nodeId) {
-             return action.waysForNode(nodeId, graph);
-           })));
-         };
+         return out;
 
-         action.disabled = function (graph) {
-           for (var i = 0; i < nodeIds.length; i++) {
-             var nodeId = nodeIds[i];
-             var candidates = action.waysForNode(nodeId, graph);
+         function cleanValue(k, v) {
+           function keepSpaces(k) {
+             return /_hours|_times|:conditional$/.test(k);
+           }
 
-             if (candidates.length === 0 || _wayIDs && _wayIDs.length !== candidates.length) {
-               return 'not_eligible';
-             }
+           function skip(k) {
+             return /^(description|note|fixme)$/.test(k);
            }
-         };
 
-         action.limitWays = function (val) {
-           if (!arguments.length) return _wayIDs;
-           _wayIDs = val;
-           return action;
-         };
+           if (skip(k)) return v;
+           var cleaned = v.split(';').map(function (s) {
+             return s.trim();
+           }).join(keepSpaces(k) ? '; ' : ';'); // The code below is not intended to validate websites and emails.
+           // It is only intended to prevent obvious copy-paste errors. (#2323)
+           // clean website- and email-like tags
 
-         action.keepHistoryOn = function (val) {
-           if (!arguments.length) return _keepHistoryOn;
-           _keepHistoryOn = val;
-           return action;
-         };
+           if (k.indexOf('website') !== -1 || k.indexOf('email') !== -1 || cleaned.indexOf('http') === 0) {
+             cleaned = cleaned.replace(/[\u200B-\u200F\uFEFF]/g, ''); // strip LRM and other zero width chars
+           }
 
-         return action;
+           return cleaned;
+         }
        }
 
-       function coreGraph(other, mutable) {
-         if (!(this instanceof coreGraph)) return new coreGraph(other, mutable);
+       var _detected;
 
-         if (other instanceof coreGraph) {
-           var base = other.base();
-           this.entities = Object.assign(Object.create(base.entities), other.entities);
-           this._parentWays = Object.assign(Object.create(base.parentWays), other._parentWays);
-           this._parentRels = Object.assign(Object.create(base.parentRels), other._parentRels);
-         } else {
-           this.entities = Object.create({});
-           this._parentWays = Object.create({});
-           this._parentRels = Object.create({});
-           this.rebase(other || [], [this]);
+       function utilDetect(refresh) {
+         if (_detected && !refresh) return _detected;
+         _detected = {};
+         var ua = navigator.userAgent;
+         var m = null;
+         /* Browser */
+
+         m = ua.match(/(edge)\/?\s*(\.?\d+(\.\d+)*)/i); // Edge
+
+         if (m !== null) {
+           _detected.browser = m[1];
+           _detected.version = m[2];
          }
 
-         this.transients = {};
-         this._childNodes = {};
-         this.frozen = !mutable;
-       }
-       coreGraph.prototype = {
-         hasEntity: function hasEntity(id) {
-           return this.entities[id];
-         },
-         entity: function entity(id) {
-           var entity = this.entities[id]; //https://github.com/openstreetmap/iD/issues/3973#issuecomment-307052376
+         if (!_detected.browser) {
+           m = ua.match(/Trident\/.*rv:([0-9]{1,}[\.0-9]{0,})/i); // IE11
 
-           if (!entity) {
-             entity = this.entities.__proto__[id]; // eslint-disable-line no-proto
+           if (m !== null) {
+             _detected.browser = 'msie';
+             _detected.version = m[1];
            }
+         }
 
-           if (!entity) {
-             throw new Error('entity ' + id + ' not found');
+         if (!_detected.browser) {
+           m = ua.match(/(opr)\/?\s*(\.?\d+(\.\d+)*)/i); // Opera 15+
+
+           if (m !== null) {
+             _detected.browser = 'Opera';
+             _detected.version = m[2];
            }
+         }
 
-           return entity;
-         },
-         geometry: function geometry(id) {
-           return this.entity(id).geometry(this);
-         },
-         "transient": function transient(entity, key, fn) {
-           var id = entity.id;
-           var transients = this.transients[id] || (this.transients[id] = {});
+         if (!_detected.browser) {
+           m = ua.match(/(opera|chrome|safari|firefox|msie)\/?\s*(\.?\d+(\.\d+)*)/i);
 
-           if (transients[key] !== undefined) {
-             return transients[key];
+           if (m !== null) {
+             _detected.browser = m[1];
+             _detected.version = m[2];
+             m = ua.match(/version\/([\.\d]+)/i);
+             if (m !== null) _detected.version = m[1];
            }
+         }
 
-           transients[key] = fn.call(entity);
-           return transients[key];
-         },
-         parentWays: function parentWays(entity) {
-           var parents = this._parentWays[entity.id];
-           var result = [];
+         if (!_detected.browser) {
+           _detected.browser = navigator.appName;
+           _detected.version = navigator.appVersion;
+         } // keep major.minor version only..
 
-           if (parents) {
-             parents.forEach(function (id) {
-               result.push(this.entity(id));
-             }, this);
-           }
 
-           return result;
-         },
-         isPoi: function isPoi(entity) {
-           var parents = this._parentWays[entity.id];
-           return !parents || parents.size === 0;
-         },
-         isShared: function isShared(entity) {
-           var parents = this._parentWays[entity.id];
-           return parents && parents.size > 1;
-         },
-         parentRelations: function parentRelations(entity) {
-           var parents = this._parentRels[entity.id];
-           var result = [];
+         _detected.version = _detected.version.split(/\W/).slice(0, 2).join('.'); // detect other browser capabilities
+         // Legacy Opera has incomplete svg style support. See #715
 
-           if (parents) {
-             parents.forEach(function (id) {
-               result.push(this.entity(id));
-             }, this);
-           }
+         _detected.opera = _detected.browser.toLowerCase() === 'opera' && parseFloat(_detected.version) < 15;
 
-           return result;
-         },
-         parentMultipolygons: function parentMultipolygons(entity) {
-           return this.parentRelations(entity).filter(function (relation) {
-             return relation.isMultipolygon();
-           });
-         },
-         childNodes: function childNodes(entity) {
-           if (this._childNodes[entity.id]) return this._childNodes[entity.id];
-           if (!entity.nodes) return [];
-           var nodes = [];
+         if (_detected.browser.toLowerCase() === 'msie') {
+           _detected.ie = true;
+           _detected.browser = 'Internet Explorer';
+           _detected.support = parseFloat(_detected.version) >= 11;
+         } else {
+           _detected.ie = false;
+           _detected.support = true;
+         }
 
-           for (var i = 0; i < entity.nodes.length; i++) {
-             nodes[i] = this.entity(entity.nodes[i]);
-           }
-           this._childNodes[entity.id] = nodes;
-           return this._childNodes[entity.id];
-         },
-         base: function base() {
-           return {
-             'entities': Object.getPrototypeOf(this.entities),
-             'parentWays': Object.getPrototypeOf(this._parentWays),
-             'parentRels': Object.getPrototypeOf(this._parentRels)
-           };
-         },
-         // Unlike other graph methods, rebase mutates in place. This is because it
-         // is used only during the history operation that merges newly downloaded
-         // data into each state. To external consumers, it should appear as if the
-         // graph always contained the newly downloaded data.
-         rebase: function rebase(entities, stack, force) {
-           var base = this.base();
-           var i, j, k, id;
+         _detected.filedrop = window.FileReader && 'ondrop' in window;
+         _detected.download = !(_detected.ie || _detected.browser.toLowerCase() === 'edge');
+         _detected.cssfilters = !(_detected.ie || _detected.browser.toLowerCase() === 'edge');
+         /* Platform */
 
-           for (i = 0; i < entities.length; i++) {
-             var entity = entities[i];
-             if (!entity.visible || !force && base.entities[entity.id]) continue; // Merging data into the base graph
+         if (/Win/.test(ua)) {
+           _detected.os = 'win';
+           _detected.platform = 'Windows';
+         } else if (/Mac/.test(ua)) {
+           _detected.os = 'mac';
+           _detected.platform = 'Macintosh';
+         } else if (/X11/.test(ua) || /Linux/.test(ua)) {
+           _detected.os = 'linux';
+           _detected.platform = 'Linux';
+         } else {
+           _detected.os = 'win';
+           _detected.platform = 'Unknown';
+         }
 
-             base.entities[entity.id] = entity;
+         _detected.isMobileWebKit = (/\b(iPad|iPhone|iPod)\b/.test(ua) || // HACK: iPadOS 13+ requests desktop sites by default by using a Mac user agent,
+         // so assume any "mac" with multitouch is actually iOS
+         navigator.platform === 'MacIntel' && 'maxTouchPoints' in navigator && navigator.maxTouchPoints > 1) && /WebKit/.test(ua) && !/Edge/.test(ua) && !window.MSStream;
+         /* Locale */
+         // An array of locales requested by the browser in priority order.
 
-             this._updateCalculated(undefined, entity, base.parentWays, base.parentRels); // Restore provisionally-deleted nodes that are discovered to have an extant parent
+         _detected.browserLocales = Array.from(new Set( // remove duplicates
+         [navigator.language].concat(navigator.languages || []).concat([// old property for backwards compatibility
+         navigator.userLanguage]) // remove any undefined values
+         .filter(Boolean)));
+         /* Host */
 
+         var loc = window.top.location;
+         var origin = loc.origin;
 
-             if (entity.type === 'way') {
-               for (j = 0; j < entity.nodes.length; j++) {
-                 id = entity.nodes[j];
+         if (!origin) {
+           // for unpatched IE11
+           origin = loc.protocol + '//' + loc.hostname + (loc.port ? ':' + loc.port : '');
+         }
 
-                 for (k = 1; k < stack.length; k++) {
-                   var ents = stack[k].entities;
+         _detected.host = origin + loc.pathname;
+         return _detected;
+       }
 
-                   if (ents.hasOwnProperty(id) && ents[id] === undefined) {
-                     delete ents[id];
-                   }
-                 }
-               }
-             }
+       // Like selection.property('value', ...), but avoids no-op value sets,
+       // which can result in layout/repaint thrashing in some situations.
+       function utilGetSetValue(selection, value) {
+         function d3_selection_value(value) {
+           function valueNull() {
+             delete this.value;
            }
 
-           for (i = 0; i < stack.length; i++) {
-             stack[i]._updateRebased();
-           }
-         },
-         _updateRebased: function _updateRebased() {
-           var base = this.base();
-           Object.keys(this._parentWays).forEach(function (child) {
-             if (base.parentWays[child]) {
-               base.parentWays[child].forEach(function (id) {
-                 if (!this.entities.hasOwnProperty(id)) {
-                   this._parentWays[child].add(id);
-                 }
-               }, this);
-             }
-           }, this);
-           Object.keys(this._parentRels).forEach(function (child) {
-             if (base.parentRels[child]) {
-               base.parentRels[child].forEach(function (id) {
-                 if (!this.entities.hasOwnProperty(id)) {
-                   this._parentRels[child].add(id);
-                 }
-               }, this);
+           function valueConstant() {
+             if (this.value !== value) {
+               this.value = value;
              }
-           }, this);
-           this.transients = {}; // this._childNodes is not updated, under the assumption that
-           // ways are always downloaded with their child nodes.
-         },
-         // Updates calculated properties (parentWays, parentRels) for the specified change
-         _updateCalculated: function _updateCalculated(oldentity, entity, parentWays, parentRels) {
-           parentWays = parentWays || this._parentWays;
-           parentRels = parentRels || this._parentRels;
-           var type = entity && entity.type || oldentity && oldentity.type;
-           var removed, added, i;
+           }
 
-           if (type === 'way') {
-             // Update parentWays
-             if (oldentity && entity) {
-               removed = utilArrayDifference(oldentity.nodes, entity.nodes);
-               added = utilArrayDifference(entity.nodes, oldentity.nodes);
-             } else if (oldentity) {
-               removed = oldentity.nodes;
-               added = [];
-             } else if (entity) {
-               removed = [];
-               added = entity.nodes;
-             }
+           function valueFunction() {
+             var x = value.apply(this, arguments);
 
-             for (i = 0; i < removed.length; i++) {
-               // make a copy of prototype property, store as own property, and update..
-               parentWays[removed[i]] = new Set(parentWays[removed[i]]);
-               parentWays[removed[i]]["delete"](oldentity.id);
+             if (x === null || x === undefined) {
+               delete this.value;
+             } else if (this.value !== x) {
+               this.value = x;
              }
+           }
 
-             for (i = 0; i < added.length; i++) {
-               // make a copy of prototype property, store as own property, and update..
-               parentWays[added[i]] = new Set(parentWays[added[i]]);
-               parentWays[added[i]].add(entity.id);
-             }
-           } else if (type === 'relation') {
-             // Update parentRels
-             // diff only on the IDs since the same entity can be a member multiple times with different roles
-             var oldentityMemberIDs = oldentity ? oldentity.members.map(function (m) {
-               return m.id;
-             }) : [];
-             var entityMemberIDs = entity ? entity.members.map(function (m) {
-               return m.id;
-             }) : [];
+           return value === null || value === undefined ? valueNull : typeof value === 'function' ? valueFunction : valueConstant;
+         }
 
-             if (oldentity && entity) {
-               removed = utilArrayDifference(oldentityMemberIDs, entityMemberIDs);
-               added = utilArrayDifference(entityMemberIDs, oldentityMemberIDs);
-             } else if (oldentity) {
-               removed = oldentityMemberIDs;
-               added = [];
-             } else if (entity) {
-               removed = [];
-               added = entityMemberIDs;
-             }
+         if (arguments.length === 1) {
+           return selection.property('value');
+         }
 
-             for (i = 0; i < removed.length; i++) {
-               // make a copy of prototype property, store as own property, and update..
-               parentRels[removed[i]] = new Set(parentRels[removed[i]]);
-               parentRels[removed[i]]["delete"](oldentity.id);
-             }
+         return selection.each(d3_selection_value(value));
+       }
 
-             for (i = 0; i < added.length; i++) {
-               // make a copy of prototype property, store as own property, and update..
-               parentRels[added[i]] = new Set(parentRels[added[i]]);
-               parentRels[added[i]].add(entity.id);
-             }
-           }
-         },
-         replace: function replace(entity) {
-           if (this.entities[entity.id] === entity) return this;
-           return this.update(function () {
-             this._updateCalculated(this.entities[entity.id], entity);
+       function utilKeybinding(namespace) {
+         var _keybindings = {};
 
-             this.entities[entity.id] = entity;
+         function testBindings(d3_event, isCapturing) {
+           var didMatch = false;
+           var bindings = Object.keys(_keybindings).map(function (id) {
+             return _keybindings[id];
            });
-         },
-         remove: function remove(entity) {
-           return this.update(function () {
-             this._updateCalculated(entity, undefined);
+           var i, binding; // Most key shortcuts will accept either lower or uppercase ('h' or 'H'),
+           // so we don't strictly match on the shift key, but we prioritize
+           // shifted keybindings first, and fallback to unshifted only if no match.
+           // (This lets us differentiate between '←'/'⇧←' or '⌘Z'/'⌘⇧Z')
+           // priority match shifted keybindings first
 
-             this.entities[entity.id] = undefined;
-           });
-         },
-         revert: function revert(id) {
-           var baseEntity = this.base().entities[id];
-           var headEntity = this.entities[id];
-           if (headEntity === baseEntity) return this;
-           return this.update(function () {
-             this._updateCalculated(headEntity, baseEntity);
+           for (i = 0; i < bindings.length; i++) {
+             binding = bindings[i];
+             if (!binding.event.modifiers.shiftKey) continue; // no shift
 
-             delete this.entities[id];
-           });
-         },
-         update: function update() {
-           var graph = this.frozen ? coreGraph(this, true) : this;
+             if (!!binding.capture !== isCapturing) continue;
+
+             if (matches(d3_event, binding, true)) {
+               binding.callback(d3_event);
+               didMatch = true; // match a max of one binding per event
+
+               break;
+             }
+           }
+
+           if (didMatch) return; // then unshifted keybindings
+
+           for (i = 0; i < bindings.length; i++) {
+             binding = bindings[i];
+             if (binding.event.modifiers.shiftKey) continue; // shift
 
-           for (var i = 0; i < arguments.length; i++) {
-             arguments[i].call(graph, graph);
+             if (!!binding.capture !== isCapturing) continue;
+
+             if (matches(d3_event, binding, false)) {
+               binding.callback(d3_event);
+               break;
+             }
            }
 
-           if (this.frozen) graph.frozen = true;
-           return graph;
-         },
-         // Obliterates any existing entities
-         load: function load(entities) {
-           var base = this.base();
-           this.entities = Object.create(base.entities);
+           function matches(d3_event, binding, testShift) {
+             var event = d3_event;
+             var isMatch = false;
+             var tryKeyCode = true; // Prefer a match on `KeyboardEvent.key`
 
-           for (var i in entities) {
-             this.entities[i] = entities[i];
+             if (event.key !== undefined) {
+               tryKeyCode = event.key.charCodeAt(0) > 255; // outside ISO-Latin-1
 
-             this._updateCalculated(base.entities[i], this.entities[i]);
-           }
+               isMatch = true;
 
-           return this;
-         }
-       };
+               if (binding.event.key === undefined) {
+                 isMatch = false;
+               } else if (Array.isArray(binding.event.key)) {
+                 if (binding.event.key.map(function (s) {
+                   return s.toLowerCase();
+                 }).indexOf(event.key.toLowerCase()) === -1) {
+                   isMatch = false;
+                 }
+               } else {
+                 if (event.key.toLowerCase() !== binding.event.key.toLowerCase()) {
+                   isMatch = false;
+                 }
+               }
+             } // Fallback match on `KeyboardEvent.keyCode`, can happen if:
+             // - browser doesn't support `KeyboardEvent.key`
+             // - `KeyboardEvent.key` is outside ISO-Latin-1 range (cyrillic?)
 
-       function osmTurn(turn) {
-         if (!(this instanceof osmTurn)) {
-           return new osmTurn(turn);
-         }
 
-         Object.assign(this, turn);
-       }
-       function osmIntersection(graph, startVertexId, maxDistance) {
-         maxDistance = maxDistance || 30; // in meters
+             if (!isMatch && tryKeyCode) {
+               isMatch = event.keyCode === binding.event.keyCode;
+             }
 
-         var vgraph = coreGraph(); // virtual graph
+             if (!isMatch) return false; // test modifier keys
 
-         var i, j, k;
+             if (!(event.ctrlKey && event.altKey)) {
+               // if both are set, assume AltGr and skip it - #4096
+               if (event.ctrlKey !== binding.event.modifiers.ctrlKey) return false;
+               if (event.altKey !== binding.event.modifiers.altKey) return false;
+             }
 
-         function memberOfRestriction(entity) {
-           return graph.parentRelations(entity).some(function (r) {
-             return r.isRestriction();
-           });
+             if (event.metaKey !== binding.event.modifiers.metaKey) return false;
+             if (testShift && event.shiftKey !== binding.event.modifiers.shiftKey) return false;
+             return true;
+           }
          }
 
-         function isRoad(way) {
-           if (way.isArea() || way.isDegenerate()) return false;
-           var roads = {
-             'motorway': true,
-             'motorway_link': true,
-             'trunk': true,
-             'trunk_link': true,
-             'primary': true,
-             'primary_link': true,
-             'secondary': true,
-             'secondary_link': true,
-             'tertiary': true,
-             'tertiary_link': true,
-             'residential': true,
-             'unclassified': true,
-             'living_street': true,
-             'service': true,
-             'road': true,
-             'track': true
-           };
-           return roads[way.tags.highway];
+         function capture(d3_event) {
+           testBindings(d3_event, true);
          }
 
-         var startNode = graph.entity(startVertexId);
-         var checkVertices = [startNode];
-         var checkWays;
-         var vertices = [];
-         var vertexIds = [];
-         var vertex;
-         var ways = [];
-         var wayIds = [];
-         var way;
-         var nodes = [];
-         var node;
-         var parents = [];
-         var parent; // `actions` will store whatever actions must be performed to satisfy
-         // preconditions for adding a turn restriction to this intersection.
-         //  - Remove any existing degenerate turn restrictions (missing from/to, etc)
-         //  - Reverse oneways so that they are drawn in the forward direction
-         //  - Split ways on key vertices
+         function bubble(d3_event) {
+           var tagName = select(d3_event.target).node().tagName;
 
-         var actions = []; // STEP 1:  walk the graph outwards from starting vertex to search
-         //  for more key vertices and ways to include in the intersection..
+           if (tagName === 'INPUT' || tagName === 'SELECT' || tagName === 'TEXTAREA') {
+             return;
+           }
 
-         while (checkVertices.length) {
-           vertex = checkVertices.pop(); // check this vertex for parent ways that are roads
+           testBindings(d3_event, false);
+         }
 
-           checkWays = graph.parentWays(vertex);
-           var hasWays = false;
+         function keybinding(selection) {
+           selection = selection || select(document);
+           selection.on('keydown.capture.' + namespace, capture, true);
+           selection.on('keydown.bubble.' + namespace, bubble, false);
+           return keybinding;
+         } // was: keybinding.off()
 
-           for (i = 0; i < checkWays.length; i++) {
-             way = checkWays[i];
-             if (!isRoad(way) && !memberOfRestriction(way)) continue;
-             ways.push(way); // it's a road, or it's already in a turn restriction
 
-             hasWays = true; // check the way's children for more key vertices
+         keybinding.unbind = function (selection) {
+           _keybindings = [];
+           selection = selection || select(document);
+           selection.on('keydown.capture.' + namespace, null);
+           selection.on('keydown.bubble.' + namespace, null);
+           return keybinding;
+         };
 
-             nodes = utilArrayUniq(graph.childNodes(way));
+         keybinding.clear = function () {
+           _keybindings = {};
+           return keybinding;
+         }; // Remove one or more keycode bindings.
 
-             for (j = 0; j < nodes.length; j++) {
-               node = nodes[j];
-               if (node === vertex) continue; // same thing
 
-               if (vertices.indexOf(node) !== -1) continue; // seen it already
+         keybinding.off = function (codes, capture) {
+           var arr = utilArrayUniq([].concat(codes));
 
-               if (geoSphericalDistance(node.loc, startNode.loc) > maxDistance) continue; // too far from start
-               // a key vertex will have parents that are also roads
+           for (var i = 0; i < arr.length; i++) {
+             var id = arr[i] + (capture ? '-capture' : '-bubble');
+             delete _keybindings[id];
+           }
 
-               var hasParents = false;
-               parents = graph.parentWays(node);
+           return keybinding;
+         }; // Add one or more keycode bindings.
 
-               for (k = 0; k < parents.length; k++) {
-                 parent = parents[k];
-                 if (parent === way) continue; // same thing
 
-                 if (ways.indexOf(parent) !== -1) continue; // seen it already
+         keybinding.on = function (codes, callback, capture) {
+           if (typeof callback !== 'function') {
+             return keybinding.off(codes, capture);
+           }
 
-                 if (!isRoad(parent)) continue; // not a road
+           var arr = utilArrayUniq([].concat(codes));
 
-                 hasParents = true;
-                 break;
+           for (var i = 0; i < arr.length; i++) {
+             var id = arr[i] + (capture ? '-capture' : '-bubble');
+             var binding = {
+               id: id,
+               capture: capture,
+               callback: callback,
+               event: {
+                 key: undefined,
+                 // preferred
+                 keyCode: 0,
+                 // fallback
+                 modifiers: {
+                   shiftKey: false,
+                   ctrlKey: false,
+                   altKey: false,
+                   metaKey: false
+                 }
                }
+             };
 
-               if (hasParents) {
-                 checkVertices.push(node);
-               }
+             if (_keybindings[id]) {
+               console.warn('warning: duplicate keybinding for "' + id + '"'); // eslint-disable-line no-console
              }
-           }
 
-           if (hasWays) {
-             vertices.push(vertex);
-           }
-         }
+             _keybindings[id] = binding;
+             var matches = arr[i].toLowerCase().match(/(?:(?:[^+⇧⌃⌥⌘])+|[⇧⌃⌥⌘]|\+\+|^\+$)/g);
 
-         vertices = utilArrayUniq(vertices);
-         ways = utilArrayUniq(ways); // STEP 2:  Build a virtual graph containing only the entities in the intersection..
-         // Everything done after this step should act on the virtual graph
-         // Any actions that must be performed later to the main graph go in `actions` array
+             for (var j = 0; j < matches.length; j++) {
+               // Normalise matching errors
+               if (matches[j] === '++') matches[j] = '+';
 
-         ways.forEach(function (way) {
-           graph.childNodes(way).forEach(function (node) {
-             vgraph = vgraph.replace(node);
-           });
-           vgraph = vgraph.replace(way);
-           graph.parentRelations(way).forEach(function (relation) {
-             if (relation.isRestriction()) {
-               if (relation.isValidRestriction(graph)) {
-                 vgraph = vgraph.replace(relation);
-               } else if (relation.isComplete(graph)) {
-                 actions.push(actionDeleteRelation(relation.id));
+               if (matches[j] in utilKeybinding.modifierCodes) {
+                 var prop = utilKeybinding.modifierProperties[utilKeybinding.modifierCodes[matches[j]]];
+                 binding.event.modifiers[prop] = true;
+               } else {
+                 binding.event.key = utilKeybinding.keys[matches[j]] || matches[j];
+
+                 if (matches[j] in utilKeybinding.keyCodes) {
+                   binding.event.keyCode = utilKeybinding.keyCodes[matches[j]];
+                 }
                }
              }
-           });
-         }); // STEP 3:  Force all oneways to be drawn in the forward direction
+           }
+
+           return keybinding;
+         };
+
+         return keybinding;
+       }
+       /*
+        * See https://github.com/keithamus/jwerty
+        */
+
+       utilKeybinding.modifierCodes = {
+         // Shift key, ⇧
+         '⇧': 16,
+         shift: 16,
+         // CTRL key, on Mac: ⌃
+         '⌃': 17,
+         ctrl: 17,
+         // ALT key, on Mac: ⌥ (Alt)
+         '⌥': 18,
+         alt: 18,
+         option: 18,
+         // META, on Mac: ⌘ (CMD), on Windows (Win), on Linux (Super)
+         '⌘': 91,
+         meta: 91,
+         cmd: 91,
+         'super': 91,
+         win: 91
+       };
+       utilKeybinding.modifierProperties = {
+         16: 'shiftKey',
+         17: 'ctrlKey',
+         18: 'altKey',
+         91: 'metaKey'
+       };
+       utilKeybinding.plusKeys = ['plus', 'ffplus', '=', 'ffequals', '≠', '±'];
+       utilKeybinding.minusKeys = ['_', '-', 'ffminus', 'dash', '–', '—'];
+       utilKeybinding.keys = {
+         // Backspace key, on Mac: ⌫ (Backspace)
+         '⌫': 'Backspace',
+         backspace: 'Backspace',
+         // Tab Key, on Mac: ⇥ (Tab), on Windows ⇥⇥
+         '⇥': 'Tab',
+         '⇆': 'Tab',
+         tab: 'Tab',
+         // Return key, ↩
+         '↩': 'Enter',
+         '↵': 'Enter',
+         '⏎': 'Enter',
+         'return': 'Enter',
+         enter: 'Enter',
+         '⌅': 'Enter',
+         // Pause/Break key
+         'pause': 'Pause',
+         'pause-break': 'Pause',
+         // Caps Lock key, ⇪
+         '⇪': 'CapsLock',
+         caps: 'CapsLock',
+         'caps-lock': 'CapsLock',
+         // Escape key, on Mac: ⎋, on Windows: Esc
+         '⎋': ['Escape', 'Esc'],
+         escape: ['Escape', 'Esc'],
+         esc: ['Escape', 'Esc'],
+         // Space key
+         space: [' ', 'Spacebar'],
+         // Page-Up key, or pgup, on Mac: ↖
+         '↖': 'PageUp',
+         pgup: 'PageUp',
+         'page-up': 'PageUp',
+         // Page-Down key, or pgdown, on Mac: ↘
+         '↘': 'PageDown',
+         pgdown: 'PageDown',
+         'page-down': 'PageDown',
+         // END key, on Mac: ⇟
+         '⇟': 'End',
+         end: 'End',
+         // HOME key, on Mac: ⇞
+         '⇞': 'Home',
+         home: 'Home',
+         // Insert key, or ins
+         ins: 'Insert',
+         insert: 'Insert',
+         // Delete key, on Mac: ⌦ (Delete)
+         '⌦': ['Delete', 'Del'],
+         del: ['Delete', 'Del'],
+         'delete': ['Delete', 'Del'],
+         // Left Arrow Key, or ←
+         '←': ['ArrowLeft', 'Left'],
+         left: ['ArrowLeft', 'Left'],
+         'arrow-left': ['ArrowLeft', 'Left'],
+         // Up Arrow Key, or ↑
+         '↑': ['ArrowUp', 'Up'],
+         up: ['ArrowUp', 'Up'],
+         'arrow-up': ['ArrowUp', 'Up'],
+         // Right Arrow Key, or →
+         '→': ['ArrowRight', 'Right'],
+         right: ['ArrowRight', 'Right'],
+         'arrow-right': ['ArrowRight', 'Right'],
+         // Up Arrow Key, or ↓
+         '↓': ['ArrowDown', 'Down'],
+         down: ['ArrowDown', 'Down'],
+         'arrow-down': ['ArrowDown', 'Down'],
+         // odities, stuff for backward compatibility (browsers and code):
+         // Num-Multiply, or *
+         '*': ['*', 'Multiply'],
+         star: ['*', 'Multiply'],
+         asterisk: ['*', 'Multiply'],
+         multiply: ['*', 'Multiply'],
+         // Num-Plus or +
+         '+': ['+', 'Add'],
+         'plus': ['+', 'Add'],
+         // Num-Subtract, or -
+         '-': ['-', 'Subtract'],
+         subtract: ['-', 'Subtract'],
+         'dash': ['-', 'Subtract'],
+         // Semicolon
+         semicolon: ';',
+         // = or equals
+         equals: '=',
+         // Comma, or ,
+         comma: ',',
+         // Period, or ., or full-stop
+         period: '.',
+         'full-stop': '.',
+         // Slash, or /, or forward-slash
+         slash: '/',
+         'forward-slash': '/',
+         // Tick, or `, or back-quote
+         tick: '`',
+         'back-quote': '`',
+         // Open bracket, or [
+         'open-bracket': '[',
+         // Back slash, or \
+         'back-slash': '\\',
+         // Close backet, or ]
+         'close-bracket': ']',
+         // Apostrophe, or Quote, or '
+         quote: '\'',
+         apostrophe: '\'',
+         // NUMPAD 0-9
+         'num-0': '0',
+         'num-1': '1',
+         'num-2': '2',
+         'num-3': '3',
+         'num-4': '4',
+         'num-5': '5',
+         'num-6': '6',
+         'num-7': '7',
+         'num-8': '8',
+         'num-9': '9',
+         // F1-F25
+         f1: 'F1',
+         f2: 'F2',
+         f3: 'F3',
+         f4: 'F4',
+         f5: 'F5',
+         f6: 'F6',
+         f7: 'F7',
+         f8: 'F8',
+         f9: 'F9',
+         f10: 'F10',
+         f11: 'F11',
+         f12: 'F12',
+         f13: 'F13',
+         f14: 'F14',
+         f15: 'F15',
+         f16: 'F16',
+         f17: 'F17',
+         f18: 'F18',
+         f19: 'F19',
+         f20: 'F20',
+         f21: 'F21',
+         f22: 'F22',
+         f23: 'F23',
+         f24: 'F24',
+         f25: 'F25'
+       };
+       utilKeybinding.keyCodes = {
+         // Backspace key, on Mac: ⌫ (Backspace)
+         '⌫': 8,
+         backspace: 8,
+         // Tab Key, on Mac: ⇥ (Tab), on Windows ⇥⇥
+         '⇥': 9,
+         '⇆': 9,
+         tab: 9,
+         // Return key, ↩
+         '↩': 13,
+         '↵': 13,
+         '⏎': 13,
+         'return': 13,
+         enter: 13,
+         '⌅': 13,
+         // Pause/Break key
+         'pause': 19,
+         'pause-break': 19,
+         // Caps Lock key, ⇪
+         '⇪': 20,
+         caps: 20,
+         'caps-lock': 20,
+         // Escape key, on Mac: ⎋, on Windows: Esc
+         '⎋': 27,
+         escape: 27,
+         esc: 27,
+         // Space key
+         space: 32,
+         // Page-Up key, or pgup, on Mac: ↖
+         '↖': 33,
+         pgup: 33,
+         'page-up': 33,
+         // Page-Down key, or pgdown, on Mac: ↘
+         '↘': 34,
+         pgdown: 34,
+         'page-down': 34,
+         // END key, on Mac: ⇟
+         '⇟': 35,
+         end: 35,
+         // HOME key, on Mac: ⇞
+         '⇞': 36,
+         home: 36,
+         // Insert key, or ins
+         ins: 45,
+         insert: 45,
+         // Delete key, on Mac: ⌦ (Delete)
+         '⌦': 46,
+         del: 46,
+         'delete': 46,
+         // Left Arrow Key, or ←
+         '←': 37,
+         left: 37,
+         'arrow-left': 37,
+         // Up Arrow Key, or ↑
+         '↑': 38,
+         up: 38,
+         'arrow-up': 38,
+         // Right Arrow Key, or →
+         '→': 39,
+         right: 39,
+         'arrow-right': 39,
+         // Up Arrow Key, or ↓
+         '↓': 40,
+         down: 40,
+         'arrow-down': 40,
+         // odities, printing characters that come out wrong:
+         // Firefox Equals
+         'ffequals': 61,
+         // Num-Multiply, or *
+         '*': 106,
+         star: 106,
+         asterisk: 106,
+         multiply: 106,
+         // Num-Plus or +
+         '+': 107,
+         'plus': 107,
+         // Num-Subtract, or -
+         '-': 109,
+         subtract: 109,
+         // Vertical Bar / Pipe
+         '|': 124,
+         // Firefox Plus
+         'ffplus': 171,
+         // Firefox Minus
+         'ffminus': 173,
+         // Semicolon
+         ';': 186,
+         semicolon: 186,
+         // = or equals
+         '=': 187,
+         'equals': 187,
+         // Comma, or ,
+         ',': 188,
+         comma: 188,
+         // Dash / Underscore key
+         'dash': 189,
+         // Period, or ., or full-stop
+         '.': 190,
+         period: 190,
+         'full-stop': 190,
+         // Slash, or /, or forward-slash
+         '/': 191,
+         slash: 191,
+         'forward-slash': 191,
+         // Tick, or `, or back-quote
+         '`': 192,
+         tick: 192,
+         'back-quote': 192,
+         // Open bracket, or [
+         '[': 219,
+         'open-bracket': 219,
+         // Back slash, or \
+         '\\': 220,
+         'back-slash': 220,
+         // Close backet, or ]
+         ']': 221,
+         'close-bracket': 221,
+         // Apostrophe, or Quote, or '
+         '\'': 222,
+         quote: 222,
+         apostrophe: 222
+       }; // NUMPAD 0-9
 
-         ways.forEach(function (w) {
-           var way = vgraph.entity(w.id);
+       var i = 95,
+           n = 0;
 
-           if (way.tags.oneway === '-1') {
-             var action = actionReverse(way.id, {
-               reverseOneway: true
-             });
-             actions.push(action);
-             vgraph = action(vgraph);
-           }
-         }); // STEP 4:  Split ways on key vertices
+       while (++i < 106) {
+         utilKeybinding.keyCodes['num-' + n] = i;
+         ++n;
+       } // 0-9
 
-         var origCount = osmEntity.id.next.way;
-         vertices.forEach(function (v) {
-           // This is an odd way to do it, but we need to find all the ways that
-           // will be split here, then split them one at a time to ensure that these
-           // actions can be replayed on the main graph exactly in the same order.
-           // (It is unintuitive, but the order of ways returned from graph.parentWays()
-           // is arbitrary, depending on how the main graph and vgraph were built)
-           var splitAll = actionSplit([v.id]).keepHistoryOn('first');
 
-           if (!splitAll.disabled(vgraph)) {
-             splitAll.ways(vgraph).forEach(function (way) {
-               var splitOne = actionSplit([v.id]).limitWays([way.id]).keepHistoryOn('first');
-               actions.push(splitOne);
-               vgraph = splitOne(vgraph);
-             });
-           }
-         }); // In here is where we should also split the intersection at nearby junction.
-         //   for https://github.com/mapbox/iD-internal/issues/31
-         // nearbyVertices.forEach(function(v) {
-         // });
-         // Reasons why we reset the way id count here:
-         //  1. Continuity with way ids created by the splits so that we can replay
-         //     these actions later if the user decides to create a turn restriction
-         //  2. Avoids churning way ids just by hovering over a vertex
-         //     and displaying the turn restriction editor
+       i = 47;
+       n = 0;
 
-         osmEntity.id.next.way = origCount; // STEP 5:  Update arrays to point to vgraph entities
+       while (++i < 58) {
+         utilKeybinding.keyCodes[n] = i;
+         ++n;
+       } // F1-F25
 
-         vertexIds = vertices.map(function (v) {
-           return v.id;
-         });
-         vertices = [];
-         ways = [];
-         vertexIds.forEach(function (id) {
-           var vertex = vgraph.entity(id);
-           var parents = vgraph.parentWays(vertex);
-           vertices.push(vertex);
-           ways = ways.concat(parents);
-         });
-         vertices = utilArrayUniq(vertices);
-         ways = utilArrayUniq(ways);
-         vertexIds = vertices.map(function (v) {
-           return v.id;
-         });
-         wayIds = ways.map(function (w) {
-           return w.id;
-         }); // STEP 6:  Update the ways with some metadata that will be useful for
-         // walking the intersection graph later and rendering turn arrows.
 
-         function withMetadata(way, vertexIds) {
-           var __oneWay = way.isOneWay(); // which affixes are key vertices?
+       i = 111;
+       n = 1;
 
+       while (++i < 136) {
+         utilKeybinding.keyCodes['f' + n] = i;
+         ++n;
+       } // a-z
 
-           var __first = vertexIds.indexOf(way.first()) !== -1;
 
-           var __last = vertexIds.indexOf(way.last()) !== -1; // what roles is this way eligible for?
+       i = 64;
 
+       while (++i < 91) {
+         utilKeybinding.keyCodes[String.fromCharCode(i).toLowerCase()] = i;
+       }
 
-           var __via = __first && __last;
+       function utilObjectOmit(obj, omitKeys) {
+         return Object.keys(obj).reduce(function (result, key) {
+           if (omitKeys.indexOf(key) === -1) {
+             result[key] = obj[key]; // keep
+           }
 
-           var __from = __first && !__oneWay || __last;
+           return result;
+         }, {});
+       }
 
-           var __to = __first || __last && !__oneWay;
+       // Copies a variable number of methods from source to target.
+       function utilRebind(target, source) {
+         var i = 1,
+             n = arguments.length,
+             method;
 
-           return way.update({
-             __first: __first,
-             __last: __last,
-             __from: __from,
-             __via: __via,
-             __to: __to,
-             __oneWay: __oneWay
-           });
+         while (++i < n) {
+           target[method = arguments[i]] = d3_rebind(target, source, source[method]);
          }
 
-         ways = [];
-         wayIds.forEach(function (id) {
-           var way = withMetadata(vgraph.entity(id), vertexIds);
-           vgraph = vgraph.replace(way);
-           ways.push(way);
-         }); // STEP 7:  Simplify - This is an iterative process where we:
-         //  1. Find trivial vertices with only 2 parents
-         //  2. trim off the leaf way from those vertices and remove from vgraph
+         return target;
+       } // Method is assumed to be a standard D3 getter-setter:
+       // If passed with no arguments, gets the value.
+       // If passed with arguments, sets the value and returns the target.
 
-         var keepGoing;
-         var removeWayIds = [];
-         var removeVertexIds = [];
+       function d3_rebind(target, source, method) {
+         return function () {
+           var value = method.apply(source, arguments);
+           return value === source ? target : value;
+         };
+       }
 
-         do {
-           keepGoing = false;
-           checkVertices = vertexIds.slice();
+       // A per-domain session mutex backed by a cookie and dead man's
+       // switch. If the session crashes, the mutex will auto-release
+       // after 5 seconds.
+       // This accepts a string and returns an object that complies with utilSessionMutexType
+       function utilSessionMutex(name) {
+         var mutex = {};
+         var intervalID;
 
-           for (i = 0; i < checkVertices.length; i++) {
-             var vertexId = checkVertices[i];
-             vertex = vgraph.hasEntity(vertexId);
+         function renew() {
+           var expires = new Date();
+           expires.setSeconds(expires.getSeconds() + 5);
+           document.cookie = name + '=1; expires=' + expires.toUTCString() + '; sameSite=strict';
+         }
 
-             if (!vertex) {
-               if (vertexIds.indexOf(vertexId) !== -1) {
-                 vertexIds.splice(vertexIds.indexOf(vertexId), 1); // stop checking this one
-               }
+         mutex.lock = function () {
+           if (intervalID) return true;
+           var cookie = document.cookie.replace(new RegExp('(?:(?:^|.*;)\\s*' + name + '\\s*\\=\\s*([^;]*).*$)|^.*$'), '$1');
+           if (cookie) return false;
+           renew();
+           intervalID = window.setInterval(renew, 4000);
+           return true;
+         };
 
-               removeVertexIds.push(vertexId);
-               continue;
-             }
+         mutex.unlock = function () {
+           if (!intervalID) return;
+           document.cookie = name + '=; expires=Thu, 01 Jan 1970 00:00:00 GMT; sameSite=strict';
+           clearInterval(intervalID);
+           intervalID = null;
+         };
 
-             parents = vgraph.parentWays(vertex);
+         mutex.locked = function () {
+           return !!intervalID;
+         };
 
-             if (parents.length < 3) {
-               if (vertexIds.indexOf(vertexId) !== -1) {
-                 vertexIds.splice(vertexIds.indexOf(vertexId), 1); // stop checking this one
-               }
-             }
+         return mutex;
+       }
 
-             if (parents.length === 2) {
-               // vertex with 2 parents is trivial
-               var a = parents[0];
-               var b = parents[1];
-               var aIsLeaf = a && !a.__via;
-               var bIsLeaf = b && !b.__via;
-               var leaf, survivor;
+       function utilTiler() {
+         var _size = [256, 256];
+         var _scale = 256;
+         var _tileSize = 256;
+         var _zoomExtent = [0, 20];
+         var _translate = [_size[0] / 2, _size[1] / 2];
+         var _margin = 0;
+         var _skipNullIsland = false;
 
-               if (aIsLeaf && !bIsLeaf) {
-                 leaf = a;
-                 survivor = b;
-               } else if (!aIsLeaf && bIsLeaf) {
-                 leaf = b;
-                 survivor = a;
-               }
+         function clamp(num, min, max) {
+           return Math.max(min, Math.min(num, max));
+         }
 
-               if (leaf && survivor) {
-                 survivor = withMetadata(survivor, vertexIds); // update survivor way
+         function nearNullIsland(tile) {
+           var x = tile[0];
+           var y = tile[1];
+           var z = tile[2];
 
-                 vgraph = vgraph.replace(survivor).remove(leaf); // update graph
+           if (z >= 7) {
+             var center = Math.pow(2, z - 1);
+             var width = Math.pow(2, z - 6);
+             var min = center - width / 2;
+             var max = center + width / 2 - 1;
+             return x >= min && x <= max && y >= min && y <= max;
+           }
 
-                 removeWayIds.push(leaf.id);
-                 keepGoing = true;
-               }
-             }
+           return false;
+         }
 
-             parents = vgraph.parentWays(vertex);
+         function tiler() {
+           var z = geoScaleToZoom(_scale / (2 * Math.PI), _tileSize);
+           var z0 = clamp(Math.round(z), _zoomExtent[0], _zoomExtent[1]);
+           var tileMin = 0;
+           var tileMax = Math.pow(2, z0) - 1;
+           var log2ts = Math.log(_tileSize) * Math.LOG2E;
+           var k = Math.pow(2, z - z0 + log2ts);
+           var origin = [(_translate[0] - _scale / 2) / k, (_translate[1] - _scale / 2) / k];
+           var cols = range$1(clamp(Math.floor(-origin[0]) - _margin, tileMin, tileMax + 1), clamp(Math.ceil(_size[0] / k - origin[0]) + _margin, tileMin, tileMax + 1));
+           var rows = range$1(clamp(Math.floor(-origin[1]) - _margin, tileMin, tileMax + 1), clamp(Math.ceil(_size[1] / k - origin[1]) + _margin, tileMin, tileMax + 1));
+           var tiles = [];
 
-             if (parents.length < 2) {
-               // vertex is no longer a key vertex
-               if (vertexIds.indexOf(vertexId) !== -1) {
-                 vertexIds.splice(vertexIds.indexOf(vertexId), 1); // stop checking this one
-               }
+           for (var i = 0; i < rows.length; i++) {
+             var y = rows[i];
 
-               removeVertexIds.push(vertexId);
-               keepGoing = true;
-             }
+             for (var j = 0; j < cols.length; j++) {
+               var x = cols[j];
 
-             if (parents.length < 1) {
-               // vertex is no longer attached to anything
-               vgraph = vgraph.remove(vertex);
+               if (i >= _margin && i <= rows.length - _margin && j >= _margin && j <= cols.length - _margin) {
+                 tiles.unshift([x, y, z0]); // tiles in view at beginning
+               } else {
+                 tiles.push([x, y, z0]); // tiles in margin at the end
+               }
              }
            }
-         } while (keepGoing);
-
-         vertices = vertices.filter(function (vertex) {
-           return removeVertexIds.indexOf(vertex.id) === -1;
-         }).map(function (vertex) {
-           return vgraph.entity(vertex.id);
-         });
-         ways = ways.filter(function (way) {
-           return removeWayIds.indexOf(way.id) === -1;
-         }).map(function (way) {
-           return vgraph.entity(way.id);
-         }); // OK!  Here is our intersection..
-
-         var intersection = {
-           graph: vgraph,
-           actions: actions,
-           vertices: vertices,
-           ways: ways
-         }; // Get all the valid turns through this intersection given a starting way id.
-         // This operates on the virtual graph for everything.
-         //
-         // Basically, walk through all possible paths from starting way,
-         //   honoring the existing turn restrictions as we go (watch out for loops!)
-         //
-         // For each path found, generate and return a `osmTurn` datastructure.
-         //
-
-         intersection.turns = function (fromWayId, maxViaWay) {
-           if (!fromWayId) return [];
-           if (!maxViaWay) maxViaWay = 0;
-           var vgraph = intersection.graph;
-           var keyVertexIds = intersection.vertices.map(function (v) {
-             return v.id;
-           });
-           var start = vgraph.entity(fromWayId);
-           if (!start || !(start.__from || start.__via)) return []; // maxViaWay=0   from-*-to              (0 vias)
-           // maxViaWay=1   from-*-via-*-to        (1 via max)
-           // maxViaWay=2   from-*-via-*-via-*-to  (2 vias max)
 
-           var maxPathLength = maxViaWay * 2 + 3;
-           var turns = [];
-           step(start);
-           return turns; // traverse the intersection graph and find all the valid paths
+           tiles.translate = origin;
+           tiles.scale = k;
+           return tiles;
+         }
+         /**
+          * getTiles() returns an array of tiles that cover the map view
+          */
 
-           function step(entity, currPath, currRestrictions, matchedRestriction) {
-             currPath = (currPath || []).slice(); // shallow copy
 
-             if (currPath.length >= maxPathLength) return;
-             currPath.push(entity.id);
-             currRestrictions = (currRestrictions || []).slice(); // shallow copy
+         tiler.getTiles = function (projection) {
+           var origin = [projection.scale() * Math.PI - projection.translate()[0], projection.scale() * Math.PI - projection.translate()[1]];
+           this.size(projection.clipExtent()[1]).scale(projection.scale() * 2 * Math.PI).translate(projection.translate());
+           var tiles = tiler();
+           var ts = tiles.scale;
+           return tiles.map(function (tile) {
+             if (_skipNullIsland && nearNullIsland(tile)) {
+               return false;
+             }
 
-             var i, j;
+             var x = tile[0] * ts - origin[0];
+             var y = tile[1] * ts - origin[1];
+             return {
+               id: tile.toString(),
+               xyz: tile,
+               extent: geoExtent(projection.invert([x, y + ts]), projection.invert([x + ts, y]))
+             };
+           }).filter(Boolean);
+         };
+         /**
+          * getGeoJSON() returns a FeatureCollection for debugging tiles
+          */
 
-             if (entity.type === 'node') {
-               var parents = vgraph.parentWays(entity);
-               var nextWays = []; // which ways can we step into?
 
-               for (i = 0; i < parents.length; i++) {
-                 var way = parents[i]; // if next way is a oneway incoming to this vertex, skip
+         tiler.getGeoJSON = function (projection) {
+           var features = tiler.getTiles(projection).map(function (tile) {
+             return {
+               type: 'Feature',
+               properties: {
+                 id: tile.id,
+                 name: tile.id
+               },
+               geometry: {
+                 type: 'Polygon',
+                 coordinates: [tile.extent.polygon()]
+               }
+             };
+           });
+           return {
+             type: 'FeatureCollection',
+             features: features
+           };
+         };
 
-                 if (way.__oneWay && way.nodes[0] !== entity.id) continue; // if we have seen it before (allowing for an initial u-turn), skip
+         tiler.tileSize = function (val) {
+           if (!arguments.length) return _tileSize;
+           _tileSize = val;
+           return tiler;
+         };
 
-                 if (currPath.indexOf(way.id) !== -1 && currPath.length >= 3) continue; // Check all "current" restrictions (where we've already walked the `FROM`)
+         tiler.zoomExtent = function (val) {
+           if (!arguments.length) return _zoomExtent;
+           _zoomExtent = val;
+           return tiler;
+         };
 
-                 var restrict = null;
+         tiler.size = function (val) {
+           if (!arguments.length) return _size;
+           _size = val;
+           return tiler;
+         };
 
-                 for (j = 0; j < currRestrictions.length; j++) {
-                   var restriction = currRestrictions[j];
-                   var f = restriction.memberByRole('from');
-                   var v = restriction.membersByRole('via');
-                   var t = restriction.memberByRole('to');
-                   var isOnly = /^only_/.test(restriction.tags.restriction); // Does the current path match this turn restriction?
+         tiler.scale = function (val) {
+           if (!arguments.length) return _scale;
+           _scale = val;
+           return tiler;
+         };
 
-                   var matchesFrom = f.id === fromWayId;
-                   var matchesViaTo = false;
-                   var isAlongOnlyPath = false;
+         tiler.translate = function (val) {
+           if (!arguments.length) return _translate;
+           _translate = val;
+           return tiler;
+         }; // number to extend the rows/columns beyond those covering the viewport
 
-                   if (t.id === way.id) {
-                     // match TO
-                     if (v.length === 1 && v[0].type === 'node') {
-                       // match VIA node
-                       matchesViaTo = v[0].id === entity.id && (matchesFrom && currPath.length === 2 || !matchesFrom && currPath.length > 2);
-                     } else {
-                       // match all VIA ways
-                       var pathVias = [];
 
-                       for (k = 2; k < currPath.length; k += 2) {
-                         // k = 2 skips FROM
-                         pathVias.push(currPath[k]); // (path goes way-node-way...)
-                       }
+         tiler.margin = function (val) {
+           if (!arguments.length) return _margin;
+           _margin = +val;
+           return tiler;
+         };
 
-                       var restrictionVias = [];
+         tiler.skipNullIsland = function (val) {
+           if (!arguments.length) return _skipNullIsland;
+           _skipNullIsland = val;
+           return tiler;
+         };
 
-                       for (k = 0; k < v.length; k++) {
-                         if (v[k].type === 'way') {
-                           restrictionVias.push(v[k].id);
-                         }
-                       }
+         return tiler;
+       }
 
-                       var diff = utilArrayDifference(pathVias, restrictionVias);
-                       matchesViaTo = !diff.length;
-                     }
-                   } else if (isOnly) {
-                     for (k = 0; k < v.length; k++) {
-                       // way doesn't match TO, but is one of the via ways along the path of an "only"
-                       if (v[k].type === 'way' && v[k].id === way.id) {
-                         isAlongOnlyPath = true;
-                         break;
-                       }
-                     }
-                   }
+       function utilTriggerEvent(target, type) {
+         target.each(function () {
+           var evt = document.createEvent('HTMLEvents');
+           evt.initEvent(type, true, true);
+           this.dispatchEvent(evt);
+         });
+       }
 
-                   if (matchesViaTo) {
-                     if (isOnly) {
-                       restrict = {
-                         id: restriction.id,
-                         direct: matchesFrom,
-                         from: f.id,
-                         only: true,
-                         end: true
-                       };
-                     } else {
-                       restrict = {
-                         id: restriction.id,
-                         direct: matchesFrom,
-                         from: f.id,
-                         no: true,
-                         end: true
-                       };
-                     }
-                   } else {
-                     // indirect - caused by a different nearby restriction
-                     if (isAlongOnlyPath) {
-                       restrict = {
-                         id: restriction.id,
-                         direct: false,
-                         from: f.id,
-                         only: true,
-                         end: false
-                       };
-                     } else if (isOnly) {
-                       restrict = {
-                         id: restriction.id,
-                         direct: false,
-                         from: f.id,
-                         no: true,
-                         end: true
-                       };
-                     }
-                   } // stop looking if we find a "direct" restriction (matching FROM, VIA, TO)
+       var _mainLocations = coreLocations(); // singleton
+       // `coreLocations` maintains an internal index of all the boundaries/geofences used by iD.
+       // It's used by presets, community index, background imagery, to know where in the world these things are valid.
+       // These geofences should be defined by `locationSet` objects:
+       //
+       // let locationSet = {
+       //   include: [ Array of locations ],
+       //   exclude: [ Array of locations ]
+       // };
+       //
+       // For more info see the location-conflation and country-coder projects, see:
+       // https://github.com/ideditor/location-conflation
+       // https://github.com/ideditor/country-coder
+       //
 
+       function coreLocations() {
+         var _this = {};
+         var _resolvedFeatures = {}; // cache of *resolved* locationSet features
 
-                   if (restrict && restrict.direct) break;
-                 }
+         var _loco = new _default(); // instance of a location-conflation resolver
 
-                 nextWays.push({
-                   way: way,
-                   restrict: restrict
-                 });
-               }
 
-               nextWays.forEach(function (nextWay) {
-                 step(nextWay.way, currPath, currRestrictions, nextWay.restrict);
-               });
-             } else {
-               // entity.type === 'way'
-               if (currPath.length >= 3) {
-                 // this is a "complete" path..
-                 var turnPath = currPath.slice(); // shallow copy
-                 // an indirect restriction - only include the partial path (starting at FROM)
+         var _wp; // instance of a which-polygon index
+         // pre-resolve the worldwide locationSet
 
-                 if (matchedRestriction && matchedRestriction.direct === false) {
-                   for (i = 0; i < turnPath.length; i++) {
-                     if (turnPath[i] === matchedRestriction.from) {
-                       turnPath = turnPath.slice(i);
-                       break;
-                     }
-                   }
-                 }
 
-                 var turn = pathToTurn(turnPath);
+         var world = {
+           locationSet: {
+             include: ['Q2']
+           }
+         };
+         resolveLocationSet(world);
+         rebuildIndex();
+         var _queue = [];
 
-                 if (turn) {
-                   if (matchedRestriction) {
-                     turn.restrictionID = matchedRestriction.id;
-                     turn.no = matchedRestriction.no;
-                     turn.only = matchedRestriction.only;
-                     turn.direct = matchedRestriction.direct;
-                   }
+         var _deferred = new Set();
 
-                   turns.push(osmTurn(turn));
-                 }
+         var _inProcess; // Returns a Promise to process the queue
 
-                 if (currPath[0] === currPath[2]) return; // if we made a u-turn - stop here
-               }
 
-               if (matchedRestriction && matchedRestriction.end) return; // don't advance any further
-               // which nodes can we step into?
+         function processQueue() {
+           if (!_queue.length) return Promise.resolve(); // console.log(`queue length ${_queue.length}`);
 
-               var n1 = vgraph.entity(entity.first());
-               var n2 = vgraph.entity(entity.last());
-               var dist = geoSphericalDistance(n1.loc, n2.loc);
-               var nextNodes = [];
+           var chunk = _queue.pop();
 
-               if (currPath.length > 1) {
-                 if (dist > maxDistance) return; // the next node is too far
+           return new Promise(function (resolvePromise) {
+             var handle = window.requestIdleCallback(function () {
+               _deferred["delete"](handle); // const t0 = performance.now();
 
-                 if (!entity.__via) return; // this way is a leaf / can't be a via
-               }
 
-               if (!entity.__oneWay && // bidirectional..
-               keyVertexIds.indexOf(n1.id) !== -1 && // key vertex..
-               currPath.indexOf(n1.id) === -1) {
-                 // haven't seen it yet..
-                 nextNodes.push(n1); // can advance to first node
-               }
+               chunk.forEach(resolveLocationSet); // const t1 = performance.now();
+               // console.log('chunk processed in ' + (t1 - t0) + ' ms');
 
-               if (keyVertexIds.indexOf(n2.id) !== -1 && // key vertex..
-               currPath.indexOf(n2.id) === -1) {
-                 // haven't seen it yet..
-                 nextNodes.push(n2); // can advance to last node
-               }
+               resolvePromise();
+             });
 
-               nextNodes.forEach(function (nextNode) {
-                 // gather restrictions FROM this way
-                 var fromRestrictions = vgraph.parentRelations(entity).filter(function (r) {
-                   if (!r.isRestriction()) return false;
-                   var f = r.memberByRole('from');
-                   if (!f || f.id !== entity.id) return false;
-                   var isOnly = /^only_/.test(r.tags.restriction);
-                   if (!isOnly) return true; // `only_` restrictions only matter along the direction of the VIA - #4849
+             _deferred.add(handle);
+           }).then(function () {
+             return processQueue();
+           });
+         } // Pass an Object with a `locationSet` property,
+         // Performs the locationSet resolution, caches the result, and sets a `locationSetID` property on the object.
 
-                   var isOnlyVia = false;
-                   var v = r.membersByRole('via');
 
-                   if (v.length === 1 && v[0].type === 'node') {
-                     // via node
-                     isOnlyVia = v[0].id === nextNode.id;
-                   } else {
-                     // via way(s)
-                     for (var i = 0; i < v.length; i++) {
-                       if (v[i].type !== 'way') continue;
-                       var viaWay = vgraph.entity(v[i].id);
+         function resolveLocationSet(obj) {
+           if (obj.locationSetID) return; // work was done already
 
-                       if (viaWay.first() === nextNode.id || viaWay.last() === nextNode.id) {
-                         isOnlyVia = true;
-                         break;
-                       }
-                     }
-                   }
+           try {
+             var locationSet = obj.locationSet;
 
-                   return isOnlyVia;
-                 });
-                 step(nextNode, currPath, currRestrictions.concat(fromRestrictions), false);
-               });
+             if (!locationSet) {
+               throw new Error('object missing locationSet property');
              }
-           } // assumes path is alternating way-node-way of odd length
 
+             if (!locationSet.include) {
+               // missing `include`, default to worldwide include
+               locationSet.include = ['Q2']; // https://github.com/openstreetmap/iD/pull/8305#discussion_r662344647
+             }
 
-           function pathToTurn(path) {
-             if (path.length < 3) return;
-             var fromWayId, fromNodeId, fromVertexId;
-             var toWayId, toNodeId, toVertexId;
-             var viaWayIds, viaNodeId, isUturn;
-             fromWayId = path[0];
-             toWayId = path[path.length - 1];
+             var resolved = _loco.resolveLocationSet(locationSet);
 
-             if (path.length === 3 && fromWayId === toWayId) {
-               // u turn
-               var way = vgraph.entity(fromWayId);
-               if (way.__oneWay) return null;
-               isUturn = true;
-               viaNodeId = fromVertexId = toVertexId = path[1];
-               fromNodeId = toNodeId = adjacentNode(fromWayId, viaNodeId);
-             } else {
-               isUturn = false;
-               fromVertexId = path[1];
-               fromNodeId = adjacentNode(fromWayId, fromVertexId);
-               toVertexId = path[path.length - 2];
-               toNodeId = adjacentNode(toWayId, toVertexId);
+             var locationSetID = resolved.id;
+             obj.locationSetID = locationSetID;
 
-               if (path.length === 3) {
-                 viaNodeId = path[1];
-               } else {
-                 viaWayIds = path.filter(function (entityId) {
-                   return entityId[0] === 'w';
-                 });
-                 viaWayIds = viaWayIds.slice(1, viaWayIds.length - 1); // remove first, last
-               }
+             if (!resolved.feature.geometry.coordinates.length || !resolved.feature.properties.area) {
+               throw new Error("locationSet ".concat(locationSetID, " resolves to an empty feature."));
              }
 
-             return {
-               key: path.join('_'),
-               path: path,
-               from: {
-                 node: fromNodeId,
-                 way: fromWayId,
-                 vertex: fromVertexId
-               },
-               via: {
-                 node: viaNodeId,
-                 ways: viaWayIds
-               },
-               to: {
-                 node: toNodeId,
-                 way: toWayId,
-                 vertex: toVertexId
-               },
-               u: isUturn
-             };
+             if (!_resolvedFeatures[locationSetID]) {
+               // First time seeing this locationSet feature
+               var feature = JSON.parse(JSON.stringify(resolved.feature)); // deep clone
 
-             function adjacentNode(wayId, affixId) {
-               var nodes = vgraph.entity(wayId).nodes;
-               return affixId === nodes[0] ? nodes[1] : nodes[nodes.length - 2];
+               feature.id = locationSetID; // Important: always use the locationSet `id` (`+[Q30]`), not the feature `id` (`Q30`)
+
+               feature.properties.id = locationSetID;
+               _resolvedFeatures[locationSetID] = feature; // insert into cache
              }
+           } catch (err) {
+             obj.locationSet = {
+               include: ['Q2']
+             }; // default worldwide
+
+             obj.locationSetID = '+[Q2]';
            }
-         };
+         } // Rebuilds the whichPolygon index with whatever features have been resolved.
 
-         return intersection;
-       }
-       function osmInferRestriction(graph, turn, projection) {
-         var fromWay = graph.entity(turn.from.way);
-         var fromNode = graph.entity(turn.from.node);
-         var fromVertex = graph.entity(turn.from.vertex);
-         var toWay = graph.entity(turn.to.way);
-         var toNode = graph.entity(turn.to.node);
-         var toVertex = graph.entity(turn.to.vertex);
-         var fromOneWay = fromWay.tags.oneway === 'yes';
-         var toOneWay = toWay.tags.oneway === 'yes';
-         var angle = (geoAngle(fromVertex, fromNode, projection) - geoAngle(toVertex, toNode, projection)) * 180 / Math.PI;
 
-         while (angle < 0) {
-           angle += 360;
-         }
+         function rebuildIndex() {
+           _wp = whichPolygon_1({
+             features: Object.values(_resolvedFeatures)
+           });
+         } //
+         // `mergeCustomGeoJSON`
+         //  Accepts an FeatureCollection-like object containing custom locations
+         //  Each feature must have a filename-like `id`, for example: `something.geojson`
+         //
+         //  {
+         //    "type": "FeatureCollection"
+         //    "features": [
+         //      {
+         //        "type": "Feature",
+         //        "id": "philly_metro.geojson",
+         //        "properties": { … },
+         //        "geometry": { … }
+         //      }
+         //    ]
+         //  }
+         //
 
-         if (fromNode === toNode) return 'no_u_turn';
-         if ((angle < 23 || angle > 336) && fromOneWay && toOneWay) return 'no_u_turn'; // wider tolerance for u-turn if both ways are oneway
 
-         if ((angle < 40 || angle > 319) && fromOneWay && toOneWay && turn.from.vertex !== turn.to.vertex) return 'no_u_turn'; // even wider tolerance for u-turn if there is a via way (from !== to)
+         _this.mergeCustomGeoJSON = function (fc) {
+           if (fc && fc.type === 'FeatureCollection' && Array.isArray(fc.features)) {
+             fc.features.forEach(function (feature) {
+               feature.properties = feature.properties || {};
+               var props = feature.properties; // Get `id` from either `id` or `properties`
 
-         if (angle < 158) return 'no_right_turn';
-         if (angle > 202) return 'no_left_turn';
-         return 'no_straight_on';
-       }
+               var id = feature.id || props.id;
+               if (!id || !/^\S+\.geojson$/i.test(id)) return; // Ensure `id` exists and is lowercase
 
-       function actionMergePolygon(ids, newRelationId) {
-         function groupEntities(graph) {
-           var entities = ids.map(function (id) {
-             return graph.entity(id);
-           });
-           var geometryGroups = utilArrayGroupBy(entities, function (entity) {
-             if (entity.type === 'way' && entity.isClosed()) {
-               return 'closedWay';
-             } else if (entity.type === 'relation' && entity.isMultipolygon()) {
-               return 'multipolygon';
-             } else {
-               return 'other';
-             }
-           });
-           return Object.assign({
-             closedWay: [],
-             multipolygon: [],
-             other: []
-           }, geometryGroups);
-         }
+               id = id.toLowerCase();
+               feature.id = id;
+               props.id = id; // Ensure `area` property exists
 
-         var action = function action(graph) {
-           var entities = groupEntities(graph); // An array representing all the polygons that are part of the multipolygon.
-           //
-           // Each element is itself an array of objects with an id property, and has a
-           // locs property which is an array of the locations forming the polygon.
+               if (!props.area) {
+                 var area = geojsonArea.geometry(feature.geometry) / 1e6; // m² to km²
 
-           var polygons = entities.multipolygon.reduce(function (polygons, m) {
-             return polygons.concat(osmJoinWays(m.members, graph));
-           }, []).concat(entities.closedWay.map(function (d) {
-             var member = [{
-               id: d.id
-             }];
-             member.nodes = graph.childNodes(d);
-             return member;
-           })); // contained is an array of arrays of boolean values,
-           // where contained[j][k] is true iff the jth way is
-           // contained by the kth way.
+                 props.area = Number(area.toFixed(2));
+               }
 
-           var contained = polygons.map(function (w, i) {
-             return polygons.map(function (d, n) {
-               if (i === n) return null;
-               return geoPolygonContainsPolygon(d.nodes.map(function (n) {
-                 return n.loc;
-               }), w.nodes.map(function (n) {
-                 return n.loc;
-               }));
+               _loco._cache[id] = feature;
              });
-           }); // Sort all polygons as either outer or inner ways
+           }
+         }; //
+         // `mergeLocationSets`
+         //  Accepts an Array of Objects containing `locationSet` properties.
+         //  The locationSets will be resolved and indexed in the background.
+         //  [
+         //   { id: 'preset1', locationSet: {…} },
+         //   { id: 'preset2', locationSet: {…} },
+         //   { id: 'preset3', locationSet: {…} },
+         //   …
+         //  ]
+         //  After resolving and indexing, the Objects will be decorated with a
+         //  `locationSetID` property.
+         //  [
+         //   { id: 'preset1', locationSet: {…}, locationSetID: '+[Q2]' },
+         //   { id: 'preset2', locationSet: {…}, locationSetID: '+[Q30]' },
+         //   { id: 'preset3', locationSet: {…}, locationSetID: '+[Q2]' },
+         //   …
+         //  ]
+         //
+         //  Returns a Promise fulfilled when the resolving/indexing has been completed
+         //  This will take some seconds but happen in the background during browser idle time.
+         //
 
-           var members = [];
-           var outer = true;
 
-           while (polygons.length) {
-             extractUncontained(polygons);
-             polygons = polygons.filter(isContained);
-             contained = contained.filter(isContained).map(filterContained);
-           }
+         _this.mergeLocationSets = function (objects) {
+           if (!Array.isArray(objects)) return Promise.reject('nothing to do'); // Resolve all locationSets -> geojson, processing data in chunks
+           //
+           // Because this will happen during idle callbacks, we want to choose a chunk size
+           // that won't make the browser stutter too badly.  LocationSets that are a simple
+           // country coder include will resolve instantly, but ones that involve complex
+           // include/exclude operations will take some milliseconds longer.
+           //
+           // Some discussion and performance results on these tickets:
+           // https://github.com/ideditor/location-conflation/issues/26
+           // https://github.com/osmlab/name-suggestion-index/issues/4784#issuecomment-742003434
 
-           function isContained(d, i) {
-             return contained[i].some(function (val) {
-               return val;
+           _queue = _queue.concat(utilArrayChunk(objects, 200));
+
+           if (!_inProcess) {
+             _inProcess = processQueue().then(function () {
+               rebuildIndex();
+               _inProcess = null;
+               return objects;
              });
            }
 
-           function filterContained(d) {
-             return d.filter(isContained);
+           return _inProcess;
+         }; //
+         // `locationSetID`
+         // Returns a locationSetID for a given locationSet (fallback to `+[Q2]`, world)
+         // (The locationset doesn't necessarily need to be resolved to compute its `id`)
+         //
+         // Arguments
+         //   `locationSet`: A locationSet, e.g. `{ include: ['us'] }`
+         // Returns
+         //   The locationSetID, e.g. `+[Q30]`
+         //
+
+
+         _this.locationSetID = function (locationSet) {
+           var locationSetID;
+
+           try {
+             locationSetID = _loco.validateLocationSet(locationSet).id;
+           } catch (err) {
+             locationSetID = '+[Q2]'; // the world
            }
 
-           function extractUncontained(polygons) {
-             polygons.forEach(function (d, i) {
-               if (!isContained(d, i)) {
-                 d.forEach(function (member) {
-                   members.push({
-                     type: 'way',
-                     id: member.id,
-                     role: outer ? 'outer' : 'inner'
-                   });
-                 });
-               }
-             });
-             outer = !outer;
-           } // Move all tags to one relation
+           return locationSetID;
+         }; //
+         // `feature`
+         // Returns the resolved GeoJSON feature for a given locationSetID (fallback to 'world')
+         //
+         // Arguments
+         //   `locationSetID`: id of the form like `+[Q30]`  (United States)
+         // Returns
+         //   A GeoJSON feature:
+         //   {
+         //     type: 'Feature',
+         //     id: '+[Q30]',
+         //     properties: { id: '+[Q30]', area: 21817019.17, … },
+         //     geometry: { … }
+         //   }
 
 
-           var relation = entities.multipolygon[0] || osmRelation({
-             id: newRelationId,
-             tags: {
-               type: 'multipolygon'
-             }
-           });
-           entities.multipolygon.slice(1).forEach(function (m) {
-             relation = relation.mergeTags(m.tags);
-             graph = graph.remove(m);
-           });
-           entities.closedWay.forEach(function (way) {
-             function isThisOuter(m) {
-               return m.id === way.id && m.role !== 'inner';
-             }
+         _this.feature = function (locationSetID) {
+           return _resolvedFeatures[locationSetID] || _resolvedFeatures['+[Q2]'];
+         }; //
+         // `locationsAt`
+         // Find all the resolved locationSets valid at the given location.
+         // Results include the area (in km²) to facilitate sorting.
+         //
+         // Arguments
+         //   `loc`: the [lon,lat] location to query, e.g. `[-74.4813, 40.7967]`
+         // Returns
+         //   Object of locationSetIDs to areas (in km²)
+         //   {
+         //     "+[Q2]": 511207893.3958111,
+         //     "+[Q30]": 21817019.17,
+         //     "+[new_jersey.geojson]": 22390.77,
+         //     …
+         //   }
+         //
 
-             if (members.some(isThisOuter)) {
-               relation = relation.mergeTags(way.tags);
-               graph = graph.replace(way.update({
-                 tags: {}
-               }));
-             }
+
+         _this.locationsAt = function (loc) {
+           var result = {};
+           (_wp(loc, true) || []).forEach(function (prop) {
+             return result[prop.id] = prop.area;
            });
-           return graph.replace(relation.update({
-             members: members,
-             tags: utilObjectOmit(relation.tags, ['area'])
-           }));
-         };
+           return result;
+         }; //
+         // `query`
+         // Execute a query directly against which-polygon
+         // https://github.com/mapbox/which-polygon
+         //
+         // Arguments
+         //   `loc`: the [lon,lat] location to query,
+         //   `multi`: `true` to return all results, `false` to return first result
+         // Returns
+         //   Array of GeoJSON *properties* for the locationSet features that exist at `loc`
+         //
 
-         action.disabled = function (graph) {
-           var entities = groupEntities(graph);
 
-           if (entities.other.length > 0 || entities.closedWay.length + entities.multipolygon.length < 2) {
-             return 'not_eligible';
-           }
+         _this.query = function (loc, multi) {
+           return _wp(loc, multi);
+         }; // Direct access to the location-conflation resolver
 
-           if (!entities.multipolygon.every(function (r) {
-             return r.isComplete(graph);
-           })) {
-             return 'incomplete_relation';
-           }
 
-           if (!entities.multipolygon.length) {
-             var sharedMultipolygons = [];
-             entities.closedWay.forEach(function (way, i) {
-               if (i === 0) {
-                 sharedMultipolygons = graph.parentMultipolygons(way);
-               } else {
-                 sharedMultipolygons = utilArrayIntersection(sharedMultipolygons, graph.parentMultipolygons(way));
-               }
-             });
-             sharedMultipolygons = sharedMultipolygons.filter(function (relation) {
-               return relation.members.length === entities.closedWay.length;
-             });
+         _this.loco = function () {
+           return _loco;
+         }; // Direct access to the which-polygon index
 
-             if (sharedMultipolygons.length) {
-               // don't create a new multipolygon if it'd be redundant
-               return 'not_eligible';
-             }
-           } else if (entities.closedWay.some(function (way) {
-             return utilArrayIntersection(graph.parentMultipolygons(way), entities.multipolygon).length;
-           })) {
-             // don't add a way to a multipolygon again if it's already a member
-             return 'not_eligible';
-           }
+
+         _this.wp = function () {
+           return _wp;
          };
 
-         return action;
+         return _this;
        }
 
-       var UNSUPPORTED_Y$3 = regexpStickyHelpers.UNSUPPORTED_Y;
-
-       // `RegExp.prototype.flags` getter
-       // https://tc39.es/ecma262/#sec-get-regexp.prototype.flags
-       if (descriptors && (/./g.flags != 'g' || UNSUPPORTED_Y$3)) {
-         objectDefineProperty.f(RegExp.prototype, 'flags', {
-           configurable: true,
-           get: regexpFlags
-         });
-       }
+       var $findIndex = arrayIteration.findIndex;
 
-       var fastDeepEqual = function equal(a, b) {
-         if (a === b) return true;
 
-         if (a && b && _typeof(a) == 'object' && _typeof(b) == 'object') {
-           if (a.constructor !== b.constructor) return false;
-           var length, i, keys;
+       var FIND_INDEX = 'findIndex';
+       var SKIPS_HOLES = true;
 
-           if (Array.isArray(a)) {
-             length = a.length;
-             if (length != b.length) return false;
+       // Shouldn't skip holes
+       if (FIND_INDEX in []) Array(1)[FIND_INDEX](function () { SKIPS_HOLES = false; });
 
-             for (i = length; i-- !== 0;) {
-               if (!equal(a[i], b[i])) return false;
-             }
+       // `Array.prototype.findIndex` method
+       // https://tc39.es/ecma262/#sec-array.prototype.findindex
+       _export({ target: 'Array', proto: true, forced: SKIPS_HOLES }, {
+         findIndex: function findIndex(callbackfn /* , that = undefined */) {
+           return $findIndex(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
+         }
+       });
 
-             return true;
-           }
+       // https://tc39.es/ecma262/#sec-array.prototype-@@unscopables
+       addToUnscopables(FIND_INDEX);
 
-           if (a.constructor === RegExp) return a.source === b.source && a.flags === b.flags;
-           if (a.valueOf !== Object.prototype.valueOf) return a.valueOf() === b.valueOf();
-           if (a.toString !== Object.prototype.toString) return a.toString() === b.toString();
-           keys = Object.keys(a);
-           length = keys.length;
-           if (length !== Object.keys(b).length) return false;
+       var notARegexp = function (it) {
+         if (isRegexp(it)) {
+           throw TypeError("The method doesn't accept regular expressions");
+         } return it;
+       };
 
-           for (i = length; i-- !== 0;) {
-             if (!Object.prototype.hasOwnProperty.call(b, keys[i])) return false;
-           }
+       var MATCH = wellKnownSymbol('match');
 
-           for (i = length; i-- !== 0;) {
-             var key = keys[i];
-             if (!equal(a[key], b[key])) return false;
-           }
+       var correctIsRegexpLogic = function (METHOD_NAME) {
+         var regexp = /./;
+         try {
+           '/./'[METHOD_NAME](regexp);
+         } catch (error1) {
+           try {
+             regexp[MATCH] = false;
+             return '/./'[METHOD_NAME](regexp);
+           } catch (error2) { /* empty */ }
+         } return false;
+       };
 
-           return true;
-         } // true if both NaN, false otherwise
+       // `String.prototype.includes` method
+       // https://tc39.es/ecma262/#sec-string.prototype.includes
+       _export({ target: 'String', proto: true, forced: !correctIsRegexpLogic('includes') }, {
+         includes: function includes(searchString /* , position = 0 */) {
+           return !!~String(requireObjectCoercible(this))
+             .indexOf(notARegexp(searchString), arguments.length > 1 ? arguments[1] : undefined);
+         }
+       });
 
+       var _mainLocalizer = coreLocalizer(); // singleton
 
-         return a !== a && b !== b;
-       };
 
-       // J. W. Hunt and M. D. McIlroy, An algorithm for differential buffer
-       // comparison, Bell Telephone Laboratories CSTR #41 (1976)
-       // http://www.cs.dartmouth.edu/~doug/
-       // https://en.wikipedia.org/wiki/Longest_common_subsequence_problem
+       var _t = _mainLocalizer.t;
+       // coreLocalizer manages language and locale parameters including translated strings
        //
-       // Expects two arrays, finds longest common sequence
 
-       function LCS(buffer1, buffer2) {
-         var equivalenceClasses = {};
+       function coreLocalizer() {
+         var localizer = {};
+         var _dataLanguages = {}; // `_dataLocales` is an object containing all _supported_ locale codes -> language info.
+         // * `rtl` - right-to-left or left-to-right text direction
+         // * `pct` - the percent of strings translated; 1 = 100%, full coverage
+         //
+         // {
+         // en: { rtl: false, pct: {…} },
+         // de: { rtl: false, pct: {…} },
+         // …
+         // }
 
-         for (var j = 0; j < buffer2.length; j++) {
-           var item = buffer2[j];
+         var _dataLocales = {}; // `localeStrings` is an object containing all _loaded_ locale codes -> string data.
+         // {
+         // en: { icons: {…}, toolbar: {…}, modes: {…}, operations: {…}, … },
+         // de: { icons: {…}, toolbar: {…}, modes: {…}, operations: {…}, … },
+         // …
+         // }
 
-           if (equivalenceClasses[item]) {
-             equivalenceClasses[item].push(j);
-           } else {
-             equivalenceClasses[item] = [j];
-           }
-         }
+         var _localeStrings = {}; // the current locale
 
-         var NULLRESULT = {
-           buffer1index: -1,
-           buffer2index: -1,
-           chain: null
+         var _localeCode = 'en-US'; // `_localeCodes` must contain `_localeCode` first, optionally followed by fallbacks
+
+         var _localeCodes = ['en-US', 'en'];
+         var _languageCode = 'en';
+         var _textDirection = 'ltr';
+         var _usesMetric = false;
+         var _languageNames = {};
+         var _scriptNames = {}; // getters for the current locale parameters
+
+         localizer.localeCode = function () {
+           return _localeCode;
          };
-         var candidates = [NULLRESULT];
 
-         for (var i = 0; i < buffer1.length; i++) {
-           var _item = buffer1[i];
-           var buffer2indices = equivalenceClasses[_item] || [];
-           var r = 0;
-           var c = candidates[0];
+         localizer.localeCodes = function () {
+           return _localeCodes;
+         };
 
-           for (var jx = 0; jx < buffer2indices.length; jx++) {
-             var _j = buffer2indices[jx];
-             var s = void 0;
+         localizer.languageCode = function () {
+           return _languageCode;
+         };
 
-             for (s = r; s < candidates.length; s++) {
-               if (candidates[s].buffer2index < _j && (s === candidates.length - 1 || candidates[s + 1].buffer2index > _j)) {
-                 break;
-               }
-             }
+         localizer.textDirection = function () {
+           return _textDirection;
+         };
 
-             if (s < candidates.length) {
-               var newCandidate = {
-                 buffer1index: i,
-                 buffer2index: _j,
-                 chain: candidates[s]
-               };
+         localizer.usesMetric = function () {
+           return _usesMetric;
+         };
 
-               if (r === candidates.length) {
-                 candidates.push(c);
-               } else {
-                 candidates[r] = c;
-               }
+         localizer.languageNames = function () {
+           return _languageNames;
+         };
 
-               r = s + 1;
-               c = newCandidate;
+         localizer.scriptNames = function () {
+           return _scriptNames;
+         }; // The client app may want to manually set the locale, regardless of the
+         // settings provided by the browser
 
-               if (r === candidates.length) {
-                 break; // no point in examining further (j)s
-               }
-             }
+
+         var _preferredLocaleCodes = [];
+
+         localizer.preferredLocaleCodes = function (codes) {
+           if (!arguments.length) return _preferredLocaleCodes;
+
+           if (typeof codes === 'string') {
+             // be generous and accept delimited strings as input
+             _preferredLocaleCodes = codes.split(/,|;| /gi).filter(Boolean);
+           } else {
+             _preferredLocaleCodes = codes;
            }
 
-           candidates[r] = c;
-         } // At this point, we know the LCS: it's in the reverse of the
-         // linked-list through .chain of candidates[candidates.length - 1].
+           return localizer;
+         };
+
+         var _loadPromise;
+
+         localizer.ensureLoaded = function () {
+           if (_loadPromise) return _loadPromise;
+           var filesToFetch = [// load the list of languages
+           'languages', // load the list of supported locales
+           'locales'];
+           var localeDirs = {
+             general: 'locales',
+             tagging: 'https://cdn.jsdelivr.net/npm/@openstreetmap/id-tagging-schema@3/dist/translations'
+           };
+           var fileMap = _mainFileFetcher.fileMap();
+
+           for (var scopeId in localeDirs) {
+             var key = "locales_index_".concat(scopeId);
+             fileMap[key] = localeDirs[scopeId] + '/index.min.json';
+             filesToFetch.push(key);
+           }
 
+           return _loadPromise = Promise.all(filesToFetch.map(function (key) {
+             return _mainFileFetcher.get(key);
+           })).then(function (results) {
+             _dataLanguages = results[0];
+             _dataLocales = results[1];
+             var indexes = results.slice(2);
 
-         return candidates[candidates.length - 1];
-       } // We apply the LCS to build a 'comm'-style picture of the
-       // offsets and lengths of mismatched chunks in the input
-       // buffers. This is used by diff3MergeRegions.
+             var requestedLocales = (_preferredLocaleCodes || []). // List of locales preferred by the browser in priority order.
+             concat(utilDetect().browserLocales) // fallback to English since it's the only guaranteed complete language
+             .concat(['en']);
 
+             _localeCodes = localesToUseFrom(requestedLocales); // Run iD in the highest-priority locale; the rest are fallbacks
 
-       function diffIndices(buffer1, buffer2) {
-         var lcs = LCS(buffer1, buffer2);
-         var result = [];
-         var tail1 = buffer1.length;
-         var tail2 = buffer2.length;
+             _localeCode = _localeCodes[0];
+             var loadStringsPromises = [];
+             indexes.forEach(function (index, i) {
+               // Will always return the index for `en` if nothing else
+               var fullCoverageIndex = _localeCodes.findIndex(function (locale) {
+                 return index[locale] && index[locale].pct === 1;
+               }); // We only need to load locales up until we find one with full coverage
 
-         for (var candidate = lcs; candidate !== null; candidate = candidate.chain) {
-           var mismatchLength1 = tail1 - candidate.buffer1index - 1;
-           var mismatchLength2 = tail2 - candidate.buffer2index - 1;
-           tail1 = candidate.buffer1index;
-           tail2 = candidate.buffer2index;
 
-           if (mismatchLength1 || mismatchLength2) {
-             result.push({
-               buffer1: [tail1 + 1, mismatchLength1],
-               buffer1Content: buffer1.slice(tail1 + 1, tail1 + 1 + mismatchLength1),
-               buffer2: [tail2 + 1, mismatchLength2],
-               buffer2Content: buffer2.slice(tail2 + 1, tail2 + 1 + mismatchLength2)
+               _localeCodes.slice(0, fullCoverageIndex + 1).forEach(function (code) {
+                 var scopeId = Object.keys(localeDirs)[i];
+                 var directory = Object.values(localeDirs)[i];
+                 if (index[code]) loadStringsPromises.push(localizer.loadLocale(code, scopeId, directory));
+               });
              });
-           }
-         }
+             return Promise.all(loadStringsPromises);
+           }).then(function () {
+             updateForCurrentLocale();
+           })["catch"](function (err) {
+             return console.error(err);
+           }); // eslint-disable-line
+         }; // Returns the locales from `requestedLocales` supported by iD that we should use
 
-         result.reverse();
-         return result;
-       } // We apply the LCS to build a JSON representation of a
-       // independently derived from O, returns a fairly complicated
-       // internal representation of merge decisions it's taken. The
-       // interested reader may wish to consult
-       //
-       // Sanjeev Khanna, Keshav Kunal, and Benjamin C. Pierce.
-       // 'A Formal Investigation of ' In Arvind and Prasad,
-       // editors, Foundations of Software Technology and Theoretical
-       // Computer Science (FSTTCS), December 2007.
-       //
-       // (http://www.cis.upenn.edu/~bcpierce/papers/diff3-short.pdf)
-       //
 
+         function localesToUseFrom(requestedLocales) {
+           var supportedLocales = _dataLocales;
+           var toUse = [];
 
-       function diff3MergeRegions(a, o, b) {
-         // "hunks" are array subsets where `a` or `b` are different from `o`
-         // https://www.gnu.org/software/diffutils/manual/html_node/diff3-Hunks.html
-         var hunks = [];
+           for (var i in requestedLocales) {
+             var locale = requestedLocales[i];
+             if (supportedLocales[locale]) toUse.push(locale);
 
-         function addHunk(h, ab) {
-           hunks.push({
-             ab: ab,
-             oStart: h.buffer1[0],
-             oLength: h.buffer1[1],
-             // length of o to remove
-             abStart: h.buffer2[0],
-             abLength: h.buffer2[1] // length of a/b to insert
-             // abContent: (ab === 'a' ? a : b).slice(h.buffer2[0], h.buffer2[0] + h.buffer2[1])
+             if (locale.includes('-')) {
+               // Full locale ('es-ES'), add fallback to the base ('es')
+               var langPart = locale.split('-')[0];
+               if (supportedLocales[langPart]) toUse.push(langPart);
+             }
+           } // remove duplicates
 
-           });
+
+           return utilArrayUniq(toUse);
          }
 
-         diffIndices(o, a).forEach(function (item) {
-           return addHunk(item, 'a');
-         });
-         diffIndices(o, b).forEach(function (item) {
-           return addHunk(item, 'b');
-         });
-         hunks.sort(function (x, y) {
-           return x.oStart - y.oStart;
-         });
-         var results = [];
-         var currOffset = 0;
+         function updateForCurrentLocale() {
+           if (!_localeCode) return;
+           _languageCode = _localeCode.split('-')[0];
+           var currentData = _dataLocales[_localeCode] || _dataLocales[_languageCode];
+           var hash = utilStringQs(window.location.hash);
 
-         function advanceTo(endOffset) {
-           if (endOffset > currOffset) {
-             results.push({
-               stable: true,
-               buffer: 'o',
-               bufferStart: currOffset,
-               bufferLength: endOffset - currOffset,
-               bufferContent: o.slice(currOffset, endOffset)
-             });
-             currOffset = endOffset;
+           if (hash.rtl === 'true') {
+             _textDirection = 'rtl';
+           } else if (hash.rtl === 'false') {
+             _textDirection = 'ltr';
+           } else {
+             _textDirection = currentData && currentData.rtl ? 'rtl' : 'ltr';
            }
+
+           var locale = _localeCode;
+           if (locale.toLowerCase() === 'en-us') locale = 'en';
+           _languageNames = _localeStrings.general[locale].languageNames;
+           _scriptNames = _localeStrings.general[locale].scriptNames;
+           _usesMetric = _localeCode.slice(-3).toLowerCase() !== '-us';
          }
+         /* Locales */
+         // Returns a Promise to load the strings for the requested locale
 
-         while (hunks.length) {
-           var hunk = hunks.shift();
-           var regionStart = hunk.oStart;
-           var regionEnd = hunk.oStart + hunk.oLength;
-           var regionHunks = [hunk];
-           advanceTo(regionStart); // Try to pull next overlapping hunk into this region
 
-           while (hunks.length) {
-             var nextHunk = hunks[0];
-             var nextHunkStart = nextHunk.oStart;
-             if (nextHunkStart > regionEnd) break; // no overlap
+         localizer.loadLocale = function (locale, scopeId, directory) {
+           // US English is the default
+           if (locale.toLowerCase() === 'en-us') locale = 'en';
 
-             regionEnd = Math.max(regionEnd, nextHunkStart + nextHunk.oLength);
-             regionHunks.push(hunks.shift());
+           if (_localeStrings[scopeId] && _localeStrings[scopeId][locale]) {
+             // already loaded
+             return Promise.resolve(locale);
            }
 
-           if (regionHunks.length === 1) {
-             // Only one hunk touches this region, meaning that there is no conflict here.
-             // Either `a` or `b` is inserting into a region of `o` unchanged by the other.
-             if (hunk.abLength > 0) {
-               var buffer = hunk.ab === 'a' ? a : b;
-               results.push({
-                 stable: true,
-                 buffer: hunk.ab,
-                 bufferStart: hunk.abStart,
-                 bufferLength: hunk.abLength,
-                 bufferContent: buffer.slice(hunk.abStart, hunk.abStart + hunk.abLength)
-               });
-             }
-           } else {
-             // A true a/b conflict. Determine the bounds involved from `a`, `o`, and `b`.
-             // Effectively merge all the `a` hunks into one giant hunk, then do the
-             // same for the `b` hunks; then, correct for skew in the regions of `o`
-             // that each side changed, and report appropriate spans for the three sides.
-             var bounds = {
-               a: [a.length, -1, o.length, -1],
-               b: [b.length, -1, o.length, -1]
-             };
-
-             while (regionHunks.length) {
-               hunk = regionHunks.shift();
-               var oStart = hunk.oStart;
-               var oEnd = oStart + hunk.oLength;
-               var abStart = hunk.abStart;
-               var abEnd = abStart + hunk.abLength;
-               var _b = bounds[hunk.ab];
-               _b[0] = Math.min(abStart, _b[0]);
-               _b[1] = Math.max(abEnd, _b[1]);
-               _b[2] = Math.min(oStart, _b[2]);
-               _b[3] = Math.max(oEnd, _b[3]);
-             }
-
-             var aStart = bounds.a[0] + (regionStart - bounds.a[2]);
-             var aEnd = bounds.a[1] + (regionEnd - bounds.a[3]);
-             var bStart = bounds.b[0] + (regionStart - bounds.b[2]);
-             var bEnd = bounds.b[1] + (regionEnd - bounds.b[3]);
-             var result = {
-               stable: false,
-               aStart: aStart,
-               aLength: aEnd - aStart,
-               aContent: a.slice(aStart, aEnd),
-               oStart: regionStart,
-               oLength: regionEnd - regionStart,
-               oContent: o.slice(regionStart, regionEnd),
-               bStart: bStart,
-               bLength: bEnd - bStart,
-               bContent: b.slice(bStart, bEnd)
-             };
-             results.push(result);
-           }
+           var fileMap = _mainFileFetcher.fileMap();
+           var key = "locale_".concat(scopeId, "_").concat(locale);
+           fileMap[key] = "".concat(directory, "/").concat(locale, ".min.json");
+           return _mainFileFetcher.get(key).then(function (d) {
+             if (!_localeStrings[scopeId]) _localeStrings[scopeId] = {};
+             _localeStrings[scopeId][locale] = d[locale];
+             return locale;
+           });
+         };
 
-           currOffset = regionEnd;
-         }
+         localizer.pluralRule = function (number) {
+           return pluralRule(number, _localeCode);
+         }; // Returns the plural rule for the given `number` with the given `localeCode`.
+         // One of: `zero`, `one`, `two`, `few`, `many`, `other`
 
-         advanceTo(o.length);
-         return results;
-       } // Applies the output of diff3MergeRegions to actually
-       // construct the merged buffer; the returned result alternates
-       // between 'ok' and 'conflict' blocks.
-       // A "false conflict" is where `a` and `b` both change the same from `o`
 
+         function pluralRule(number, localeCode) {
+           // modern browsers have this functionality built-in
+           var rules = 'Intl' in window && Intl.PluralRules && new Intl.PluralRules(localeCode);
 
-       function diff3Merge(a, o, b, options) {
-         var defaults = {
-           excludeFalseConflicts: true,
-           stringSeparator: /\s+/
-         };
-         options = Object.assign(defaults, options);
-         var aString = typeof a === 'string';
-         var oString = typeof o === 'string';
-         var bString = typeof b === 'string';
-         if (aString) a = a.split(options.stringSeparator);
-         if (oString) o = o.split(options.stringSeparator);
-         if (bString) b = b.split(options.stringSeparator);
-         var results = [];
-         var regions = diff3MergeRegions(a, o, b);
-         var okBuffer = [];
+           if (rules) {
+             return rules.select(number);
+           } // fallback to basic one/other, as in English
 
-         function flushOk() {
-           if (okBuffer.length) {
-             results.push({
-               ok: okBuffer
-             });
-           }
 
-           okBuffer = [];
+           if (number === 1) return 'one';
+           return 'other';
          }
+         /**
+         * Try to find that string in `locale` or the current `_localeCode` matching
+         * the given `stringId`. If no string can be found in the requested locale,
+         * we'll recurse down all the `_localeCodes` until one is found.
+         *
+         * @param  {string}   stringId      string identifier
+         * @param  {object?}  replacements  token replacements and default string
+         * @param  {string?}  locale        locale to use (defaults to currentLocale)
+         * @return {string?}  localized string
+         */
 
-         function isFalseConflict(a, b) {
-           if (a.length !== b.length) return false;
 
-           for (var i = 0; i < a.length; i++) {
-             if (a[i] !== b[i]) return false;
-           }
+         localizer.tInfo = function (origStringId, replacements, locale) {
+           var stringId = origStringId.trim();
+           var scopeId = 'general';
 
-           return true;
-         }
+           if (stringId[0] === '_') {
+             var split = stringId.split('.');
+             scopeId = split[0].slice(1);
+             stringId = split.slice(1).join('.');
+           }
 
-         regions.forEach(function (region) {
-           if (region.stable) {
-             var _okBuffer;
+           locale = locale || _localeCode;
+           var path = stringId.split('.').map(function (s) {
+             return s.replace(/<TX_DOT>/g, '.');
+           }).reverse();
+           var stringsKey = locale; // US English is the default
 
-             (_okBuffer = okBuffer).push.apply(_okBuffer, _toConsumableArray(region.bufferContent));
-           } else {
-             if (options.excludeFalseConflicts && isFalseConflict(region.aContent, region.bContent)) {
-               var _okBuffer2;
+           if (stringsKey.toLowerCase() === 'en-us') stringsKey = 'en';
+           var result = _localeStrings && _localeStrings[scopeId] && _localeStrings[scopeId][stringsKey];
 
-               (_okBuffer2 = okBuffer).push.apply(_okBuffer2, _toConsumableArray(region.aContent));
-             } else {
-               flushOk();
-               results.push({
-                 conflict: {
-                   a: region.aContent,
-                   aIndex: region.aStart,
-                   o: region.oContent,
-                   oIndex: region.oStart,
-                   b: region.bContent,
-                   bIndex: region.bStart
-                 }
-               });
-             }
+           while (result !== undefined && path.length) {
+             result = result[path.pop()];
            }
-         });
-         flushOk();
-         return results;
-       }
 
-       function actionMergeRemoteChanges(id, localGraph, remoteGraph, discardTags, formatUser) {
-         discardTags = discardTags || {};
-         var _option = 'safe'; // 'safe', 'force_local', 'force_remote'
+           if (result !== undefined) {
+             if (replacements) {
+               if (_typeof(result) === 'object' && Object.keys(result).length) {
+                 // If plural forms are provided, dig one level deeper based on the
+                 // first numeric token replacement provided.
+                 var number = Object.values(replacements).find(function (value) {
+                   return typeof value === 'number';
+                 });
 
-         var _conflicts = [];
+                 if (number !== undefined) {
+                   var rule = pluralRule(number, locale);
 
-         function user(d) {
-           return typeof formatUser === 'function' ? formatUser(d) : d;
-         }
+                   if (result[rule]) {
+                     result = result[rule];
+                   } else {
+                     // We're pretty sure this should be a plural but no string
+                     // could be found for the given rule. Just pick the first
+                     // string and hope it makes sense.
+                     result = Object.values(result)[0];
+                   }
+                 }
+               }
 
-         function mergeLocation(remote, target) {
-           function pointEqual(a, b) {
-             var epsilon = 1e-6;
-             return Math.abs(a[0] - b[0]) < epsilon && Math.abs(a[1] - b[1]) < epsilon;
-           }
+               if (typeof result === 'string') {
+                 for (var key in replacements) {
+                   var value = replacements[key];
 
-           if (_option === 'force_local' || pointEqual(target.loc, remote.loc)) {
-             return target;
-           }
+                   if (typeof value === 'number') {
+                     if (value.toLocaleString) {
+                       // format numbers for the locale
+                       value = value.toLocaleString(locale, {
+                         style: 'decimal',
+                         useGrouping: true,
+                         minimumFractionDigits: 0
+                       });
+                     } else {
+                       value = value.toString();
+                     }
+                   }
 
-           if (_option === 'force_remote') {
-             return target.update({
-               loc: remote.loc
-             });
-           }
+                   var token = "{".concat(key, "}");
+                   var regex = new RegExp(token, 'g');
+                   result = result.replace(regex, value);
+                 }
+               }
+             }
 
-           _conflicts.push(_t('merge_remote_changes.conflict.location', {
-             user: user(remote.user)
-           }));
+             if (typeof result === 'string') {
+               // found a localized string!
+               return {
+                 text: result,
+                 locale: locale
+               };
+             }
+           } // no localized string found...
+           // attempt to fallback to a lower-priority language
 
-           return target;
-         }
 
-         function mergeNodes(base, remote, target) {
-           if (_option === 'force_local' || fastDeepEqual(target.nodes, remote.nodes)) {
-             return target;
+           var index = _localeCodes.indexOf(locale);
+
+           if (index >= 0 && index < _localeCodes.length - 1) {
+             // eventually this will be 'en' or another locale with 100% coverage
+             var fallback = _localeCodes[index + 1];
+             return localizer.tInfo(origStringId, replacements, fallback);
            }
 
-           if (_option === 'force_remote') {
-             return target.update({
-               nodes: remote.nodes
-             });
+           if (replacements && 'default' in replacements) {
+             // Fallback to a default value if one is specified in `replacements`
+             return {
+               text: replacements["default"],
+               locale: null
+             };
            }
 
-           var ccount = _conflicts.length;
-           var o = base.nodes || [];
-           var a = target.nodes || [];
-           var b = remote.nodes || [];
-           var nodes = [];
-           var hunks = diff3Merge(a, o, b, {
-             excludeFalseConflicts: true
-           });
+           var missing = "Missing ".concat(locale, " translation: ").concat(origStringId);
+           if (typeof console !== 'undefined') console.error(missing); // eslint-disable-line
 
-           for (var i = 0; i < hunks.length; i++) {
-             var hunk = hunks[i];
+           return {
+             text: missing,
+             locale: 'en'
+           };
+         };
 
-             if (hunk.ok) {
-               nodes.push.apply(nodes, hunk.ok);
-             } else {
-               // for all conflicts, we can assume c.a !== c.b
-               // because `diff3Merge` called with `true` option to exclude false conflicts..
-               var c = hunk.conflict;
+         localizer.hasTextForStringId = function (stringId) {
+           return !!localizer.tInfo(stringId, {
+             "default": 'nothing found'
+           }).locale;
+         }; // Returns only the localized text, discarding the locale info
 
-               if (fastDeepEqual(c.o, c.a)) {
-                 // only changed remotely
-                 nodes.push.apply(nodes, c.b);
-               } else if (fastDeepEqual(c.o, c.b)) {
-                 // only changed locally
-                 nodes.push.apply(nodes, c.a);
-               } else {
-                 // changed both locally and remotely
-                 _conflicts.push(_t('merge_remote_changes.conflict.nodelist', {
-                   user: user(remote.user)
-                 }));
 
-                 break;
-               }
-             }
-           }
+         localizer.t = function (stringId, replacements, locale) {
+           return localizer.tInfo(stringId, replacements, locale).text;
+         }; // Returns the localized text wrapped in an HTML element encoding the locale info
 
-           return _conflicts.length === ccount ? target.update({
-             nodes: nodes
-           }) : target;
-         }
 
-         function mergeChildren(targetWay, children, updates, graph) {
-           function isUsed(node, targetWay) {
-             var hasInterestingParent = graph.parentWays(node).some(function (way) {
-               return way.id !== targetWay.id;
-             });
-             return node.hasInterestingTags() || hasInterestingParent || graph.parentRelations(node).length > 0;
-           }
+         localizer.t.html = function (stringId, replacements, locale) {
+           var info = localizer.tInfo(stringId, replacements, locale); // text may be empty or undefined if `replacements.default` is
 
-           var ccount = _conflicts.length;
+           return info.text ? localizer.htmlForLocalizedText(info.text, info.locale) : '';
+         };
 
-           for (var i = 0; i < children.length; i++) {
-             var id = children[i];
-             var node = graph.hasEntity(id); // remove unused childNodes..
+         localizer.htmlForLocalizedText = function (text, localeCode) {
+           return "<span class=\"localized-text\" lang=\"".concat(localeCode || 'unknown', "\">").concat(text, "</span>");
+         };
 
-             if (targetWay.nodes.indexOf(id) === -1) {
-               if (node && !isUsed(node, targetWay)) {
-                 updates.removeIds.push(id);
-               }
+         localizer.languageName = function (code, options) {
+           if (_languageNames[code]) {
+             // name in locale language
+             // e.g. "German"
+             return _languageNames[code];
+           } // sometimes we only want the local name
 
-               continue;
-             } // restore used childNodes..
 
+           if (options && options.localOnly) return null;
+           var langInfo = _dataLanguages[code];
 
-             var local = localGraph.hasEntity(id);
-             var remote = remoteGraph.hasEntity(id);
-             var target;
+           if (langInfo) {
+             if (langInfo.nativeName) {
+               // name in native language
+               // e.g. "Deutsch (de)"
+               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 language this is based on
 
-             if (_option === 'force_remote' && remote && remote.visible) {
-               updates.replacements.push(remote);
-             } else if (_option === 'force_local' && local) {
-               target = osmEntity(local);
+               if (_languageNames[base]) {
+                 // base language name in locale language
+                 var scriptCode = langInfo.script;
+                 var script = _scriptNames[scriptCode] || scriptCode; // e.g. "Serbian (Cyrillic)"
 
-               if (remote) {
-                 target = target.update({
-                   version: remote.version
+                 return localizer.t('translate.language_and_code', {
+                   language: _languageNames[base],
+                   code: script
+                 });
+               } else if (_dataLanguages[base] && _dataLanguages[base].nativeName) {
+                 // e.g. "српски (sr-Cyrl)"
+                 return localizer.t('translate.language_and_code', {
+                   language: _dataLanguages[base].nativeName,
+                   code: code
                  });
                }
+             }
+           }
 
-               updates.replacements.push(target);
-             } else if (_option === 'safe' && local && remote && local.version !== remote.version) {
-               target = osmEntity(local, {
-                 version: remote.version
-               });
+           return code; // if not found, use the code
+         };
 
-               if (remote.visible) {
-                 target = mergeLocation(remote, target);
-               } else {
-                 _conflicts.push(_t('merge_remote_changes.conflict.deleted', {
-                   user: user(remote.user)
-                 }));
-               }
+         return localizer;
+       }
 
-               if (_conflicts.length !== ccount) break;
-               updates.replacements.push(target);
-             }
-           }
+       // `presetCollection` is a wrapper around an `Array` of presets `collection`,
+       // and decorated with some extra methods for searching and matching geometry
+       //
+
+       function presetCollection(collection) {
+         var MAXRESULTS = 50;
+         var _this = {};
+         var _memo = {};
+         _this.collection = collection;
+
+         _this.item = function (id) {
+           if (_memo[id]) return _memo[id];
 
-           return targetWay;
-         }
+           var found = _this.collection.find(function (d) {
+             return d.id === id;
+           });
 
-         function updateChildren(updates, graph) {
-           for (var i = 0; i < updates.replacements.length; i++) {
-             graph = graph.replace(updates.replacements[i]);
-           }
+           if (found) _memo[id] = found;
+           return found;
+         };
 
-           if (updates.removeIds.length) {
-             graph = actionDeleteMultiple(updates.removeIds)(graph);
-           }
+         _this.index = function (id) {
+           return _this.collection.findIndex(function (d) {
+             return d.id === id;
+           });
+         };
 
-           return graph;
-         }
+         _this.matchGeometry = function (geometry) {
+           return presetCollection(_this.collection.filter(function (d) {
+             return d.matchGeometry(geometry);
+           }));
+         };
 
-         function mergeMembers(remote, target) {
-           if (_option === 'force_local' || fastDeepEqual(target.members, remote.members)) {
-             return target;
-           }
+         _this.matchAllGeometry = function (geometries) {
+           return presetCollection(_this.collection.filter(function (d) {
+             return d && d.matchAllGeometry(geometries);
+           }));
+         };
 
-           if (_option === 'force_remote') {
-             return target.update({
-               members: remote.members
+         _this.matchAnyGeometry = function (geometries) {
+           return presetCollection(_this.collection.filter(function (d) {
+             return geometries.some(function (geom) {
+               return d.matchGeometry(geom);
              });
-           }
-
-           _conflicts.push(_t('merge_remote_changes.conflict.memberlist', {
-             user: user(remote.user)
            }));
+         };
 
-           return target;
-         }
+         _this.fallback = function (geometry) {
+           var id = geometry;
+           if (id === 'vertex') id = 'point';
+           return _this.item(id);
+         };
 
-         function mergeTags(base, remote, target) {
-           if (_option === 'force_local' || fastDeepEqual(target.tags, remote.tags)) {
-             return target;
-           }
+         _this.search = function (value, geometry, loc) {
+           if (!value) return _this; // don't remove diacritical characters since we're assuming the user is being intentional
 
-           if (_option === 'force_remote') {
-             return target.update({
-               tags: remote.tags
-             });
+           value = value.toLowerCase().trim(); // match at name beginning or just after a space (e.g. "office" -> match "Law Office")
+
+           function leading(a) {
+             var index = a.indexOf(value);
+             return index === 0 || a[index - 1] === ' ';
+           } // match at name beginning only
+
+
+           function leadingStrict(a) {
+             var index = a.indexOf(value);
+             return index === 0;
            }
 
-           var ccount = _conflicts.length;
-           var o = base.tags || {};
-           var a = target.tags || {};
-           var b = remote.tags || {};
-           var keys = utilArrayUnion(utilArrayUnion(Object.keys(o), Object.keys(a)), Object.keys(b)).filter(function (k) {
-             return !discardTags[k];
-           });
-           var tags = Object.assign({}, a); // shallow copy
+           function sortPresets(nameProp) {
+             return function sortNames(a, b) {
+               var aCompare = a[nameProp]();
+               var bCompare = b[nameProp](); // priority if search string matches preset name exactly - #4325
 
-           var changed = false;
+               if (value === aCompare) return -1;
+               if (value === bCompare) return 1; // priority for higher matchScore
 
-           for (var i = 0; i < keys.length; i++) {
-             var k = keys[i];
+               var i = b.originalScore - a.originalScore;
+               if (i !== 0) return i; // priority if search string appears earlier in preset name
 
-             if (o[k] !== b[k] && a[k] !== b[k]) {
-               // changed remotely..
-               if (o[k] !== a[k]) {
-                 // changed locally..
-                 _conflicts.push(_t('merge_remote_changes.conflict.tags', {
-                   tag: k,
-                   local: a[k],
-                   remote: b[k],
-                   user: user(remote.user)
-                 }));
-               } else {
-                 // unchanged locally, accept remote change..
-                 if (b.hasOwnProperty(k)) {
-                   tags[k] = b[k];
-                 } else {
-                   delete tags[k];
-                 }
+               i = aCompare.indexOf(value) - bCompare.indexOf(value);
+               if (i !== 0) return i; // priority for shorter preset names
 
-                 changed = true;
-               }
-             }
+               return aCompare.length - bCompare.length;
+             };
            }
 
-           return changed && _conflicts.length === ccount ? target.update({
-             tags: tags
-           }) : target;
-         } //  `graph.base()` is the common ancestor of the two graphs.
-         //  `localGraph` contains user's edits up to saving
-         //  `remoteGraph` contains remote edits to modified nodes
-         //  `graph` must be a descendent of `localGraph` and may include
-         //      some conflict resolution actions performed on it.
-         //
-         //                  --- ... --- `localGraph` -- ... -- `graph`
-         //                 /
-         //  `graph.base()` --- ... --- `remoteGraph`
-         //
-
+           var pool = _this.collection;
 
-         var action = function action(graph) {
-           var updates = {
-             replacements: [],
-             removeIds: []
-           };
-           var base = graph.base().entities[id];
-           var local = localGraph.entity(id);
-           var remote = remoteGraph.entity(id);
-           var target = osmEntity(local, {
-             version: remote.version
-           }); // delete/undelete
+           if (Array.isArray(loc)) {
+             var validLocations = _mainLocations.locationsAt(loc);
+             pool = pool.filter(function (a) {
+               return !a.locationSetID || validLocations[a.locationSetID];
+             });
+           }
 
-           if (!remote.visible) {
-             if (_option === 'force_remote') {
-               return actionDeleteMultiple([id])(graph);
-             } else if (_option === 'force_local') {
-               if (target.type === 'way') {
-                 target = mergeChildren(target, utilArrayUniq(local.nodes), updates, graph);
-                 graph = updateChildren(updates, graph);
-               }
+           var searchable = pool.filter(function (a) {
+             return a.searchable !== false && a.suggestion !== true;
+           });
+           var suggestions = pool.filter(function (a) {
+             return a.suggestion === true;
+           }); // matches value to preset.name
 
-               return graph.replace(target);
-             } else {
-               _conflicts.push(_t('merge_remote_changes.conflict.deleted', {
-                 user: user(remote.user)
-               }));
+           var leadingNames = searchable.filter(function (a) {
+             return leading(a.searchName());
+           }).sort(sortPresets('searchName')); // matches value to preset suggestion name
+
+           var leadingSuggestions = suggestions.filter(function (a) {
+             return leadingStrict(a.searchName());
+           }).sort(sortPresets('searchName'));
+           var leadingNamesStripped = searchable.filter(function (a) {
+             return leading(a.searchNameStripped());
+           }).sort(sortPresets('searchNameStripped'));
+           var leadingSuggestionsStripped = suggestions.filter(function (a) {
+             return leadingStrict(a.searchNameStripped());
+           }).sort(sortPresets('searchNameStripped')); // matches value to preset.terms values
+
+           var leadingTerms = searchable.filter(function (a) {
+             return (a.terms() || []).some(leading);
+           });
+           var leadingSuggestionTerms = suggestions.filter(function (a) {
+             return (a.terms() || []).some(leading);
+           }); // matches value to preset.tags values
 
-               return graph; // do nothing
-             }
-           } // merge
+           var leadingTagValues = searchable.filter(function (a) {
+             return Object.values(a.tags || {}).filter(function (val) {
+               return val !== '*';
+             }).some(leading);
+           }); // finds close matches to value in preset.name
 
+           var similarName = searchable.map(function (a) {
+             return {
+               preset: a,
+               dist: utilEditDistance(value, a.searchName())
+             };
+           }).filter(function (a) {
+             return a.dist + Math.min(value.length - a.preset.searchName().length, 0) < 3;
+           }).sort(function (a, b) {
+             return a.dist - b.dist;
+           }).map(function (a) {
+             return a.preset;
+           }); // finds close matches to value to preset suggestion name
 
-           if (target.type === 'node') {
-             target = mergeLocation(remote, target);
-           } else if (target.type === 'way') {
-             // pull in any child nodes that may not be present locally..
-             graph.rebase(remoteGraph.childNodes(remote), [graph], false);
-             target = mergeNodes(base, remote, target);
-             target = mergeChildren(target, utilArrayUnion(local.nodes, remote.nodes), updates, graph);
-           } else if (target.type === 'relation') {
-             target = mergeMembers(remote, target);
-           }
+           var similarSuggestions = suggestions.map(function (a) {
+             return {
+               preset: a,
+               dist: utilEditDistance(value, a.searchName())
+             };
+           }).filter(function (a) {
+             return a.dist + Math.min(value.length - a.preset.searchName().length, 0) < 1;
+           }).sort(function (a, b) {
+             return a.dist - b.dist;
+           }).map(function (a) {
+             return a.preset;
+           }); // finds close matches to value in preset.terms
 
-           target = mergeTags(base, remote, target);
+           var similarTerms = searchable.filter(function (a) {
+             return (a.terms() || []).some(function (b) {
+               return utilEditDistance(value, b) + Math.min(value.length - b.length, 0) < 3;
+             });
+           });
+           var results = leadingNames.concat(leadingSuggestions, leadingNamesStripped, leadingSuggestionsStripped, leadingTerms, leadingSuggestionTerms, leadingTagValues, similarName, similarSuggestions, similarTerms).slice(0, MAXRESULTS - 1);
 
-           if (!_conflicts.length) {
-             graph = updateChildren(updates, graph).replace(target);
+           if (geometry) {
+             if (typeof geometry === 'string') {
+               results.push(_this.fallback(geometry));
+             } else {
+               geometry.forEach(function (geom) {
+                 return results.push(_this.fallback(geom));
+               });
+             }
            }
 
-           return graph;
+           return presetCollection(utilArrayUniq(results));
          };
 
-         action.withOption = function (opt) {
-           _option = opt;
-           return action;
-         };
+         return _this;
+       }
 
-         action.conflicts = function () {
-           return _conflicts;
-         };
+       // `presetCategory` builds a `presetCollection` of member presets,
+       // decorated with some extra methods for searching and matching geometry
+       //
 
-         return action;
-       }
+       function presetCategory(categoryID, category, allPresets) {
+         var _this = Object.assign({}, category); // shallow copy
 
-       // https://github.com/openstreetmap/potlatch2/blob/master/net/systemeD/halcyon/connection/actions/MoveNodeAction.as
 
-       function actionMove(moveIDs, tryDelta, projection, cache) {
-         var _delta = tryDelta;
+         var _searchName; // cache
 
-         function setupCache(graph) {
-           function canMove(nodeID) {
-             // Allow movement of any node that is in the selectedIDs list..
-             if (moveIDs.indexOf(nodeID) !== -1) return true; // Allow movement of a vertex where 2 ways meet..
 
-             var parents = graph.parentWays(graph.entity(nodeID));
-             if (parents.length < 3) return true; // Restrict movement of a vertex where >2 ways meet, unless all parentWays are moving too..
+         var _searchNameStripped; // cache
 
-             var parentsMoving = parents.every(function (way) {
-               return cache.moving[way.id];
-             });
-             if (!parentsMoving) delete cache.moving[nodeID];
-             return parentsMoving;
-           }
 
-           function cacheEntities(ids) {
-             for (var i = 0; i < ids.length; i++) {
-               var id = ids[i];
-               if (cache.moving[id]) continue;
-               cache.moving[id] = true;
-               var entity = graph.hasEntity(id);
-               if (!entity) continue;
+         _this.id = categoryID;
+         _this.members = presetCollection((category.members || []).map(function (presetID) {
+           return allPresets[presetID];
+         }).filter(Boolean));
+         _this.geometry = _this.members.collection.reduce(function (acc, preset) {
+           for (var i in preset.geometry) {
+             var geometry = preset.geometry[i];
 
-               if (entity.type === 'node') {
-                 cache.nodes.push(id);
-                 cache.startLoc[id] = entity.loc;
-               } else if (entity.type === 'way') {
-                 cache.ways.push(id);
-                 cacheEntities(entity.nodes);
-               } else {
-                 cacheEntities(entity.members.map(function (member) {
-                   return member.id;
-                 }));
-               }
+             if (acc.indexOf(geometry) === -1) {
+               acc.push(geometry);
              }
            }
 
-           function cacheIntersections(ids) {
-             function isEndpoint(way, id) {
-               return !way.isClosed() && !!way.affix(id);
-             }
-
-             for (var i = 0; i < ids.length; i++) {
-               var id = ids[i]; // consider only intersections with 1 moved and 1 unmoved way.
-
-               var childNodes = graph.childNodes(graph.entity(id));
+           return acc;
+         }, []);
 
-               for (var j = 0; j < childNodes.length; j++) {
-                 var node = childNodes[j];
-                 var parents = graph.parentWays(node);
-                 if (parents.length !== 2) continue;
-                 var moved = graph.entity(id);
-                 var unmoved = null;
+         _this.matchGeometry = function (geom) {
+           return _this.geometry.indexOf(geom) >= 0;
+         };
 
-                 for (var k = 0; k < parents.length; k++) {
-                   var way = parents[k];
+         _this.matchAllGeometry = function (geometries) {
+           return _this.members.collection.some(function (preset) {
+             return preset.matchAllGeometry(geometries);
+           });
+         };
 
-                   if (!cache.moving[way.id]) {
-                     unmoved = way;
-                     break;
-                   }
-                 }
+         _this.matchScore = function () {
+           return -1;
+         };
 
-                 if (!unmoved) continue; // exclude ways that are overly connected..
+         _this.name = function () {
+           return _t("_tagging.presets.categories.".concat(categoryID, ".name"), {
+             'default': categoryID
+           });
+         };
 
-                 if (utilArrayIntersection(moved.nodes, unmoved.nodes).length > 2) continue;
-                 if (moved.isArea() || unmoved.isArea()) continue;
-                 cache.intersections.push({
-                   nodeId: node.id,
-                   movedId: moved.id,
-                   unmovedId: unmoved.id,
-                   movedIsEP: isEndpoint(moved, node.id),
-                   unmovedIsEP: isEndpoint(unmoved, node.id)
-                 });
-               }
-             }
-           }
+         _this.nameLabel = function () {
+           return _t.html("_tagging.presets.categories.".concat(categoryID, ".name"), {
+             'default': categoryID
+           });
+         };
 
-           if (!cache) {
-             cache = {};
-           }
+         _this.terms = function () {
+           return [];
+         };
 
-           if (!cache.ok) {
-             cache.moving = {};
-             cache.intersections = [];
-             cache.replacedVertex = {};
-             cache.startLoc = {};
-             cache.nodes = [];
-             cache.ways = [];
-             cacheEntities(moveIDs);
-             cacheIntersections(cache.ways);
-             cache.nodes = cache.nodes.filter(canMove);
-             cache.ok = true;
+         _this.searchName = function () {
+           if (!_searchName) {
+             _searchName = (_this.suggestion ? _this.originalName : _this.name()).toLowerCase();
            }
-         } // Place a vertex where the moved vertex used to be, to preserve way shape..
-         //
-         //  Start:
-         //      b ---- e
-         //     / \
-         //    /   \
-         //   /     \
-         //  a       c
-         //
-         //      *               node '*' added to preserve shape
-         //     / \
-         //    /   b ---- e      way `b,e` moved here:
-         //   /     \
-         //  a       c
-         //
-         //
-
-
-         function replaceMovedVertex(nodeId, wayId, graph, delta) {
-           var way = graph.entity(wayId);
-           var moved = graph.entity(nodeId);
-           var movedIndex = way.nodes.indexOf(nodeId);
-           var len, prevIndex, nextIndex;
 
-           if (way.isClosed()) {
-             len = way.nodes.length - 1;
-             prevIndex = (movedIndex + len - 1) % len;
-             nextIndex = (movedIndex + len + 1) % len;
-           } else {
-             len = way.nodes.length;
-             prevIndex = movedIndex - 1;
-             nextIndex = movedIndex + 1;
-           }
+           return _searchName;
+         };
 
-           var prev = graph.hasEntity(way.nodes[prevIndex]);
-           var next = graph.hasEntity(way.nodes[nextIndex]); // Don't add orig vertex at endpoint..
+         _this.searchNameStripped = function () {
+           if (!_searchNameStripped) {
+             _searchNameStripped = _this.searchName(); // split combined diacritical characters into their parts
 
-           if (!prev || !next) return graph;
-           var key = wayId + '_' + nodeId;
-           var orig = cache.replacedVertex[key];
+             if (_searchNameStripped.normalize) _searchNameStripped = _searchNameStripped.normalize('NFD'); // remove diacritics
 
-           if (!orig) {
-             orig = osmNode();
-             cache.replacedVertex[key] = orig;
-             cache.startLoc[orig.id] = cache.startLoc[nodeId];
+             _searchNameStripped = _searchNameStripped.replace(/[\u0300-\u036f]/g, '');
            }
 
-           var start, end;
+           return _searchNameStripped;
+         };
 
-           if (delta) {
-             start = projection(cache.startLoc[nodeId]);
-             end = projection.invert(geoVecAdd(start, delta));
-           } else {
-             end = cache.startLoc[nodeId];
-           }
+         return _this;
+       }
 
-           orig = orig.move(end);
-           var angle = Math.abs(geoAngle(orig, prev, projection) - geoAngle(orig, next, projection)) * 180 / Math.PI; // Don't add orig vertex if it would just make a straight line..
+       // `presetField` decorates a given `field` Object
+       // with some extra methods for searching and matching geometry
+       //
 
-           if (angle > 175 && angle < 185) return graph; // moving forward or backward along way?
+       function presetField(fieldID, field) {
+         var _this = Object.assign({}, field); // shallow copy
 
-           var p1 = [prev.loc, orig.loc, moved.loc, next.loc].map(projection);
-           var p2 = [prev.loc, moved.loc, orig.loc, next.loc].map(projection);
-           var d1 = geoPathLength(p1);
-           var d2 = geoPathLength(p2);
-           var insertAt = d1 <= d2 ? movedIndex : nextIndex; // moving around closed loop?
 
-           if (way.isClosed() && insertAt === 0) insertAt = len;
-           way = way.addNode(orig.id, insertAt);
-           return graph.replace(orig).replace(way);
-         } // Remove duplicate vertex that might have been added by
-         // replaceMovedVertex.  This is done after the unzorro checks.
+         _this.id = fieldID; // for use in classes, element ids, css selectors
 
+         _this.safeid = utilSafeClassName(fieldID);
 
-         function removeDuplicateVertices(wayId, graph) {
-           var way = graph.entity(wayId);
-           var epsilon = 1e-6;
-           var prev, curr;
+         _this.matchGeometry = function (geom) {
+           return !_this.geometry || _this.geometry.indexOf(geom) !== -1;
+         };
 
-           function isInteresting(node, graph) {
-             return graph.parentWays(node).length > 1 || graph.parentRelations(node).length || node.hasInterestingTags();
-           }
+         _this.matchAllGeometry = function (geometries) {
+           return !_this.geometry || geometries.every(function (geom) {
+             return _this.geometry.indexOf(geom) !== -1;
+           });
+         };
 
-           for (var i = 0; i < way.nodes.length; i++) {
-             curr = graph.entity(way.nodes[i]);
+         _this.t = function (scope, options) {
+           return _t("_tagging.presets.fields.".concat(fieldID, ".").concat(scope), options);
+         };
 
-             if (prev && curr && geoVecEqual(prev.loc, curr.loc, epsilon)) {
-               if (!isInteresting(prev, graph)) {
-                 way = way.removeNode(prev.id);
-                 graph = graph.replace(way).remove(prev);
-               } else if (!isInteresting(curr, graph)) {
-                 way = way.removeNode(curr.id);
-                 graph = graph.replace(way).remove(curr);
-               }
-             }
+         _this.t.html = function (scope, options) {
+           return _t.html("_tagging.presets.fields.".concat(fieldID, ".").concat(scope), options);
+         };
 
-             prev = curr;
-           }
+         _this.hasTextForStringId = function (scope) {
+           return _mainLocalizer.hasTextForStringId("_tagging.presets.fields.".concat(fieldID, ".").concat(scope));
+         };
 
-           return graph;
-         } // Reorder nodes around intersections that have moved..
-         //
-         //  Start:                way1.nodes: b,e         (moving)
-         //  a - b - c ----- d     way2.nodes: a,b,c,d     (static)
-         //      |                 vertex: b
-         //      e                 isEP1: true,  isEP2, false
-         //
-         //  way1 `b,e` moved here:
-         //  a ----- c = b - d
-         //              |
-         //              e
-         //
-         //  reorder nodes         way1.nodes: b,e
-         //  a ----- c - b - d     way2.nodes: a,c,b,d
-         //              |
-         //              e
-         //
+         _this.title = function () {
+           return _this.overrideLabel || _this.t('label', {
+             'default': fieldID
+           });
+         };
 
+         _this.label = function () {
+           return _this.overrideLabel || _this.t.html('label', {
+             'default': fieldID
+           });
+         };
 
-         function unZorroIntersection(intersection, graph) {
-           var vertex = graph.entity(intersection.nodeId);
-           var way1 = graph.entity(intersection.movedId);
-           var way2 = graph.entity(intersection.unmovedId);
-           var isEP1 = intersection.movedIsEP;
-           var isEP2 = intersection.unmovedIsEP; // don't move the vertex if it is the endpoint of both ways.
+         var _placeholder = _this.placeholder;
 
-           if (isEP1 && isEP2) return graph;
-           var nodes1 = graph.childNodes(way1).filter(function (n) {
-             return n !== vertex;
-           });
-           var nodes2 = graph.childNodes(way2).filter(function (n) {
-             return n !== vertex;
+         _this.placeholder = function () {
+           return _this.t('placeholder', {
+             'default': _placeholder
            });
-           if (way1.isClosed() && way1.first() === vertex.id) nodes1.push(nodes1[0]);
-           if (way2.isClosed() && way2.first() === vertex.id) nodes2.push(nodes2[0]);
-           var edge1 = !isEP1 && geoChooseEdge(nodes1, projection(vertex.loc), projection);
-           var edge2 = !isEP2 && geoChooseEdge(nodes2, projection(vertex.loc), projection);
-           var loc; // snap vertex to nearest edge (or some point between them)..
+         };
 
-           if (!isEP1 && !isEP2) {
-             var epsilon = 1e-6,
-                 maxIter = 10;
+         _this.originalTerms = (_this.terms || []).join();
 
-             for (var i = 0; i < maxIter; i++) {
-               loc = geoVecInterp(edge1.loc, edge2.loc, 0.5);
-               edge1 = geoChooseEdge(nodes1, projection(loc), projection);
-               edge2 = geoChooseEdge(nodes2, projection(loc), projection);
-               if (Math.abs(edge1.distance - edge2.distance) < epsilon) break;
-             }
-           } else if (!isEP1) {
-             loc = edge1.loc;
-           } else {
-             loc = edge2.loc;
-           }
+         _this.terms = function () {
+           return _this.t('terms', {
+             'default': _this.originalTerms
+           }).toLowerCase().trim().split(/\s*,+\s*/);
+         };
 
-           graph = graph.replace(vertex.move(loc)); // if zorro happened, reorder nodes..
+         _this.increment = _this.type === 'number' ? _this.increment || 1 : undefined;
+         return _this;
+       }
 
-           if (!isEP1 && edge1.index !== way1.nodes.indexOf(vertex.id)) {
-             way1 = way1.removeNode(vertex.id).addNode(vertex.id, edge1.index);
-             graph = graph.replace(way1);
-           }
+       // `Array.prototype.lastIndexOf` method
+       // https://tc39.es/ecma262/#sec-array.prototype.lastindexof
+       // eslint-disable-next-line es/no-array-prototype-lastindexof -- required for testing
+       _export({ target: 'Array', proto: true, forced: arrayLastIndexOf !== [].lastIndexOf }, {
+         lastIndexOf: arrayLastIndexOf
+       });
 
-           if (!isEP2 && edge2.index !== way2.nodes.indexOf(vertex.id)) {
-             way2 = way2.removeNode(vertex.id).addNode(vertex.id, edge2.index);
-             graph = graph.replace(way2);
-           }
+       // `presetPreset` decorates a given `preset` Object
+       // with some extra methods for searching and matching geometry
+       //
 
-           return graph;
-         }
+       function presetPreset(presetID, preset, addable, allFields, allPresets) {
+         allFields = allFields || {};
+         allPresets = allPresets || {};
 
-         function cleanupIntersections(graph) {
-           for (var i = 0; i < cache.intersections.length; i++) {
-             var obj = cache.intersections[i];
-             graph = replaceMovedVertex(obj.nodeId, obj.movedId, graph, _delta);
-             graph = replaceMovedVertex(obj.nodeId, obj.unmovedId, graph, null);
-             graph = unZorroIntersection(obj, graph);
-             graph = removeDuplicateVertices(obj.movedId, graph);
-             graph = removeDuplicateVertices(obj.unmovedId, graph);
-           }
+         var _this = Object.assign({}, preset); // shallow copy
 
-           return graph;
-         } // check if moving way endpoint can cross an unmoved way, if so limit delta..
 
+         var _addable = addable || false;
 
-         function limitDelta(graph) {
-           function moveNode(loc) {
-             return geoVecAdd(projection(loc), _delta);
-           }
+         var _resolvedFields; // cache
 
-           for (var i = 0; i < cache.intersections.length; i++) {
-             var obj = cache.intersections[i]; // Don't limit movement if this is vertex joins 2 endpoints..
 
-             if (obj.movedIsEP && obj.unmovedIsEP) continue; // Don't limit movement if this vertex is not an endpoint anyway..
+         var _resolvedMoreFields; // cache
 
-             if (!obj.movedIsEP) continue;
-             var node = graph.entity(obj.nodeId);
-             var start = projection(node.loc);
-             var end = geoVecAdd(start, _delta);
-             var movedNodes = graph.childNodes(graph.entity(obj.movedId));
-             var movedPath = movedNodes.map(function (n) {
-               return moveNode(n.loc);
-             });
-             var unmovedNodes = graph.childNodes(graph.entity(obj.unmovedId));
-             var unmovedPath = unmovedNodes.map(function (n) {
-               return projection(n.loc);
-             });
-             var hits = geoPathIntersections(movedPath, unmovedPath);
 
-             for (var j = 0; i < hits.length; i++) {
-               if (geoVecEqual(hits[j], end)) continue;
-               var edge = geoChooseEdge(unmovedNodes, end, projection);
-               _delta = geoVecSubtract(projection(edge.loc), start);
-             }
-           }
-         }
+         var _searchName; // cache
 
-         var action = function action(graph) {
-           if (_delta[0] === 0 && _delta[1] === 0) return graph;
-           setupCache(graph);
 
-           if (cache.intersections.length) {
-             limitDelta(graph);
-           }
+         var _searchNameStripped; // cache
 
-           for (var i = 0; i < cache.nodes.length; i++) {
-             var node = graph.entity(cache.nodes[i]);
-             var start = projection(node.loc);
-             var end = geoVecAdd(start, _delta);
-             graph = graph.replace(node.move(projection.invert(end)));
-           }
 
-           if (cache.intersections.length) {
-             graph = cleanupIntersections(graph);
-           }
+         _this.id = presetID;
+         _this.safeid = utilSafeClassName(presetID); // for use in css classes, selectors, element ids
 
-           return graph;
-         };
+         _this.originalTerms = (_this.terms || []).join();
+         _this.originalName = _this.name || '';
+         _this.originalScore = _this.matchScore || 1;
+         _this.originalReference = _this.reference || {};
+         _this.originalFields = _this.fields || [];
+         _this.originalMoreFields = _this.moreFields || [];
 
-         action.delta = function () {
-           return _delta;
+         _this.fields = function () {
+           return _resolvedFields || (_resolvedFields = resolve('fields'));
          };
 
-         return action;
-       }
-
-       function actionMoveMember(relationId, fromIndex, toIndex) {
-         return function (graph) {
-           return graph.replace(graph.entity(relationId).moveMember(fromIndex, toIndex));
+         _this.moreFields = function () {
+           return _resolvedMoreFields || (_resolvedMoreFields = resolve('moreFields'));
          };
-       }
 
-       function actionMoveNode(nodeID, toLoc) {
-         var action = function action(graph, t) {
-           if (t === null || !isFinite(t)) t = 1;
-           t = Math.min(Math.max(+t, 0), 1);
-           var node = graph.entity(nodeID);
-           return graph.replace(node.move(geoVecInterp(node.loc, toLoc, t)));
+         _this.resetFields = function () {
+           return _resolvedFields = _resolvedMoreFields = null;
          };
 
-         action.transitionable = true;
-         return action;
-       }
+         _this.tags = _this.tags || {};
+         _this.addTags = _this.addTags || _this.tags;
+         _this.removeTags = _this.removeTags || _this.addTags;
+         _this.geometry = _this.geometry || [];
 
-       function actionNoop() {
-         return function (graph) {
-           return graph;
+         _this.matchGeometry = function (geom) {
+           return _this.geometry.indexOf(geom) >= 0;
          };
-       }
 
-       function actionOrthogonalize(wayID, projection, vertexID, degThresh, ep) {
-         var epsilon = ep || 1e-4;
-         var threshold = degThresh || 13; // degrees within right or straight to alter
-         // We test normalized dot products so we can compare as cos(angle)
+         _this.matchAllGeometry = function (geoms) {
+           return geoms.every(_this.matchGeometry);
+         };
 
-         var lowerThreshold = Math.cos((90 - threshold) * Math.PI / 180);
-         var upperThreshold = Math.cos(threshold * Math.PI / 180);
+         _this.matchScore = function (entityTags) {
+           var tags = _this.tags;
+           var seen = {};
+           var score = 0; // match on tags
 
-         var action = function action(graph, t) {
-           if (t === null || !isFinite(t)) t = 1;
-           t = Math.min(Math.max(+t, 0), 1);
-           var way = graph.entity(wayID);
-           way = way.removeNode(''); // sanity check - remove any consecutive duplicates
+           for (var k in tags) {
+             seen[k] = true;
 
-           if (way.tags.nonsquare) {
-             var tags = Object.assign({}, way.tags); // since we're squaring, remove indication that this is physically unsquare
+             if (entityTags[k] === tags[k]) {
+               score += _this.originalScore;
+             } else if (tags[k] === '*' && k in entityTags) {
+               score += _this.originalScore / 2;
+             } else {
+               return -1;
+             }
+           } // boost score for additional matches in addTags - #6802
 
-             delete tags.nonsquare;
-             way = way.update({
-               tags: tags
-             });
-           }
 
-           graph = graph.replace(way);
-           var isClosed = way.isClosed();
-           var nodes = graph.childNodes(way).slice(); // shallow copy
+           var addTags = _this.addTags;
 
-           if (isClosed) nodes.pop();
+           for (var _k in addTags) {
+             if (!seen[_k] && entityTags[_k] === addTags[_k]) {
+               score += _this.originalScore;
+             }
+           }
 
-           if (vertexID !== undefined) {
-             nodes = nodeSubset(nodes, vertexID, isClosed);
-             if (nodes.length !== 3) return graph;
-           } // note: all geometry functions here use the unclosed node/point/coord list
+           return score;
+         };
 
+         _this.t = function (scope, options) {
+           var textID = "_tagging.presets.presets.".concat(presetID, ".").concat(scope);
+           return _t(textID, options);
+         };
 
-           var nodeCount = {};
-           var points = [];
-           var corner = {
-             i: 0,
-             dotp: 1
-           };
-           var node, point, loc, score, motions, i, j;
+         _this.t.html = function (scope, options) {
+           var textID = "_tagging.presets.presets.".concat(presetID, ".").concat(scope);
+           return _t.html(textID, options);
+         };
 
-           for (i = 0; i < nodes.length; i++) {
-             node = nodes[i];
-             nodeCount[node.id] = (nodeCount[node.id] || 0) + 1;
-             points.push({
-               id: node.id,
-               coord: projection(node.loc)
-             });
-           }
+         _this.name = function () {
+           return _this.t('name', {
+             'default': _this.originalName
+           });
+         };
 
-           if (points.length === 3) {
-             // move only one vertex for right triangle
-             for (i = 0; i < 1000; i++) {
-               motions = points.map(calcMotion);
-               points[corner.i].coord = geoVecAdd(points[corner.i].coord, motions[corner.i]);
-               score = corner.dotp;
+         _this.nameLabel = function () {
+           return _this.t.html('name', {
+             'default': _this.originalName
+           });
+         };
 
-               if (score < epsilon) {
-                 break;
-               }
-             }
+         _this.subtitle = function () {
+           if (_this.suggestion) {
+             var path = presetID.split('/');
+             path.pop(); // remove brand name
 
-             node = graph.entity(nodes[corner.i].id);
-             loc = projection.invert(points[corner.i].coord);
-             graph = graph.replace(node.move(geoVecInterp(node.loc, loc, t)));
-           } else {
-             var straights = [];
-             var simplified = []; // Remove points from nearly straight sections..
-             // This produces a simplified shape to orthogonalize
+             return _t('_tagging.presets.presets.' + path.join('/') + '.name');
+           }
 
-             for (i = 0; i < points.length; i++) {
-               point = points[i];
-               var dotp = 0;
+           return null;
+         };
 
-               if (isClosed || i > 0 && i < points.length - 1) {
-                 var a = points[(i - 1 + points.length) % points.length];
-                 var b = points[(i + 1) % points.length];
-                 dotp = Math.abs(geoOrthoNormalizedDotProduct(a.coord, b.coord, point.coord));
-               }
+         _this.subtitleLabel = function () {
+           if (_this.suggestion) {
+             var path = presetID.split('/');
+             path.pop(); // remove brand name
 
-               if (dotp > upperThreshold) {
-                 straights.push(point);
-               } else {
-                 simplified.push(point);
-               }
-             } // Orthogonalize the simplified shape
+             return _t.html('_tagging.presets.presets.' + path.join('/') + '.name');
+           }
 
+           return null;
+         };
 
-             var bestPoints = clonePoints(simplified);
-             var originalPoints = clonePoints(simplified);
-             score = Infinity;
+         _this.terms = function () {
+           return _this.t('terms', {
+             'default': _this.originalTerms
+           }).toLowerCase().trim().split(/\s*,+\s*/);
+         };
 
-             for (i = 0; i < 1000; i++) {
-               motions = simplified.map(calcMotion);
+         _this.searchName = function () {
+           if (!_searchName) {
+             _searchName = (_this.suggestion ? _this.originalName : _this.name()).toLowerCase();
+           }
 
-               for (j = 0; j < motions.length; j++) {
-                 simplified[j].coord = geoVecAdd(simplified[j].coord, motions[j]);
-               }
+           return _searchName;
+         };
 
-               var newScore = geoOrthoCalcScore(simplified, isClosed, epsilon, threshold);
+         _this.searchNameStripped = function () {
+           if (!_searchNameStripped) {
+             _searchNameStripped = _this.searchName(); // split combined diacritical characters into their parts
 
-               if (newScore < score) {
-                 bestPoints = clonePoints(simplified);
-                 score = newScore;
-               }
+             if (_searchNameStripped.normalize) _searchNameStripped = _searchNameStripped.normalize('NFD'); // remove diacritics
 
-               if (score < epsilon) {
-                 break;
-               }
-             }
+             _searchNameStripped = _searchNameStripped.replace(/[\u0300-\u036f]/g, '');
+           }
 
-             var bestCoords = bestPoints.map(function (p) {
-               return p.coord;
-             });
-             if (isClosed) bestCoords.push(bestCoords[0]); // move the nodes that should move
+           return _searchNameStripped;
+         };
 
-             for (i = 0; i < bestPoints.length; i++) {
-               point = bestPoints[i];
+         _this.isFallback = function () {
+           var tagCount = Object.keys(_this.tags).length;
+           return tagCount === 0 || tagCount === 1 && _this.tags.hasOwnProperty('area');
+         };
 
-               if (!geoVecEqual(originalPoints[i].coord, point.coord)) {
-                 node = graph.entity(point.id);
-                 loc = projection.invert(point.coord);
-                 graph = graph.replace(node.move(geoVecInterp(node.loc, loc, t)));
-               }
-             } // move the nodes along straight segments
+         _this.addable = function (val) {
+           if (!arguments.length) return _addable;
+           _addable = val;
+           return _this;
+         };
 
+         _this.reference = function () {
+           // Lookup documentation on Wikidata...
+           var qid = _this.tags.wikidata || _this.tags['flag:wikidata'] || _this.tags['brand:wikidata'] || _this.tags['network:wikidata'] || _this.tags['operator:wikidata'];
 
-             for (i = 0; i < straights.length; i++) {
-               point = straights[i];
-               if (nodeCount[point.id] > 1) continue; // skip self-intersections
+           if (qid) {
+             return {
+               qid: qid
+             };
+           } // Lookup documentation on OSM Wikibase...
 
-               node = graph.entity(point.id);
 
-               if (t === 1 && graph.parentWays(node).length === 1 && graph.parentRelations(node).length === 0 && !node.hasInterestingTags()) {
-                 // remove uninteresting points..
-                 graph = actionDeleteNode(node.id)(graph);
-               } else {
-                 // move interesting points to the nearest edge..
-                 var choice = geoVecProject(point.coord, bestCoords);
+           var key = _this.originalReference.key || Object.keys(utilObjectOmit(_this.tags, 'name'))[0];
+           var value = _this.originalReference.value || _this.tags[key];
 
-                 if (choice) {
-                   loc = projection.invert(choice.target);
-                   graph = graph.replace(node.move(geoVecInterp(node.loc, loc, t)));
-                 }
-               }
-             }
+           if (value === '*') {
+             return {
+               key: key
+             };
+           } else {
+             return {
+               key: key,
+               value: value
+             };
            }
+         };
 
-           return graph;
+         _this.unsetTags = function (tags, geometry, ignoringKeys, skipFieldDefaults) {
+           // allow manually keeping some tags
+           var removeTags = ignoringKeys ? utilObjectOmit(_this.removeTags, ignoringKeys) : _this.removeTags;
+           tags = utilObjectOmit(tags, Object.keys(removeTags));
 
-           function clonePoints(array) {
-             return array.map(function (p) {
-               return {
-                 id: p.id,
-                 coord: [p.coord[0], p.coord[1]]
-               };
+           if (geometry && !skipFieldDefaults) {
+             _this.fields().forEach(function (field) {
+               if (field.matchGeometry(geometry) && field.key && field["default"] === tags[field.key]) {
+                 delete tags[field.key];
+               }
              });
            }
 
-           function calcMotion(point, i, array) {
-             // don't try to move the endpoints of a non-closed way.
-             if (!isClosed && (i === 0 || i === array.length - 1)) return [0, 0]; // don't try to move a node that appears more than once (self intersection)
+           delete tags.area;
+           return tags;
+         };
 
-             if (nodeCount[array[i].id] > 1) return [0, 0];
-             var a = array[(i - 1 + array.length) % array.length].coord;
-             var origin = point.coord;
-             var b = array[(i + 1) % array.length].coord;
-             var p = geoVecSubtract(a, origin);
-             var q = geoVecSubtract(b, origin);
-             var scale = 2 * Math.min(geoVecLength(p), geoVecLength(q));
-             p = geoVecNormalize(p);
-             q = geoVecNormalize(q);
-             var dotp = p[0] * q[0] + p[1] * q[1];
-             var val = Math.abs(dotp);
+         _this.setTags = function (tags, geometry, skipFieldDefaults) {
+           var addTags = _this.addTags;
+           tags = Object.assign({}, tags); // shallow copy
 
-             if (val < lowerThreshold) {
-               // nearly orthogonal
-               corner.i = i;
-               corner.dotp = val;
-               var vec = geoVecNormalize(geoVecAdd(p, q));
-               return geoVecScale(vec, 0.1 * dotp * scale);
+           for (var k in addTags) {
+             if (addTags[k] === '*') {
+               // if this tag is ancillary, don't override an existing value since any value is okay
+               if (_this.tags[k] || !tags[k] || tags[k] === 'no') {
+                 tags[k] = 'yes';
+               }
+             } else {
+               tags[k] = addTags[k];
              }
+           } // Add area=yes if necessary.
+           // This is necessary if the geometry is already an area (e.g. user drew an area) AND any of:
+           // 1. chosen preset could be either an area or a line (`barrier=city_wall`)
+           // 2. chosen preset doesn't have a key in osmAreaKeys (`railway=station`)
 
-             return [0, 0]; // do nothing
-           }
-         }; // if we are only orthogonalizing one vertex,
-         // get that vertex and the previous and next
 
+           if (!addTags.hasOwnProperty('area')) {
+             delete tags.area;
 
-         function nodeSubset(nodes, vertexID, isClosed) {
-           var first = isClosed ? 0 : 1;
-           var last = isClosed ? nodes.length : nodes.length - 1;
+             if (geometry === 'area') {
+               var needsAreaTag = true;
 
-           for (var i = first; i < last; i++) {
-             if (nodes[i].id === vertexID) {
-               return [nodes[(i - 1 + nodes.length) % nodes.length], nodes[i], nodes[(i + 1) % nodes.length]];
+               if (_this.geometry.indexOf('line') === -1) {
+                 for (var _k2 in addTags) {
+                   if (_k2 in osmAreaKeys) {
+                     needsAreaTag = false;
+                     break;
+                   }
+                 }
+               }
+
+               if (needsAreaTag) {
+                 tags.area = 'yes';
+               }
              }
            }
 
-           return [];
-         }
-
-         action.disabled = function (graph) {
-           var way = graph.entity(wayID);
-           way = way.removeNode(''); // sanity check - remove any consecutive duplicates
-
-           graph = graph.replace(way);
-           var isClosed = way.isClosed();
-           var nodes = graph.childNodes(way).slice(); // shallow copy
-
-           if (isClosed) nodes.pop();
-           var allowStraightAngles = false;
-
-           if (vertexID !== undefined) {
-             allowStraightAngles = true;
-             nodes = nodeSubset(nodes, vertexID, isClosed);
-             if (nodes.length !== 3) return 'end_vertex';
+           if (geometry && !skipFieldDefaults) {
+             _this.fields().forEach(function (field) {
+               if (field.matchGeometry(geometry) && field.key && !tags[field.key] && field["default"]) {
+                 tags[field.key] = field["default"];
+               }
+             });
            }
 
-           var coords = nodes.map(function (n) {
-             return projection(n.loc);
-           });
-           var score = geoOrthoCanOrthogonalize(coords, isClosed, epsilon, threshold, allowStraightAngles);
+           return tags;
+         }; // For a preset without fields, use the fields of the parent preset.
+         // Replace {preset} placeholders with the fields of the specified presets.
 
-           if (score === null) {
-             return 'not_squarish';
-           } else if (score === 0) {
-             return 'square_enough';
-           } else {
-             return false;
-           }
-         };
 
-         action.transitionable = true;
-         return action;
-       }
+         function resolve(which) {
+           var fieldIDs = which === 'fields' ? _this.originalFields : _this.originalMoreFields;
+           var resolved = [];
+           fieldIDs.forEach(function (fieldID) {
+             var match = fieldID.match(/\{(.*)\}/);
 
-       //
-       // `turn` must be an `osmTurn` object
-       // see osm/intersection.js, pathToTurn()
-       //
-       // This specifies a restriction of type `restriction` when traveling from
-       // `turn.from.way` toward `turn.to.way` via `turn.via.node` OR `turn.via.ways`.
-       // (The action does not check that these entities form a valid intersection.)
-       //
-       // From, to, and via ways should be split before calling this action.
-       // (old versions of the code would split the ways here, but we no longer do it)
-       //
-       // For testing convenience, accepts a restrictionID to assign to the new
-       // relation. Normally, this will be undefined and the relation will
-       // automatically be assigned a new ID.
-       //
+             if (match !== null) {
+               // a presetID wrapped in braces {}
+               resolved = resolved.concat(inheritFields(match[1], which));
+             } else if (allFields[fieldID]) {
+               // a normal fieldID
+               resolved.push(allFields[fieldID]);
+             } else {
+               console.log("Cannot resolve \"".concat(fieldID, "\" found in ").concat(_this.id, ".").concat(which)); // eslint-disable-line no-console
+             }
+           }); // no fields resolved, so use the parent's if possible
 
-       function actionRestrictTurn(turn, restrictionType, restrictionID) {
-         return function (graph) {
-           var fromWay = graph.entity(turn.from.way);
-           var toWay = graph.entity(turn.to.way);
-           var viaNode = turn.via.node && graph.entity(turn.via.node);
-           var viaWays = turn.via.ways && turn.via.ways.map(function (id) {
-             return graph.entity(id);
-           });
-           var members = [];
-           members.push({
-             id: fromWay.id,
-             type: 'way',
-             role: 'from'
-           });
+           if (!resolved.length) {
+             var endIndex = _this.id.lastIndexOf('/');
 
-           if (viaNode) {
-             members.push({
-               id: viaNode.id,
-               type: 'node',
-               role: 'via'
-             });
-           } else if (viaWays) {
-             viaWays.forEach(function (viaWay) {
-               members.push({
-                 id: viaWay.id,
-                 type: 'way',
-                 role: 'via'
-               });
-             });
-           }
+             var parentID = endIndex && _this.id.substring(0, endIndex);
 
-           members.push({
-             id: toWay.id,
-             type: 'way',
-             role: 'to'
-           });
-           return graph.replace(osmRelation({
-             id: restrictionID,
-             tags: {
-               type: 'restriction',
-               restriction: restrictionType
-             },
-             members: members
-           }));
-         };
-       }
+             if (parentID) {
+               resolved = inheritFields(parentID, which);
+             }
+           }
 
-       function actionRevert(id) {
-         var action = function action(graph) {
-           var entity = graph.hasEntity(id),
-               base = graph.base().entities[id];
+           return utilArrayUniq(resolved); // returns an array of fields to inherit from the given presetID, if found
 
-           if (entity && !base) {
-             // entity will be removed..
-             if (entity.type === 'node') {
-               graph.parentWays(entity).forEach(function (parent) {
-                 parent = parent.removeNode(id);
-                 graph = graph.replace(parent);
+           function inheritFields(presetID, which) {
+             var parent = allPresets[presetID];
+             if (!parent) return [];
 
-                 if (parent.isDegenerate()) {
-                   graph = actionDeleteWay(parent.id)(graph);
-                 }
-               });
+             if (which === 'fields') {
+               return parent.fields().filter(shouldInherit);
+             } else if (which === 'moreFields') {
+               return parent.moreFields();
+             } else {
+               return [];
              }
+           } // Skip `fields` for the keys which define the preset.
+           // These are usually `typeCombo` fields like `shop=*`
 
-             graph.parentRelations(entity).forEach(function (parent) {
-               parent = parent.removeMembersWithID(id);
-               graph = graph.replace(parent);
 
-               if (parent.isDegenerate()) {
-                 graph = actionDeleteRelation(parent.id)(graph);
-               }
-             });
+           function shouldInherit(f) {
+             if (f.key && _this.tags[f.key] !== undefined && // inherit anyway if multiple values are allowed or just a checkbox
+             f.type !== 'multiCombo' && f.type !== 'semiCombo' && f.type !== 'manyCombo' && f.type !== 'check') return false;
+             return true;
            }
+         }
 
-           return graph.revert(id);
+         return _this;
+       }
+
+       var _mainPresetIndex = presetIndex(); // singleton
+       // `presetIndex` wraps a `presetCollection`
+       // with methods for loading new data and returning defaults
+       //
+
+       function presetIndex() {
+         var dispatch = dispatch$8('favoritePreset', 'recentsChange');
+         var MAXRECENTS = 30; // seed the preset lists with geometry fallbacks
+
+         var POINT = presetPreset('point', {
+           name: 'Point',
+           tags: {},
+           geometry: ['point', 'vertex'],
+           matchScore: 0.1
+         });
+         var LINE = presetPreset('line', {
+           name: 'Line',
+           tags: {},
+           geometry: ['line'],
+           matchScore: 0.1
+         });
+         var AREA = presetPreset('area', {
+           name: 'Area',
+           tags: {
+             area: 'yes'
+           },
+           geometry: ['area'],
+           matchScore: 0.1
+         });
+         var RELATION = presetPreset('relation', {
+           name: 'Relation',
+           tags: {},
+           geometry: ['relation'],
+           matchScore: 0.1
+         });
+
+         var _this = presetCollection([POINT, LINE, AREA, RELATION]);
+
+         var _presets = {
+           point: POINT,
+           line: LINE,
+           area: AREA,
+           relation: RELATION
+         };
+         var _defaults = {
+           point: presetCollection([POINT]),
+           vertex: presetCollection([POINT]),
+           line: presetCollection([LINE]),
+           area: presetCollection([AREA]),
+           relation: presetCollection([RELATION])
          };
+         var _fields = {};
+         var _categories = {};
+         var _universal = [];
+         var _addablePresetIDs = null; // Set of preset IDs that the user can add
 
-         return action;
-       }
+         var _recents;
 
-       function actionRotate(rotateIds, pivot, angle, projection) {
-         var action = function action(graph) {
-           return graph.update(function (graph) {
-             utilGetAllNodes(rotateIds, graph).forEach(function (node) {
-               var point = geoRotate([projection(node.loc)], angle, pivot)[0];
-               graph = graph.replace(node.move(projection.invert(point)));
-             });
-           });
+         var _favorites; // Index of presets by (geometry, tag key).
+
+
+         var _geometryIndex = {
+           point: {},
+           vertex: {},
+           line: {},
+           area: {},
+           relation: {}
          };
 
-         return action;
-       }
+         var _loadPromise;
 
-       function actionScale(ids, pivotLoc, scaleFactor, projection) {
-         return function (graph) {
-           return graph.update(function (graph) {
-             var point, radial;
-             utilGetAllNodes(ids, graph).forEach(function (node) {
-               point = projection(node.loc);
-               radial = [point[0] - pivotLoc[0], point[1] - pivotLoc[1]];
-               point = [pivotLoc[0] + scaleFactor * radial[0], pivotLoc[1] + scaleFactor * radial[1]];
-               graph = graph.replace(node.move(projection.invert(point)));
+         _this.ensureLoaded = function () {
+           if (_loadPromise) return _loadPromise;
+           return _loadPromise = Promise.all([_mainFileFetcher.get('preset_categories'), _mainFileFetcher.get('preset_defaults'), _mainFileFetcher.get('preset_presets'), _mainFileFetcher.get('preset_fields')]).then(function (vals) {
+             _this.merge({
+               categories: vals[0],
+               defaults: vals[1],
+               presets: vals[2],
+               fields: vals[3]
              });
+
+             osmSetAreaKeys(_this.areaKeys());
+             osmSetPointTags(_this.pointTags());
+             osmSetVertexTags(_this.vertexTags());
            });
-         };
-       }
+         }; // `merge` accepts an object containing new preset data (all properties optional):
+         // {
+         //   fields: {},
+         //   presets: {},
+         //   categories: {},
+         //   defaults: {},
+         //   featureCollection: {}
+         //}
 
-       /* Align nodes along their common axis */
 
-       function actionStraightenNodes(nodeIDs, projection) {
-         function positionAlongWay(a, o, b) {
-           return geoVecDot(a, b, o) / geoVecDot(b, b, o);
-         } // returns the endpoints of the long axis of symmetry of the `points` bounding rect
+         _this.merge = function (d) {
+           var newLocationSets = []; // Merge Fields
 
+           if (d.fields) {
+             Object.keys(d.fields).forEach(function (fieldID) {
+               var f = d.fields[fieldID];
 
-         function getEndpoints(points) {
-           var ssr = geoGetSmallestSurroundingRectangle(points); // Choose line pq = axis of symmetry.
-           // The shape's surrounding rectangle has 2 axes of symmetry.
-           // Snap points to the long axis
+               if (f) {
+                 // add or replace
+                 f = presetField(fieldID, f);
+                 if (f.locationSet) newLocationSets.push(f);
+                 _fields[fieldID] = f;
+               } else {
+                 // remove
+                 delete _fields[fieldID];
+               }
+             });
+           } // Merge Presets
 
-           var p1 = [(ssr.poly[0][0] + ssr.poly[1][0]) / 2, (ssr.poly[0][1] + ssr.poly[1][1]) / 2];
-           var q1 = [(ssr.poly[2][0] + ssr.poly[3][0]) / 2, (ssr.poly[2][1] + ssr.poly[3][1]) / 2];
-           var p2 = [(ssr.poly[3][0] + ssr.poly[4][0]) / 2, (ssr.poly[3][1] + ssr.poly[4][1]) / 2];
-           var q2 = [(ssr.poly[1][0] + ssr.poly[2][0]) / 2, (ssr.poly[1][1] + ssr.poly[2][1]) / 2];
-           var isLong = geoVecLength(p1, q1) > geoVecLength(p2, q2);
 
-           if (isLong) {
-             return [p1, q1];
-           }
+           if (d.presets) {
+             Object.keys(d.presets).forEach(function (presetID) {
+               var p = d.presets[presetID];
 
-           return [p2, q2];
-         }
+               if (p) {
+                 // add or replace
+                 var isAddable = !_addablePresetIDs || _addablePresetIDs.has(presetID);
 
-         var action = function action(graph, t) {
-           if (t === null || !isFinite(t)) t = 1;
-           t = Math.min(Math.max(+t, 0), 1);
-           var nodes = nodeIDs.map(function (id) {
-             return graph.entity(id);
-           });
-           var points = nodes.map(function (n) {
-             return projection(n.loc);
-           });
-           var endpoints = getEndpoints(points);
-           var startPoint = endpoints[0];
-           var endPoint = endpoints[1]; // Move points onto the line connecting the endpoints
+                 p = presetPreset(presetID, p, isAddable, _fields, _presets);
+                 if (p.locationSet) newLocationSets.push(p);
+                 _presets[presetID] = p;
+               } else {
+                 // remove (but not if it's a fallback)
+                 var existing = _presets[presetID];
 
-           for (var i = 0; i < points.length; i++) {
-             var node = nodes[i];
-             var point = points[i];
-             var u = positionAlongWay(point, startPoint, endPoint);
-             var point2 = geoVecInterp(startPoint, endPoint, u);
-             var loc2 = projection.invert(point2);
-             graph = graph.replace(node.move(geoVecInterp(node.loc, loc2, t)));
-           }
+                 if (existing && !existing.isFallback()) {
+                   delete _presets[presetID];
+                 }
+               }
+             });
+           } // Merge Categories
 
-           return graph;
-         };
 
-         action.disabled = function (graph) {
-           var nodes = nodeIDs.map(function (id) {
-             return graph.entity(id);
-           });
-           var points = nodes.map(function (n) {
-             return projection(n.loc);
-           });
-           var endpoints = getEndpoints(points);
-           var startPoint = endpoints[0];
-           var endPoint = endpoints[1];
-           var maxDistance = 0;
+           if (d.categories) {
+             Object.keys(d.categories).forEach(function (categoryID) {
+               var c = d.categories[categoryID];
 
-           for (var i = 0; i < points.length; i++) {
-             var point = points[i];
-             var u = positionAlongWay(point, startPoint, endPoint);
-             var p = geoVecInterp(startPoint, endPoint, u);
-             var dist = geoVecLength(p, point);
+               if (c) {
+                 // add or replace
+                 c = presetCategory(categoryID, c, _presets);
+                 if (c.locationSet) newLocationSets.push(c);
+                 _categories[categoryID] = c;
+               } else {
+                 // remove
+                 delete _categories[categoryID];
+               }
+             });
+           } // Rebuild _this.collection after changing presets and categories
 
-             if (!isNaN(dist) && dist > maxDistance) {
-               maxDistance = dist;
-             }
-           }
 
-           if (maxDistance < 0.0001) {
-             return 'straight_enough';
-           }
-         };
+           _this.collection = Object.values(_presets).concat(Object.values(_categories)); // Merge Defaults
 
-         action.transitionable = true;
-         return action;
-       }
+           if (d.defaults) {
+             Object.keys(d.defaults).forEach(function (geometry) {
+               var def = d.defaults[geometry];
 
-       /*
-        * Based on https://github.com/openstreetmap/potlatch2/net/systemeD/potlatch2/tools/Straighten.as
-        */
+               if (Array.isArray(def)) {
+                 // add or replace
+                 _defaults[geometry] = presetCollection(def.map(function (id) {
+                   return _presets[id] || _categories[id];
+                 }).filter(Boolean));
+               } else {
+                 // remove
+                 delete _defaults[geometry];
+               }
+             });
+           } // Rebuild universal fields array
 
-       function actionStraightenWay(selectedIDs, projection) {
-         function positionAlongWay(a, o, b) {
-           return geoVecDot(a, b, o) / geoVecDot(b, b, o);
-         } // Return all selected ways as a continuous, ordered array of nodes
 
+           _universal = Object.values(_fields).filter(function (field) {
+             return field.universal;
+           }); // Reset all the preset fields - they'll need to be resolved again
 
-         function allNodes(graph) {
-           var nodes = [];
-           var startNodes = [];
-           var endNodes = [];
-           var remainingWays = [];
-           var selectedWays = selectedIDs.filter(function (w) {
-             return graph.entity(w).type === 'way';
-           });
-           var selectedNodes = selectedIDs.filter(function (n) {
-             return graph.entity(n).type === 'node';
-           });
+           Object.values(_presets).forEach(function (preset) {
+             return preset.resetFields();
+           }); // Rebuild geometry index
 
-           for (var i = 0; i < selectedWays.length; i++) {
-             var way = graph.entity(selectedWays[i]);
-             nodes = way.nodes.slice(0);
-             remainingWays.push(nodes);
-             startNodes.push(nodes[0]);
-             endNodes.push(nodes[nodes.length - 1]);
-           } // Remove duplicate end/startNodes (duplicate nodes cannot be at the line end,
-           //   and need to be removed so currNode difference calculation below works)
-           // i.e. ["n-1", "n-1", "n-2"] => ["n-2"]
+           _geometryIndex = {
+             point: {},
+             vertex: {},
+             line: {},
+             area: {},
+             relation: {}
+           };
 
+           _this.collection.forEach(function (preset) {
+             (preset.geometry || []).forEach(function (geometry) {
+               var g = _geometryIndex[geometry];
 
-           startNodes = startNodes.filter(function (n) {
-             return startNodes.indexOf(n) === startNodes.lastIndexOf(n);
-           });
-           endNodes = endNodes.filter(function (n) {
-             return endNodes.indexOf(n) === endNodes.lastIndexOf(n);
-           }); // Choose the initial endpoint to start from
+               for (var key in preset.tags) {
+                 (g[key] = g[key] || []).push(preset);
+               }
+             });
+           }); // Merge Custom Features
 
-           var currNode = utilArrayDifference(startNodes, endNodes).concat(utilArrayDifference(endNodes, startNodes))[0];
-           var nextWay = [];
-           nodes = []; // Create nested function outside of loop to avoid "function in loop" lint error
 
-           var getNextWay = function getNextWay(currNode, remainingWays) {
-             return remainingWays.filter(function (way) {
-               return way[0] === currNode || way[way.length - 1] === currNode;
-             })[0];
-           }; // Add nodes to end of nodes array, until all ways are added
+           if (d.featureCollection && Array.isArray(d.featureCollection.features)) {
+             _mainLocations.mergeCustomGeoJSON(d.featureCollection);
+           } // Resolve all locationSet features.
 
 
-           while (remainingWays.length) {
-             nextWay = getNextWay(currNode, remainingWays);
-             remainingWays = utilArrayDifference(remainingWays, [nextWay]);
+           if (newLocationSets.length) {
+             _mainLocations.mergeLocationSets(newLocationSets);
+           }
 
-             if (nextWay[0] !== currNode) {
-               nextWay.reverse();
+           return _this;
+         };
+
+         _this.match = function (entity, resolver) {
+           return resolver["transient"](entity, 'presetMatch', function () {
+             var geometry = entity.geometry(resolver); // Treat entities on addr:interpolation lines as points, not vertices - #3241
+
+             if (geometry === 'vertex' && entity.isOnAddressLine(resolver)) {
+               geometry = 'point';
              }
 
-             nodes = nodes.concat(nextWay);
-             currNode = nodes[nodes.length - 1];
-           } // If user selected 2 nodes to straighten between, then slice nodes array to those nodes
+             var entityExtent = entity.extent(resolver);
+             return _this.matchTags(entity.tags, geometry, entityExtent.center());
+           });
+         };
 
+         _this.matchTags = function (tags, geometry, loc) {
+           var geometryMatches = _geometryIndex[geometry];
+           var address;
+           var best = -1;
+           var match;
+           var validLocations;
 
-           if (selectedNodes.length === 2) {
-             var startNodeIdx = nodes.indexOf(selectedNodes[0]);
-             var endNodeIdx = nodes.indexOf(selectedNodes[1]);
-             var sortedStartEnd = [startNodeIdx, endNodeIdx];
-             sortedStartEnd.sort(function (a, b) {
-               return a - b;
-             });
-             nodes = nodes.slice(sortedStartEnd[0], sortedStartEnd[1] + 1);
+           if (Array.isArray(loc)) {
+             validLocations = _mainLocations.locationsAt(loc);
            }
 
-           return nodes.map(function (n) {
-             return graph.entity(n);
-           });
-         }
+           for (var k in tags) {
+             // If any part of an address is present, allow fallback to "Address" preset - #4353
+             if (/^addr:/.test(k) && geometryMatches['addr:*']) {
+               address = geometryMatches['addr:*'][0];
+             }
 
-         function shouldKeepNode(node, graph) {
-           return graph.parentWays(node).length > 1 || graph.parentRelations(node).length || node.hasInterestingTags();
-         }
+             var keyMatches = geometryMatches[k];
+             if (!keyMatches) continue;
 
-         var action = function action(graph, t) {
-           if (t === null || !isFinite(t)) t = 1;
-           t = Math.min(Math.max(+t, 0), 1);
-           var nodes = allNodes(graph);
-           var points = nodes.map(function (n) {
-             return projection(n.loc);
-           });
-           var startPoint = points[0];
-           var endPoint = points[points.length - 1];
-           var toDelete = [];
-           var i;
+             for (var i = 0; i < keyMatches.length; i++) {
+               var candidate = keyMatches[i]; // discard candidate preset if location is not valid at `loc`
 
-           for (i = 1; i < points.length - 1; i++) {
-             var node = nodes[i];
-             var point = points[i];
+               if (validLocations && candidate.locationSetID) {
+                 if (!validLocations[candidate.locationSetID]) continue;
+               }
 
-             if (t < 1 || shouldKeepNode(node, graph)) {
-               var u = positionAlongWay(point, startPoint, endPoint);
-               var p = geoVecInterp(startPoint, endPoint, u);
-               var loc2 = projection.invert(p);
-               graph = graph.replace(node.move(geoVecInterp(node.loc, loc2, t)));
-             } else {
-               // safe to delete
-               if (toDelete.indexOf(node) === -1) {
-                 toDelete.push(node);
+               var score = candidate.matchScore(tags);
+
+               if (score > best) {
+                 best = score;
+                 match = candidate;
                }
              }
            }
 
-           for (i = 0; i < toDelete.length; i++) {
-             graph = actionDeleteNode(toDelete[i].id)(graph);
+           if (address && (!match || match.isFallback())) {
+             match = address;
            }
 
-           return graph;
+           return match || _this.fallback(geometry);
          };
 
-         action.disabled = function (graph) {
-           // check way isn't too bendy
-           var nodes = allNodes(graph);
-           var points = nodes.map(function (n) {
-             return projection(n.loc);
+         _this.allowsVertex = function (entity, resolver) {
+           if (entity.type !== 'node') return false;
+           if (Object.keys(entity.tags).length === 0) return true;
+           return resolver["transient"](entity, 'vertexMatch', function () {
+             // address lines allow vertices to act as standalone points
+             if (entity.isOnAddressLine(resolver)) return true;
+             var geometries = osmNodeGeometriesForTags(entity.tags);
+             if (geometries.vertex) return true;
+             if (geometries.point) return false; // allow vertices for unspecified points
+
+             return true;
            });
-           var startPoint = points[0];
-           var endPoint = points[points.length - 1];
-           var threshold = 0.2 * geoVecLength(startPoint, endPoint);
-           var i;
+         }; // Because of the open nature of tagging, iD will never have a complete
+         // list of tags used in OSM, so we want it to have logic like "assume
+         // that a closed way with an amenity tag is an area, unless the amenity
+         // is one of these specific types". This function computes a structure
+         // that allows testing of such conditions, based on the presets designated
+         // as as supporting (or not supporting) the area geometry.
+         //
+         // The returned object L is a keeplist/discardlist of tags. A closed way
+         // with a tag (k, v) is considered to be an area if `k in L && !(v in L[k])`
+         // (see `Way#isArea()`). In other words, the keys of L form the keeplist,
+         // and the subkeys form the discardlist.
 
-           if (threshold === 0) {
-             return 'too_bendy';
-           }
 
-           var maxDistance = 0;
+         _this.areaKeys = function () {
+           // The ignore list is for keys that imply lines. (We always add `area=yes` for exceptions)
+           var ignore = ['barrier', 'highway', 'footway', 'railway', 'junction', 'type'];
+           var areaKeys = {}; // ignore name-suggestion-index and deprecated presets
 
-           for (i = 1; i < points.length - 1; i++) {
-             var point = points[i];
-             var u = positionAlongWay(point, startPoint, endPoint);
-             var p = geoVecInterp(startPoint, endPoint, u);
-             var dist = geoVecLength(p, point); // to bendy if point is off by 20% of total start/end distance in projected space
+           var presets = _this.collection.filter(function (p) {
+             return !p.suggestion && !p.replacement;
+           }); // keeplist
 
-             if (isNaN(dist) || dist > threshold) {
-               return 'too_bendy';
-             } else if (dist > maxDistance) {
-               maxDistance = dist;
-             }
-           }
 
-           var keepingAllNodes = nodes.every(function (node, i) {
-             return i === 0 || i === nodes.length - 1 || shouldKeepNode(node, graph);
-           });
+           presets.forEach(function (p) {
+             var keys = p.tags && Object.keys(p.tags);
+             var key = keys && keys.length && keys[0]; // pick the first tag
 
-           if (maxDistance < 0.0001 && // Allow straightening even if already straight in order to remove extraneous nodes
-           keepingAllNodes) {
-             return 'straight_enough';
-           }
-         };
+             if (!key) return;
+             if (ignore.indexOf(key) !== -1) return;
 
-         action.transitionable = true;
-         return action;
-       }
+             if (p.geometry.indexOf('area') !== -1) {
+               // probably an area..
+               areaKeys[key] = areaKeys[key] || {};
+             }
+           }); // discardlist
 
-       //
-       // `turn` must be an `osmTurn` object with a `restrictionID` property.
-       // see osm/intersection.js, pathToTurn()
-       //
+           presets.forEach(function (p) {
+             var key;
 
-       function actionUnrestrictTurn(turn) {
-         return function (graph) {
-           return actionDeleteRelation(turn.restrictionID)(graph);
+             for (key in p.addTags) {
+               // examine all addTags to get a better sense of what can be tagged on lines - #6800
+               var value = p.addTags[key];
+
+               if (key in areaKeys && // probably an area...
+               p.geometry.indexOf('line') !== -1 && // but sometimes a line
+               value !== '*') {
+                 areaKeys[key][value] = true;
+               }
+             }
+           });
+           return areaKeys;
          };
-       }
 
-       /* Reflect the given area around its axis of symmetry */
+         _this.pointTags = function () {
+           return _this.collection.reduce(function (pointTags, d) {
+             // ignore name-suggestion-index, deprecated, and generic presets
+             if (d.suggestion || d.replacement || d.searchable === false) return pointTags; // only care about the primary tag
 
-       function actionReflect(reflectIds, projection) {
-         var _useLongAxis = true;
+             var keys = d.tags && Object.keys(d.tags);
+             var key = keys && keys.length && keys[0]; // pick the first tag
 
-         var action = function action(graph, t) {
-           if (t === null || !isFinite(t)) t = 1;
-           t = Math.min(Math.max(+t, 0), 1);
-           var nodes = utilGetAllNodes(reflectIds, graph);
-           var points = nodes.map(function (n) {
-             return projection(n.loc);
-           });
-           var ssr = geoGetSmallestSurroundingRectangle(points); // Choose line pq = axis of symmetry.
-           // The shape's surrounding rectangle has 2 axes of symmetry.
-           // Reflect across the longer axis by default.
+             if (!key) return pointTags; // if this can be a point
 
-           var p1 = [(ssr.poly[0][0] + ssr.poly[1][0]) / 2, (ssr.poly[0][1] + ssr.poly[1][1]) / 2];
-           var q1 = [(ssr.poly[2][0] + ssr.poly[3][0]) / 2, (ssr.poly[2][1] + ssr.poly[3][1]) / 2];
-           var p2 = [(ssr.poly[3][0] + ssr.poly[4][0]) / 2, (ssr.poly[3][1] + ssr.poly[4][1]) / 2];
-           var q2 = [(ssr.poly[1][0] + ssr.poly[2][0]) / 2, (ssr.poly[1][1] + ssr.poly[2][1]) / 2];
-           var p, q;
-           var isLong = geoVecLength(p1, q1) > geoVecLength(p2, q2);
+             if (d.geometry.indexOf('point') !== -1) {
+               pointTags[key] = pointTags[key] || {};
+               pointTags[key][d.tags[key]] = true;
+             }
 
-           if (_useLongAxis && isLong || !_useLongAxis && !isLong) {
-             p = p1;
-             q = q1;
-           } else {
-             p = p2;
-             q = q2;
-           } // reflect c across pq
-           // http://math.stackexchange.com/questions/65503/point-reflection-over-a-line
+             return pointTags;
+           }, {});
+         };
 
+         _this.vertexTags = function () {
+           return _this.collection.reduce(function (vertexTags, d) {
+             // ignore name-suggestion-index, deprecated, and generic presets
+             if (d.suggestion || d.replacement || d.searchable === false) return vertexTags; // only care about the primary tag
 
-           var dx = q[0] - p[0];
-           var dy = q[1] - p[1];
-           var a = (dx * dx - dy * dy) / (dx * dx + dy * dy);
-           var b = 2 * dx * dy / (dx * dx + dy * dy);
+             var keys = d.tags && Object.keys(d.tags);
+             var key = keys && keys.length && keys[0]; // pick the first tag
 
-           for (var i = 0; i < nodes.length; i++) {
-             var node = nodes[i];
-             var c = projection(node.loc);
-             var c2 = [a * (c[0] - p[0]) + b * (c[1] - p[1]) + p[0], b * (c[0] - p[0]) - a * (c[1] - p[1]) + p[1]];
-             var loc2 = projection.invert(c2);
-             node = node.move(geoVecInterp(node.loc, loc2, t));
-             graph = graph.replace(node);
-           }
+             if (!key) return vertexTags; // if this can be a vertex
 
-           return graph;
+             if (d.geometry.indexOf('vertex') !== -1) {
+               vertexTags[key] = vertexTags[key] || {};
+               vertexTags[key][d.tags[key]] = true;
+             }
+
+             return vertexTags;
+           }, {});
          };
 
-         action.useLongAxis = function (val) {
-           if (!arguments.length) return _useLongAxis;
-           _useLongAxis = val;
-           return action;
+         _this.field = function (id) {
+           return _fields[id];
          };
 
-         action.transitionable = true;
-         return action;
-       }
+         _this.universal = function () {
+           return _universal;
+         };
 
-       function actionUpgradeTags(entityId, oldTags, replaceTags) {
-         return function (graph) {
-           var entity = graph.entity(entityId);
-           var tags = Object.assign({}, entity.tags); // shallow copy
+         _this.defaults = function (geometry, n, startWithRecents, loc) {
+           var recents = [];
 
-           var transferValue;
-           var semiIndex;
+           if (startWithRecents) {
+             recents = _this.recent().matchGeometry(geometry).collection.slice(0, 4);
+           }
 
-           for (var oldTagKey in oldTags) {
-             if (!(oldTagKey in tags)) continue; // wildcard match
+           var defaults;
 
-             if (oldTags[oldTagKey] === '*') {
-               // note the value since we might need to transfer it
-               transferValue = tags[oldTagKey];
-               delete tags[oldTagKey]; // exact match
-             } else if (oldTags[oldTagKey] === tags[oldTagKey]) {
-               delete tags[oldTagKey]; // match is within semicolon-delimited values
-             } else {
-               var vals = tags[oldTagKey].split(';').filter(Boolean);
-               var oldIndex = vals.indexOf(oldTags[oldTagKey]);
+           if (_addablePresetIDs) {
+             defaults = Array.from(_addablePresetIDs).map(function (id) {
+               var preset = _this.item(id);
 
-               if (vals.length === 1 || oldIndex === -1) {
-                 delete tags[oldTagKey];
-               } else {
-                 if (replaceTags && replaceTags[oldTagKey]) {
-                   // replacing a value within a semicolon-delimited value, note the index
-                   semiIndex = oldIndex;
-                 }
+               if (preset && preset.matchGeometry(geometry)) return preset;
+               return null;
+             }).filter(Boolean);
+           } else {
+             defaults = _defaults[geometry].collection.concat(_this.fallback(geometry));
+           }
 
-                 vals.splice(oldIndex, 1);
-                 tags[oldTagKey] = vals.join(';');
-               }
-             }
+           var result = presetCollection(utilArrayUniq(recents.concat(defaults)).slice(0, n - 1));
+
+           if (Array.isArray(loc)) {
+             var validLocations = _mainLocations.locationsAt(loc);
+             result.collection = result.collection.filter(function (a) {
+               return !a.locationSetID || validLocations[a.locationSetID];
+             });
            }
 
-           if (replaceTags) {
-             for (var replaceKey in replaceTags) {
-               var replaceValue = replaceTags[replaceKey];
+           return result;
+         }; // pass a Set of addable preset ids
 
-               if (replaceValue === '*') {
-                 if (tags[replaceKey] && tags[replaceKey] !== 'no') {
-                   // allow any pre-existing value except `no` (troll tag)
-                   continue;
-                 } else {
-                   // otherwise assume `yes` is okay
-                   tags[replaceKey] = 'yes';
-                 }
-               } else if (replaceValue === '$1') {
-                 tags[replaceKey] = transferValue;
-               } else {
-                 if (tags[replaceKey] && oldTags[replaceKey] && semiIndex !== undefined) {
-                   // don't override preexisting values
-                   var existingVals = tags[replaceKey].split(';').filter(Boolean);
 
-                   if (existingVals.indexOf(replaceValue) === -1) {
-                     existingVals.splice(semiIndex, 0, replaceValue);
-                     tags[replaceKey] = existingVals.join(';');
-                   }
-                 } else {
-                   tags[replaceKey] = replaceValue;
-                 }
-               }
-             }
+         _this.addablePresetIDs = function (val) {
+           if (!arguments.length) return _addablePresetIDs; // accept and convert arrays
+
+           if (Array.isArray(val)) val = new Set(val);
+           _addablePresetIDs = val;
+
+           if (_addablePresetIDs) {
+             // reset all presets
+             _this.collection.forEach(function (p) {
+               // categories aren't addable
+               if (p.addable) p.addable(_addablePresetIDs.has(p.id));
+             });
+           } else {
+             _this.collection.forEach(function (p) {
+               if (p.addable) p.addable(true);
+             });
            }
 
-           return graph.replace(entity.update({
-             tags: tags
-           }));
+           return _this;
          };
-       }
-
-       function behaviorEdit(context) {
-         function behavior() {
-           context.map().minzoom(context.minEditableZoom());
-         }
 
-         behavior.off = function () {
-           context.map().minzoom(0);
+         _this.recent = function () {
+           return presetCollection(utilArrayUniq(_this.getRecents().map(function (d) {
+             return d.preset;
+           })));
          };
 
-         return behavior;
-       }
+         function RibbonItem(preset, source) {
+           var item = {};
+           item.preset = preset;
+           item.source = source;
 
-       /*
-          The hover behavior adds the `.hover` class on pointerover to all elements to which
-          the identical datum is bound, and removes it on pointerout.
+           item.isFavorite = function () {
+             return item.source === 'favorite';
+           };
 
-          The :hover pseudo-class is insufficient for iD's purposes because a datum's visual
-          representation may consist of several elements scattered throughout the DOM hierarchy.
-          Only one of these elements can have the :hover pseudo-class, but all of them will
-          have the .hover class.
-        */
+           item.isRecent = function () {
+             return item.source === 'recent';
+           };
 
-       function behaviorHover(context) {
-         var dispatch$1 = dispatch('hover');
+           item.matches = function (preset) {
+             return item.preset.id === preset.id;
+           };
 
-         var _selection = select(null);
+           item.minified = function () {
+             return {
+               pID: item.preset.id
+             };
+           };
 
-         var _newNodeId = null;
-         var _initialNodeID = null;
+           return item;
+         }
 
-         var _altDisables;
+         function ribbonItemForMinified(d, source) {
+           if (d && d.pID) {
+             var preset = _this.item(d.pID);
 
-         var _ignoreVertex;
+             if (!preset) return null;
+             return RibbonItem(preset, source);
+           }
 
-         var _targets = []; // use pointer events on supported platforms; fallback to mouse events
+           return null;
+         }
 
-         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
+         _this.getGenericRibbonItems = function () {
+           return ['point', 'line', 'area'].map(function (id) {
+             return RibbonItem(_this.item(id), 'generic');
+           });
+         };
 
-         function keydown(d3_event) {
-           if (_altDisables && d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
-             _selection.selectAll('.hover').classed('hover-suppressed', true).classed('hover', false);
+         _this.getAddable = function () {
+           if (!_addablePresetIDs) return [];
+           return _addablePresetIDs.map(function (id) {
+             var preset = _this.item(id);
 
-             _selection.classed('hover-disabled', true);
+             if (preset) return RibbonItem(preset, 'addable');
+             return null;
+           }).filter(Boolean);
+         };
 
-             dispatch$1.call('hover', this, null);
-           }
+         function setRecents(items) {
+           _recents = items;
+           var minifiedItems = items.map(function (d) {
+             return d.minified();
+           });
+           corePreferences('preset_recents', JSON.stringify(minifiedItems));
+           dispatch.call('recentsChange');
          }
 
-         function keyup(d3_event) {
-           if (_altDisables && d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
-             _selection.selectAll('.hover-suppressed').classed('hover-suppressed', false).classed('hover', true);
-
-             _selection.classed('hover-disabled', false);
-
-             dispatch$1.call('hover', this, _targets);
+         _this.getRecents = function () {
+           if (!_recents) {
+             // fetch from local storage
+             _recents = (JSON.parse(corePreferences('preset_recents')) || []).reduce(function (acc, d) {
+               var item = ribbonItemForMinified(d, 'recent');
+               if (item && item.preset.addable()) acc.push(item);
+               return acc;
+             }, []);
            }
-         }
 
-         function behavior(selection) {
-           _selection = selection;
-           _targets = [];
+           return _recents;
+         };
 
-           if (_initialNodeID) {
-             _newNodeId = _initialNodeID;
-             _initialNodeID = null;
-           } else {
-             _newNodeId = null;
-           }
+         _this.addRecent = function (preset, besidePreset, after) {
+           var recents = _this.getRecents();
 
-           _selection.on(_pointerPrefix + 'over.hover', pointerover).on(_pointerPrefix + 'out.hover', pointerout) // treat pointerdown as pointerover for touch devices
-           .on(_pointerPrefix + 'down.hover', pointerover);
+           var beforeItem = _this.recentMatching(besidePreset);
 
-           select(window).on(_pointerPrefix + 'up.hover pointercancel.hover', pointerout, true).on('keydown.hover', keydown).on('keyup.hover', keyup);
+           var toIndex = recents.indexOf(beforeItem);
+           if (after) toIndex += 1;
+           var newItem = RibbonItem(preset, 'recent');
+           recents.splice(toIndex, 0, newItem);
+           setRecents(recents);
+         };
 
-           function eventTarget(d3_event) {
-             var datum = d3_event.target && d3_event.target.__data__;
-             if (_typeof(datum) !== 'object') return null;
+         _this.removeRecent = function (preset) {
+           var item = _this.recentMatching(preset);
 
-             if (!(datum instanceof osmEntity) && datum.properties && datum.properties.entity instanceof osmEntity) {
-               return datum.properties.entity;
-             }
+           if (item) {
+             var items = _this.getRecents();
 
-             return datum;
+             items.splice(items.indexOf(item), 1);
+             setRecents(items);
            }
+         };
 
-           function pointerover(d3_event) {
-             // ignore mouse hovers with buttons pressed unless dragging
-             if (context.mode().id.indexOf('drag') === -1 && (!d3_event.pointerType || d3_event.pointerType === 'mouse') && d3_event.buttons) return;
-             var target = eventTarget(d3_event);
-
-             if (target && _targets.indexOf(target) === -1) {
-               _targets.push(target);
+         _this.recentMatching = function (preset) {
+           var items = _this.getRecents();
 
-               updateHover(d3_event, _targets);
+           for (var i in items) {
+             if (items[i].matches(preset)) {
+               return items[i];
              }
            }
 
-           function pointerout(d3_event) {
-             var target = eventTarget(d3_event);
+           return null;
+         };
 
-             var index = _targets.indexOf(target);
+         _this.moveItem = function (items, fromIndex, toIndex) {
+           if (fromIndex === toIndex || fromIndex < 0 || toIndex < 0 || fromIndex >= items.length || toIndex >= items.length) return null;
+           items.splice(toIndex, 0, items.splice(fromIndex, 1)[0]);
+           return items;
+         };
 
-             if (index !== -1) {
-               _targets.splice(index);
+         _this.moveRecent = function (item, beforeItem) {
+           var recents = _this.getRecents();
 
-               updateHover(d3_event, _targets);
-             }
-           }
+           var fromIndex = recents.indexOf(item);
+           var toIndex = recents.indexOf(beforeItem);
 
-           function allowsVertex(d) {
-             return d.geometry(context.graph()) === 'vertex' || _mainPresetIndex.allowsVertex(d, context.graph());
-           }
+           var items = _this.moveItem(recents, fromIndex, toIndex);
 
-           function modeAllowsHover(target) {
-             var mode = context.mode();
+           if (items) setRecents(items);
+         };
 
-             if (mode.id === 'add-point') {
-               return mode.preset.matchGeometry('vertex') || target.type !== 'way' && target.geometry(context.graph()) !== 'vertex';
-             }
+         _this.setMostRecent = function (preset) {
+           if (preset.searchable === false) return;
 
-             return true;
-           }
+           var items = _this.getRecents();
 
-           function updateHover(d3_event, targets) {
-             _selection.selectAll('.hover').classed('hover', false);
+           var item = _this.recentMatching(preset);
 
-             _selection.selectAll('.hover-suppressed').classed('hover-suppressed', false);
+           if (item) {
+             items.splice(items.indexOf(item), 1);
+           } else {
+             item = RibbonItem(preset, 'recent');
+           } // remove the last recent (first in, first out)
 
-             var mode = context.mode();
 
-             if (!_newNodeId && (mode.id === 'draw-line' || mode.id === 'draw-area')) {
-               var node = targets.find(function (target) {
-                 return target instanceof osmEntity && target.type === 'node';
-               });
-               _newNodeId = node && node.id;
-             }
+           while (items.length >= MAXRECENTS) {
+             items.pop();
+           } // prepend array
 
-             targets = targets.filter(function (datum) {
-               if (datum instanceof osmEntity) {
-                 // If drawing a way, don't hover on a node that was just placed. #3974
-                 return datum.id !== _newNodeId && (datum.type !== 'node' || !_ignoreVertex || allowsVertex(datum)) && modeAllowsHover(datum);
-               }
 
-               return true;
-             });
-             var selector = '';
+           items.unshift(item);
+           setRecents(items);
+         };
 
-             for (var i in targets) {
-               var datum = targets[i]; // What are we hovering over?
+         function setFavorites(items) {
+           _favorites = items;
+           var minifiedItems = items.map(function (d) {
+             return d.minified();
+           });
+           corePreferences('preset_favorites', JSON.stringify(minifiedItems)); // call update
 
-               if (datum.__featurehash__) {
-                 // hovering custom data
-                 selector += ', .data' + datum.__featurehash__;
-               } else if (datum instanceof QAItem) {
-                 selector += ', .' + datum.service + '.itemId-' + datum.id;
-               } else if (datum instanceof osmNote) {
-                 selector += ', .note-' + datum.id;
-               } else if (datum instanceof osmEntity) {
-                 selector += ', .' + datum.id;
+           dispatch.call('favoritePreset');
+         }
 
-                 if (datum.type === 'relation') {
-                   for (var j in datum.members) {
-                     selector += ', .' + datum.members[j].id;
-                   }
-                 }
-               }
-             }
+         _this.addFavorite = function (preset, besidePreset, after) {
+           var favorites = _this.getFavorites();
 
-             var suppressed = _altDisables && d3_event && d3_event.altKey;
+           var beforeItem = _this.favoriteMatching(besidePreset);
 
-             if (selector.trim().length) {
-               // remove the first comma
-               selector = selector.slice(1);
+           var toIndex = favorites.indexOf(beforeItem);
+           if (after) toIndex += 1;
+           var newItem = RibbonItem(preset, 'favorite');
+           favorites.splice(toIndex, 0, newItem);
+           setFavorites(favorites);
+         };
 
-               _selection.selectAll(selector).classed(suppressed ? 'hover-suppressed' : 'hover', true);
-             }
+         _this.toggleFavorite = function (preset) {
+           var favs = _this.getFavorites();
 
-             dispatch$1.call('hover', this, !suppressed && targets);
+           var favorite = _this.favoriteMatching(preset);
+
+           if (favorite) {
+             favs.splice(favs.indexOf(favorite), 1);
+           } else {
+             // only allow 10 favorites
+             if (favs.length === 10) {
+               // remove the last favorite (last in, first out)
+               favs.pop();
+             } // append array
+
+
+             favs.push(RibbonItem(preset, 'favorite'));
            }
-         }
 
-         behavior.off = function (selection) {
-           selection.selectAll('.hover').classed('hover', false);
-           selection.selectAll('.hover-suppressed').classed('hover-suppressed', false);
-           selection.classed('hover-disabled', false);
-           selection.on(_pointerPrefix + 'over.hover', null).on(_pointerPrefix + 'out.hover', null).on(_pointerPrefix + 'down.hover', null);
-           select(window).on(_pointerPrefix + 'up.hover pointercancel.hover', null, true).on('keydown.hover', null).on('keyup.hover', null);
+           setFavorites(favs);
          };
 
-         behavior.altDisables = function (val) {
-           if (!arguments.length) return _altDisables;
-           _altDisables = val;
-           return behavior;
-         };
+         _this.removeFavorite = function (preset) {
+           var item = _this.favoriteMatching(preset);
 
-         behavior.ignoreVertex = function (val) {
-           if (!arguments.length) return _ignoreVertex;
-           _ignoreVertex = val;
-           return behavior;
-         };
+           if (item) {
+             var items = _this.getFavorites();
 
-         behavior.initialNodeID = function (nodeId) {
-           _initialNodeID = nodeId;
-           return behavior;
+             items.splice(items.indexOf(item), 1);
+             setFavorites(items);
+           }
          };
 
-         return utilRebind(behavior, dispatch$1, 'on');
-       }
+         _this.getFavorites = function () {
+           if (!_favorites) {
+             // fetch from local storage
+             var rawFavorites = JSON.parse(corePreferences('preset_favorites'));
 
-       var _disableSpace = false;
-       var _lastSpace = null;
-       function behaviorDraw(context) {
-         var dispatch$1 = dispatch('move', 'down', 'downcancel', 'click', 'clickWay', 'clickNode', 'undo', 'cancel', 'finish');
-         var keybinding = utilKeybinding('draw');
+             if (!rawFavorites) {
+               rawFavorites = [];
+               corePreferences('preset_favorites', JSON.stringify(rawFavorites));
+             }
 
-         var _hover = behaviorHover(context).altDisables(true).ignoreVertex(true).on('hover', context.ui().sidebar.hover);
+             _favorites = rawFavorites.reduce(function (output, d) {
+               var item = ribbonItemForMinified(d, 'favorite');
+               if (item && item.preset.addable()) output.push(item);
+               return output;
+             }, []);
+           }
 
-         var _edit = behaviorEdit(context);
+           return _favorites;
+         };
 
-         var _closeTolerance = 4;
-         var _tolerance = 12;
-         var _mouseLeave = false;
-         var _lastMouse = null;
+         _this.favoriteMatching = function (preset) {
+           var favs = _this.getFavorites();
 
-         var _lastPointerUpEvent;
+           for (var index in favs) {
+             if (favs[index].matches(preset)) {
+               return favs[index];
+             }
+           }
+
+           return null;
+         };
+
+         return utilRebind(_this, dispatch, 'on');
+       }
 
-         var _downPointer; // use pointer events on supported platforms; fallback to mouse events
+       function utilTagText(entity) {
+         var obj = entity && entity.tags || {};
+         return Object.keys(obj).map(function (k) {
+           return k + '=' + obj[k];
+         }).join(', ');
+       }
+       function utilTotalExtent(array, graph) {
+         var extent = geoExtent();
+         var val, entity;
 
+         for (var i = 0; i < array.length; i++) {
+           val = array[i];
+           entity = typeof val === 'string' ? graph.hasEntity(val) : val;
 
-         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse'; // related code
-         // - `mode/drag_node.js` `datum()`
+           if (entity) {
+             extent._extend(entity.extent(graph));
+           }
+         }
 
+         return extent;
+       }
+       function utilTagDiff(oldTags, newTags) {
+         var tagDiff = [];
+         var keys = utilArrayUnion(Object.keys(oldTags), Object.keys(newTags)).sort();
+         keys.forEach(function (k) {
+           var oldVal = oldTags[k];
+           var newVal = newTags[k];
 
-         function datum(d3_event) {
-           var mode = context.mode();
-           var isNote = mode && mode.id.indexOf('note') !== -1;
-           if (d3_event.altKey || isNote) return {};
-           var element;
+           if ((oldVal || oldVal === '') && (newVal === undefined || newVal !== oldVal)) {
+             tagDiff.push({
+               type: '-',
+               key: k,
+               oldVal: oldVal,
+               newVal: newVal,
+               display: '- ' + k + '=' + oldVal
+             });
+           }
 
-           if (d3_event.type === 'keydown') {
-             element = _lastMouse && _lastMouse.target;
-           } else {
-             element = d3_event.target;
-           } // When drawing, snap only to touch targets..
-           // (this excludes area fills and active drawing elements)
+           if ((newVal || newVal === '') && (oldVal === undefined || newVal !== oldVal)) {
+             tagDiff.push({
+               type: '+',
+               key: k,
+               oldVal: oldVal,
+               newVal: newVal,
+               display: '+ ' + k + '=' + newVal
+             });
+           }
+         });
+         return tagDiff;
+       }
+       function utilEntitySelector(ids) {
+         return ids.length ? '.' + ids.join(',.') : 'nothing';
+       } // returns an selector to select entity ids for:
+       //  - entityIDs passed in
+       //  - shallow descendant entityIDs for any of those entities that are relations
 
+       function utilEntityOrMemberSelector(ids, graph) {
+         var seen = new Set(ids);
+         ids.forEach(collectShallowDescendants);
+         return utilEntitySelector(Array.from(seen));
 
-           var d = element.__data__;
-           return d && d.properties && d.properties.target ? d : {};
+         function collectShallowDescendants(id) {
+           var entity = graph.hasEntity(id);
+           if (!entity || entity.type !== 'relation') return;
+           entity.members.map(function (member) {
+             return member.id;
+           }).forEach(function (id) {
+             seen.add(id);
+           });
          }
+       } // returns an selector to select entity ids for:
+       //  - entityIDs passed in
+       //  - deep descendant entityIDs for any of those entities that are relations
 
-         function pointerdown(d3_event) {
-           if (_downPointer) return;
-           var pointerLocGetter = utilFastMouse(this);
-           _downPointer = {
-             id: d3_event.pointerId || 'mouse',
-             pointerLocGetter: pointerLocGetter,
-             downTime: +new Date(),
-             downLoc: pointerLocGetter(d3_event)
-           };
-           dispatch$1.call('down', this, d3_event, datum(d3_event));
-         }
+       function utilEntityOrDeepMemberSelector(ids, graph) {
+         return utilEntitySelector(utilEntityAndDeepMemberIDs(ids, graph));
+       } // returns an selector to select entity ids for:
+       //  - entityIDs passed in
+       //  - deep descendant entityIDs for any of those entities that are relations
 
-         function pointerup(d3_event) {
-           if (!_downPointer || _downPointer.id !== (d3_event.pointerId || 'mouse')) return;
-           var downPointer = _downPointer;
-           _downPointer = null;
-           _lastPointerUpEvent = d3_event;
-           if (downPointer.isCancelled) return;
-           var t2 = +new Date();
-           var p2 = downPointer.pointerLocGetter(d3_event);
-           var dist = geoVecLength(downPointer.downLoc, p2);
+       function utilEntityAndDeepMemberIDs(ids, graph) {
+         var seen = new Set();
+         ids.forEach(collectDeepDescendants);
+         return Array.from(seen);
 
-           if (dist < _closeTolerance || dist < _tolerance && t2 - downPointer.downTime < 500) {
-             // Prevent a quick second click
-             select(window).on('click.draw-block', function () {
-               d3_event.stopPropagation();
-             }, true);
-             context.map().dblclickZoomEnable(false);
-             window.setTimeout(function () {
-               context.map().dblclickZoomEnable(true);
-               select(window).on('click.draw-block', null);
-             }, 500);
-             click(d3_event, p2);
-           }
+         function collectDeepDescendants(id) {
+           if (seen.has(id)) return;
+           seen.add(id);
+           var entity = graph.hasEntity(id);
+           if (!entity || entity.type !== 'relation') return;
+           entity.members.map(function (member) {
+             return member.id;
+           }).forEach(collectDeepDescendants); // recurse
          }
+       } // returns an selector to select entity ids for:
+       //  - deep descendant entityIDs for any of those entities that are relations
 
-         function pointermove(d3_event) {
-           if (_downPointer && _downPointer.id === (d3_event.pointerId || 'mouse') && !_downPointer.isCancelled) {
-             var p2 = _downPointer.pointerLocGetter(d3_event);
+       function utilDeepMemberSelector(ids, graph, skipMultipolgonMembers) {
+         var idsSet = new Set(ids);
+         var seen = new Set();
+         var returners = new Set();
+         ids.forEach(collectDeepDescendants);
+         return utilEntitySelector(Array.from(returners));
 
-             var dist = geoVecLength(_downPointer.downLoc, p2);
+         function collectDeepDescendants(id) {
+           if (seen.has(id)) return;
+           seen.add(id);
 
-             if (dist >= _closeTolerance) {
-               _downPointer.isCancelled = true;
-               dispatch$1.call('downcancel', this);
-             }
+           if (!idsSet.has(id)) {
+             returners.add(id);
            }
 
-           if (d3_event.pointerType && d3_event.pointerType !== 'mouse' || d3_event.buttons || _downPointer) return; // HACK: Mobile Safari likes to send one or more `mouse` type pointermove
-           // events immediately after non-mouse pointerup events; detect and ignore them.
-
-           if (_lastPointerUpEvent && _lastPointerUpEvent.pointerType !== 'mouse' && d3_event.timeStamp - _lastPointerUpEvent.timeStamp < 100) return;
-           _lastMouse = d3_event;
-           dispatch$1.call('move', this, d3_event, datum(d3_event));
+           var entity = graph.hasEntity(id);
+           if (!entity || entity.type !== 'relation') return;
+           if (skipMultipolgonMembers && entity.isMultipolygon()) return;
+           entity.members.map(function (member) {
+             return member.id;
+           }).forEach(collectDeepDescendants); // recurse
          }
+       } // Adds or removes highlight styling for the specified entities
 
-         function pointercancel(d3_event) {
-           if (_downPointer && _downPointer.id === (d3_event.pointerId || 'mouse')) {
-             if (!_downPointer.isCancelled) {
-               dispatch$1.call('downcancel', this);
-             }
+       function utilHighlightEntities(ids, highlighted, context) {
+         context.surface().selectAll(utilEntityOrDeepMemberSelector(ids, context.graph())).classed('highlighted', highlighted);
+       } // returns an Array that is the union of:
+       //  - nodes for any nodeIDs passed in
+       //  - child nodes of any wayIDs passed in
+       //  - descendant member and child nodes of relationIDs passed in
 
-             _downPointer = null;
-           }
-         }
+       function utilGetAllNodes(ids, graph) {
+         var seen = new Set();
+         var nodes = new Set();
+         ids.forEach(collectNodes);
+         return Array.from(nodes);
 
-         function mouseenter() {
-           _mouseLeave = false;
-         }
+         function collectNodes(id) {
+           if (seen.has(id)) return;
+           seen.add(id);
+           var entity = graph.hasEntity(id);
+           if (!entity) return;
 
-         function mouseleave() {
-           _mouseLeave = true;
+           if (entity.type === 'node') {
+             nodes.add(entity);
+           } else if (entity.type === 'way') {
+             entity.nodes.forEach(collectNodes);
+           } else {
+             entity.members.map(function (member) {
+               return member.id;
+             }).forEach(collectNodes); // recurse
+           }
          }
+       }
+       function utilDisplayName(entity) {
+         var localizedNameKey = 'name:' + _mainLocalizer.languageCode().toLowerCase();
+         var name = entity.tags[localizedNameKey] || entity.tags.name || '';
+         if (name) return name;
+         var tags = {
+           direction: entity.tags.direction,
+           from: entity.tags.from,
+           network: entity.tags.cycle_network || entity.tags.network,
+           ref: entity.tags.ref,
+           to: entity.tags.to,
+           via: entity.tags.via
+         };
+         var keyComponents = [];
 
-         function allowsVertex(d) {
-           return d.geometry(context.graph()) === 'vertex' || _mainPresetIndex.allowsVertex(d, context.graph());
-         } // related code
-         // - `mode/drag_node.js`     `doMove()`
-         // - `behavior/draw.js`      `click()`
-         // - `behavior/draw_way.js`  `move()`
+         if (tags.network) {
+           keyComponents.push('network');
+         }
 
+         if (tags.ref) {
+           keyComponents.push('ref');
+         } // Routes may need more disambiguation based on direction or destination
 
-         function click(d3_event, loc) {
-           var d = datum(d3_event);
-           var target = d && d.properties && d.properties.entity;
-           var mode = context.mode();
 
-           if (target && target.type === 'node' && allowsVertex(target)) {
-             // Snap to a node
-             dispatch$1.call('clickNode', this, target, d);
-             return;
-           } else if (target && target.type === 'way' && (mode.id !== 'add-point' || mode.preset.matchGeometry('vertex'))) {
-             // Snap to a way
-             var choice = geoChooseEdge(context.graph().childNodes(target), loc, context.projection, context.activeID());
+         if (entity.tags.route) {
+           if (tags.direction) {
+             keyComponents.push('direction');
+           } else if (tags.from && tags.to) {
+             keyComponents.push('from');
+             keyComponents.push('to');
 
-             if (choice) {
-               var edge = [target.nodes[choice.index - 1], target.nodes[choice.index]];
-               dispatch$1.call('clickWay', this, choice.loc, edge, d);
-               return;
+             if (tags.via) {
+               keyComponents.push('via');
              }
-           } else if (mode.id !== 'add-point' || mode.preset.matchGeometry('point')) {
-             var locLatLng = context.projection.invert(loc);
-             dispatch$1.call('click', this, locLatLng, d);
            }
-         } // treat a spacebar press like a click
+         }
 
+         if (keyComponents.length) {
+           name = _t('inspector.display_name.' + keyComponents.join('_'), tags);
+         }
 
-         function space(d3_event) {
-           d3_event.preventDefault();
-           d3_event.stopPropagation();
-           var currSpace = context.map().mouse();
+         return name;
+       }
+       function utilDisplayNameForPath(entity) {
+         var name = utilDisplayName(entity);
+         var isFirefox = utilDetect().browser.toLowerCase().indexOf('firefox') > -1;
 
-           if (_disableSpace && _lastSpace) {
-             var dist = geoVecLength(_lastSpace, currSpace);
+         if (!isFirefox && name && rtlRegex.test(name)) {
+           name = fixRTLTextForSvg(name);
+         }
 
-             if (dist > _tolerance) {
-               _disableSpace = false;
-             }
-           }
+         return name;
+       }
+       function utilDisplayType(id) {
+         return {
+           n: _t('inspector.node'),
+           w: _t('inspector.way'),
+           r: _t('inspector.relation')
+         }[id.charAt(0)];
+       } // `utilDisplayLabel`
+       // Returns a string suitable for display
+       // By default returns something like name/ref, fallback to preset type, fallback to OSM type
+       //   "Main Street" or "Tertiary Road"
+       // If `verbose=true`, include both preset name and feature name.
+       //   "Tertiary Road Main Street"
+       //
 
-           if (_disableSpace || _mouseLeave || !_lastMouse) return; // user must move mouse or release space bar to allow another click
+       function utilDisplayLabel(entity, graphOrGeometry, verbose) {
+         var result;
+         var displayName = utilDisplayName(entity);
+         var preset = typeof graphOrGeometry === 'string' ? _mainPresetIndex.matchTags(entity.tags, graphOrGeometry) : _mainPresetIndex.match(entity, graphOrGeometry);
+         var presetName = preset && (preset.suggestion ? preset.subtitle() : preset.name());
 
-           _lastSpace = currSpace;
-           _disableSpace = true;
-           select(window).on('keyup.space-block', function () {
-             d3_event.preventDefault();
-             d3_event.stopPropagation();
-             _disableSpace = false;
-             select(window).on('keyup.space-block', null);
-           }); // get the current mouse position
+         if (verbose) {
+           result = [presetName, displayName].filter(Boolean).join(' ');
+         } else {
+           result = displayName || presetName;
+         } // Fallback to the OSM type (node/way/relation)
 
-           var loc = context.map().mouse() || // or the map center if the mouse has never entered the map
-           context.projection(context.map().center());
-           click(d3_event, loc);
-         }
 
-         function backspace(d3_event) {
-           d3_event.preventDefault();
-           dispatch$1.call('undo');
-         }
+         return result || utilDisplayType(entity.id);
+       }
+       function utilEntityRoot(entityType) {
+         return {
+           node: 'n',
+           way: 'w',
+           relation: 'r'
+         }[entityType];
+       } // Returns a single object containing the tags of all the given entities.
+       // Example:
+       // {
+       //   highway: 'service',
+       //   service: 'parking_aisle'
+       // }
+       //           +
+       // {
+       //   highway: 'service',
+       //   service: 'driveway',
+       //   width: '3'
+       // }
+       //           =
+       // {
+       //   highway: 'service',
+       //   service: [ 'driveway', 'parking_aisle' ],
+       //   width: [ '3', undefined ]
+       // }
 
-         function del(d3_event) {
-           d3_event.preventDefault();
-           dispatch$1.call('cancel');
-         }
+       function utilCombinedTags(entityIDs, graph) {
+         var tags = {};
+         var tagCounts = {};
+         var allKeys = new Set();
+         var entities = entityIDs.map(function (entityID) {
+           return graph.hasEntity(entityID);
+         }).filter(Boolean); // gather the aggregate keys
 
-         function ret(d3_event) {
-           d3_event.preventDefault();
-           dispatch$1.call('finish');
-         }
+         entities.forEach(function (entity) {
+           var keys = Object.keys(entity.tags).filter(Boolean);
+           keys.forEach(function (key) {
+             allKeys.add(key);
+           });
+         });
+         entities.forEach(function (entity) {
+           allKeys.forEach(function (key) {
+             var value = entity.tags[key]; // purposely allow `undefined`
 
-         function behavior(selection) {
-           context.install(_hover);
-           context.install(_edit);
-           _downPointer = null;
-           keybinding.on('⌫', backspace).on('⌦', del).on('⎋', ret).on('↩', ret).on('space', space).on('⌥space', space);
-           selection.on('mouseenter.draw', mouseenter).on('mouseleave.draw', mouseleave).on(_pointerPrefix + 'down.draw', pointerdown).on(_pointerPrefix + 'move.draw', pointermove);
-           select(window).on(_pointerPrefix + 'up.draw', pointerup, true).on('pointercancel.draw', pointercancel, true);
-           select(document).call(keybinding);
-           return behavior;
-         }
+             if (!tags.hasOwnProperty(key)) {
+               // first value, set as raw
+               tags[key] = value;
+             } else {
+               if (!Array.isArray(tags[key])) {
+                 if (tags[key] !== value) {
+                   // first alternate value, replace single value with array
+                   tags[key] = [tags[key], value];
+                 }
+               } else {
+                 // type is array
+                 if (tags[key].indexOf(value) === -1) {
+                   // subsequent alternate value, add to array
+                   tags[key].push(value);
+                 }
+               }
+             }
 
-         behavior.off = function (selection) {
-           context.ui().sidebar.hover.cancel();
-           context.uninstall(_hover);
-           context.uninstall(_edit);
-           selection.on('mouseenter.draw', null).on('mouseleave.draw', null).on(_pointerPrefix + 'down.draw', null).on(_pointerPrefix + 'move.draw', null);
-           select(window).on(_pointerPrefix + 'up.draw', null).on('pointercancel.draw', null); // note: keyup.space-block, click.draw-block should remain
+             var tagHash = key + '=' + value;
+             if (!tagCounts[tagHash]) tagCounts[tagHash] = 0;
+             tagCounts[tagHash] += 1;
+           });
+         });
 
-           select(document).call(keybinding.unbind);
-         };
+         for (var key in tags) {
+           if (!Array.isArray(tags[key])) continue; // sort values by frequency then alphabetically
 
-         behavior.hover = function () {
-           return _hover;
-         };
+           tags[key] = tags[key].sort(function (val1, val2) {
+             var key = key; // capture
 
-         return utilRebind(behavior, dispatch$1, 'on');
-       }
+             var count2 = tagCounts[key + '=' + val2];
+             var count1 = tagCounts[key + '=' + val1];
 
-       function initRange(domain, range) {
-         switch (arguments.length) {
-           case 0:
-             break;
+             if (count2 !== count1) {
+               return count2 - count1;
+             }
 
-           case 1:
-             this.range(domain);
-             break;
+             if (val2 && val1) {
+               return val1.localeCompare(val2);
+             }
 
-           default:
-             this.range(range).domain(domain);
-             break;
+             return val1 ? 1 : -1;
+           });
          }
 
-         return this;
+         return tags;
        }
+       function utilStringQs(str) {
+         var i = 0; // advance past any leading '?' or '#' characters
 
-       function constants(x) {
-         return function () {
-           return x;
-         };
-       }
+         while (i < str.length && (str[i] === '?' || str[i] === '#')) {
+           i++;
+         }
 
-       function number$1(x) {
-         return +x;
-       }
+         str = str.slice(i);
+         return str.split('&').reduce(function (obj, pair) {
+           var parts = pair.split('=');
 
-       var unit = [0, 1];
-       function identity$3(x) {
-         return x;
-       }
+           if (parts.length === 2) {
+             obj[parts[0]] = null === parts[1] ? '' : decodeURIComponent(parts[1]);
+           }
 
-       function normalize$1(a, b) {
-         return (b -= a = +a) ? function (x) {
-           return (x - a) / b;
-         } : constants(isNaN(b) ? NaN : 0.5);
+           return obj;
+         }, {});
        }
+       function utilQsString(obj, noencode) {
+         // encode everything except special characters used in certain hash parameters:
+         // "/" in map states, ":", ",", {" and "}" in background
+         function softEncode(s) {
+           return encodeURIComponent(s).replace(/(%2F|%3A|%2C|%7B|%7D)/g, decodeURIComponent);
+         }
 
-       function clamper(a, b) {
-         var t;
-         if (a > b) t = a, a = b, b = t;
-         return function (x) {
-           return Math.max(a, Math.min(b, x));
-         };
-       } // normalize(a, b)(x) takes a domain value x in [a,b] and returns the corresponding parameter t in [0,1].
-       // interpolate(a, b)(t) takes a parameter t in [0,1] and returns the corresponding range value x in [a,b].
+         return Object.keys(obj).sort().map(function (key) {
+           return encodeURIComponent(key) + '=' + (noencode ? softEncode(obj[key]) : encodeURIComponent(obj[key]));
+         }).join('&');
+       }
+       function utilPrefixDOMProperty(property) {
+         var prefixes = ['webkit', 'ms', 'moz', 'o'];
+         var i = -1;
+         var n = prefixes.length;
+         var s = document.body;
+         if (property in s) return property;
+         property = property.substr(0, 1).toUpperCase() + property.substr(1);
 
+         while (++i < n) {
+           if (prefixes[i] + property in s) {
+             return prefixes[i] + property;
+           }
+         }
 
-       function bimap(domain, range, interpolate) {
-         var d0 = domain[0],
-             d1 = domain[1],
-             r0 = range[0],
-             r1 = range[1];
-         if (d1 < d0) d0 = normalize$1(d1, d0), r0 = interpolate(r1, r0);else d0 = normalize$1(d0, d1), r0 = interpolate(r0, r1);
-         return function (x) {
-           return r0(d0(x));
-         };
+         return false;
        }
+       function utilPrefixCSSProperty(property) {
+         var prefixes = ['webkit', 'ms', 'Moz', 'O'];
+         var i = -1;
+         var n = prefixes.length;
+         var s = document.body.style;
 
-       function polymap(domain, range, interpolate) {
-         var j = Math.min(domain.length, range.length) - 1,
-             d = new Array(j),
-             r = new Array(j),
-             i = -1; // Reverse descending domains.
-
-         if (domain[j] < domain[0]) {
-           domain = domain.slice().reverse();
-           range = range.slice().reverse();
+         if (property.toLowerCase() in s) {
+           return property.toLowerCase();
          }
 
-         while (++i < j) {
-           d[i] = normalize$1(domain[i], domain[i + 1]);
-           r[i] = interpolate(range[i], range[i + 1]);
+         while (++i < n) {
+           if (prefixes[i] + property in s) {
+             return '-' + prefixes[i].toLowerCase() + property.replace(/([A-Z])/g, '-$1').toLowerCase();
+           }
          }
 
-         return function (x) {
-           var i = bisectRight(domain, x, 1, j) - 1;
-           return r[i](d[i](x));
-         };
-       }
-
-       function copy(source, target) {
-         return target.domain(source.domain()).range(source.range()).interpolate(source.interpolate()).clamp(source.clamp()).unknown(source.unknown());
+         return false;
        }
-       function transformer$1() {
-         var domain = unit,
-             range = unit,
-             interpolate$1 = interpolate,
-             transform,
-             untransform,
-             unknown,
-             clamp = identity$3,
-             piecewise,
-             output,
-             input;
+       var transformProperty;
+       function utilSetTransform(el, x, y, scale) {
+         var prop = transformProperty = transformProperty || utilPrefixCSSProperty('Transform');
+         var translate = utilDetect().opera ? 'translate(' + x + 'px,' + y + 'px)' : 'translate3d(' + x + 'px,' + y + 'px,0)';
+         return el.style(prop, translate + (scale ? ' scale(' + scale + ')' : ''));
+       } // Calculates Levenshtein distance between two strings
+       // see:  https://en.wikipedia.org/wiki/Levenshtein_distance
+       // first converts the strings to lowercase and replaces diacritic marks with ascii equivalents.
 
-         function rescale() {
-           var n = Math.min(domain.length, range.length);
-           if (clamp !== identity$3) clamp = clamper(domain[0], domain[n - 1]);
-           piecewise = n > 2 ? polymap : bimap;
-           output = input = null;
-           return scale;
-         }
+       function utilEditDistance(a, b) {
+         a = remove$6(a.toLowerCase());
+         b = remove$6(b.toLowerCase());
+         if (a.length === 0) return b.length;
+         if (b.length === 0) return a.length;
+         var matrix = [];
+         var i, j;
 
-         function scale(x) {
-           return isNaN(x = +x) ? unknown : (output || (output = piecewise(domain.map(transform), range, interpolate$1)))(transform(clamp(x)));
+         for (i = 0; i <= b.length; i++) {
+           matrix[i] = [i];
          }
 
-         scale.invert = function (y) {
-           return clamp(untransform((input || (input = piecewise(range, domain.map(transform), d3_interpolateNumber)))(y)));
-         };
-
-         scale.domain = function (_) {
-           return arguments.length ? (domain = Array.from(_, number$1), rescale()) : domain.slice();
-         };
+         for (j = 0; j <= a.length; j++) {
+           matrix[0][j] = j;
+         }
 
-         scale.range = function (_) {
-           return arguments.length ? (range = Array.from(_), rescale()) : range.slice();
-         };
+         for (i = 1; i <= b.length; i++) {
+           for (j = 1; j <= a.length; j++) {
+             if (b.charAt(i - 1) === a.charAt(j - 1)) {
+               matrix[i][j] = matrix[i - 1][j - 1];
+             } else {
+               matrix[i][j] = Math.min(matrix[i - 1][j - 1] + 1, // substitution
+               Math.min(matrix[i][j - 1] + 1, // insertion
+               matrix[i - 1][j] + 1)); // deletion
+             }
+           }
+         }
 
-         scale.rangeRound = function (_) {
-           return range = Array.from(_), interpolate$1 = interpolateRound, rescale();
-         };
+         return matrix[b.length][a.length];
+       } // a d3.mouse-alike which
+       // 1. Only works on HTML elements, not SVG
+       // 2. Does not cause style recalculation
 
-         scale.clamp = function (_) {
-           return arguments.length ? (clamp = _ ? true : identity$3, rescale()) : clamp !== identity$3;
+       function utilFastMouse(container) {
+         var rect = container.getBoundingClientRect();
+         var rectLeft = rect.left;
+         var rectTop = rect.top;
+         var clientLeft = +container.clientLeft;
+         var clientTop = +container.clientTop;
+         return function (e) {
+           return [e.clientX - rectLeft - clientLeft, e.clientY - rectTop - clientTop];
          };
+       }
+       function utilAsyncMap(inputs, func, callback) {
+         var remaining = inputs.length;
+         var results = [];
+         var errors = [];
+         inputs.forEach(function (d, i) {
+           func(d, function done(err, data) {
+             errors[i] = err;
+             results[i] = data;
+             remaining--;
+             if (!remaining) callback(errors, results);
+           });
+         });
+       } // wraps an index to an interval [0..length-1]
 
-         scale.interpolate = function (_) {
-           return arguments.length ? (interpolate$1 = _, rescale()) : interpolate$1;
-         };
+       function utilWrap(index, length) {
+         if (index < 0) {
+           index += Math.ceil(-index / length) * length;
+         }
 
-         scale.unknown = function (_) {
-           return arguments.length ? (unknown = _, scale) : unknown;
-         };
+         return index % length;
+       }
+       /**
+        * a replacement for functor
+        *
+        * @param {*} value any value
+        * @returns {Function} a function that returns that value or the value if it's a function
+        */
 
-         return function (t, u) {
-           transform = t, untransform = u;
-           return rescale();
+       function utilFunctor(value) {
+         if (typeof value === 'function') return value;
+         return function () {
+           return value;
          };
        }
-       function continuous() {
-         return transformer$1()(identity$3, identity$3);
-       }
+       function utilNoAuto(selection) {
+         var isText = selection.size() && selection.node().tagName.toLowerCase() === 'textarea';
+         return selection // assign 'new-password' even for non-password fields to prevent browsers (Chrome) ignoring 'off'
+         .attr('autocomplete', 'new-password').attr('autocorrect', 'off').attr('autocapitalize', 'off').attr('spellcheck', isText ? 'true' : 'false');
+       } // https://stackoverflow.com/questions/194846/is-there-any-kind-of-hash-code-function-in-javascript
+       // https://werxltd.com/wp/2010/05/13/javascript-implementation-of-javas-string-hashcode-method/
 
-       function formatDecimal (x) {
-         return Math.abs(x = Math.round(x)) >= 1e21 ? x.toLocaleString("en").replace(/,/g, "") : x.toString(10);
-       } // Computes the decimal coefficient and exponent of the specified number x with
-       // significant digits p, where x is positive and p is in [1, 21] or undefined.
-       // For example, formatDecimalParts(1.23) returns ["123", 0].
+       function utilHashcode(str) {
+         var hash = 0;
 
-       function formatDecimalParts(x, p) {
-         if ((i = (x = p ? x.toExponential(p - 1) : x.toExponential()).indexOf("e")) < 0) return null; // NaN, ±Infinity
+         if (str.length === 0) {
+           return hash;
+         }
 
-         var i,
-             coefficient = x.slice(0, i); // The string returned by toExponential either has the form \d\.\d+e[-+]\d+
-         // (e.g., 1.2e+3) or the form \de[-+]\d+ (e.g., 1e+3).
+         for (var i = 0; i < str.length; i++) {
+           var _char = str.charCodeAt(i);
 
-         return [coefficient.length > 1 ? coefficient[0] + coefficient.slice(2) : coefficient, +x.slice(i + 1)];
-       }
+           hash = (hash << 5) - hash + _char;
+           hash = hash & hash; // Convert to 32bit integer
+         }
 
-       function exponent (x) {
-         return x = formatDecimalParts(Math.abs(x)), x ? x[1] : NaN;
-       }
+         return hash;
+       } // Returns version of `str` with all runs of special characters replaced by `_`;
+       // suitable for HTML ids, classes, selectors, etc.
 
-       function formatGroup (grouping, thousands) {
-         return function (value, width) {
-           var i = value.length,
-               t = [],
-               j = 0,
-               g = grouping[0],
-               length = 0;
+       function utilSafeClassName(str) {
+         return str.toLowerCase().replace(/[^a-z0-9]+/g, '_');
+       } // Returns string based on `val` that is highly unlikely to collide with an id
+       // used previously or that's present elsewhere in the document. Useful for preventing
+       // browser-provided autofills or when embedding iD on pages with unknown elements.
 
-           while (i > 0 && g > 0) {
-             if (length + g + 1 > width) g = Math.max(1, width - length);
-             t.push(value.substring(i -= g, i + g));
-             if ((length += g + 1) > width) break;
-             g = grouping[j = (j + 1) % grouping.length];
-           }
+       function utilUniqueDomId(val) {
+         return 'ideditor-' + utilSafeClassName(val.toString()) + '-' + new Date().getTime().toString();
+       } // Returns the length of `str` in unicode characters. This can be less than
+       // `String.length()` since a single unicode character can be composed of multiple
+       // JavaScript UTF-16 code units.
 
-           return t.reverse().join(thousands);
-         };
-       }
+       function utilUnicodeCharsCount(str) {
+         // Native ES2015 implementations of `Array.from` split strings into unicode characters
+         return Array.from(str).length;
+       } // Returns a new string representing `str` cut from its start to `limit` length
+       // in unicode characters. Note that this runs the risk of splitting graphemes.
 
-       function formatNumerals (numerals) {
-         return function (value) {
-           return value.replace(/[0-9]/g, function (i) {
-             return numerals[+i];
-           });
-         };
-       }
+       function utilUnicodeCharsTruncated(str, limit) {
+         return Array.from(str).slice(0, limit).join('');
+       } // Variation of d3.json (https://github.com/d3/d3-fetch/blob/master/src/json.js)
 
-       // [[fill]align][sign][symbol][0][width][,][.precision][~][type]
-       var re = /^(?:(.)?([<>=^]))?([+\-( ])?([$#])?(0)?(\d+)?(,)?(\.\d+)?(~)?([a-z%])?$/i;
-       function formatSpecifier(specifier) {
-         if (!(match = re.exec(specifier))) throw new Error("invalid format: " + specifier);
-         var match;
-         return new FormatSpecifier({
-           fill: match[1],
-           align: match[2],
-           sign: match[3],
-           symbol: match[4],
-           zero: match[5],
-           width: match[6],
-           comma: match[7],
-           precision: match[8] && match[8].slice(1),
-           trim: match[9],
-           type: match[10]
+       function utilFetchJson(resourse, init) {
+         return fetch(resourse, init).then(function (response) {
+           // fetch in PhantomJS tests may return ok=false and status=0 even if it's okay
+           if (!response.ok && response.status !== 0 || !response.json) throw new Error(response.status + ' ' + response.statusText);
+           if (response.status === 204 || response.status === 205) return;
+           return response.json();
          });
        }
-       formatSpecifier.prototype = FormatSpecifier.prototype; // instanceof
-
-       function FormatSpecifier(specifier) {
-         this.fill = specifier.fill === undefined ? " " : specifier.fill + "";
-         this.align = specifier.align === undefined ? ">" : specifier.align + "";
-         this.sign = specifier.sign === undefined ? "-" : specifier.sign + "";
-         this.symbol = specifier.symbol === undefined ? "" : specifier.symbol + "";
-         this.zero = !!specifier.zero;
-         this.width = specifier.width === undefined ? undefined : +specifier.width;
-         this.comma = !!specifier.comma;
-         this.precision = specifier.precision === undefined ? undefined : +specifier.precision;
-         this.trim = !!specifier.trim;
-         this.type = specifier.type === undefined ? "" : specifier.type + "";
-       }
-
-       FormatSpecifier.prototype.toString = function () {
-         return this.fill + this.align + this.sign + this.symbol + (this.zero ? "0" : "") + (this.width === undefined ? "" : Math.max(1, this.width | 0)) + (this.comma ? "," : "") + (this.precision === undefined ? "" : "." + Math.max(0, this.precision | 0)) + (this.trim ? "~" : "") + this.type;
-       };
 
-       // Trims insignificant zeros, e.g., replaces 1.2000k with 1.2k.
-       function formatTrim (s) {
-         out: for (var n = s.length, i = 1, i0 = -1, i1; i < n; ++i) {
-           switch (s[i]) {
-             case ".":
-               i0 = i1 = i;
-               break;
+       function osmEntity(attrs) {
+         // For prototypal inheritance.
+         if (this instanceof osmEntity) return; // Create the appropriate subtype.
 
-             case "0":
-               if (i0 === 0) i0 = i;
-               i1 = i;
-               break;
+         if (attrs && attrs.type) {
+           return osmEntity[attrs.type].apply(this, arguments);
+         } else if (attrs && attrs.id) {
+           return osmEntity[osmEntity.id.type(attrs.id)].apply(this, arguments);
+         } // Initialize a generic Entity (used only in tests).
 
-             default:
-               if (!+s[i]) break out;
-               if (i0 > 0) i0 = 0;
-               break;
-           }
-         }
 
-         return i0 > 0 ? s.slice(0, i0) + s.slice(i1 + 1) : s;
+         return new osmEntity().initialize(arguments);
        }
 
-       // `thisNumberValue` abstract operation
-       // https://tc39.es/ecma262/#sec-thisnumbervalue
-       var thisNumberValue = function (value) {
-         if (typeof value != 'number' && classofRaw(value) != 'Number') {
-           throw TypeError('Incorrect invocation');
-         }
-         return +value;
+       osmEntity.id = function (type) {
+         return osmEntity.id.fromOSM(type, osmEntity.id.next[type]--);
        };
 
-       // `String.prototype.repeat` method implementation
-       // https://tc39.es/ecma262/#sec-string.prototype.repeat
-       var stringRepeat = ''.repeat || function repeat(count) {
-         var str = String(requireObjectCoercible(this));
-         var result = '';
-         var n = toInteger(count);
-         if (n < 0 || n == Infinity) throw RangeError('Wrong number of repetitions');
-         for (;n > 0; (n >>>= 1) && (str += str)) if (n & 1) result += str;
-         return result;
+       osmEntity.id.next = {
+         changeset: -1,
+         node: -1,
+         way: -1,
+         relation: -1
        };
 
-       var nativeToFixed = 1.0.toFixed;
-       var floor$6 = Math.floor;
-
-       var pow$2 = function (x, n, acc) {
-         return n === 0 ? acc : n % 2 === 1 ? pow$2(x, n - 1, acc * x) : pow$2(x * x, n / 2, acc);
+       osmEntity.id.fromOSM = function (type, id) {
+         return type[0] + id;
        };
 
-       var log$2 = function (x) {
-         var n = 0;
-         var x2 = x;
-         while (x2 >= 4096) {
-           n += 12;
-           x2 /= 4096;
-         }
-         while (x2 >= 2) {
-           n += 1;
-           x2 /= 2;
-         } return n;
+       osmEntity.id.toOSM = function (id) {
+         return id.slice(1);
        };
 
-       var multiply = function (data, n, c) {
-         var index = -1;
-         var c2 = c;
-         while (++index < 6) {
-           c2 += n * data[index];
-           data[index] = c2 % 1e7;
-           c2 = floor$6(c2 / 1e7);
-         }
-       };
+       osmEntity.id.type = function (id) {
+         return {
+           'c': 'changeset',
+           'n': 'node',
+           'w': 'way',
+           'r': 'relation'
+         }[id[0]];
+       }; // A function suitable for use as the second argument to d3.selection#data().
 
-       var divide = function (data, n) {
-         var index = 6;
-         var c = 0;
-         while (--index >= 0) {
-           c += data[index];
-           data[index] = floor$6(c / n);
-           c = (c % n) * 1e7;
-         }
-       };
 
-       var dataToString = function (data) {
-         var index = 6;
-         var s = '';
-         while (--index >= 0) {
-           if (s !== '' || index === 0 || data[index] !== 0) {
-             var t = String(data[index]);
-             s = s === '' ? t : s + stringRepeat.call('0', 7 - t.length) + t;
-           }
-         } return s;
+       osmEntity.key = function (entity) {
+         return entity.id + 'v' + (entity.v || 0);
        };
 
-       var FORCED$c = nativeToFixed && (
-         0.00008.toFixed(3) !== '0.000' ||
-         0.9.toFixed(0) !== '1' ||
-         1.255.toFixed(2) !== '1.25' ||
-         1000000000000000128.0.toFixed(0) !== '1000000000000000128'
-       ) || !fails(function () {
-         // V8 ~ Android 4.3-
-         nativeToFixed.call({});
-       });
+       var _deprecatedTagValuesByKey;
 
-       // `Number.prototype.toFixed` method
-       // https://tc39.es/ecma262/#sec-number.prototype.tofixed
-       _export({ target: 'Number', proto: true, forced: FORCED$c }, {
-         toFixed: function toFixed(fractionDigits) {
-           var number = thisNumberValue(this);
-           var fractDigits = toInteger(fractionDigits);
-           var data = [0, 0, 0, 0, 0, 0];
-           var sign = '';
-           var result = '0';
-           var e, z, j, k;
+       osmEntity.deprecatedTagValuesByKey = function (dataDeprecated) {
+         if (!_deprecatedTagValuesByKey) {
+           _deprecatedTagValuesByKey = {};
+           dataDeprecated.forEach(function (d) {
+             var oldKeys = Object.keys(d.old);
 
-           if (fractDigits < 0 || fractDigits > 20) throw RangeError('Incorrect fraction digits');
-           // eslint-disable-next-line no-self-compare -- NaN check
-           if (number != number) return 'NaN';
-           if (number <= -1e21 || number >= 1e21) return String(number);
-           if (number < 0) {
-             sign = '-';
-             number = -number;
-           }
-           if (number > 1e-21) {
-             e = log$2(number * pow$2(2, 69, 1)) - 69;
-             z = e < 0 ? number * pow$2(2, -e, 1) : number / pow$2(2, e, 1);
-             z *= 0x10000000000000;
-             e = 52 - e;
-             if (e > 0) {
-               multiply(data, 0, z);
-               j = fractDigits;
-               while (j >= 7) {
-                 multiply(data, 1e7, 0);
-                 j -= 7;
-               }
-               multiply(data, pow$2(10, j, 1), 0);
-               j = e - 1;
-               while (j >= 23) {
-                 divide(data, 1 << 23);
-                 j -= 23;
+             if (oldKeys.length === 1) {
+               var oldKey = oldKeys[0];
+               var oldValue = d.old[oldKey];
+
+               if (oldValue !== '*') {
+                 if (!_deprecatedTagValuesByKey[oldKey]) {
+                   _deprecatedTagValuesByKey[oldKey] = [oldValue];
+                 } else {
+                   _deprecatedTagValuesByKey[oldKey].push(oldValue);
+                 }
                }
-               divide(data, 1 << j);
-               multiply(data, 1, 1);
-               divide(data, 2);
-               result = dataToString(data);
-             } else {
-               multiply(data, 0, z);
-               multiply(data, 1 << -e, 0);
-               result = dataToString(data) + stringRepeat.call('0', fractDigits);
              }
-           }
-           if (fractDigits > 0) {
-             k = result.length;
-             result = sign + (k <= fractDigits
-               ? '0.' + stringRepeat.call('0', fractDigits - k) + result
-               : result.slice(0, k - fractDigits) + '.' + result.slice(k - fractDigits));
-           } else {
-             result = sign + result;
-           } return result;
-         }
-       });
-
-       var nativeToPrecision = 1.0.toPrecision;
-
-       var FORCED$d = fails(function () {
-         // IE7-
-         return nativeToPrecision.call(1, undefined) !== '1';
-       }) || !fails(function () {
-         // V8 ~ Android 4.3-
-         nativeToPrecision.call({});
-       });
-
-       // `Number.prototype.toPrecision` method
-       // https://tc39.es/ecma262/#sec-number.prototype.toprecision
-       _export({ target: 'Number', proto: true, forced: FORCED$d }, {
-         toPrecision: function toPrecision(precision) {
-           return precision === undefined
-             ? nativeToPrecision.call(thisNumberValue(this))
-             : nativeToPrecision.call(thisNumberValue(this), precision);
+           });
          }
-       });
-
-       var prefixExponent;
-       function formatPrefixAuto (x, p) {
-         var d = formatDecimalParts(x, p);
-         if (!d) return x + "";
-         var coefficient = d[0],
-             exponent = d[1],
-             i = exponent - (prefixExponent = Math.max(-8, Math.min(8, Math.floor(exponent / 3))) * 3) + 1,
-             n = coefficient.length;
-         return i === n ? coefficient : i > n ? coefficient + new Array(i - n + 1).join("0") : i > 0 ? coefficient.slice(0, i) + "." + coefficient.slice(i) : "0." + new Array(1 - i).join("0") + formatDecimalParts(x, Math.max(0, p + i - 1))[0]; // less than 1y!
-       }
-
-       function formatRounded (x, p) {
-         var d = formatDecimalParts(x, p);
-         if (!d) return x + "";
-         var coefficient = d[0],
-             exponent = d[1];
-         return exponent < 0 ? "0." + new Array(-exponent).join("0") + coefficient : coefficient.length > exponent + 1 ? coefficient.slice(0, exponent + 1) + "." + coefficient.slice(exponent + 1) : coefficient + new Array(exponent - coefficient.length + 2).join("0");
-       }
 
-       var formatTypes = {
-         "%": function _(x, p) {
-           return (x * 100).toFixed(p);
-         },
-         "b": function b(x) {
-           return Math.round(x).toString(2);
-         },
-         "c": function c(x) {
-           return x + "";
-         },
-         "d": formatDecimal,
-         "e": function e(x, p) {
-           return x.toExponential(p);
-         },
-         "f": function f(x, p) {
-           return x.toFixed(p);
-         },
-         "g": function g(x, p) {
-           return x.toPrecision(p);
-         },
-         "o": function o(x) {
-           return Math.round(x).toString(8);
-         },
-         "p": function p(x, _p) {
-           return formatRounded(x * 100, _p);
-         },
-         "r": formatRounded,
-         "s": formatPrefixAuto,
-         "X": function X(x) {
-           return Math.round(x).toString(16).toUpperCase();
-         },
-         "x": function x(_x) {
-           return Math.round(_x).toString(16);
-         }
+         return _deprecatedTagValuesByKey;
        };
 
-       function identity$4 (x) {
-         return x;
-       }
-
-       var map = Array.prototype.map,
-           prefixes = ["y", "z", "a", "f", "p", "n", "µ", "m", "", "k", "M", "G", "T", "P", "E", "Z", "Y"];
-       function formatLocale (locale) {
-         var group = locale.grouping === undefined || locale.thousands === undefined ? identity$4 : formatGroup(map.call(locale.grouping, Number), locale.thousands + ""),
-             currencyPrefix = locale.currency === undefined ? "" : locale.currency[0] + "",
-             currencySuffix = locale.currency === undefined ? "" : locale.currency[1] + "",
-             decimal = locale.decimal === undefined ? "." : locale.decimal + "",
-             numerals = locale.numerals === undefined ? identity$4 : formatNumerals(map.call(locale.numerals, String)),
-             percent = locale.percent === undefined ? "%" : locale.percent + "",
-             minus = locale.minus === undefined ? "−" : locale.minus + "",
-             nan = locale.nan === undefined ? "NaN" : locale.nan + "";
-
-         function newFormat(specifier) {
-           specifier = formatSpecifier(specifier);
-           var fill = specifier.fill,
-               align = specifier.align,
-               sign = specifier.sign,
-               symbol = specifier.symbol,
-               zero = specifier.zero,
-               width = specifier.width,
-               comma = specifier.comma,
-               precision = specifier.precision,
-               trim = specifier.trim,
-               type = specifier.type; // The "n" type is an alias for ",g".
+       osmEntity.prototype = {
+         tags: {},
+         initialize: function initialize(sources) {
+           for (var i = 0; i < sources.length; ++i) {
+             var source = sources[i];
 
-           if (type === "n") comma = true, type = "g"; // The "" type, and any invalid type, is an alias for ".12~g".
-           else if (!formatTypes[type]) precision === undefined && (precision = 12), trim = true, type = "g"; // If zero fill is specified, padding goes after sign and before digits.
+             for (var prop in source) {
+               if (Object.prototype.hasOwnProperty.call(source, prop)) {
+                 if (source[prop] === undefined) {
+                   delete this[prop];
+                 } else {
+                   this[prop] = source[prop];
+                 }
+               }
+             }
+           }
 
-           if (zero || fill === "0" && align === "=") zero = true, fill = "0", align = "="; // Compute the prefix and suffix.
-           // For SI-prefix, the suffix is lazily computed.
+           if (!this.id && this.type) {
+             this.id = osmEntity.id(this.type);
+           }
 
-           var prefix = symbol === "$" ? currencyPrefix : symbol === "#" && /[boxX]/.test(type) ? "0" + type.toLowerCase() : "",
-               suffix = symbol === "$" ? currencySuffix : /[%p]/.test(type) ? percent : ""; // What format function should we use?
-           // Is this an integer type?
-           // Can this type generate exponential notation?
+           if (!this.hasOwnProperty('visible')) {
+             this.visible = true;
+           }
 
-           var formatType = formatTypes[type],
-               maybeSuffix = /[defgprs%]/.test(type); // Set the default precision if not specified,
-           // or clamp the specified precision to the supported range.
-           // For significant precision, it must be in [1, 21].
-           // For fixed precision, it must be in [0, 20].
+           if (debug) {
+             Object.freeze(this);
+             Object.freeze(this.tags);
+             if (this.loc) Object.freeze(this.loc);
+             if (this.nodes) Object.freeze(this.nodes);
+             if (this.members) Object.freeze(this.members);
+           }
 
-           precision = precision === undefined ? 6 : /[gprs]/.test(type) ? Math.max(1, Math.min(21, precision)) : Math.max(0, Math.min(20, precision));
+           return this;
+         },
+         copy: function copy(resolver, copies) {
+           if (copies[this.id]) return copies[this.id];
+           var copy = osmEntity(this, {
+             id: undefined,
+             user: undefined,
+             version: undefined
+           });
+           copies[this.id] = copy;
+           return copy;
+         },
+         osmId: function osmId() {
+           return osmEntity.id.toOSM(this.id);
+         },
+         isNew: function isNew() {
+           return this.osmId() < 0;
+         },
+         update: function update(attrs) {
+           return osmEntity(this, attrs, {
+             v: 1 + (this.v || 0)
+           });
+         },
+         mergeTags: function mergeTags(tags) {
+           var merged = Object.assign({}, this.tags); // shallow copy
 
-           function format(value) {
-             var valuePrefix = prefix,
-                 valueSuffix = suffix,
-                 i,
-                 n,
-                 c;
+           var changed = false;
 
-             if (type === "c") {
-               valueSuffix = formatType(value) + valueSuffix;
-               value = "";
-             } else {
-               value = +value; // Determine the sign. -0 is not less than 0, but 1 / -0 is!
+           for (var k in tags) {
+             var t1 = merged[k];
+             var t2 = tags[k];
 
-               var valueNegative = value < 0 || 1 / value < 0; // Perform the initial formatting.
+             if (!t1) {
+               changed = true;
+               merged[k] = t2;
+             } else if (t1 !== t2) {
+               changed = true;
+               merged[k] = utilUnicodeCharsTruncated(utilArrayUnion(t1.split(/;\s*/), t2.split(/;\s*/)).join(';'), 255 // avoid exceeding character limit; see also services/osm.js -> maxCharsForTagValue()
+               );
+             }
+           }
 
-               value = isNaN(value) ? nan : formatType(Math.abs(value), precision); // Trim insignificant zeros.
+           return changed ? this.update({
+             tags: merged
+           }) : this;
+         },
+         intersects: function intersects(extent, resolver) {
+           return this.extent(resolver).intersects(extent);
+         },
+         hasNonGeometryTags: function hasNonGeometryTags() {
+           return Object.keys(this.tags).some(function (k) {
+             return k !== 'area';
+           });
+         },
+         hasParentRelations: function hasParentRelations(resolver) {
+           return resolver.parentRelations(this).length > 0;
+         },
+         hasInterestingTags: function hasInterestingTags() {
+           return Object.keys(this.tags).some(osmIsInterestingTag);
+         },
+         isHighwayIntersection: function isHighwayIntersection() {
+           return false;
+         },
+         isDegenerate: function isDegenerate() {
+           return true;
+         },
+         deprecatedTags: function deprecatedTags(dataDeprecated) {
+           var tags = this.tags; // if there are no tags, none can be deprecated
 
-               if (trim) value = formatTrim(value); // If a negative value rounds to zero after formatting, and no explicit positive sign is requested, hide the sign.
+           if (Object.keys(tags).length === 0) return [];
+           var deprecated = [];
+           dataDeprecated.forEach(function (d) {
+             var oldKeys = Object.keys(d.old);
 
-               if (valueNegative && +value === 0 && sign !== "+") valueNegative = false; // Compute the prefix and suffix.
+             if (d.replace) {
+               var hasExistingValues = Object.keys(d.replace).some(function (replaceKey) {
+                 if (!tags[replaceKey] || d.old[replaceKey]) return false;
+                 var replaceValue = d.replace[replaceKey];
+                 if (replaceValue === '*') return false;
+                 if (replaceValue === tags[replaceKey]) return false;
+                 return true;
+               }); // don't flag deprecated tags if the upgrade path would overwrite existing data - #7843
 
-               valuePrefix = (valueNegative ? sign === "(" ? sign : minus : sign === "-" || sign === "(" ? "" : sign) + valuePrefix;
-               valueSuffix = (type === "s" ? prefixes[8 + prefixExponent / 3] : "") + valueSuffix + (valueNegative && sign === "(" ? ")" : ""); // Break the formatted value into the integer “value” part that can be
-               // grouped, and fractional or exponential “suffix” part that is not.
+               if (hasExistingValues) return;
+             }
 
-               if (maybeSuffix) {
-                 i = -1, n = value.length;
+             var matchesDeprecatedTags = oldKeys.every(function (oldKey) {
+               if (!tags[oldKey]) return false;
+               if (d.old[oldKey] === '*') return true;
+               if (d.old[oldKey] === tags[oldKey]) return true;
+               var vals = tags[oldKey].split(';').filter(Boolean);
 
-                 while (++i < n) {
-                   if (c = value.charCodeAt(i), 48 > c || c > 57) {
-                     valueSuffix = (c === 46 ? decimal + value.slice(i + 1) : value.slice(i)) + valueSuffix;
-                     value = value.slice(0, i);
-                     break;
+               if (vals.length === 0) {
+                 return false;
+               } else if (vals.length > 1) {
+                 return vals.indexOf(d.old[oldKey]) !== -1;
+               } else {
+                 if (tags[oldKey] === d.old[oldKey]) {
+                   if (d.replace && d.old[oldKey] === d.replace[oldKey]) {
+                     var replaceKeys = Object.keys(d.replace);
+                     return !replaceKeys.every(function (replaceKey) {
+                       return tags[replaceKey] === d.replace[replaceKey];
+                     });
+                   } else {
+                     return true;
                    }
                  }
                }
-             } // If the fill character is not "0", grouping is applied before padding.
-
 
-             if (comma && !zero) value = group(value, Infinity); // Compute the padding.
+               return false;
+             });
 
-             var length = valuePrefix.length + value.length + valueSuffix.length,
-                 padding = length < width ? new Array(width - length + 1).join(fill) : ""; // If the fill character is "0", grouping is applied after padding.
+             if (matchesDeprecatedTags) {
+               deprecated.push(d);
+             }
+           });
+           return deprecated;
+         }
+       };
 
-             if (comma && zero) value = group(padding + value, padding.length ? width - valueSuffix.length : Infinity), padding = ""; // Reconstruct the final output based on the desired alignment.
+       function osmLanes(entity) {
+         if (entity.type !== 'way') return null;
+         if (!entity.tags.highway) return null;
+         var tags = entity.tags;
+         var isOneWay = entity.isOneWay();
+         var laneCount = getLaneCount(tags, isOneWay);
+         var maxspeed = parseMaxspeed(tags);
+         var laneDirections = parseLaneDirections(tags, isOneWay, laneCount);
+         var forward = laneDirections.forward;
+         var backward = laneDirections.backward;
+         var bothways = laneDirections.bothways; // parse the piped string 'x|y|z' format
 
-             switch (align) {
-               case "<":
-                 value = valuePrefix + value + valueSuffix + padding;
-                 break;
+         var turnLanes = {};
+         turnLanes.unspecified = parseTurnLanes(tags['turn:lanes']);
+         turnLanes.forward = parseTurnLanes(tags['turn:lanes:forward']);
+         turnLanes.backward = parseTurnLanes(tags['turn:lanes:backward']);
+         var maxspeedLanes = {};
+         maxspeedLanes.unspecified = parseMaxspeedLanes(tags['maxspeed:lanes'], maxspeed);
+         maxspeedLanes.forward = parseMaxspeedLanes(tags['maxspeed:lanes:forward'], maxspeed);
+         maxspeedLanes.backward = parseMaxspeedLanes(tags['maxspeed:lanes:backward'], maxspeed);
+         var psvLanes = {};
+         psvLanes.unspecified = parseMiscLanes(tags['psv:lanes']);
+         psvLanes.forward = parseMiscLanes(tags['psv:lanes:forward']);
+         psvLanes.backward = parseMiscLanes(tags['psv:lanes:backward']);
+         var busLanes = {};
+         busLanes.unspecified = parseMiscLanes(tags['bus:lanes']);
+         busLanes.forward = parseMiscLanes(tags['bus:lanes:forward']);
+         busLanes.backward = parseMiscLanes(tags['bus:lanes:backward']);
+         var taxiLanes = {};
+         taxiLanes.unspecified = parseMiscLanes(tags['taxi:lanes']);
+         taxiLanes.forward = parseMiscLanes(tags['taxi:lanes:forward']);
+         taxiLanes.backward = parseMiscLanes(tags['taxi:lanes:backward']);
+         var hovLanes = {};
+         hovLanes.unspecified = parseMiscLanes(tags['hov:lanes']);
+         hovLanes.forward = parseMiscLanes(tags['hov:lanes:forward']);
+         hovLanes.backward = parseMiscLanes(tags['hov:lanes:backward']);
+         var hgvLanes = {};
+         hgvLanes.unspecified = parseMiscLanes(tags['hgv:lanes']);
+         hgvLanes.forward = parseMiscLanes(tags['hgv:lanes:forward']);
+         hgvLanes.backward = parseMiscLanes(tags['hgv:lanes:backward']);
+         var bicyclewayLanes = {};
+         bicyclewayLanes.unspecified = parseBicycleWay(tags['bicycleway:lanes']);
+         bicyclewayLanes.forward = parseBicycleWay(tags['bicycleway:lanes:forward']);
+         bicyclewayLanes.backward = parseBicycleWay(tags['bicycleway:lanes:backward']);
+         var lanesObj = {
+           forward: [],
+           backward: [],
+           unspecified: []
+         }; // map forward/backward/unspecified of each lane type to lanesObj
 
-               case "=":
-                 value = valuePrefix + padding + value + valueSuffix;
-                 break;
+         mapToLanesObj(lanesObj, turnLanes, 'turnLane');
+         mapToLanesObj(lanesObj, maxspeedLanes, 'maxspeed');
+         mapToLanesObj(lanesObj, psvLanes, 'psv');
+         mapToLanesObj(lanesObj, busLanes, 'bus');
+         mapToLanesObj(lanesObj, taxiLanes, 'taxi');
+         mapToLanesObj(lanesObj, hovLanes, 'hov');
+         mapToLanesObj(lanesObj, hgvLanes, 'hgv');
+         mapToLanesObj(lanesObj, bicyclewayLanes, 'bicycleway');
+         return {
+           metadata: {
+             count: laneCount,
+             oneway: isOneWay,
+             forward: forward,
+             backward: backward,
+             bothways: bothways,
+             turnLanes: turnLanes,
+             maxspeed: maxspeed,
+             maxspeedLanes: maxspeedLanes,
+             psvLanes: psvLanes,
+             busLanes: busLanes,
+             taxiLanes: taxiLanes,
+             hovLanes: hovLanes,
+             hgvLanes: hgvLanes,
+             bicyclewayLanes: bicyclewayLanes
+           },
+           lanes: lanesObj
+         };
+       }
 
-               case "^":
-                 value = padding.slice(0, length = padding.length >> 1) + valuePrefix + value + valueSuffix + padding.slice(length);
-                 break;
+       function getLaneCount(tags, isOneWay) {
+         var count;
 
-               default:
-                 value = padding + valuePrefix + value + valueSuffix;
-                 break;
-             }
+         if (tags.lanes) {
+           count = parseInt(tags.lanes, 10);
 
-             return numerals(value);
+           if (count > 0) {
+             return count;
            }
+         }
 
-           format.toString = function () {
-             return specifier + "";
-           };
+         switch (tags.highway) {
+           case 'trunk':
+           case 'motorway':
+             count = isOneWay ? 2 : 4;
+             break;
 
-           return format;
+           default:
+             count = isOneWay ? 1 : 2;
+             break;
          }
 
-         function formatPrefix(specifier, value) {
-           var f = newFormat((specifier = formatSpecifier(specifier), specifier.type = "f", specifier)),
-               e = Math.max(-8, Math.min(8, Math.floor(exponent(value) / 3))) * 3,
-               k = Math.pow(10, -e),
-               prefix = prefixes[8 + e / 3];
-           return function (value) {
-             return f(k * value) + prefix;
-           };
+         return count;
+       }
+
+       function parseMaxspeed(tags) {
+         var maxspeed = tags.maxspeed;
+         if (!maxspeed) return;
+         var maxspeedRegex = /^([0-9][\.0-9]+?)(?:[ ]?(?:km\/h|kmh|kph|mph|knots))?$/;
+         if (!maxspeedRegex.test(maxspeed)) return;
+         return parseInt(maxspeed, 10);
+       }
+
+       function parseLaneDirections(tags, isOneWay, laneCount) {
+         var forward = parseInt(tags['lanes:forward'], 10);
+         var backward = parseInt(tags['lanes:backward'], 10);
+         var bothways = parseInt(tags['lanes:both_ways'], 10) > 0 ? 1 : 0;
+
+         if (parseInt(tags.oneway, 10) === -1) {
+           forward = 0;
+           bothways = 0;
+           backward = laneCount;
+         } else if (isOneWay) {
+           forward = laneCount;
+           bothways = 0;
+           backward = 0;
+         } else if (isNaN(forward) && isNaN(backward)) {
+           backward = Math.floor((laneCount - bothways) / 2);
+           forward = laneCount - bothways - backward;
+         } else if (isNaN(forward)) {
+           if (backward > laneCount - bothways) {
+             backward = laneCount - bothways;
+           }
+
+           forward = laneCount - bothways - backward;
+         } else if (isNaN(backward)) {
+           if (forward > laneCount - bothways) {
+             forward = laneCount - bothways;
+           }
+
+           backward = laneCount - bothways - forward;
          }
 
          return {
-           format: newFormat,
-           formatPrefix: formatPrefix
+           forward: forward,
+           backward: backward,
+           bothways: bothways
          };
        }
 
-       var locale;
-       var format;
-       var formatPrefix;
-       defaultLocale({
-         thousands: ",",
-         grouping: [3],
-         currency: ["$", ""]
-       });
-       function defaultLocale(definition) {
-         locale = formatLocale(definition);
-         format = locale.format;
-         formatPrefix = locale.formatPrefix;
-         return locale;
+       function parseTurnLanes(tag) {
+         if (!tag) return;
+         var validValues = ['left', 'slight_left', 'sharp_left', 'through', 'right', 'slight_right', 'sharp_right', 'reverse', 'merge_to_left', 'merge_to_right', 'none'];
+         return tag.split('|').map(function (s) {
+           if (s === '') s = 'none';
+           return s.split(';').map(function (d) {
+             return validValues.indexOf(d) === -1 ? 'unknown' : d;
+           });
+         });
        }
 
-       function precisionFixed (step) {
-         return Math.max(0, -exponent(Math.abs(step)));
+       function parseMaxspeedLanes(tag, maxspeed) {
+         if (!tag) return;
+         return tag.split('|').map(function (s) {
+           if (s === 'none') return s;
+           var m = parseInt(s, 10);
+           if (s === '' || m === maxspeed) return null;
+           return isNaN(m) ? 'unknown' : m;
+         });
        }
 
-       function precisionPrefix (step, value) {
-         return Math.max(0, Math.max(-8, Math.min(8, Math.floor(exponent(value) / 3))) * 3 - exponent(Math.abs(step)));
+       function parseMiscLanes(tag) {
+         if (!tag) return;
+         var validValues = ['yes', 'no', 'designated'];
+         return tag.split('|').map(function (s) {
+           if (s === '') s = 'no';
+           return validValues.indexOf(s) === -1 ? 'unknown' : s;
+         });
        }
 
-       function precisionRound (step, max) {
-         step = Math.abs(step), max = Math.abs(max) - step;
-         return Math.max(0, exponent(max) - exponent(step)) + 1;
+       function parseBicycleWay(tag) {
+         if (!tag) return;
+         var validValues = ['yes', 'no', 'designated', 'lane'];
+         return tag.split('|').map(function (s) {
+           if (s === '') s = 'no';
+           return validValues.indexOf(s) === -1 ? 'unknown' : s;
+         });
        }
 
-       function tickFormat(start, stop, count, specifier) {
-         var step = tickStep(start, stop, count),
-             precision;
-         specifier = formatSpecifier(specifier == null ? ",f" : specifier);
-
-         switch (specifier.type) {
-           case "s":
-             {
-               var value = Math.max(Math.abs(start), Math.abs(stop));
-               if (specifier.precision == null && !isNaN(precision = precisionPrefix(step, value))) specifier.precision = precision;
-               return formatPrefix(specifier, value);
-             }
+       function mapToLanesObj(lanesObj, data, key) {
+         if (data.forward) {
+           data.forward.forEach(function (l, i) {
+             if (!lanesObj.forward[i]) lanesObj.forward[i] = {};
+             lanesObj.forward[i][key] = l;
+           });
+         }
 
-           case "":
-           case "e":
-           case "g":
-           case "p":
-           case "r":
-             {
-               if (specifier.precision == null && !isNaN(precision = precisionRound(step, Math.max(Math.abs(start), Math.abs(stop))))) specifier.precision = precision - (specifier.type === "e");
-               break;
-             }
+         if (data.backward) {
+           data.backward.forEach(function (l, i) {
+             if (!lanesObj.backward[i]) lanesObj.backward[i] = {};
+             lanesObj.backward[i][key] = l;
+           });
+         }
 
-           case "f":
-           case "%":
-             {
-               if (specifier.precision == null && !isNaN(precision = precisionFixed(step))) specifier.precision = precision - (specifier.type === "%") * 2;
-               break;
-             }
+         if (data.unspecified) {
+           data.unspecified.forEach(function (l, i) {
+             if (!lanesObj.unspecified[i]) lanesObj.unspecified[i] = {};
+             lanesObj.unspecified[i][key] = l;
+           });
          }
+       }
 
-         return format(specifier);
+       function osmWay() {
+         if (!(this instanceof osmWay)) {
+           return new osmWay().initialize(arguments);
+         } else if (arguments.length) {
+           this.initialize(arguments);
+         }
        }
+       osmEntity.way = osmWay;
+       osmWay.prototype = Object.create(osmEntity.prototype);
+       Object.assign(osmWay.prototype, {
+         type: 'way',
+         nodes: [],
+         copy: function copy(resolver, copies) {
+           if (copies[this.id]) return copies[this.id];
+           var copy = osmEntity.prototype.copy.call(this, resolver, copies);
+           var nodes = this.nodes.map(function (id) {
+             return resolver.entity(id).copy(resolver, copies).id;
+           });
+           copy = copy.update({
+             nodes: nodes
+           });
+           copies[this.id] = copy;
+           return copy;
+         },
+         extent: function extent(resolver) {
+           return resolver["transient"](this, 'extent', function () {
+             var extent = geoExtent();
 
-       function linearish(scale) {
-         var domain = scale.domain;
+             for (var i = 0; i < this.nodes.length; i++) {
+               var node = resolver.hasEntity(this.nodes[i]);
 
-         scale.ticks = function (count) {
-           var d = domain();
-           return ticks(d[0], d[d.length - 1], count == null ? 10 : count);
-         };
+               if (node) {
+                 extent._extend(node.extent());
+               }
+             }
 
-         scale.tickFormat = function (count, specifier) {
-           var d = domain();
-           return tickFormat(d[0], d[d.length - 1], count == null ? 10 : count, specifier);
-         };
+             return extent;
+           });
+         },
+         first: function first() {
+           return this.nodes[0];
+         },
+         last: function last() {
+           return this.nodes[this.nodes.length - 1];
+         },
+         contains: function contains(node) {
+           return this.nodes.indexOf(node) >= 0;
+         },
+         affix: function affix(node) {
+           if (this.nodes[0] === node) return 'prefix';
+           if (this.nodes[this.nodes.length - 1] === node) return 'suffix';
+         },
+         layer: function layer() {
+           // explicit layer tag, clamp between -10, 10..
+           if (isFinite(this.tags.layer)) {
+             return Math.max(-10, Math.min(+this.tags.layer, 10));
+           } // implied layer tag..
 
-         scale.nice = function (count) {
-           if (count == null) count = 10;
-           var d = domain();
-           var i0 = 0;
-           var i1 = d.length - 1;
-           var start = d[i0];
-           var stop = d[i1];
-           var prestep;
-           var step;
-           var maxIter = 10;
 
-           if (stop < start) {
-             step = start, start = stop, stop = step;
-             step = i0, i0 = i1, i1 = step;
-           }
+           if (this.tags.covered === 'yes') return -1;
+           if (this.tags.location === 'overground') return 1;
+           if (this.tags.location === 'underground') return -1;
+           if (this.tags.location === 'underwater') return -10;
+           if (this.tags.power === 'line') return 10;
+           if (this.tags.power === 'minor_line') return 10;
+           if (this.tags.aerialway) return 10;
+           if (this.tags.bridge) return 1;
+           if (this.tags.cutting) return -1;
+           if (this.tags.tunnel) return -1;
+           if (this.tags.waterway) return -1;
+           if (this.tags.man_made === 'pipeline') return -10;
+           if (this.tags.boundary) return -10;
+           return 0;
+         },
+         // the approximate width of the line based on its tags except its `width` tag
+         impliedLineWidthMeters: function impliedLineWidthMeters() {
+           var averageWidths = {
+             highway: {
+               // width is for single lane
+               motorway: 5,
+               motorway_link: 5,
+               trunk: 4.5,
+               trunk_link: 4.5,
+               primary: 4,
+               secondary: 4,
+               tertiary: 4,
+               primary_link: 4,
+               secondary_link: 4,
+               tertiary_link: 4,
+               unclassified: 4,
+               road: 4,
+               living_street: 4,
+               bus_guideway: 4,
+               pedestrian: 4,
+               residential: 3.5,
+               service: 3.5,
+               track: 3,
+               cycleway: 2.5,
+               bridleway: 2,
+               corridor: 2,
+               steps: 2,
+               path: 1.5,
+               footway: 1.5
+             },
+             railway: {
+               // width includes ties and rail bed, not just track gauge
+               rail: 2.5,
+               light_rail: 2.5,
+               tram: 2.5,
+               subway: 2.5,
+               monorail: 2.5,
+               funicular: 2.5,
+               disused: 2.5,
+               preserved: 2.5,
+               miniature: 1.5,
+               narrow_gauge: 1.5
+             },
+             waterway: {
+               river: 50,
+               canal: 25,
+               stream: 5,
+               tidal_channel: 5,
+               fish_pass: 2.5,
+               drain: 2.5,
+               ditch: 1.5
+             }
+           };
+
+           for (var key in averageWidths) {
+             if (this.tags[key] && averageWidths[key][this.tags[key]]) {
+               var width = averageWidths[key][this.tags[key]];
 
-           while (maxIter-- > 0) {
-             step = tickIncrement(start, stop, count);
+               if (key === 'highway') {
+                 var laneCount = this.tags.lanes && parseInt(this.tags.lanes, 10);
+                 if (!laneCount) laneCount = this.isOneWay() ? 1 : 2;
+                 return width * laneCount;
+               }
 
-             if (step === prestep) {
-               d[i0] = start;
-               d[i1] = stop;
-               return domain(d);
-             } else if (step > 0) {
-               start = Math.floor(start / step) * step;
-               stop = Math.ceil(stop / step) * step;
-             } else if (step < 0) {
-               start = Math.ceil(start * step) / step;
-               stop = Math.floor(stop * step) / step;
-             } else {
-               break;
+               return width;
              }
-
-             prestep = step;
            }
 
-           return scale;
-         };
-
-         return scale;
-       }
-       function linear$2() {
-         var scale = continuous();
-
-         scale.copy = function () {
-           return copy(scale, linear$2());
-         };
-
-         initRange.apply(scale, arguments);
-         return linearish(scale);
-       }
-
-       var nativeExpm1 = Math.expm1;
-       var exp$1 = Math.exp;
-
-       // `Math.expm1` method implementation
-       // https://tc39.es/ecma262/#sec-math.expm1
-       var mathExpm1 = (!nativeExpm1
-         // Old FF bug
-         || nativeExpm1(10) > 22025.465794806719 || nativeExpm1(10) < 22025.4657948067165168
-         // Tor Browser bug
-         || nativeExpm1(-2e-17) != -2e-17
-       ) ? function expm1(x) {
-         return (x = +x) == 0 ? x : x > -1e-6 && x < 1e-6 ? x + x * x / 2 : exp$1(x) - 1;
-       } : nativeExpm1;
-
-       function quantize() {
-         var x0 = 0,
-             x1 = 1,
-             n = 1,
-             domain = [0.5],
-             range = [0, 1],
-             unknown;
+           return null;
+         },
+         isOneWay: function isOneWay() {
+           // explicit oneway tag..
+           var values = {
+             'yes': true,
+             '1': true,
+             '-1': true,
+             'reversible': true,
+             'alternating': true,
+             'no': false,
+             '0': false
+           };
 
-         function scale(x) {
-           return x <= x ? range[bisectRight(domain, x, 0, n)] : unknown;
-         }
+           if (values[this.tags.oneway] !== undefined) {
+             return values[this.tags.oneway];
+           } // implied oneway tag..
 
-         function rescale() {
-           var i = -1;
-           domain = new Array(n);
 
-           while (++i < n) {
-             domain[i] = ((i + 1) * x1 - (i - n) * x0) / (n + 1);
+           for (var key in this.tags) {
+             if (key in osmOneWayTags && this.tags[key] in osmOneWayTags[key]) {
+               return true;
+             }
            }
 
-           return scale;
-         }
-
-         scale.domain = function (_) {
-           var _ref, _ref2;
-
-           return arguments.length ? ((_ref = _, _ref2 = _slicedToArray(_ref, 2), x0 = _ref2[0], x1 = _ref2[1], _ref), x0 = +x0, x1 = +x1, rescale()) : [x0, x1];
-         };
-
-         scale.range = function (_) {
-           return arguments.length ? (n = (range = Array.from(_)).length - 1, rescale()) : range.slice();
-         };
-
-         scale.invertExtent = function (y) {
-           var i = range.indexOf(y);
-           return i < 0 ? [NaN, NaN] : i < 1 ? [x0, domain[0]] : i >= n ? [domain[n - 1], x1] : [domain[i - 1], domain[i]];
-         };
-
-         scale.unknown = function (_) {
-           return arguments.length ? (unknown = _, scale) : scale;
-         };
-
-         scale.thresholds = function () {
-           return domain.slice();
-         };
-
-         scale.copy = function () {
-           return quantize().domain([x0, x1]).range(range).unknown(unknown);
-         };
-
-         return initRange.apply(linearish(scale), arguments);
-       }
-
-       // https://github.com/tc39/proposal-string-pad-start-end
-
-
+           return false;
+         },
+         // Some identifier for tag that implies that this way is "sided",
+         // i.e. the right side is the 'inside' (e.g. the right side of a
+         // natural=cliff is lower).
+         sidednessIdentifier: function sidednessIdentifier() {
+           for (var key in this.tags) {
+             var value = this.tags[key];
 
+             if (key in osmRightSideIsInsideTags && value in osmRightSideIsInsideTags[key]) {
+               if (osmRightSideIsInsideTags[key][value] === true) {
+                 return key;
+               } else {
+                 // if the map's value is something other than a
+                 // literal true, we should use it so we can
+                 // special case some keys (e.g. natural=coastline
+                 // is handled differently to other naturals).
+                 return osmRightSideIsInsideTags[key][value];
+               }
+             }
+           }
 
-       var ceil$1 = Math.ceil;
+           return null;
+         },
+         isSided: function isSided() {
+           if (this.tags.two_sided === 'yes') {
+             return false;
+           }
 
-       // `String.prototype.{ padStart, padEnd }` methods implementation
-       var createMethod$6 = function (IS_END) {
-         return function ($this, maxLength, fillString) {
-           var S = String(requireObjectCoercible($this));
-           var stringLength = S.length;
-           var fillStr = fillString === undefined ? ' ' : String(fillString);
-           var intMaxLength = toLength(maxLength);
-           var fillLen, stringFiller;
-           if (intMaxLength <= stringLength || fillStr == '') return S;
-           fillLen = intMaxLength - stringLength;
-           stringFiller = stringRepeat.call(fillStr, ceil$1(fillLen / fillStr.length));
-           if (stringFiller.length > fillLen) stringFiller = stringFiller.slice(0, fillLen);
-           return IS_END ? S + stringFiller : stringFiller + S;
-         };
-       };
+           return this.sidednessIdentifier() !== null;
+         },
+         lanes: function lanes() {
+           return osmLanes(this);
+         },
+         isClosed: function isClosed() {
+           return this.nodes.length > 1 && this.first() === this.last();
+         },
+         isConvex: function isConvex(resolver) {
+           if (!this.isClosed() || this.isDegenerate()) return null;
+           var nodes = utilArrayUniq(resolver.childNodes(this));
+           var coords = nodes.map(function (n) {
+             return n.loc;
+           });
+           var curr = 0;
+           var prev = 0;
 
-       var stringPad = {
-         // `String.prototype.padStart` method
-         // https://tc39.es/ecma262/#sec-string.prototype.padstart
-         start: createMethod$6(false),
-         // `String.prototype.padEnd` method
-         // https://tc39.es/ecma262/#sec-string.prototype.padend
-         end: createMethod$6(true)
-       };
+           for (var i = 0; i < coords.length; i++) {
+             var o = coords[(i + 1) % coords.length];
+             var a = coords[i];
+             var b = coords[(i + 2) % coords.length];
+             var res = geoVecCross(a, b, o);
+             curr = res > 0 ? 1 : res < 0 ? -1 : 0;
 
-       var padStart = stringPad.start;
+             if (curr === 0) {
+               continue;
+             } else if (prev && curr !== prev) {
+               return false;
+             }
 
-       var abs$3 = Math.abs;
-       var DatePrototype$1 = Date.prototype;
-       var getTime$1 = DatePrototype$1.getTime;
-       var nativeDateToISOString = DatePrototype$1.toISOString;
+             prev = curr;
+           }
 
-       // `Date.prototype.toISOString` method implementation
-       // https://tc39.es/ecma262/#sec-date.prototype.toisostring
-       // PhantomJS / old WebKit fails here:
-       var dateToIsoString = (fails(function () {
-         return nativeDateToISOString.call(new Date(-5e13 - 1)) != '0385-07-25T07:06:39.999Z';
-       }) || !fails(function () {
-         nativeDateToISOString.call(new Date(NaN));
-       })) ? function toISOString() {
-         if (!isFinite(getTime$1.call(this))) throw RangeError('Invalid time value');
-         var date = this;
-         var year = date.getUTCFullYear();
-         var milliseconds = date.getUTCMilliseconds();
-         var sign = year < 0 ? '-' : year > 9999 ? '+' : '';
-         return sign + padStart(abs$3(year), sign ? 6 : 4, 0) +
-           '-' + padStart(date.getUTCMonth() + 1, 2, 0) +
-           '-' + padStart(date.getUTCDate(), 2, 0) +
-           'T' + padStart(date.getUTCHours(), 2, 0) +
-           ':' + padStart(date.getUTCMinutes(), 2, 0) +
-           ':' + padStart(date.getUTCSeconds(), 2, 0) +
-           '.' + padStart(milliseconds, 3, 0) +
-           'Z';
-       } : nativeDateToISOString;
+           return true;
+         },
+         // returns an object with the tag that implies this is an area, if any
+         tagSuggestingArea: function tagSuggestingArea() {
+           return osmTagSuggestingArea(this.tags);
+         },
+         isArea: function isArea() {
+           if (this.tags.area === 'yes') return true;
+           if (!this.isClosed() || this.tags.area === 'no') return false;
+           return this.tagSuggestingArea() !== null;
+         },
+         isDegenerate: function isDegenerate() {
+           return new Set(this.nodes).size < (this.isArea() ? 3 : 2);
+         },
+         areAdjacent: function areAdjacent(n1, n2) {
+           for (var i = 0; i < this.nodes.length; i++) {
+             if (this.nodes[i] === n1) {
+               if (this.nodes[i - 1] === n2) return true;
+               if (this.nodes[i + 1] === n2) return true;
+             }
+           }
 
-       // `Date.prototype.toISOString` method
-       // https://tc39.es/ecma262/#sec-date.prototype.toisostring
-       // PhantomJS / old WebKit has a broken implementations
-       _export({ target: 'Date', proto: true, forced: Date.prototype.toISOString !== dateToIsoString }, {
-         toISOString: dateToIsoString
-       });
+           return false;
+         },
+         geometry: function geometry(graph) {
+           return graph["transient"](this, 'geometry', function () {
+             return this.isArea() ? 'area' : 'line';
+           });
+         },
+         // returns an array of objects representing the segments between the nodes in this way
+         segments: function segments(graph) {
+           function segmentExtent(graph) {
+             var n1 = graph.hasEntity(this.nodes[0]);
+             var n2 = graph.hasEntity(this.nodes[1]);
+             return n1 && n2 && geoExtent([[Math.min(n1.loc[0], n2.loc[0]), Math.min(n1.loc[1], n2.loc[1])], [Math.max(n1.loc[0], n2.loc[0]), Math.max(n1.loc[1], n2.loc[1])]]);
+           }
 
-       function behaviorBreathe() {
-         var duration = 800;
-         var steps = 4;
-         var selector = '.selected.shadow, .selected .shadow';
+           return graph["transient"](this, 'segments', function () {
+             var segments = [];
 
-         var _selected = select(null);
+             for (var i = 0; i < this.nodes.length - 1; i++) {
+               segments.push({
+                 id: this.id + '-' + i,
+                 wayId: this.id,
+                 index: i,
+                 nodes: [this.nodes[i], this.nodes[i + 1]],
+                 extent: segmentExtent
+               });
+             }
 
-         var _classed = '';
-         var _params = {};
-         var _done = false;
+             return segments;
+           });
+         },
+         // If this way is not closed, append the beginning node to the end of the nodelist to close it.
+         close: function close() {
+           if (this.isClosed() || !this.nodes.length) return this;
+           var nodes = this.nodes.slice();
+           nodes = nodes.filter(noRepeatNodes);
+           nodes.push(nodes[0]);
+           return this.update({
+             nodes: nodes
+           });
+         },
+         // If this way is closed, remove any connector nodes from the end of the nodelist to unclose it.
+         unclose: function unclose() {
+           if (!this.isClosed()) return this;
+           var nodes = this.nodes.slice();
+           var connector = this.first();
+           var i = nodes.length - 1; // remove trailing connectors..
 
-         var _timer;
+           while (i > 0 && nodes.length > 1 && nodes[i] === connector) {
+             nodes.splice(i, 1);
+             i = nodes.length - 1;
+           }
 
-         function ratchetyInterpolator(a, b, steps, units) {
-           a = parseFloat(a);
-           b = parseFloat(b);
-           var sample = quantize().domain([0, 1]).range(d3_quantize(d3_interpolateNumber(a, b), steps));
-           return function (t) {
-             return String(sample(t)) + (units || '');
-           };
-         }
+           nodes = nodes.filter(noRepeatNodes);
+           return this.update({
+             nodes: nodes
+           });
+         },
+         // Adds a node (id) in front of the node which is currently at position index.
+         // If index is undefined, the node will be added to the end of the way for linear ways,
+         //   or just before the final connecting node for circular ways.
+         // Consecutive duplicates are eliminated including existing ones.
+         // Circularity is always preserved when adding a node.
+         addNode: function addNode(id, index) {
+           var nodes = this.nodes.slice();
+           var isClosed = this.isClosed();
+           var max = isClosed ? nodes.length - 1 : nodes.length;
 
-         function reset(selection) {
-           selection.style('stroke-opacity', null).style('stroke-width', null).style('fill-opacity', null).style('r', null);
-         }
+           if (index === undefined) {
+             index = max;
+           }
 
-         function setAnimationParams(transition, fromTo) {
-           var toFrom = fromTo === 'from' ? 'to' : 'from';
-           transition.styleTween('stroke-opacity', function (d) {
-             return ratchetyInterpolator(_params[d.id][toFrom].opacity, _params[d.id][fromTo].opacity, steps);
-           }).styleTween('stroke-width', function (d) {
-             return ratchetyInterpolator(_params[d.id][toFrom].width, _params[d.id][fromTo].width, steps, 'px');
-           }).styleTween('fill-opacity', function (d) {
-             return ratchetyInterpolator(_params[d.id][toFrom].opacity, _params[d.id][fromTo].opacity, steps);
-           }).styleTween('r', function (d) {
-             return ratchetyInterpolator(_params[d.id][toFrom].width, _params[d.id][fromTo].width, steps, 'px');
-           });
-         }
+           if (index < 0 || index > max) {
+             throw new RangeError('index ' + index + ' out of range 0..' + max);
+           } // If this is a closed way, remove all connector nodes except the first one
+           // (there may be duplicates) and adjust index if necessary..
 
-         function calcAnimationParams(selection) {
-           selection.call(reset).each(function (d) {
-             var s = select(this);
-             var tag = s.node().tagName;
-             var p = {
-               'from': {},
-               'to': {}
-             };
-             var opacity;
-             var width; // determine base opacity and width
 
-             if (tag === 'circle') {
-               opacity = parseFloat(s.style('fill-opacity') || 0.5);
-               width = parseFloat(s.style('r') || 15.5);
-             } else {
-               opacity = parseFloat(s.style('stroke-opacity') || 0.7);
-               width = parseFloat(s.style('stroke-width') || 10);
-             } // calculate from/to interpolation params..
+           if (isClosed) {
+             var connector = this.first(); // leading connectors..
 
+             var i = 1;
 
-             p.tag = tag;
-             p.from.opacity = opacity * 0.6;
-             p.to.opacity = opacity * 1.25;
-             p.from.width = width * 0.7;
-             p.to.width = width * (tag === 'circle' ? 1.5 : 1);
-             _params[d.id] = p;
-           });
-         }
+             while (i < nodes.length && nodes.length > 2 && nodes[i] === connector) {
+               nodes.splice(i, 1);
+               if (index > i) index--;
+             } // trailing connectors..
 
-         function run(surface, fromTo) {
-           var toFrom = fromTo === 'from' ? 'to' : 'from';
-           var currSelected = surface.selectAll(selector);
-           var currClassed = surface.attr('class');
 
-           if (_done || currSelected.empty()) {
-             _selected.call(reset);
+             i = nodes.length - 1;
 
-             _selected = select(null);
-             return;
+             while (i > 0 && nodes.length > 1 && nodes[i] === connector) {
+               nodes.splice(i, 1);
+               if (index > i) index--;
+               i = nodes.length - 1;
+             }
            }
 
-           if (!fastDeepEqual(currSelected.data(), _selected.data()) || currClassed !== _classed) {
-             _selected.call(reset);
+           nodes.splice(index, 0, id);
+           nodes = nodes.filter(noRepeatNodes); // If the way was closed before, append a connector node to keep it closed..
 
-             _classed = currClassed;
-             _selected = currSelected.call(calcAnimationParams);
+           if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
+             nodes.push(nodes[0]);
            }
 
-           var didCallNextRun = false;
+           return this.update({
+             nodes: nodes
+           });
+         },
+         // Replaces the node which is currently at position index with the given node (id).
+         // Consecutive duplicates are eliminated including existing ones.
+         // Circularity is preserved when updating a node.
+         updateNode: function updateNode(id, index) {
+           var nodes = this.nodes.slice();
+           var isClosed = this.isClosed();
+           var max = nodes.length - 1;
 
-           _selected.transition().duration(duration).call(setAnimationParams, fromTo).on('end', function () {
-             // `end` event is called for each selected element, but we want
-             // it to run only once
-             if (!didCallNextRun) {
-               surface.call(run, toFrom);
-               didCallNextRun = true;
-             } // if entity was deselected, remove breathe styling
+           if (index === undefined || index < 0 || index > max) {
+             throw new RangeError('index ' + index + ' out of range 0..' + max);
+           } // If this is a closed way, remove all connector nodes except the first one
+           // (there may be duplicates) and adjust index if necessary..
 
 
-             if (!select(this).classed('selected')) {
-               reset(select(this));
-             }
-           });
-         }
+           if (isClosed) {
+             var connector = this.first(); // leading connectors..
 
-         function behavior(surface) {
-           _done = false;
-           _timer = timer(function () {
-             // wait for elements to actually become selected
-             if (surface.selectAll(selector).empty()) {
-               return false;
-             }
+             var i = 1;
 
-             surface.call(run, 'from');
+             while (i < nodes.length && nodes.length > 2 && nodes[i] === connector) {
+               nodes.splice(i, 1);
+               if (index > i) index--;
+             } // trailing connectors..
 
-             _timer.stop();
 
-             return true;
-           }, 20);
-         }
+             i = nodes.length - 1;
 
-         behavior.restartIfNeeded = function (surface) {
-           if (_selected.empty()) {
-             surface.call(run, 'from');
+             while (i > 0 && nodes.length > 1 && nodes[i] === connector) {
+               nodes.splice(i, 1);
+               if (index === i) index = 0; // update leading connector instead
 
-             if (_timer) {
-               _timer.stop();
+               i = nodes.length - 1;
              }
            }
-         };
 
-         behavior.off = function () {
-           _done = true;
+           nodes.splice(index, 1, id);
+           nodes = nodes.filter(noRepeatNodes); // If the way was closed before, append a connector node to keep it closed..
 
-           if (_timer) {
-             _timer.stop();
+           if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
+             nodes.push(nodes[0]);
            }
 
-           _selected.interrupt().call(reset);
-         };
-
-         return behavior;
-       }
-
-       /* Creates a keybinding behavior for an operation */
-       function behaviorOperation(context) {
-         var _operation;
-
-         function keypress(d3_event) {
-           // prevent operations during low zoom selection
-           if (!context.map().withinEditableZoom()) return;
-           if (_operation.availableForKeypress && !_operation.availableForKeypress()) return;
-           d3_event.preventDefault();
-
-           var disabled = _operation.disabled();
-
-           if (disabled) {
-             context.ui().flash.duration(4000).iconName('#iD-operation-' + _operation.id).iconClass('operation disabled').label(_operation.tooltip)();
-           } else {
-             context.ui().flash.duration(2000).iconName('#iD-operation-' + _operation.id).iconClass('operation').label(_operation.annotation() || _operation.title)();
-             if (_operation.point) _operation.point(null);
-
-             _operation();
-           }
-         }
+           return this.update({
+             nodes: nodes
+           });
+         },
+         // Replaces each occurrence of node id needle with replacement.
+         // Consecutive duplicates are eliminated including existing ones.
+         // Circularity is preserved.
+         replaceNode: function replaceNode(needleID, replacementID) {
+           var nodes = this.nodes.slice();
+           var isClosed = this.isClosed();
 
-         function behavior() {
-           if (_operation && _operation.available()) {
-             context.keybinding().on(_operation.keys, keypress);
+           for (var i = 0; i < nodes.length; i++) {
+             if (nodes[i] === needleID) {
+               nodes[i] = replacementID;
+             }
            }
 
-           return behavior;
-         }
-
-         behavior.off = function () {
-           context.keybinding().off(_operation.keys);
-         };
+           nodes = nodes.filter(noRepeatNodes); // If the way was closed before, append a connector node to keep it closed..
 
-         behavior.which = function (_) {
-           if (!arguments.length) return _operation;
-           _operation = _;
-           return behavior;
-         };
+           if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
+             nodes.push(nodes[0]);
+           }
 
-         return behavior;
-       }
+           return this.update({
+             nodes: nodes
+           });
+         },
+         // Removes each occurrence of node id.
+         // Consecutive duplicates are eliminated including existing ones.
+         // Circularity is preserved.
+         removeNode: function removeNode(id) {
+           var nodes = this.nodes.slice();
+           var isClosed = this.isClosed();
+           nodes = nodes.filter(function (node) {
+             return node !== id;
+           }).filter(noRepeatNodes); // If the way was closed before, append a connector node to keep it closed..
 
-       function operationCircularize(context, selectedIDs) {
-         var _extent;
+           if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
+             nodes.push(nodes[0]);
+           }
 
-         var _actions = selectedIDs.map(getAction).filter(Boolean);
+           return this.update({
+             nodes: nodes
+           });
+         },
+         asJXON: function asJXON(changeset_id) {
+           var r = {
+             way: {
+               '@id': this.osmId(),
+               '@version': this.version || 0,
+               nd: this.nodes.map(function (id) {
+                 return {
+                   keyAttributes: {
+                     ref: osmEntity.id.toOSM(id)
+                   }
+                 };
+               }, this),
+               tag: Object.keys(this.tags).map(function (k) {
+                 return {
+                   keyAttributes: {
+                     k: k,
+                     v: this.tags[k]
+                   }
+                 };
+               }, this)
+             }
+           };
 
-         var _amount = _actions.length === 1 ? 'single' : 'multiple';
+           if (changeset_id) {
+             r.way['@changeset'] = changeset_id;
+           }
 
-         var _coords = utilGetAllNodes(selectedIDs, context.graph()).map(function (n) {
-           return n.loc;
-         });
+           return r;
+         },
+         asGeoJSON: function asGeoJSON(resolver) {
+           return resolver["transient"](this, 'GeoJSON', function () {
+             var coordinates = resolver.childNodes(this).map(function (n) {
+               return n.loc;
+             });
 
-         function getAction(entityID) {
-           var entity = context.entity(entityID);
-           if (entity.type !== 'way' || new Set(entity.nodes).size <= 1) return null;
+             if (this.isArea() && this.isClosed()) {
+               return {
+                 type: 'Polygon',
+                 coordinates: [coordinates]
+               };
+             } else {
+               return {
+                 type: 'LineString',
+                 coordinates: coordinates
+               };
+             }
+           });
+         },
+         area: function area(resolver) {
+           return resolver["transient"](this, 'area', function () {
+             var nodes = resolver.childNodes(this);
+             var json = {
+               type: 'Polygon',
+               coordinates: [nodes.map(function (n) {
+                 return n.loc;
+               })]
+             };
 
-           if (!_extent) {
-             _extent = entity.extent(context.graph());
-           } else {
-             _extent = _extent.extend(entity.extent(context.graph()));
-           }
+             if (!this.isClosed() && nodes.length) {
+               json.coordinates[0].push(nodes[0].loc);
+             }
 
-           return actionCircularize(entityID, context.projection);
-         }
+             var area = d3_geoArea(json); // Heuristic for detecting counterclockwise winding order. Assumes
+             // that OpenStreetMap polygons are not hemisphere-spanning.
 
-         var operation = function operation() {
-           if (!_actions.length) return;
+             if (area > 2 * Math.PI) {
+               json.coordinates[0] = json.coordinates[0].reverse();
+               area = d3_geoArea(json);
+             }
 
-           var combinedAction = function combinedAction(graph, t) {
-             _actions.forEach(function (action) {
-               if (!action.disabled(graph)) {
-                 graph = action(graph, t);
-               }
-             });
+             return isNaN(area) ? 0 : area;
+           });
+         }
+       }); // Filter function to eliminate consecutive duplicates.
 
-             return graph;
-           };
+       function noRepeatNodes(node, i, arr) {
+         return i === 0 || node !== arr[i - 1];
+       }
 
-           combinedAction.transitionable = true;
-           context.perform(combinedAction, operation.annotation());
-           window.setTimeout(function () {
-             context.validator().validate();
-           }, 300); // after any transition
-         };
+       //
+       // 1. Relation tagged with `type=multipolygon` and no interesting tags.
+       // 2. One and only one member with the `outer` role. Must be a way with interesting tags.
+       // 3. No members without a role.
+       //
+       // Old multipolygons are no longer recommended but are still rendered as areas by iD.
 
-         operation.available = function () {
-           return _actions.length && selectedIDs.length === _actions.length;
-         }; // don't cache this because the visible extent could change
+       function osmOldMultipolygonOuterMemberOfRelation(entity, graph) {
+         if (entity.type !== 'relation' || !entity.isMultipolygon() || Object.keys(entity.tags).filter(osmIsInterestingTag).length > 1) {
+           return false;
+         }
 
+         var outerMember;
 
-         operation.disabled = function () {
-           if (!_actions.length) return '';
+         for (var memberIndex in entity.members) {
+           var member = entity.members[memberIndex];
 
-           var actionDisableds = _actions.map(function (action) {
-             return action.disabled(context.graph());
-           }).filter(Boolean);
+           if (!member.role || member.role === 'outer') {
+             if (outerMember) return false;
+             if (member.type !== 'way') return false;
+             if (!graph.hasEntity(member.id)) return false;
+             outerMember = graph.entity(member.id);
 
-           if (actionDisableds.length === _actions.length) {
-             // none of the features can be circularized
-             if (new Set(actionDisableds).size > 1) {
-               return 'multiple_blockers';
+             if (Object.keys(outerMember.tags).filter(osmIsInterestingTag).length === 0) {
+               return false;
              }
-
-             return actionDisableds[0];
-           } else if (_extent.percentContainedIn(context.map().extent()) < 0.8) {
-             return 'too_large';
-           } else if (someMissing()) {
-             return 'not_downloaded';
-           } else if (selectedIDs.some(context.hasHiddenConnections)) {
-             return 'connected_to_hidden';
            }
+         }
 
+         return outerMember;
+       } // For fixing up rendering of multipolygons with tags on the outer member.
+       // https://github.com/openstreetmap/iD/issues/613
+
+       function osmIsOldMultipolygonOuterMember(entity, graph) {
+         if (entity.type !== 'way' || Object.keys(entity.tags).filter(osmIsInterestingTag).length === 0) {
            return false;
+         }
 
-           function someMissing() {
-             if (context.inIntro()) return false;
-             var osm = context.connection();
+         var parents = graph.parentRelations(entity);
+         if (parents.length !== 1) return false;
+         var parent = parents[0];
 
-             if (osm) {
-               var missing = _coords.filter(function (loc) {
-                 return !osm.isDataLoaded(loc);
-               });
+         if (!parent.isMultipolygon() || Object.keys(parent.tags).filter(osmIsInterestingTag).length > 1) {
+           return false;
+         }
 
-               if (missing.length) {
-                 missing.forEach(function (loc) {
-                   context.loadTileAtLoc(loc);
-                 });
-                 return true;
-               }
-             }
+         var members = parent.members,
+             member;
+
+         for (var i = 0; i < members.length; i++) {
+           member = members[i];
 
+           if (member.id === entity.id && member.role && member.role !== 'outer') {
+             // Not outer member
              return false;
            }
-         };
-
-         operation.tooltip = function () {
-           var disable = operation.disabled();
-           return disable ? _t('operations.circularize.' + disable + '.' + _amount) : _t('operations.circularize.description.' + _amount);
-         };
 
-         operation.annotation = function () {
-           return _t('operations.circularize.annotation.feature', {
-             n: _actions.length
-           });
-         };
+           if (member.id !== entity.id && (!member.role || member.role === 'outer')) {
+             // Not a simple multipolygon
+             return false;
+           }
+         }
 
-         operation.id = 'circularize';
-         operation.keys = [_t('operations.circularize.key')];
-         operation.title = _t('operations.circularize.title');
-         operation.behavior = behaviorOperation(context).which(operation);
-         return operation;
+         return parent;
        }
+       function osmOldMultipolygonOuterMember(entity, graph) {
+         if (entity.type !== 'way') return false;
+         var parents = graph.parentRelations(entity);
+         if (parents.length !== 1) return false;
+         var parent = parents[0];
 
-       // For example, ⌘Z -> Ctrl+Z
-
-       var uiCmd = function uiCmd(code) {
-         var detected = utilDetect();
-
-         if (detected.os === 'mac') {
-           return code;
+         if (!parent.isMultipolygon() || Object.keys(parent.tags).filter(osmIsInterestingTag).length > 1) {
+           return false;
          }
 
-         if (detected.os === 'win') {
-           if (code === '⌘⇧Z') return 'Ctrl+Y';
-         }
+         var members = parent.members,
+             member,
+             outerMember;
 
-         var result = '',
-             replacements = {
-           '⌘': 'Ctrl',
-           '⇧': 'Shift',
-           '⌥': 'Alt',
-           '⌫': 'Backspace',
-           '⌦': 'Delete'
-         };
+         for (var i = 0; i < members.length; i++) {
+           member = members[i];
 
-         for (var i = 0; i < code.length; i++) {
-           if (code[i] in replacements) {
-             result += replacements[code[i]] + (i < code.length - 1 ? '+' : '');
-           } else {
-             result += code[i];
+           if (!member.role || member.role === 'outer') {
+             if (outerMember) return false; // Not a simple multipolygon
+
+             outerMember = member;
            }
          }
 
-         return result;
-       }; // return a display-focused string for a given keyboard code
-
-       uiCmd.display = function (code) {
-         if (code.length !== 1) return code;
-         var detected = utilDetect();
-         var mac = detected.os === 'mac';
-         var replacements = {
-           '⌘': mac ? '⌘ ' + _t('shortcuts.key.cmd') : _t('shortcuts.key.ctrl'),
-           '⇧': mac ? '⇧ ' + _t('shortcuts.key.shift') : _t('shortcuts.key.shift'),
-           '⌥': mac ? '⌥ ' + _t('shortcuts.key.option') : _t('shortcuts.key.alt'),
-           '⌃': mac ? '⌃ ' + _t('shortcuts.key.ctrl') : _t('shortcuts.key.ctrl'),
-           '⌫': mac ? '⌫ ' + _t('shortcuts.key.delete') : _t('shortcuts.key.backspace'),
-           '⌦': mac ? '⌦ ' + _t('shortcuts.key.del') : _t('shortcuts.key.del'),
-           '↖': mac ? '↖ ' + _t('shortcuts.key.pgup') : _t('shortcuts.key.pgup'),
-           '↘': mac ? '↘ ' + _t('shortcuts.key.pgdn') : _t('shortcuts.key.pgdn'),
-           '⇞': mac ? '⇞ ' + _t('shortcuts.key.home') : _t('shortcuts.key.home'),
-           '⇟': mac ? '⇟ ' + _t('shortcuts.key.end') : _t('shortcuts.key.end'),
-           '↵': mac ? '⏎ ' + _t('shortcuts.key.return') : _t('shortcuts.key.enter'),
-           '⎋': mac ? '⎋ ' + _t('shortcuts.key.esc') : _t('shortcuts.key.esc'),
-           '☰': mac ? '☰ ' + _t('shortcuts.key.menu') : _t('shortcuts.key.menu')
-         };
-         return replacements[code] || code;
-       };
+         if (!outerMember) return false;
+         var outerEntity = graph.hasEntity(outerMember.id);
 
-       function operationDelete(context, selectedIDs) {
-         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 = utilTotalExtent(selectedIDs, context.graph());
+         if (!outerEntity || !Object.keys(outerEntity.tags).filter(osmIsInterestingTag).length) {
+           return false;
+         }
 
-         var operation = function operation() {
-           var nextSelectedID;
-           var nextSelectedLoc;
+         return outerEntity;
+       } // Join `toJoin` array into sequences of connecting ways.
+       // Segments which share identical start/end nodes will, as much as possible,
+       // be connected with each other.
+       //
+       // The return value is a nested array. Each constituent array contains elements
+       // of `toJoin` which have been determined to connect.
+       //
+       // Each consitituent array also has a `nodes` property whose value is an
+       // ordered array of member nodes, with appropriate order reversal and
+       // start/end coordinate de-duplication.
+       //
+       // Members of `toJoin` must have, at minimum, `type` and `id` properties.
+       // Thus either an array of `osmWay`s or a relation member array may be used.
+       //
+       // If an member is an `osmWay`, its tags and childnodes may be reversed via
+       // `actionReverse` in the output.
+       //
+       // The returned sequences array also has an `actions` array property, containing
+       // any reversal actions that should be applied to the graph, should the calling
+       // code attempt to actually join the given ways.
+       //
+       // Incomplete members (those for which `graph.hasEntity(element.id)` returns
+       // false) and non-way members are ignored.
+       //
 
-           if (selectedIDs.length === 1) {
-             var id = selectedIDs[0];
-             var entity = context.entity(id);
-             var geometry = entity.geometry(context.graph());
-             var parents = context.graph().parentWays(entity);
-             var parent = parents[0]; // Select the next closest node in the way.
+       function osmJoinWays(toJoin, graph) {
+         function resolve(member) {
+           return graph.childNodes(graph.entity(member.id));
+         }
 
-             if (geometry === 'vertex') {
-               var nodes = parent.nodes;
-               var i = nodes.indexOf(id);
+         function reverse(item) {
+           var action = actionReverse(item.id, {
+             reverseOneway: true
+           });
+           sequences.actions.push(action);
+           return item instanceof osmWay ? action(graph).entity(item.id) : item;
+         } // make a copy containing only the items to join
 
-               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;
-             }
-           }
+         toJoin = toJoin.filter(function (member) {
+           return member.type === 'way' && graph.hasEntity(member.id);
+         }); // Are the things we are joining relation members or `osmWays`?
+         // If `osmWays`, skip the "prefer a forward path" code below (see #4872)
 
-           context.perform(action, operation.annotation());
-           context.validator().validate();
+         var i;
+         var joinAsMembers = true;
 
-           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));
+         for (i = 0; i < toJoin.length; i++) {
+           if (toJoin[i] instanceof osmWay) {
+             joinAsMembers = false;
+             break;
            }
-         };
-
-         operation.available = function () {
-           return true;
-         };
+         }
 
-         operation.disabled = function () {
-           if (extent.percentContainedIn(context.map().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';
-           }
+         var sequences = [];
+         sequences.actions = [];
 
-           return false;
+         while (toJoin.length) {
+           // start a new sequence
+           var item = toJoin.shift();
+           var currWays = [item];
+           var currNodes = resolve(item).slice(); // add to it
 
-           function someMissing() {
-             if (context.inIntro()) return false;
-             var osm = context.connection();
+           while (toJoin.length) {
+             var start = currNodes[0];
+             var end = currNodes[currNodes.length - 1];
+             var fn = null;
+             var nodes = null; // Find the next way/member to join.
 
-             if (osm) {
-               var missing = coords.filter(function (loc) {
-                 return !osm.isDataLoaded(loc);
-               });
+             for (i = 0; i < toJoin.length; i++) {
+               item = toJoin[i];
+               nodes = resolve(item); // (for member ordering only, not way ordering - see #4872)
+               // Strongly prefer to generate a forward path that preserves the order
+               // of the members array. For multipolygons and most relations, member
+               // order does not matter - but for routes, it does. (see #4589)
+               // If we started this sequence backwards (i.e. next member way attaches to
+               // the start node and not the end node), reverse the initial way before continuing.
 
-               if (missing.length) {
-                 missing.forEach(function (loc) {
-                   context.loadTileAtLoc(loc);
-                 });
-                 return true;
+               if (joinAsMembers && currWays.length === 1 && nodes[0] !== end && nodes[nodes.length - 1] !== end && (nodes[nodes.length - 1] === start || nodes[0] === start)) {
+                 currWays[0] = reverse(currWays[0]);
+                 currNodes.reverse();
+                 start = currNodes[0];
+                 end = currNodes[currNodes.length - 1];
                }
-             }
-
-             return false;
-           }
 
-           function hasWikidataTag(id) {
-             var entity = context.entity(id);
-             return entity.tags.wikidata && entity.tags.wikidata.trim().length > 0;
-           }
+               if (nodes[0] === end) {
+                 fn = currNodes.push; // join to end
 
-           function incompleteRelation(id) {
-             var entity = context.entity(id);
-             return entity.type === 'relation' && !entity.isComplete(context.graph());
-           }
+                 nodes = nodes.slice(1);
+                 break;
+               } else if (nodes[nodes.length - 1] === end) {
+                 fn = currNodes.push; // join to end
 
-           function protectedMember(id) {
-             var entity = context.entity(id);
-             if (entity.type !== 'way') return false;
-             var parents = context.graph().parentRelations(entity);
+                 nodes = nodes.slice(0, -1).reverse();
+                 item = reverse(item);
+                 break;
+               } else if (nodes[nodes.length - 1] === start) {
+                 fn = currNodes.unshift; // join to beginning
 
-             for (var i = 0; i < parents.length; i++) {
-               var parent = parents[i];
-               var type = parent.tags.type;
-               var role = parent.memberById(id).role || 'outer';
+                 nodes = nodes.slice(0, -1);
+                 break;
+               } else if (nodes[0] === start) {
+                 fn = currNodes.unshift; // join to beginning
 
-               if (type === 'route' || type === 'boundary' || type === 'multipolygon' && role === 'outer') {
-                 return true;
+                 nodes = nodes.slice(1).reverse();
+                 item = reverse(item);
+                 break;
+               } else {
+                 fn = nodes = null;
                }
              }
 
-             return false;
-           }
-         };
+             if (!nodes) {
+               // couldn't find a joinable way/member
+               break;
+             }
 
-         operation.tooltip = function () {
-           var disable = operation.disabled();
-           return disable ? _t('operations.delete.' + disable + '.' + multi) : _t('operations.delete.description.' + multi);
-         };
+             fn.apply(currWays, [item]);
+             fn.apply(currNodes, nodes);
+             toJoin.splice(i, 1);
+           }
 
-         operation.annotation = function () {
-           return selectedIDs.length === 1 ? _t('operations.delete.annotation.' + context.graph().geometry(selectedIDs[0])) : _t('operations.delete.annotation.feature', {
-             n: selectedIDs.length
-           });
-         };
+           currWays.nodes = currNodes;
+           sequences.push(currWays);
+         }
 
-         operation.id = 'delete';
-         operation.keys = [uiCmd('⌘⌫'), uiCmd('⌘⌦'), uiCmd('⌦')];
-         operation.title = _t('operations.delete.title');
-         operation.behavior = behaviorOperation(context).which(operation);
-         return operation;
+         return sequences;
        }
 
-       function operationOrthogonalize(context, selectedIDs) {
-         var _extent;
+       function actionAddMember(relationId, member, memberIndex, insertPair) {
+         return function action(graph) {
+           var relation = graph.entity(relationId); // There are some special rules for Public Transport v2 routes.
 
-         var _type;
+           var isPTv2 = /stop|platform/.test(member.role);
 
-         var _actions = selectedIDs.map(chooseAction).filter(Boolean);
+           if ((isNaN(memberIndex) || insertPair) && member.type === 'way' && !isPTv2) {
+             // Try to perform sensible inserts based on how the ways join together
+             graph = addWayMember(relation, graph);
+           } else {
+             // see https://wiki.openstreetmap.org/wiki/Public_transport#Service_routes
+             // Stops and Platforms for PTv2 should be ordered first.
+             // hack: We do not currently have the ability to place them in the exactly correct order.
+             if (isPTv2 && isNaN(memberIndex)) {
+               memberIndex = 0;
+             }
 
-         var _amount = _actions.length === 1 ? 'single' : 'multiple';
+             graph = graph.replace(relation.addMember(member, memberIndex));
+           }
 
-         var _coords = utilGetAllNodes(selectedIDs, context.graph()).map(function (n) {
-           return n.loc;
-         });
+           return graph;
+         }; // Add a way member into the relation "wherever it makes sense".
+         // In this situation we were not supplied a memberIndex.
 
-         function chooseAction(entityID) {
-           var entity = context.entity(entityID);
-           var geometry = entity.geometry(context.graph());
+         function addWayMember(relation, graph) {
+           var groups, tempWay, item, i, j, k; // remove PTv2 stops and platforms before doing anything.
 
-           if (!_extent) {
-             _extent = entity.extent(context.graph());
-           } else {
-             _extent = _extent.extend(entity.extent(context.graph()));
-           } // square a line/area
+           var PTv2members = [];
+           var members = [];
 
+           for (i = 0; i < relation.members.length; i++) {
+             var m = relation.members[i];
 
-           if (entity.type === 'way' && new Set(entity.nodes).size > 2) {
-             if (_type && _type !== 'feature') return null;
-             _type = 'feature';
-             return actionOrthogonalize(entityID, context.projection); // square a single vertex
-           } else if (geometry === 'vertex') {
-             if (_type && _type !== 'corner') return null;
-             _type = 'corner';
-             var graph = context.graph();
-             var parents = graph.parentWays(entity);
+             if (/stop|platform/.test(m.role)) {
+               PTv2members.push(m);
+             } else {
+               members.push(m);
+             }
+           }
 
-             if (parents.length === 1) {
-               var way = parents[0];
+           relation = relation.update({
+             members: members
+           });
 
-               if (way.nodes.indexOf(entityID) !== -1) {
-                 return actionOrthogonalize(way.id, context.projection, entityID);
-               }
-             }
+           if (insertPair) {
+             // We're adding a member that must stay paired with an existing member.
+             // (This feature is used by `actionSplit`)
+             //
+             // This is tricky because the members may exist multiple times in the
+             // member list, and with different A-B/B-A ordering and different roles.
+             // (e.g. a bus route that loops out and back - #4589).
+             //
+             // Replace the existing member with a temporary way,
+             // so that `osmJoinWays` can treat the pair like a single way.
+             tempWay = osmWay({
+               id: 'wTemp',
+               nodes: insertPair.nodes
+             });
+             graph = graph.replace(tempWay);
+             var tempMember = {
+               id: tempWay.id,
+               type: 'way',
+               role: member.role
+             };
+             var tempRelation = relation.replaceMember({
+               id: insertPair.originalID
+             }, tempMember, true);
+             groups = utilArrayGroupBy(tempRelation.members, 'type');
+             groups.way = groups.way || [];
+           } else {
+             // Add the member anywhere, one time. Just push and let `osmJoinWays` decide where to put it.
+             groups = utilArrayGroupBy(relation.members, 'type');
+             groups.way = groups.way || [];
+             groups.way.push(member);
            }
 
-           return null;
-         }
+           members = withIndex(groups.way);
+           var joined = osmJoinWays(members, graph); // `joined` might not contain all of the way members,
+           // But will contain only the completed (downloaded) members
 
-         var operation = function operation() {
-           if (!_actions.length) return;
+           for (i = 0; i < joined.length; i++) {
+             var segment = joined[i];
+             var nodes = segment.nodes.slice();
+             var startIndex = segment[0].index; // j = array index in `members` where this segment starts
 
-           var combinedAction = function combinedAction(graph, t) {
-             _actions.forEach(function (action) {
-               if (!action.disabled(graph)) {
-                 graph = action(graph, t);
+             for (j = 0; j < members.length; j++) {
+               if (members[j].index === startIndex) {
+                 break;
                }
-             });
-
-             return graph;
-           };
+             } // k = each member in segment
 
-           combinedAction.transitionable = true;
-           context.perform(combinedAction, operation.annotation());
-           window.setTimeout(function () {
-             context.validator().validate();
-           }, 300); // after any transition
-         };
 
-         operation.available = function () {
-           return _actions.length && selectedIDs.length === _actions.length;
-         }; // don't cache this because the visible extent could change
+             for (k = 0; k < segment.length; k++) {
+               item = segment[k];
+               var way = graph.entity(item.id); // If this is a paired item, generate members in correct order and role
 
+               if (tempWay && item.id === tempWay.id) {
+                 if (nodes[0].id === insertPair.nodes[0]) {
+                   item.pair = [{
+                     id: insertPair.originalID,
+                     type: 'way',
+                     role: item.role
+                   }, {
+                     id: insertPair.insertedID,
+                     type: 'way',
+                     role: item.role
+                   }];
+                 } else {
+                   item.pair = [{
+                     id: insertPair.insertedID,
+                     type: 'way',
+                     role: item.role
+                   }, {
+                     id: insertPair.originalID,
+                     type: 'way',
+                     role: item.role
+                   }];
+                 }
+               } // reorder `members` if necessary
 
-         operation.disabled = function () {
-           if (!_actions.length) return '';
 
-           var actionDisableds = _actions.map(function (action) {
-             return action.disabled(context.graph());
-           }).filter(Boolean);
+               if (k > 0) {
+                 if (j + k >= members.length || item.index !== members[j + k].index) {
+                   moveMember(members, item.index, j + k);
+                 }
+               }
 
-           if (actionDisableds.length === _actions.length) {
-             // none of the features can be squared
-             if (new Set(actionDisableds).size > 1) {
-               return 'multiple_blockers';
+               nodes.splice(0, way.nodes.length - 1);
              }
-
-             return actionDisableds[0];
-           } else if (_extent && _extent.percentContainedIn(context.map().extent()) < 0.8) {
-             return 'too_large';
-           } else if (someMissing()) {
-             return 'not_downloaded';
-           } else if (selectedIDs.some(context.hasHiddenConnections)) {
-             return 'connected_to_hidden';
            }
 
-           return false;
-
-           function someMissing() {
-             if (context.inIntro()) return false;
-             var osm = context.connection();
+           if (tempWay) {
+             graph = graph.remove(tempWay);
+           } // Final pass: skip dead items, split pairs, remove index properties
 
-             if (osm) {
-               var missing = _coords.filter(function (loc) {
-                 return !osm.isDataLoaded(loc);
-               });
 
-               if (missing.length) {
-                 missing.forEach(function (loc) {
-                   context.loadTileAtLoc(loc);
-                 });
-                 return true;
-               }
-             }
+           var wayMembers = [];
 
-             return false;
-           }
-         };
+           for (i = 0; i < members.length; i++) {
+             item = members[i];
+             if (item.index === -1) continue;
 
-         operation.tooltip = function () {
-           var disable = operation.disabled();
-           return disable ? _t('operations.orthogonalize.' + disable + '.' + _amount) : _t('operations.orthogonalize.description.' + _type + '.' + _amount);
-         };
+             if (item.pair) {
+               wayMembers.push(item.pair[0]);
+               wayMembers.push(item.pair[1]);
+             } else {
+               wayMembers.push(utilObjectOmit(item, ['index']));
+             }
+           } // Put stops and platforms first, then nodes, ways, relations
+           // This is recommended for Public Transport v2 routes:
+           // see https://wiki.openstreetmap.org/wiki/Public_transport#Service_routes
 
-         operation.annotation = function () {
-           return _t('operations.orthogonalize.annotation.' + _type, {
-             n: _actions.length
-           });
-         };
 
-         operation.id = 'orthogonalize';
-         operation.keys = [_t('operations.orthogonalize.key')];
-         operation.title = _t('operations.orthogonalize.title');
-         operation.behavior = behaviorOperation(context).which(operation);
-         return operation;
-       }
+           var newMembers = PTv2members.concat(groups.node || [], wayMembers, groups.relation || []);
+           return graph.replace(relation.update({
+             members: newMembers
+           })); // `moveMember()` changes the `members` array in place by splicing
+           // the item with `.index = findIndex` to where it belongs,
+           // and marking the old position as "dead" with `.index = -1`
+           //
+           // j=5, k=0                jk
+           // segment                 5 4 7 6
+           // members       0 1 2 3 4 5 6 7 8 9        keep 5 in j+k
+           //
+           // j=5, k=1                j k
+           // segment                 5 4 7 6
+           // members       0 1 2 3 4 5 6 7 8 9        move 4 to j+k
+           // members       0 1 2 3 x 5 4 6 7 8 9      moved
+           //
+           // j=5, k=2                j   k
+           // segment                 5 4 7 6
+           // members       0 1 2 3 x 5 4 6 7 8 9      move 7 to j+k
+           // members       0 1 2 3 x 5 4 7 6 x 8 9    moved
+           //
+           // j=5, k=3                j     k
+           // segment                 5 4 7 6
+           // members       0 1 2 3 x 5 4 7 6 x 8 9    keep 6 in j+k
+           //
 
-       function operationReflectShort(context, selectedIDs) {
-         return operationReflect(context, selectedIDs, 'short');
-       }
-       function operationReflectLong(context, selectedIDs) {
-         return operationReflect(context, selectedIDs, 'long');
-       }
-       function operationReflect(context, selectedIDs, axis) {
-         axis = axis || 'long';
-         var multi = selectedIDs.length === 1 ? 'single' : 'multiple';
-         var nodes = utilGetAllNodes(selectedIDs, context.graph());
-         var coords = nodes.map(function (n) {
-           return n.loc;
-         });
-         var extent = utilTotalExtent(selectedIDs, context.graph());
+           function moveMember(arr, findIndex, toIndex) {
+             var i;
 
-         var operation = function operation() {
-           var action = actionReflect(selectedIDs, context.projection).useLongAxis(Boolean(axis === 'long'));
-           context.perform(action, operation.annotation());
-           window.setTimeout(function () {
-             context.validator().validate();
-           }, 300); // after any transition
-         };
+             for (i = 0; i < arr.length; i++) {
+               if (arr[i].index === findIndex) {
+                 break;
+               }
+             }
 
-         operation.available = function () {
-           return nodes.length >= 3;
-         }; // don't cache this because the visible extent could change
+             var item = Object.assign({}, arr[i]); // shallow copy
 
+             arr[i].index = -1; // mark as dead
 
-         operation.disabled = function () {
-           if (extent.percentContainedIn(context.map().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(incompleteRelation)) {
-             return 'incomplete_relation';
-           }
+             item.index = toIndex;
+             arr.splice(toIndex, 0, item);
+           } // This is the same as `Relation.indexedMembers`,
+           // Except we don't want to index all the members, only the ways
 
-           return false;
 
-           function someMissing() {
-             if (context.inIntro()) return false;
-             var osm = context.connection();
+           function withIndex(arr) {
+             var result = new Array(arr.length);
 
-             if (osm) {
-               var missing = coords.filter(function (loc) {
-                 return !osm.isDataLoaded(loc);
-               });
+             for (var i = 0; i < arr.length; i++) {
+               result[i] = Object.assign({}, arr[i]); // shallow copy
 
-               if (missing.length) {
-                 missing.forEach(function (loc) {
-                   context.loadTileAtLoc(loc);
-                 });
-                 return true;
-               }
+               result[i].index = i;
              }
 
-             return false;
+             return result;
            }
+         }
+       }
 
-           function incompleteRelation(id) {
-             var entity = context.entity(id);
-             return entity.type === 'relation' && !entity.isComplete(context.graph());
-           }
-         };
+       function actionAddMidpoint(midpoint, node) {
+         return function (graph) {
+           graph = graph.replace(node.move(midpoint.loc));
+           var parents = utilArrayIntersection(graph.parentWays(graph.entity(midpoint.edge[0])), graph.parentWays(graph.entity(midpoint.edge[1])));
+           parents.forEach(function (way) {
+             for (var i = 0; i < way.nodes.length - 1; i++) {
+               if (geoEdgeEqual([way.nodes[i], way.nodes[i + 1]], midpoint.edge)) {
+                 graph = graph.replace(graph.entity(way.id).addNode(node.id, i + 1)); // Add only one midpoint on doubled-back segments,
+                 // turning them into self-intersections.
 
-         operation.tooltip = function () {
-           var disable = operation.disabled();
-           return disable ? _t('operations.reflect.' + disable + '.' + multi) : _t('operations.reflect.description.' + axis + '.' + multi);
+                 return;
+               }
+             }
+           });
+           return graph;
          };
+       }
 
-         operation.annotation = function () {
-           return _t('operations.reflect.annotation.' + axis + '.feature', {
-             n: selectedIDs.length
-           });
+       // https://github.com/openstreetmap/potlatch2/blob/master/net/systemeD/halcyon/connection/actions/AddNodeToWayAction.as
+       function actionAddVertex(wayId, nodeId, index) {
+         return function (graph) {
+           return graph.replace(graph.entity(wayId).addNode(nodeId, index));
          };
+       }
 
-         operation.id = 'reflect-' + axis;
-         operation.keys = [_t('operations.reflect.key.' + axis)];
-         operation.title = _t('operations.reflect.title.' + axis);
-         operation.behavior = behaviorOperation(context).which(operation);
-         return operation;
+       function actionChangeMember(relationId, member, memberIndex) {
+         return function (graph) {
+           return graph.replace(graph.entity(relationId).updateMember(member, memberIndex));
+         };
        }
 
-       function operationMove(context, selectedIDs) {
-         var multi = selectedIDs.length === 1 ? 'single' : 'multiple';
-         var nodes = utilGetAllNodes(selectedIDs, context.graph());
-         var coords = nodes.map(function (n) {
-           return n.loc;
-         });
-         var extent = utilTotalExtent(selectedIDs, context.graph());
+       function actionChangePreset(entityID, oldPreset, newPreset, skipFieldDefaults) {
+         return function action(graph) {
+           var entity = graph.entity(entityID);
+           var geometry = entity.geometry(graph);
+           var tags = entity.tags; // preserve tags that the new preset might care about, if any
 
-         var operation = function operation() {
-           context.enter(modeMove(context, selectedIDs));
+           if (oldPreset) tags = oldPreset.unsetTags(tags, geometry, newPreset && newPreset.addTags ? Object.keys(newPreset.addTags) : null);
+           if (newPreset) tags = newPreset.setTags(tags, geometry, skipFieldDefaults);
+           return graph.replace(entity.update({
+             tags: tags
+           }));
          };
+       }
 
-         operation.available = function () {
-           return selectedIDs.length > 1 || context.entity(selectedIDs[0]).type !== 'node';
+       function actionChangeTags(entityId, tags) {
+         return function (graph) {
+           var entity = graph.entity(entityId);
+           return graph.replace(entity.update({
+             tags: tags
+           }));
          };
+       }
 
-         operation.disabled = function () {
-           if (extent.percentContainedIn(context.map().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(incompleteRelation)) {
-             return 'incomplete_relation';
-           }
-
-           return false;
+       function osmNode() {
+         if (!(this instanceof osmNode)) {
+           return new osmNode().initialize(arguments);
+         } else if (arguments.length) {
+           this.initialize(arguments);
+         }
+       }
+       osmEntity.node = osmNode;
+       osmNode.prototype = Object.create(osmEntity.prototype);
+       Object.assign(osmNode.prototype, {
+         type: 'node',
+         loc: [9999, 9999],
+         extent: function extent() {
+           return new geoExtent(this.loc);
+         },
+         geometry: function geometry(graph) {
+           return graph["transient"](this, 'geometry', function () {
+             return graph.isPoi(this) ? 'point' : 'vertex';
+           });
+         },
+         move: function move(loc) {
+           return this.update({
+             loc: loc
+           });
+         },
+         isDegenerate: function isDegenerate() {
+           return !(Array.isArray(this.loc) && this.loc.length === 2 && this.loc[0] >= -180 && this.loc[0] <= 180 && this.loc[1] >= -90 && this.loc[1] <= 90);
+         },
+         // Inspect tags and geometry to determine which direction(s) this node/vertex points
+         directions: function directions(resolver, projection) {
+           var val;
+           var i; // which tag to use?
 
-           function someMissing() {
-             if (context.inIntro()) return false;
-             var osm = context.connection();
+           if (this.isHighwayIntersection(resolver) && (this.tags.stop || '').toLowerCase() === 'all') {
+             // all-way stop tag on a highway intersection
+             val = 'all';
+           } else {
+             // generic direction tag
+             val = (this.tags.direction || '').toLowerCase(); // better suffix-style direction tag
 
-             if (osm) {
-               var missing = coords.filter(function (loc) {
-                 return !osm.isDataLoaded(loc);
-               });
+             var re = /:direction$/i;
+             var keys = Object.keys(this.tags);
 
-               if (missing.length) {
-                 missing.forEach(function (loc) {
-                   context.loadTileAtLoc(loc);
-                 });
-                 return true;
+             for (i = 0; i < keys.length; i++) {
+               if (re.test(keys[i])) {
+                 val = this.tags[keys[i]].toLowerCase();
+                 break;
                }
              }
-
-             return false;
-           }
-
-           function incompleteRelation(id) {
-             var entity = context.entity(id);
-             return entity.type === 'relation' && !entity.isComplete(context.graph());
            }
-         };
-
-         operation.tooltip = function () {
-           var disable = operation.disabled();
-           return disable ? _t('operations.move.' + disable + '.' + multi) : _t('operations.move.description.' + multi);
-         };
 
-         operation.annotation = function () {
-           return selectedIDs.length === 1 ? _t('operations.move.annotation.' + context.graph().geometry(selectedIDs[0])) : _t('operations.move.annotation.feature', {
-             n: selectedIDs.length
-           });
-         };
+           if (val === '') return [];
+           var cardinal = {
+             north: 0,
+             n: 0,
+             northnortheast: 22,
+             nne: 22,
+             northeast: 45,
+             ne: 45,
+             eastnortheast: 67,
+             ene: 67,
+             east: 90,
+             e: 90,
+             eastsoutheast: 112,
+             ese: 112,
+             southeast: 135,
+             se: 135,
+             southsoutheast: 157,
+             sse: 157,
+             south: 180,
+             s: 180,
+             southsouthwest: 202,
+             ssw: 202,
+             southwest: 225,
+             sw: 225,
+             westsouthwest: 247,
+             wsw: 247,
+             west: 270,
+             w: 270,
+             westnorthwest: 292,
+             wnw: 292,
+             northwest: 315,
+             nw: 315,
+             northnorthwest: 337,
+             nnw: 337
+           };
+           var values = val.split(';');
+           var results = [];
+           values.forEach(function (v) {
+             // swap cardinal for numeric directions
+             if (cardinal[v] !== undefined) {
+               v = cardinal[v];
+             } // numeric direction - just add to results
 
-         operation.id = 'move';
-         operation.keys = [_t('operations.move.key')];
-         operation.title = _t('operations.move.title');
-         operation.behavior = behaviorOperation(context).which(operation);
-         operation.mouseOnly = true;
-         return operation;
-       }
 
-       function modeRotate(context, entityIDs) {
-         var mode = {
-           id: 'rotate',
-           button: 'browse'
-         };
-         var keybinding = utilKeybinding('rotate');
-         var behaviors = [behaviorEdit(context), operationCircularize(context, entityIDs).behavior, operationDelete(context, entityIDs).behavior, operationMove(context, entityIDs).behavior, operationOrthogonalize(context, entityIDs).behavior, operationReflectLong(context, entityIDs).behavior, operationReflectShort(context, entityIDs).behavior];
-         var annotation = entityIDs.length === 1 ? _t('operations.rotate.annotation.' + context.graph().geometry(entityIDs[0])) : _t('operations.rotate.annotation.feature', {
-           n: entityIDs.length
-         });
+             if (v !== '' && !isNaN(+v)) {
+               results.push(+v);
+               return;
+             } // string direction - inspect parent ways
 
-         var _prevGraph;
 
-         var _prevAngle;
+             var lookBackward = this.tags['traffic_sign:backward'] || v === 'backward' || v === 'both' || v === 'all';
+             var lookForward = this.tags['traffic_sign:forward'] || v === 'forward' || v === 'both' || v === 'all';
+             if (!lookForward && !lookBackward) return;
+             var nodeIds = {};
+             resolver.parentWays(this).forEach(function (parent) {
+               var nodes = parent.nodes;
 
-         var _prevTransform;
+               for (i = 0; i < nodes.length; i++) {
+                 if (nodes[i] === this.id) {
+                   // match current entity
+                   if (lookForward && i > 0) {
+                     nodeIds[nodes[i - 1]] = true; // look back to prev node
+                   }
 
-         var _pivot;
+                   if (lookBackward && i < nodes.length - 1) {
+                     nodeIds[nodes[i + 1]] = true; // look ahead to next node
+                   }
+                 }
+               }
+             }, this);
+             Object.keys(nodeIds).forEach(function (nodeId) {
+               // +90 because geoAngle returns angle from X axis, not Y (north)
+               results.push(geoAngle(this, resolver.entity(nodeId), projection) * (180 / Math.PI) + 90);
+             }, this);
+           }, this);
+           return utilArrayUniq(results);
+         },
+         isEndpoint: function isEndpoint(resolver) {
+           return resolver["transient"](this, 'isEndpoint', function () {
+             var id = this.id;
+             return resolver.parentWays(this).filter(function (parent) {
+               return !parent.isClosed() && !!parent.affix(id);
+             }).length > 0;
+           });
+         },
+         isConnected: function isConnected(resolver) {
+           return resolver["transient"](this, 'isConnected', function () {
+             var parents = resolver.parentWays(this);
 
-         function doRotate() {
-           var fn;
+             if (parents.length > 1) {
+               // vertex is connected to multiple parent ways
+               for (var i in parents) {
+                 if (parents[i].geometry(resolver) === 'line' && parents[i].hasInterestingTags()) return true;
+               }
+             } else if (parents.length === 1) {
+               var way = parents[0];
+               var nodes = way.nodes.slice();
 
-           if (context.graph() !== _prevGraph) {
-             fn = context.perform;
-           } else {
-             fn = context.replace;
-           } // projection changed, recalculate _pivot
+               if (way.isClosed()) {
+                 nodes.pop();
+               } // ignore connecting node if closed
+               // return true if vertex appears multiple times (way is self intersecting)
 
 
-           var projection = context.projection;
-           var currTransform = projection.transform();
+               return nodes.indexOf(this.id) !== nodes.lastIndexOf(this.id);
+             }
 
-           if (!_prevTransform || currTransform.k !== _prevTransform.k || currTransform.x !== _prevTransform.x || currTransform.y !== _prevTransform.y) {
-             var nodes = utilGetAllNodes(entityIDs, context.graph());
-             var points = nodes.map(function (n) {
-               return projection(n.loc);
+             return false;
+           });
+         },
+         parentIntersectionWays: function parentIntersectionWays(resolver) {
+           return resolver["transient"](this, 'parentIntersectionWays', function () {
+             return resolver.parentWays(this).filter(function (parent) {
+               return (parent.tags.highway || parent.tags.waterway || parent.tags.railway || parent.tags.aeroway) && parent.geometry(resolver) === 'line';
              });
-             _pivot = getPivot(points);
-             _prevAngle = undefined;
-           }
-
-           var currMouse = context.map().mouse();
-           var currAngle = Math.atan2(currMouse[1] - _pivot[1], currMouse[0] - _pivot[0]);
-           if (typeof _prevAngle === 'undefined') _prevAngle = currAngle;
-           var delta = currAngle - _prevAngle;
-           fn(actionRotate(entityIDs, _pivot, delta, projection));
-           _prevTransform = currTransform;
-           _prevAngle = currAngle;
-           _prevGraph = context.graph();
-         }
-
-         function getPivot(points) {
-           var _pivot;
-
-           if (points.length === 1) {
-             _pivot = points[0];
-           } else if (points.length === 2) {
-             _pivot = geoVecInterp(points[0], points[1], 0.5);
-           } else {
-             var polygonHull = d3_polygonHull(points);
-
-             if (polygonHull.length === 2) {
-               _pivot = geoVecInterp(points[0], points[1], 0.5);
-             } else {
-               _pivot = d3_polygonCentroid(d3_polygonHull(points));
+           });
+         },
+         isIntersection: function isIntersection(resolver) {
+           return this.parentIntersectionWays(resolver).length > 1;
+         },
+         isHighwayIntersection: function isHighwayIntersection(resolver) {
+           return resolver["transient"](this, 'isHighwayIntersection', function () {
+             return resolver.parentWays(this).filter(function (parent) {
+               return parent.tags.highway && parent.geometry(resolver) === 'line';
+             }).length > 1;
+           });
+         },
+         isOnAddressLine: function isOnAddressLine(resolver) {
+           return resolver["transient"](this, 'isOnAddressLine', function () {
+             return resolver.parentWays(this).filter(function (parent) {
+               return parent.tags.hasOwnProperty('addr:interpolation') && parent.geometry(resolver) === 'line';
+             }).length > 0;
+           });
+         },
+         asJXON: function asJXON(changeset_id) {
+           var r = {
+             node: {
+               '@id': this.osmId(),
+               '@lon': this.loc[0],
+               '@lat': this.loc[1],
+               '@version': this.version || 0,
+               tag: Object.keys(this.tags).map(function (k) {
+                 return {
+                   keyAttributes: {
+                     k: k,
+                     v: this.tags[k]
+                   }
+                 };
+               }, this)
              }
-           }
-
-           return _pivot;
-         }
-
-         function finish(d3_event) {
-           d3_event.stopPropagation();
-           context.replace(actionNoop(), annotation);
-           context.enter(modeSelect(context, entityIDs));
-         }
-
-         function cancel() {
-           context.pop();
-           context.enter(modeSelect(context, entityIDs));
-         }
-
-         function undone() {
-           context.enter(modeBrowse(context));
+           };
+           if (changeset_id) r.node['@changeset'] = changeset_id;
+           return r;
+         },
+         asGeoJSON: function asGeoJSON() {
+           return {
+             type: 'Point',
+             coordinates: this.loc
+           };
          }
+       });
 
-         mode.enter = function () {
-           context.features().forceVisible(entityIDs);
-           behaviors.forEach(context.install);
-           context.surface().on('mousemove.rotate', doRotate).on('click.rotate', finish);
-           context.history().on('undone.rotate', undone);
-           keybinding.on('⎋', cancel).on('↩', finish);
-           select(document).call(keybinding);
-         };
+       function actionCircularize(wayId, projection, maxAngle) {
+         maxAngle = (maxAngle || 20) * Math.PI / 180;
 
-         mode.exit = function () {
-           behaviors.forEach(context.uninstall);
-           context.surface().on('mousemove.rotate', null).on('click.rotate', null);
-           context.history().on('undone.rotate', null);
-           select(document).call(keybinding.unbind);
-           context.features().forceVisible([]);
-         };
+         var action = function action(graph, t) {
+           if (t === null || !isFinite(t)) t = 1;
+           t = Math.min(Math.max(+t, 0), 1);
+           var way = graph.entity(wayId);
+           var origNodes = {};
+           graph.childNodes(way).forEach(function (node) {
+             if (!origNodes[node.id]) origNodes[node.id] = node;
+           });
 
-         mode.selectedIDs = function () {
-           if (!arguments.length) return entityIDs; // no assign
+           if (!way.isConvex(graph)) {
+             graph = action.makeConvex(graph);
+           }
 
-           return mode;
-         };
+           var nodes = utilArrayUniq(graph.childNodes(way));
+           var keyNodes = nodes.filter(function (n) {
+             return graph.parentWays(n).length !== 1;
+           });
+           var points = nodes.map(function (n) {
+             return projection(n.loc);
+           });
+           var keyPoints = keyNodes.map(function (n) {
+             return projection(n.loc);
+           });
+           var centroid = points.length === 2 ? geoVecInterp(points[0], points[1], 0.5) : d3_polygonCentroid(points);
+           var radius = d3_median(points, function (p) {
+             return geoVecLength(centroid, p);
+           });
+           var sign = d3_polygonArea(points) > 0 ? 1 : -1;
+           var ids, i, j, k; // we need at least two key nodes for the algorithm to work
 
-         return mode;
-       }
+           if (!keyNodes.length) {
+             keyNodes = [nodes[0]];
+             keyPoints = [points[0]];
+           }
 
-       function operationRotate(context, selectedIDs) {
-         var multi = selectedIDs.length === 1 ? 'single' : 'multiple';
-         var nodes = utilGetAllNodes(selectedIDs, context.graph());
-         var coords = nodes.map(function (n) {
-           return n.loc;
-         });
-         var extent = utilTotalExtent(selectedIDs, context.graph());
+           if (keyNodes.length === 1) {
+             var index = nodes.indexOf(keyNodes[0]);
+             var oppositeIndex = Math.floor((index + nodes.length / 2) % nodes.length);
+             keyNodes.push(nodes[oppositeIndex]);
+             keyPoints.push(points[oppositeIndex]);
+           } // key points and nodes are those connected to the ways,
+           // 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.
 
-         var operation = function operation() {
-           context.enter(modeRotate(context, selectedIDs));
-         };
 
-         operation.available = function () {
-           return nodes.length >= 2;
-         };
+           for (i = 0; i < keyPoints.length; i++) {
+             var nextKeyNodeIndex = (i + 1) % keyNodes.length;
+             var startNode = keyNodes[i];
+             var endNode = keyNodes[nextKeyNodeIndex];
+             var startNodeIndex = nodes.indexOf(startNode);
+             var endNodeIndex = nodes.indexOf(endNode);
+             var numberNewPoints = -1;
+             var indexRange = endNodeIndex - startNodeIndex;
+             var nearNodes = {};
+             var inBetweenNodes = [];
+             var startAngle, endAngle, totalAngle, eachAngle;
+             var angle, loc, node, origNode;
 
-         operation.disabled = function () {
-           if (extent.percentContainedIn(context.map().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(incompleteRelation)) {
-             return 'incomplete_relation';
-           }
+             if (indexRange < 0) {
+               indexRange += nodes.length;
+             } // position this key node
 
-           return false;
 
-           function someMissing() {
-             if (context.inIntro()) return false;
-             var osm = context.connection();
+             var distance = geoVecLength(centroid, keyPoints[i]) || 1e-4;
+             keyPoints[i] = [centroid[0] + (keyPoints[i][0] - centroid[0]) / distance * radius, centroid[1] + (keyPoints[i][1] - centroid[1]) / distance * radius];
+             loc = projection.invert(keyPoints[i]);
+             node = keyNodes[i];
+             origNode = origNodes[node.id];
+             node = node.move(geoVecInterp(origNode.loc, loc, t));
+             graph = graph.replace(node); // figure out the between delta angle we want to match to
 
-             if (osm) {
-               var missing = coords.filter(function (loc) {
-                 return !osm.isDataLoaded(loc);
-               });
+             startAngle = Math.atan2(keyPoints[i][1] - centroid[1], keyPoints[i][0] - centroid[0]);
+             endAngle = Math.atan2(keyPoints[nextKeyNodeIndex][1] - centroid[1], keyPoints[nextKeyNodeIndex][0] - centroid[0]);
+             totalAngle = endAngle - startAngle; // detects looping around -pi/pi
 
-               if (missing.length) {
-                 missing.forEach(function (loc) {
-                   context.loadTileAtLoc(loc);
-                 });
-                 return true;
-               }
+             if (totalAngle * sign > 0) {
+               totalAngle = -sign * (2 * Math.PI - Math.abs(totalAngle));
              }
 
-             return false;
-           }
+             do {
+               numberNewPoints++;
+               eachAngle = totalAngle / (indexRange + numberNewPoints);
+             } while (Math.abs(eachAngle) > maxAngle); // move existing nodes
 
-           function incompleteRelation(id) {
-             var entity = context.entity(id);
-             return entity.type === 'relation' && !entity.isComplete(context.graph());
-           }
-         };
 
-         operation.tooltip = function () {
-           var disable = operation.disabled();
-           return disable ? _t('operations.rotate.' + disable + '.' + multi) : _t('operations.rotate.description.' + multi);
-         };
+             for (j = 1; j < indexRange; j++) {
+               angle = startAngle + j * eachAngle;
+               loc = projection.invert([centroid[0] + Math.cos(angle) * radius, centroid[1] + Math.sin(angle) * radius]);
+               node = nodes[(j + startNodeIndex) % nodes.length];
+               origNode = origNodes[node.id];
+               nearNodes[node.id] = angle;
+               node = node.move(geoVecInterp(origNode.loc, loc, t));
+               graph = graph.replace(node);
+             } // add new in between nodes if necessary
 
-         operation.annotation = function () {
-           return selectedIDs.length === 1 ? _t('operations.rotate.annotation.' + context.graph().geometry(selectedIDs[0])) : _t('operations.rotate.annotation.feature', {
-             n: selectedIDs.length
-           });
-         };
 
-         operation.id = 'rotate';
-         operation.keys = [_t('operations.rotate.key')];
-         operation.title = _t('operations.rotate.title');
-         operation.behavior = behaviorOperation(context).which(operation);
-         operation.mouseOnly = true;
-         return operation;
-       }
+             for (j = 0; j < numberNewPoints; j++) {
+               angle = startAngle + (indexRange + j) * eachAngle;
+               loc = projection.invert([centroid[0] + Math.cos(angle) * radius, centroid[1] + Math.sin(angle) * radius]); // choose a nearnode to use as the original
 
-       function modeMove(context, entityIDs, baseGraph) {
-         var mode = {
-           id: 'move',
-           button: 'browse'
-         };
-         var keybinding = utilKeybinding('move');
-         var behaviors = [behaviorEdit(context), operationCircularize(context, entityIDs).behavior, operationDelete(context, entityIDs).behavior, operationOrthogonalize(context, entityIDs).behavior, operationReflectLong(context, entityIDs).behavior, operationReflectShort(context, entityIDs).behavior, operationRotate(context, entityIDs).behavior];
-         var annotation = entityIDs.length === 1 ? _t('operations.move.annotation.' + context.graph().geometry(entityIDs[0])) : _t('operations.move.annotation.feature', {
-           n: entityIDs.length
-         });
+               var min = Infinity;
 
-         var _prevGraph;
+               for (var nodeId in nearNodes) {
+                 var nearAngle = nearNodes[nodeId];
+                 var dist = Math.abs(nearAngle - angle);
 
-         var _cache;
+                 if (dist < min) {
+                   min = dist;
+                   origNode = origNodes[nodeId];
+                 }
+               }
 
-         var _origin;
+               node = osmNode({
+                 loc: geoVecInterp(origNode.loc, loc, t)
+               });
+               graph = graph.replace(node);
+               nodes.splice(endNodeIndex + j, 0, node);
+               inBetweenNodes.push(node.id);
+             } // Check for other ways that share these keyNodes..
+             // If keyNodes are adjacent in both ways,
+             // we can add inBetweenNodes to that shared way too..
 
-         var _nudgeInterval;
 
-         function doMove(nudge) {
-           nudge = nudge || [0, 0];
-           var fn;
+             if (indexRange === 1 && inBetweenNodes.length) {
+               var startIndex1 = way.nodes.lastIndexOf(startNode.id);
+               var endIndex1 = way.nodes.lastIndexOf(endNode.id);
+               var wayDirection1 = endIndex1 - startIndex1;
 
-           if (_prevGraph !== context.graph()) {
-             _cache = {};
-             _origin = context.map().mouseCoordinates();
-             fn = context.perform;
-           } else {
-             fn = context.overwrite;
-           }
+               if (wayDirection1 < -1) {
+                 wayDirection1 = 1;
+               }
 
-           var currMouse = context.map().mouse();
-           var origMouse = context.projection(_origin);
-           var delta = geoVecSubtract(geoVecSubtract(currMouse, origMouse), nudge);
-           fn(actionMove(entityIDs, delta, context.projection, _cache));
-           _prevGraph = context.graph();
-         }
+               var parentWays = graph.parentWays(keyNodes[i]);
 
-         function startNudge(nudge) {
-           if (_nudgeInterval) window.clearInterval(_nudgeInterval);
-           _nudgeInterval = window.setInterval(function () {
-             context.map().pan(nudge);
-             doMove(nudge);
-           }, 50);
-         }
+               for (j = 0; j < parentWays.length; j++) {
+                 var sharedWay = parentWays[j];
+                 if (sharedWay === way) continue;
 
-         function stopNudge() {
-           if (_nudgeInterval) {
-             window.clearInterval(_nudgeInterval);
-             _nudgeInterval = null;
-           }
-         }
+                 if (sharedWay.areAdjacent(startNode.id, endNode.id)) {
+                   var startIndex2 = sharedWay.nodes.lastIndexOf(startNode.id);
+                   var endIndex2 = sharedWay.nodes.lastIndexOf(endNode.id);
+                   var wayDirection2 = endIndex2 - startIndex2;
+                   var insertAt = endIndex2;
 
-         function move() {
-           doMove();
-           var nudge = geoViewportEdge(context.map().mouse(), context.map().dimensions());
+                   if (wayDirection2 < -1) {
+                     wayDirection2 = 1;
+                   }
 
-           if (nudge) {
-             startNudge(nudge);
-           } else {
-             stopNudge();
-           }
-         }
+                   if (wayDirection1 !== wayDirection2) {
+                     inBetweenNodes.reverse();
+                     insertAt = startIndex2;
+                   }
 
-         function finish(d3_event) {
-           d3_event.stopPropagation();
-           context.replace(actionNoop(), annotation);
-           context.enter(modeSelect(context, entityIDs));
-           stopNudge();
-         }
+                   for (k = 0; k < inBetweenNodes.length; k++) {
+                     sharedWay = sharedWay.addNode(inBetweenNodes[k], insertAt + k);
+                   }
 
-         function cancel() {
-           if (baseGraph) {
-             while (context.graph() !== baseGraph) {
-               context.pop();
+                   graph = graph.replace(sharedWay);
+                 }
+               }
              }
+           } // update the way to have all the new nodes
 
-             context.enter(modeBrowse(context));
-           } else {
-             context.pop();
-             context.enter(modeSelect(context, entityIDs));
-           }
 
-           stopNudge();
-         }
+           ids = nodes.map(function (n) {
+             return n.id;
+           });
+           ids.push(ids[0]);
+           way = way.update({
+             nodes: ids
+           });
+           graph = graph.replace(way);
+           return graph;
+         };
 
-         function undone() {
-           context.enter(modeBrowse(context));
-         }
+         action.makeConvex = function (graph) {
+           var way = graph.entity(wayId);
+           var nodes = utilArrayUniq(graph.childNodes(way));
+           var points = nodes.map(function (n) {
+             return projection(n.loc);
+           });
+           var sign = d3_polygonArea(points) > 0 ? 1 : -1;
+           var hull = d3_polygonHull(points);
+           var i, j; // D3 convex hulls go counterclockwise..
 
-         mode.enter = function () {
-           _origin = context.map().mouseCoordinates();
-           _prevGraph = null;
-           _cache = {};
-           context.features().forceVisible(entityIDs);
-           behaviors.forEach(context.install);
-           context.surface().on('mousemove.move', move).on('click.move', finish);
-           context.history().on('undone.move', undone);
-           keybinding.on('⎋', cancel).on('↩', finish);
-           select(document).call(keybinding);
-         };
+           if (sign === -1) {
+             nodes.reverse();
+             points.reverse();
+           }
+
+           for (i = 0; i < hull.length - 1; i++) {
+             var startIndex = points.indexOf(hull[i]);
+             var endIndex = points.indexOf(hull[i + 1]);
+             var indexRange = endIndex - startIndex;
 
-         mode.exit = function () {
-           stopNudge();
-           behaviors.forEach(function (behavior) {
-             context.uninstall(behavior);
-           });
-           context.surface().on('mousemove.move', null).on('click.move', null);
-           context.history().on('undone.move', null);
-           select(document).call(keybinding.unbind);
-           context.features().forceVisible([]);
-         };
+             if (indexRange < 0) {
+               indexRange += nodes.length;
+             } // move interior nodes to the surface of the convex hull..
 
-         mode.selectedIDs = function () {
-           if (!arguments.length) return entityIDs; // no assign
 
-           return mode;
+             for (j = 1; j < indexRange; j++) {
+               var point = geoVecInterp(hull[i], hull[i + 1], j / indexRange);
+               var node = nodes[(j + startIndex) % nodes.length].move(projection.invert(point));
+               graph = graph.replace(node);
+             }
+           }
+
+           return graph;
          };
 
-         return mode;
-       }
+         action.disabled = function (graph) {
+           if (!graph.entity(wayId).isClosed()) {
+             return 'not_closed';
+           } //disable when already circular
 
-       function behaviorPaste(context) {
-         function doPaste(d3_event) {
-           // prevent paste during low zoom selection
-           if (!context.map().withinEditableZoom()) return;
-           d3_event.preventDefault();
-           var baseGraph = context.graph();
-           var mouse = context.map().mouse();
-           var projection = context.projection;
-           var viewport = geoExtent(projection.clipExtent()).polygon();
-           if (!geoPointInPolygon(mouse, viewport)) return;
-           var oldIDs = context.copyIDs();
-           if (!oldIDs.length) return;
-           var extent = geoExtent();
-           var oldGraph = context.copyGraph();
-           var newIDs = [];
-           var action = actionCopyEntities(oldIDs, oldGraph);
-           context.perform(action);
-           var copies = action.copies();
-           var originals = new Set();
-           Object.values(copies).forEach(function (entity) {
-             originals.add(entity.id);
-           });
 
-           for (var id in copies) {
-             var oldEntity = oldGraph.entity(id);
-             var newEntity = copies[id];
+           var way = graph.entity(wayId);
+           var nodes = utilArrayUniq(graph.childNodes(way));
+           var points = nodes.map(function (n) {
+             return projection(n.loc);
+           });
+           var hull = d3_polygonHull(points);
+           var epsilonAngle = Math.PI / 180;
 
-             extent._extend(oldEntity.extent(oldGraph)); // Exclude child nodes from newIDs if their parent way was also copied.
+           if (hull.length !== points.length || hull.length < 3) {
+             return false;
+           }
 
+           var centroid = d3_polygonCentroid(points);
+           var radius = geoVecLengthSquare(centroid, points[0]);
+           var i, actualPoint; // compare distances between centroid and points
 
-             var parents = context.graph().parentWays(newEntity);
-             var parentCopied = parents.some(function (parent) {
-               return originals.has(parent.id);
-             });
+           for (i = 0; i < hull.length; i++) {
+             actualPoint = hull[i];
+             var actualDist = geoVecLengthSquare(actualPoint, centroid);
+             var diff = Math.abs(actualDist - radius); //compare distances with epsilon-error (5%)
 
-             if (!parentCopied) {
-               newIDs.push(newEntity.id);
+             if (diff > 0.05 * radius) {
+               return false;
              }
-           } // Put pasted objects where mouse pointer is..
+           } //check if central angles are smaller than maxAngle
 
 
-           var copyPoint = context.copyLonLat() && projection(context.copyLonLat()) || projection(extent.center());
-           var delta = geoVecSubtract(mouse, copyPoint);
-           context.perform(actionMove(newIDs, delta, projection));
-           context.enter(modeMove(context, newIDs, baseGraph));
-         }
+           for (i = 0; i < hull.length; i++) {
+             actualPoint = hull[i];
+             var nextPoint = hull[(i + 1) % hull.length];
+             var startAngle = Math.atan2(actualPoint[1] - centroid[1], actualPoint[0] - centroid[0]);
+             var endAngle = Math.atan2(nextPoint[1] - centroid[1], nextPoint[0] - centroid[0]);
+             var angle = endAngle - startAngle;
 
-         function behavior() {
-           context.keybinding().on(uiCmd('⌘V'), doPaste);
-           return behavior;
-         }
+             if (angle < 0) {
+               angle = -angle;
+             }
 
-         behavior.off = function () {
-           context.keybinding().off(uiCmd('⌘V'));
+             if (angle > Math.PI) {
+               angle = 2 * Math.PI - angle;
+             }
+
+             if (angle > maxAngle + epsilonAngle) {
+               return false;
+             }
+           }
+
+           return 'already_circular';
          };
 
-         return behavior;
+         action.transitionable = true;
+         return action;
        }
 
-       // `String.prototype.repeat` method
-       // https://tc39.es/ecma262/#sec-string.prototype.repeat
-       _export({ target: 'String', proto: true }, {
-         repeat: stringRepeat
-       });
+       function actionDeleteWay(wayID) {
+         function canDeleteNode(node, graph) {
+           // don't delete nodes still attached to ways or relations
+           if (graph.parentWays(node).length || graph.parentRelations(node).length) return false;
+           var geometries = osmNodeGeometriesForTags(node.tags); // don't delete if this node can be a standalone point
 
-       /*
-           `behaviorDrag` is like `d3_behavior.drag`, with the following differences:
+           if (geometries.point) return false; // delete if this node only be a vertex
 
-           * The `origin` function is expected to return an [x, y] tuple rather than an
-             {x, y} object.
-           * The events are `start`, `move`, and `end`.
-             (https://github.com/mbostock/d3/issues/563)
-           * The `start` event is not dispatched until the first cursor movement occurs.
-             (https://github.com/mbostock/d3/pull/368)
-           * The `move` event has a `point` and `delta` [x, y] tuple properties rather
-             than `x`, `y`, `dx`, and `dy` properties.
-           * The `end` event is not dispatched if no movement occurs.
-           * An `off` function is available that unbinds the drag's internal event handlers.
-        */
+           if (geometries.vertex) return true; // iD doesn't know if this should be a point or vertex,
+           // so only delete if there are no interesting tags
 
-       function behaviorDrag() {
-         var dispatch$1 = dispatch('start', 'move', 'end'); // see also behaviorSelect
+           return !node.hasInterestingTags();
+         }
 
-         var _tolerancePx = 1; // keep this low to facilitate pixel-perfect micromapping
+         var action = function action(graph) {
+           var way = graph.entity(wayID);
+           graph.parentRelations(way).forEach(function (parent) {
+             parent = parent.removeMembersWithID(wayID);
+             graph = graph.replace(parent);
 
-         var _penTolerancePx = 4; // styluses can be touchy so require greater movement - #1981
+             if (parent.isDegenerate()) {
+               graph = actionDeleteRelation(parent.id)(graph);
+             }
+           });
+           new Set(way.nodes).forEach(function (nodeID) {
+             graph = graph.replace(way.removeNode(nodeID));
+             var node = graph.entity(nodeID);
 
-         var _origin = null;
-         var _selector = '';
+             if (canDeleteNode(node, graph)) {
+               graph = graph.remove(node);
+             }
+           });
+           return graph.remove(way);
+         };
 
-         var _targetNode;
+         return action;
+       }
 
-         var _targetEntity;
+       function actionDeleteMultiple(ids) {
+         var actions = {
+           way: actionDeleteWay,
+           node: actionDeleteNode,
+           relation: actionDeleteRelation
+         };
 
-         var _surface;
+         var action = function action(graph) {
+           ids.forEach(function (id) {
+             if (graph.hasEntity(id)) {
+               // It may have been deleted already.
+               graph = actions[graph.entity(id).type](id)(graph);
+             }
+           });
+           return graph;
+         };
 
-         var _pointerId; // use pointer events on supported platforms; fallback to mouse events
+         return action;
+       }
 
+       function actionDeleteRelation(relationID, allowUntaggedMembers) {
+         function canDeleteEntity(entity, graph) {
+           return !graph.parentWays(entity).length && !graph.parentRelations(entity).length && !entity.hasInterestingTags() && !allowUntaggedMembers;
+         }
 
-         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
+         var action = function action(graph) {
+           var relation = graph.entity(relationID);
+           graph.parentRelations(relation).forEach(function (parent) {
+             parent = parent.removeMembersWithID(relationID);
+             graph = graph.replace(parent);
 
-         var d3_event_userSelectProperty = utilPrefixCSSProperty('UserSelect');
+             if (parent.isDegenerate()) {
+               graph = actionDeleteRelation(parent.id)(graph);
+             }
+           });
+           var memberIDs = utilArrayUniq(relation.members.map(function (m) {
+             return m.id;
+           }));
+           memberIDs.forEach(function (memberID) {
+             graph = graph.replace(relation.removeMembersWithID(memberID));
+             var entity = graph.entity(memberID);
 
-         var d3_event_userSelectSuppress = function d3_event_userSelectSuppress() {
-           var selection$1 = selection();
-           var select = selection$1.style(d3_event_userSelectProperty);
-           selection$1.style(d3_event_userSelectProperty, 'none');
-           return function () {
-             selection$1.style(d3_event_userSelectProperty, select);
-           };
+             if (canDeleteEntity(entity, graph)) {
+               graph = actionDeleteMultiple([memberID])(graph);
+             }
+           });
+           return graph.remove(relation);
          };
 
-         function pointerdown(d3_event) {
-           if (_pointerId) return;
-           _pointerId = d3_event.pointerId || 'mouse';
-           _targetNode = this; // only force reflow once per drag
+         return action;
+       }
 
-           var pointerLocGetter = utilFastMouse(_surface || _targetNode.parentNode);
-           var offset;
-           var startOrigin = pointerLocGetter(d3_event);
-           var started = false;
-           var selectEnable = d3_event_userSelectSuppress();
-           select(window).on(_pointerPrefix + 'move.drag', pointermove).on(_pointerPrefix + 'up.drag pointercancel.drag', pointerup, true);
+       function actionDeleteNode(nodeId) {
+         var action = function action(graph) {
+           var node = graph.entity(nodeId);
+           graph.parentWays(node).forEach(function (parent) {
+             parent = parent.removeNode(nodeId);
+             graph = graph.replace(parent);
 
-           if (_origin) {
-             offset = _origin.call(_targetNode, _targetEntity);
-             offset = [offset[0] - startOrigin[0], offset[1] - startOrigin[1]];
-           } else {
-             offset = [0, 0];
-           }
+             if (parent.isDegenerate()) {
+               graph = actionDeleteWay(parent.id)(graph);
+             }
+           });
+           graph.parentRelations(node).forEach(function (parent) {
+             parent = parent.removeMembersWithID(nodeId);
+             graph = graph.replace(parent);
 
-           d3_event.stopPropagation();
+             if (parent.isDegenerate()) {
+               graph = actionDeleteRelation(parent.id)(graph);
+             }
+           });
+           return graph.remove(node);
+         };
 
-           function pointermove(d3_event) {
-             if (_pointerId !== (d3_event.pointerId || 'mouse')) return;
-             var p = pointerLocGetter(d3_event);
+         return action;
+       }
 
-             if (!started) {
-               var dist = geoVecLength(startOrigin, p);
-               var tolerance = d3_event.pointerType === 'pen' ? _penTolerancePx : _tolerancePx; // don't start until the drag has actually moved somewhat
+       //
+       // First choose a node to be the survivor, with preference given
+       // to an existing (not new) node.
+       //
+       // Tags and relation memberships of of non-surviving nodes are merged
+       // to the survivor.
+       //
+       // This is the inverse of `iD.actionDisconnect`.
+       //
+       // Reference:
+       //   https://github.com/openstreetmap/potlatch2/blob/master/net/systemeD/halcyon/connection/actions/MergeNodesAction.as
+       //   https://github.com/openstreetmap/josm/blob/mirror/src/org/openstreetmap/josm/actions/MergeNodesAction.java
+       //
 
-               if (dist < tolerance) return;
-               started = true;
-               dispatch$1.call('start', this, d3_event, _targetEntity); // Don't send a `move` event in the same cycle as `start` since dragging
-               // a midpoint will convert the target to a node.
-             } else {
-               startOrigin = p;
-               d3_event.stopPropagation();
-               d3_event.preventDefault();
-               var dx = p[0] - startOrigin[0];
-               var dy = p[1] - startOrigin[1];
-               dispatch$1.call('move', this, d3_event, _targetEntity, [p[0] + offset[0], p[1] + offset[1]], [dx, dy]);
-             }
-           }
+       function actionConnect(nodeIDs) {
+         var action = function action(graph) {
+           var survivor;
+           var node;
+           var parents;
+           var i, j; // Choose a survivor node, prefer an existing (not new) node - #4974
 
-           function pointerup(d3_event) {
-             if (_pointerId !== (d3_event.pointerId || 'mouse')) return;
-             _pointerId = null;
+           for (i = 0; i < nodeIDs.length; i++) {
+             survivor = graph.entity(nodeIDs[i]);
+             if (survivor.version) break; // found one
+           } // Replace all non-surviving nodes with the survivor and merge tags.
 
-             if (started) {
-               dispatch$1.call('end', this, d3_event, _targetEntity);
-               d3_event.preventDefault();
-             }
 
-             select(window).on(_pointerPrefix + 'move.drag', null).on(_pointerPrefix + 'up.drag pointercancel.drag', null);
-             selectEnable();
-           }
-         }
+           for (i = 0; i < nodeIDs.length; i++) {
+             node = graph.entity(nodeIDs[i]);
+             if (node.id === survivor.id) continue;
+             parents = graph.parentWays(node);
 
-         function behavior(selection) {
-           var matchesSelector = utilPrefixDOMProperty('matchesSelector');
-           var delegate = pointerdown;
+             for (j = 0; j < parents.length; j++) {
+               graph = graph.replace(parents[j].replaceNode(node.id, survivor.id));
+             }
 
-           if (_selector) {
-             delegate = function delegate(d3_event) {
-               var root = this;
-               var target = d3_event.target;
+             parents = graph.parentRelations(node);
 
-               for (; target && target !== root; target = target.parentNode) {
-                 var datum = target.__data__;
-                 _targetEntity = datum instanceof osmNote ? datum : datum && datum.properties && datum.properties.entity;
+             for (j = 0; j < parents.length; j++) {
+               graph = graph.replace(parents[j].replaceMember(node, survivor));
+             }
 
-                 if (_targetEntity && target[matchesSelector](_selector)) {
-                   return pointerdown.call(target, d3_event);
-                 }
-               }
-             };
+             survivor = survivor.mergeTags(node.tags);
+             graph = actionDeleteNode(node.id)(graph);
            }
 
-           selection.on(_pointerPrefix + 'down.drag' + _selector, delegate);
-         }
-
-         behavior.off = function (selection) {
-           selection.on(_pointerPrefix + 'down.drag' + _selector, null);
-         };
+           graph = graph.replace(survivor); // find and delete any degenerate ways created by connecting adjacent vertices
 
-         behavior.selector = function (_) {
-           if (!arguments.length) return _selector;
-           _selector = _;
-           return behavior;
-         };
+           parents = graph.parentWays(survivor);
 
-         behavior.origin = function (_) {
-           if (!arguments.length) return _origin;
-           _origin = _;
-           return behavior;
-         };
+           for (i = 0; i < parents.length; i++) {
+             if (parents[i].isDegenerate()) {
+               graph = actionDeleteWay(parents[i].id)(graph);
+             }
+           }
 
-         behavior.cancel = function () {
-           select(window).on(_pointerPrefix + 'move.drag', null).on(_pointerPrefix + 'up.drag pointercancel.drag', null);
-           return behavior;
+           return graph;
          };
 
-         behavior.targetNode = function (_) {
-           if (!arguments.length) return _targetNode;
-           _targetNode = _;
-           return behavior;
-         };
+         action.disabled = function (graph) {
+           var seen = {};
+           var restrictionIDs = [];
+           var survivor;
+           var node, way;
+           var relations, relation, role;
+           var i, j, k; // Choose a survivor node, prefer an existing (not new) node - #4974
 
-         behavior.targetEntity = function (_) {
-           if (!arguments.length) return _targetEntity;
-           _targetEntity = _;
-           return behavior;
-         };
+           for (i = 0; i < nodeIDs.length; i++) {
+             survivor = graph.entity(nodeIDs[i]);
+             if (survivor.version) break; // found one
+           } // 1. disable if the nodes being connected have conflicting relation roles
 
-         behavior.surface = function (_) {
-           if (!arguments.length) return _surface;
-           _surface = _;
-           return behavior;
-         };
 
-         return utilRebind(behavior, dispatch$1, 'on');
-       }
+           for (i = 0; i < nodeIDs.length; i++) {
+             node = graph.entity(nodeIDs[i]);
+             relations = graph.parentRelations(node);
 
-       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);
+             for (j = 0; j < relations.length; j++) {
+               relation = relations[j];
+               role = relation.memberById(node.id).role || ''; // if this node is a via node in a restriction, remember for later
 
-         var _nudgeInterval;
+               if (relation.hasFromViaTo()) {
+                 restrictionIDs.push(relation.id);
+               }
 
-         var _restoreSelectedIDs = [];
-         var _wasMidpoint = false;
-         var _isCancelled = false;
+               if (seen[relation.id] !== undefined && seen[relation.id] !== role) {
+                 return 'relation';
+               } else {
+                 seen[relation.id] = role;
+               }
+             }
+           } // gather restrictions for parent ways
 
-         var _activeEntity;
 
-         var _startLoc;
+           for (i = 0; i < nodeIDs.length; i++) {
+             node = graph.entity(nodeIDs[i]);
+             var parents = graph.parentWays(node);
 
-         var _lastLoc;
+             for (j = 0; j < parents.length; j++) {
+               var parent = parents[j];
+               relations = graph.parentRelations(parent);
 
-         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);
-         }
+               for (k = 0; k < relations.length; k++) {
+                 relation = relations[k];
 
-         function stopNudge() {
-           if (_nudgeInterval) {
-             window.clearInterval(_nudgeInterval);
-             _nudgeInterval = null;
-           }
-         }
+                 if (relation.hasFromViaTo()) {
+                   restrictionIDs.push(relation.id);
+                 }
+               }
+             }
+           } // test restrictions
 
-         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());
+           restrictionIDs = utilArrayUniq(restrictionIDs);
 
-           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
+           for (i = 0; i < restrictionIDs.length; i++) {
+             relation = graph.entity(restrictionIDs[i]);
+             if (!relation.isComplete(graph)) continue;
+             var memberWays = relation.members.filter(function (m) {
+               return m.type === 'way';
+             }).map(function (m) {
+               return graph.entity(m.id);
+             });
+             memberWays = utilArrayUniq(memberWays);
+             var f = relation.memberByRole('from');
+             var t = relation.memberByRole('to');
+             var isUturn = f.id === t.id; // 2a. disable if connection would damage a restriction
+             // (a key node is a node at the junction of ways)
 
-             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');
-               }
+             var nodes = {
+               from: [],
+               via: [],
+               to: [],
+               keyfrom: [],
+               keyto: []
+             };
 
-               return _t('operations.connect.annotation.from_vertex.to_sibling_vertex');
+             for (j = 0; j < relation.members.length; j++) {
+               collectNodes(relation.members[j], nodes);
              }
-           }
 
-           return _t('operations.connect.annotation.from_' + nodeGeometry + '.to_' + targetGeometry);
-         }
+             nodes.keyfrom = utilArrayUniq(nodes.keyfrom.filter(hasDuplicates));
+             nodes.keyto = utilArrayUniq(nodes.keyto.filter(hasDuplicates));
+             var filter = keyNodeFilter(nodes.keyfrom, nodes.keyto);
+             nodes.from = nodes.from.filter(filter);
+             nodes.via = nodes.via.filter(filter);
+             nodes.to = nodes.to.filter(filter);
+             var connectFrom = false;
+             var connectVia = false;
+             var connectTo = false;
+             var connectKeyFrom = false;
+             var connectKeyTo = false;
 
-         function shouldSnapToNode(target) {
-           if (!_activeEntity) return false;
-           return _activeEntity.geometry(context.graph()) !== 'vertex' || target.geometry(context.graph()) === 'vertex' || _mainPresetIndex.allowsVertex(target, context.graph());
-         }
+             for (j = 0; j < nodeIDs.length; j++) {
+               var n = nodeIDs[j];
+
+               if (nodes.from.indexOf(n) !== -1) {
+                 connectFrom = true;
+               }
 
-         function origin(entity) {
-           return context.projection(entity.loc);
-         }
+               if (nodes.via.indexOf(n) !== -1) {
+                 connectVia = true;
+               }
 
-         function keydown(d3_event) {
-           if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
-             if (context.surface().classed('nope')) {
-               context.surface().classed('nope-suppressed', true);
-             }
+               if (nodes.to.indexOf(n) !== -1) {
+                 connectTo = true;
+               }
 
-             context.surface().classed('nope', false).classed('nope-disabled', true);
-           }
-         }
+               if (nodes.keyfrom.indexOf(n) !== -1) {
+                 connectKeyFrom = true;
+               }
 
-         function keyup(d3_event) {
-           if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
-             if (context.surface().classed('nope-suppressed')) {
-               context.surface().classed('nope', true);
+               if (nodes.keyto.indexOf(n) !== -1) {
+                 connectKeyTo = 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 (connectFrom && connectTo && !isUturn) {
+               return 'restriction';
+             }
 
-           if (_isCancelled) {
-             if (hasHidden) {
-               context.ui().flash.duration(4000).iconName('#iD-icon-no').label(_t('modes.drag_node.connected_to_hidden'))();
+             if (connectFrom && connectVia) {
+               return 'restriction';
              }
 
-             return drag.cancel();
-           }
+             if (connectTo && connectVia) {
+               return 'restriction';
+             } // connecting to a key node -
+             // if both nodes are on a member way (i.e. part of the turn restriction),
+             // the connecting node must be adjacent to the key node.
 
-           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());
-           }
+             if (connectKeyFrom || connectKeyTo) {
+               if (nodeIDs.length !== 2) {
+                 return 'restriction';
+               }
 
-           _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()`
+               var n0 = null;
+               var n1 = null;
 
+               for (j = 0; j < memberWays.length; j++) {
+                 way = memberWays[j];
 
-         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 : {};
-           }
-         }
+                 if (way.contains(nodeIDs[0])) {
+                   n0 = nodeIDs[0];
+                 }
 
-         function doMove(d3_event, entity, nudge) {
-           nudge = nudge || [0, 0];
-           var currPoint = d3_event && d3_event.point || context.projection(_lastLoc);
-           var currMouse = geoVecSubtract(currPoint, nudge);
-           var loc = context.projection.invert(currMouse);
-           var target, edge;
+                 if (way.contains(nodeIDs[1])) {
+                   n1 = nodeIDs[1];
+                 }
+               }
 
-           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 (n0 && n1) {
+                 // both nodes are part of the restriction
+                 var ok = false;
 
-             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);
+                 for (j = 0; j < memberWays.length; j++) {
+                   way = memberWays[j];
 
-               if (edge) {
-                 loc = edge.loc;
+                   if (way.areAdjacent(n0, n1)) {
+                     ok = true;
+                     break;
+                   }
+                 }
+
+                 if (!ok) {
+                   return 'restriction';
+                 }
                }
-             }
-           }
+             } // 2b. disable if nodes being connected will destroy a member way in a restriction
+             // (to test, make a copy and try actually connecting the nodes)
 
-           context.replace(actionMoveNode(entity.id, loc)); // Below here: validations
 
-           var isInvalid = false; // Check if this connection to `target` could cause relations to break..
+             for (j = 0; j < memberWays.length; j++) {
+               way = memberWays[j].update({}); // make copy
 
-           if (target) {
-             isInvalid = hasRelationConflict(entity, target, edge, context.graph());
-           } // Check if this drag causes the geometry to break..
+               for (k = 0; k < nodeIDs.length; k++) {
+                 if (nodeIDs[k] === survivor.id) continue;
 
+                 if (way.areAdjacent(nodeIDs[k], survivor.id)) {
+                   way = way.removeNode(nodeIDs[k]);
+                 } else {
+                   way = way.replaceNode(nodeIDs[k], survivor.id);
+                 }
+               }
 
-           if (!isInvalid) {
-             isInvalid = hasInvalidGeometry(entity, context.graph());
+               if (way.isDegenerate()) {
+                 return 'restriction';
+               }
+             }
            }
 
-           var nope = context.surface().classed('nope');
+           return false; // if a key node appears multiple times (indexOf !== lastIndexOf) it's a FROM-VIA or TO-VIA junction
 
-           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: _mainPresetIndex.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('')();
-             }
+           function hasDuplicates(n, i, arr) {
+             return arr.indexOf(n) !== arr.lastIndexOf(n);
            }
 
-           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);
+           function keyNodeFilter(froms, tos) {
+             return function (n) {
+               return froms.indexOf(n) === -1 && tos.indexOf(n) === -1;
+             };
            }
 
-           _lastLoc = loc;
-         } // Uses `actionConnect.disabled()` to know whether this connection is ok..
+           function collectNodes(member, collection) {
+             var entity = graph.hasEntity(member.id);
+             if (!entity) return;
+             var role = member.role || '';
 
+             if (!collection[role]) {
+               collection[role] = [];
+             }
 
-         function hasRelationConflict(entity, target, edge, graph) {
-           var testGraph = graph.update(); // copy
-           // if snapping to way - add midpoint there and consider that the target..
+             if (member.type === 'node') {
+               collection[role].push(member.id);
 
-           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?
+               if (role === 'via') {
+                 collection.keyfrom.push(member.id);
+                 collection.keyto.push(member.id);
+               }
+             } else if (member.type === 'way') {
+               collection[role].push.apply(collection[role], entity.nodes);
 
+               if (role === 'from' || role === 'via') {
+                 collection.keyfrom.push(entity.first());
+                 collection.keyfrom.push(entity.last());
+               }
 
-           var ids = [entity.id, target.id];
-           return actionConnect(ids).disabled(testGraph);
-         }
+               if (role === 'to' || role === 'via') {
+                 collection.keyto.push(entity.first());
+                 collection.keyto.push(entity.last());
+               }
+             }
+           }
+         };
 
-         function hasInvalidGeometry(entity, graph) {
-           var parents = graph.parentWays(entity);
-           var i, j, k;
+         return action;
+       }
 
-           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
+       function actionCopyEntities(ids, fromGraph) {
+         var _copies = {};
 
-             var relations = graph.parentRelations(parent);
+         var action = function action(graph) {
+           ids.forEach(function (id) {
+             fromGraph.entity(id).copy(fromGraph, _copies);
+           });
 
-             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 (var id in _copies) {
+             graph = graph.replace(_copies[id]);
+           }
 
-               for (k = 0; k < rings.length; k++) {
-                 nodes = rings[k].nodes;
+           return graph;
+         };
 
-                 if (nodes.find(function (n) {
-                   return n.id === entity.id;
-                 })) {
-                   activeIndex = k;
+         action.copies = function () {
+           return _copies;
+         };
 
-                   if (geoHasSelfIntersections(nodes, entity.id)) {
-                     return 'multipolygonMember';
-                   }
-                 }
+         return action;
+       }
 
-                 rings[k].coords = nodes.map(function (n) {
-                   return n.loc;
-                 });
-               } // test active ring for intersections with other rings in the multipolygon
+       function actionDeleteMember(relationId, memberIndex) {
+         return function (graph) {
+           var relation = graph.entity(relationId).removeMember(memberIndex);
+           graph = graph.replace(relation);
 
+           if (relation.isDegenerate()) {
+             graph = actionDeleteRelation(relation.id)(graph);
+           }
 
-               for (k = 0; k < rings.length; k++) {
-                 if (k === activeIndex) continue; // make sure active ring doesn't cross passive rings
+           return graph;
+         };
+       }
 
-                 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.
+       function actionDiscardTags(difference, discardTags) {
+         discardTags = discardTags || {};
+         return function (graph) {
+           difference.modified().forEach(checkTags);
+           difference.created().forEach(checkTags);
+           return graph;
 
+           function checkTags(entity) {
+             var keys = Object.keys(entity.tags);
+             var didDiscard = false;
+             var tags = {};
 
-             if (activeIndex === null) {
-               nodes = parent.nodes.map(function (nodeID) {
-                 return graph.entity(nodeID);
-               });
+             for (var i = 0; i < keys.length; i++) {
+               var k = keys[i];
 
-               if (nodes.length && geoHasSelfIntersections(nodes, entity.id)) {
-                 return parent.geometry(graph);
+               if (discardTags[k] || !entity.tags[k]) {
+                 didDiscard = true;
+               } else {
+                 tags[k] = entity.tags[k];
                }
              }
-           }
-
-           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 = geoViewportEdge(point, context.map().dimensions());
 
-           if (nudge) {
-             startNudge(d3_event, entity, nudge);
-           } else {
-             stopNudge();
+             if (didDiscard) {
+               graph = graph.replace(entity.update({
+                 tags: tags
+               }));
+             }
            }
-         }
+         };
+       }
 
-         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
+       //
+       // Optionally, disconnect only the given ways.
+       //
+       // For testing convenience, accepts an ID to assign to the (first) new node.
+       // Normally, this will be undefined and the way will automatically
+       // be assigned a new ID.
+       //
+       // This is the inverse of `iD.actionConnect`.
+       //
+       // Reference:
+       //   https://github.com/openstreetmap/potlatch2/blob/master/net/systemeD/halcyon/connection/actions/UnjoinNodeAction.as
+       //   https://github.com/openstreetmap/josm/blob/mirror/src/org/openstreetmap/josm/actions/UnGlueAction.java
+       //
 
-           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));
-           }
+       function actionDisconnect(nodeId, newNodeId) {
+         var wayIds;
 
-           if (wasPoint) {
-             context.enter(modeSelect(context, [entity.id]));
-           } else {
-             var reselection = _restoreSelectedIDs.filter(function (id) {
-               return context.graph().hasEntity(id);
+         var action = function action(graph) {
+           var node = graph.entity(nodeId);
+           var connections = action.connections(graph);
+           connections.forEach(function (connection) {
+             var way = graph.entity(connection.wayID);
+             var newNode = osmNode({
+               id: newNodeId,
+               loc: node.loc,
+               tags: node.tags
              });
+             graph = graph.replace(newNode);
 
-             if (reselection.length) {
-               context.enter(modeSelect(context, reselection));
+             if (connection.index === 0 && way.isArea()) {
+               // replace shared node with shared node..
+               graph = graph.replace(way.replaceNode(way.nodes[0], newNode.id));
+             } else if (way.isClosed() && connection.index === way.nodes.length - 1) {
+               // replace closing node with new new node..
+               graph = graph.replace(way.unclose().addNode(newNode.id));
              } else {
-               context.enter(modeBrowse(context));
+               // replace shared node with multiple new nodes..
+               graph = graph.replace(way.updateNode(newNode.id, connection.index));
              }
-           }
-         }
+           });
+           return graph;
+         };
 
-         function _actionBounceBack(nodeID, toLoc) {
-           var moveNode = actionMoveNode(nodeID, toLoc);
+         action.connections = function (graph) {
+           var candidates = [];
+           var keeping = false;
+           var parentWays = graph.parentWays(graph.entity(nodeId));
+           var way, waynode;
 
-           var action = function action(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);
-           };
+           for (var i = 0; i < parentWays.length; i++) {
+             way = parentWays[i];
 
-           action.transitionable = true;
-           return action;
-         }
+             if (wayIds && wayIds.indexOf(way.id) === -1) {
+               keeping = true;
+               continue;
+             }
 
-         function cancel() {
-           drag.cancel();
-           context.enter(modeBrowse(context));
-         }
+             if (way.isArea() && way.nodes[0] === nodeId) {
+               candidates.push({
+                 wayID: way.id,
+                 index: 0
+               });
+             } else {
+               for (var j = 0; j < way.nodes.length; j++) {
+                 waynode = way.nodes[j];
 
-         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);
+                 if (waynode === nodeId) {
+                   if (way.isClosed() && parentWays.length > 1 && wayIds && wayIds.indexOf(way.id) !== -1 && j === way.nodes.length - 1) {
+                     continue;
+                   }
 
-         mode.enter = function () {
-           context.install(hover);
-           context.install(edit);
-           select(window).on('keydown.dragNode', keydown).on('keyup.dragNode', keyup);
-           context.history().on('undone.drag-node', cancel);
-         };
+                   candidates.push({
+                     wayID: way.id,
+                     index: j
+                   });
+                 }
+               }
+             }
+           }
 
-         mode.exit = function () {
-           context.ui().sidebar.hover.cancel();
-           context.uninstall(hover);
-           context.uninstall(edit);
-           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();
+           return keeping ? candidates : candidates.slice(1);
          };
 
-         mode.selectedIDs = function () {
-           if (!arguments.length) return _activeEntity ? [_activeEntity.id] : []; // no assign
+         action.disabled = function (graph) {
+           var connections = action.connections(graph);
+           if (connections.length === 0) return 'not_connected';
+           var parentWays = graph.parentWays(graph.entity(nodeId));
+           var seenRelationIds = {};
+           var sharedRelation;
+           parentWays.forEach(function (way) {
+             var relations = graph.parentRelations(way);
+             relations.forEach(function (relation) {
+               if (relation.id in seenRelationIds) {
+                 if (wayIds) {
+                   if (wayIds.indexOf(way.id) !== -1 || wayIds.indexOf(seenRelationIds[relation.id]) !== -1) {
+                     sharedRelation = relation;
+                   }
+                 } else {
+                   sharedRelation = relation;
+                 }
+               } else {
+                 seenRelationIds[relation.id] = way.id;
+               }
+             });
+           });
+           if (sharedRelation) return 'relation';
+         };
 
-           return mode;
+         action.limitWays = function (val) {
+           if (!arguments.length) return wayIds;
+           wayIds = val;
+           return action;
          };
 
-         mode.activeID = function () {
-           if (!arguments.length) return _activeEntity && _activeEntity.id; // no assign
+         return action;
+       }
 
-           return mode;
-         };
+       function actionExtract(entityID, projection) {
+         var extractedNodeID;
 
-         mode.restoreSelectedIDs = function (_) {
-           if (!arguments.length) return _restoreSelectedIDs;
-           _restoreSelectedIDs = _;
-           return mode;
-         };
+         var action = function action(graph) {
+           var entity = graph.entity(entityID);
 
-         mode.behavior = drag;
-         return mode;
-       }
+           if (entity.type === 'node') {
+             return extractFromNode(entity, graph);
+           }
 
-       // @@search logic
-       fixRegexpWellKnownSymbolLogic('search', 1, function (SEARCH, nativeSearch, maybeCallNative) {
-         return [
-           // `String.prototype.search` method
-           // https://tc39.es/ecma262/#sec-string.prototype.search
-           function search(regexp) {
-             var O = requireObjectCoercible(this);
-             var searcher = regexp == undefined ? undefined : regexp[SEARCH];
-             return searcher !== undefined ? searcher.call(regexp, O) : new RegExp(regexp)[SEARCH](String(O));
-           },
-           // `RegExp.prototype[@@search]` method
-           // https://tc39.es/ecma262/#sec-regexp.prototype-@@search
-           function (regexp) {
-             var res = maybeCallNative(nativeSearch, regexp, this);
-             if (res.done) return res.value;
+           return extractFromWayOrRelation(entity, graph);
+         };
 
-             var rx = anObject(regexp);
-             var S = String(this);
+         function extractFromNode(node, graph) {
+           extractedNodeID = node.id; // Create a new node to replace the one we will detach
 
-             var previousLastIndex = rx.lastIndex;
-             if (!sameValue(previousLastIndex, 0)) rx.lastIndex = 0;
-             var result = regexpExecAbstract(rx, S);
-             if (!sameValue(rx.lastIndex, previousLastIndex)) rx.lastIndex = previousLastIndex;
-             return result === null ? -1 : result.index;
-           }
-         ];
-       });
+           var replacement = osmNode({
+             loc: node.loc
+           });
+           graph = graph.replace(replacement); // Process each way in turn, updating the graph as we go
 
-       // Safari bug https://bugs.webkit.org/show_bug.cgi?id=200829
-       var NON_GENERIC = !!nativePromiseConstructor && fails(function () {
-         nativePromiseConstructor.prototype['finally'].call({ then: function () { /* empty */ } }, function () { /* empty */ });
-       });
+           graph = graph.parentWays(node).reduce(function (accGraph, parentWay) {
+             return accGraph.replace(parentWay.replaceNode(entityID, replacement.id));
+           }, graph); // Process any relations too
 
-       // `Promise.prototype.finally` method
-       // https://tc39.es/ecma262/#sec-promise.prototype.finally
-       _export({ target: 'Promise', proto: true, real: true, forced: NON_GENERIC }, {
-         'finally': function (onFinally) {
-           var C = speciesConstructor(this, getBuiltIn('Promise'));
-           var isFunction = typeof onFinally == 'function';
-           return this.then(
-             isFunction ? function (x) {
-               return promiseResolve(C, onFinally()).then(function () { return x; });
-             } : onFinally,
-             isFunction ? function (e) {
-               return promiseResolve(C, onFinally()).then(function () { throw e; });
-             } : onFinally
-           );
+           return graph.parentRelations(node).reduce(function (accGraph, parentRel) {
+             return accGraph.replace(parentRel.replaceMember(node, replacement));
+           }, graph);
          }
-       });
-
-       // patch native Promise.prototype for native async functions
-       if ( typeof nativePromiseConstructor == 'function' && !nativePromiseConstructor.prototype['finally']) {
-         redefine(nativePromiseConstructor.prototype, 'finally', getBuiltIn('Promise').prototype['finally']);
-       }
 
-       function quickselect$1(arr, k, left, right, compare) {
-         quickselectStep(arr, k, left || 0, right || arr.length - 1, compare || defaultCompare);
-       }
+         function extractFromWayOrRelation(entity, graph) {
+           var fromGeometry = entity.geometry(graph);
+           var keysToCopyAndRetain = ['source', 'wheelchair'];
+           var keysToRetain = ['area'];
+           var buildingKeysToRetain = ['architect', 'building', 'height', 'layer'];
+           var extractedLoc = d3_geoPath(projection).centroid(entity.asGeoJSON(graph));
+           extractedLoc = extractedLoc && projection.invert(extractedLoc);
 
-       function quickselectStep(arr, k, left, right, compare) {
-         while (right > left) {
-           if (right - left > 600) {
-             var n = right - left + 1;
-             var m = k - left + 1;
-             var z = Math.log(n);
-             var s = 0.5 * Math.exp(2 * z / 3);
-             var sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1);
-             var newLeft = Math.max(left, Math.floor(k - m * s / n + sd));
-             var newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd));
-             quickselectStep(arr, k, newLeft, newRight, compare);
+           if (!extractedLoc || !isFinite(extractedLoc[0]) || !isFinite(extractedLoc[1])) {
+             extractedLoc = entity.extent(graph).center();
            }
 
-           var t = arr[k];
-           var i = left;
-           var j = right;
-           swap$1(arr, left, k);
-           if (compare(arr[right], t) > 0) swap$1(arr, left, right);
+           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
 
-           while (i < j) {
-             swap$1(arr, i, j);
-             i++;
-             j--;
+           var pointTags = {};
 
-             while (compare(arr[i], t) < 0) {
-               i++;
+           for (var key in entityTags) {
+             if (entity.type === 'relation' && key === 'type') {
+               continue;
              }
 
-             while (compare(arr[j], t) > 0) {
-               j--;
+             if (keysToRetain.indexOf(key) !== -1) {
+               continue;
              }
+
+             if (isBuilding) {
+               // don't transfer building-related tags
+               if (buildingKeysToRetain.indexOf(key) !== -1 || 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]; // leave addresses and some other tags so they're on both features
+
+             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 (compare(arr[left], t) === 0) swap$1(arr, left, j);else {
-             j++;
-             swap$1(arr, j, right);
+           if (!isBuilding && !isIndoorArea && fromGeometry === 'area') {
+             // ensure that areas keep area geometry
+             entityTags.area = 'yes';
            }
-           if (j <= k) left = j + 1;
-           if (k <= j) right = j - 1;
+
+           var replacement = osmNode({
+             loc: extractedLoc,
+             tags: pointTags
+           });
+           graph = graph.replace(replacement);
+           extractedNodeID = replacement.id;
+           return graph.replace(entity.update({
+             tags: entityTags
+           }));
          }
-       }
 
-       function swap$1(arr, i, j) {
-         var tmp = arr[i];
-         arr[i] = arr[j];
-         arr[j] = tmp;
-       }
+         action.getExtractedNodeID = function () {
+           return extractedNodeID;
+         };
 
-       function defaultCompare(a, b) {
-         return a < b ? -1 : a > b ? 1 : 0;
+         return action;
        }
 
-       var RBush = /*#__PURE__*/function () {
-         function RBush() {
-           var maxEntries = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 9;
-
-           _classCallCheck(this, RBush);
+       //
+       // This is the inverse of `iD.actionSplit`.
+       //
+       // 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
+       //
 
-           // max entries in a node is 9 by default; min node fill is 40% for best performance
-           this._maxEntries = Math.max(4, maxEntries);
-           this._minEntries = Math.max(2, Math.ceil(this._maxEntries * 0.4));
-           this.clear();
+       function actionJoin(ids) {
+         function groupEntitiesByGeometry(graph) {
+           var entities = ids.map(function (id) {
+             return graph.entity(id);
+           });
+           return Object.assign({
+             line: []
+           }, utilArrayGroupBy(entities, function (entity) {
+             return entity.geometry(graph);
+           }));
          }
 
-         _createClass(RBush, [{
-           key: "all",
-           value: function all() {
-             return this._all(this.data, []);
-           }
-         }, {
-           key: "search",
-           value: function search(bbox) {
-             var node = this.data;
-             var result = [];
-             if (!intersects(bbox, node)) return result;
-             var toBBox = this.toBBox;
-             var nodesToSearch = [];
-
-             while (node) {
-               for (var i = 0; i < node.children.length; i++) {
-                 var child = node.children[i];
-                 var childBBox = node.leaf ? toBBox(child) : child;
+         var action = function action(graph) {
+           var ways = ids.map(graph.entity, graph);
+           var survivorID = ways[0].id; // if any of the ways are sided (e.g. coastline, cliff, kerb)
+           // sort them first so they establish the overall order - #6033
 
-                 if (intersects(bbox, childBBox)) {
-                   if (node.leaf) result.push(child);else if (contains(bbox, childBBox)) this._all(child, result);else nodesToSearch.push(child);
-                 }
-               }
+           ways.sort(function (a, b) {
+             var aSided = a.isSided();
+             var bSided = b.isSided();
+             return aSided && !bSided ? -1 : bSided && !aSided ? 1 : 0;
+           }); // Prefer to keep an existing way.
 
-               node = nodesToSearch.pop();
+           for (var i = 0; i < ways.length; i++) {
+             if (!ways[i].isNew()) {
+               survivorID = ways[i].id;
+               break;
              }
-
-             return result;
            }
-         }, {
-           key: "collides",
-           value: function collides(bbox) {
-             var node = this.data;
-             if (!intersects(bbox, node)) return false;
-             var nodesToSearch = [];
 
-             while (node) {
-               for (var i = 0; i < node.children.length; i++) {
-                 var child = node.children[i];
-                 var childBBox = node.leaf ? this.toBBox(child) : child;
+           var sequences = osmJoinWays(ways, graph);
+           var joined = sequences[0]; // We might need to reverse some of these ways before joining them.  #4688
+           // `joined.actions` property will contain any actions we need to apply.
 
-                 if (intersects(bbox, childBBox)) {
-                   if (node.leaf || contains(bbox, childBBox)) return true;
-                   nodesToSearch.push(child);
-                 }
-               }
+           graph = sequences.actions.reduce(function (g, action) {
+             return action(g);
+           }, graph);
+           var survivor = graph.entity(survivorID);
+           survivor = survivor.update({
+             nodes: joined.nodes.map(function (n) {
+               return n.id;
+             })
+           });
+           graph = graph.replace(survivor);
+           joined.forEach(function (way) {
+             if (way.id === survivorID) return;
+             graph.parentRelations(way).forEach(function (parent) {
+               graph = graph.replace(parent.replaceMember(way, survivor));
+             });
+             survivor = survivor.mergeTags(way.tags);
+             graph = graph.replace(survivor);
+             graph = actionDeleteWay(way.id)(graph);
+           }); // Finds if the join created a single-member multipolygon,
+           // and if so turns it into a basic area instead
 
-               node = nodesToSearch.pop();
+           function checkForSimpleMultipolygon() {
+             if (!survivor.isClosed()) return;
+             var multipolygons = graph.parentMultipolygons(survivor).filter(function (multipolygon) {
+               // find multipolygons where the survivor is the only member
+               return multipolygon.members.length === 1;
+             }); // skip if this is the single member of multiple multipolygons
+
+             if (multipolygons.length !== 1) return;
+             var multipolygon = multipolygons[0];
+
+             for (var key in survivor.tags) {
+               if (multipolygon.tags[key] && // don't collapse if tags cannot be cleanly merged
+               multipolygon.tags[key] !== survivor.tags[key]) return;
              }
 
-             return false;
-           }
-         }, {
-           key: "load",
-           value: function load(data) {
-             if (!(data && data.length)) return this;
+             survivor = survivor.mergeTags(multipolygon.tags);
+             graph = graph.replace(survivor);
+             graph = actionDeleteRelation(multipolygon.id, true
+             /* allow untagged members */
+             )(graph);
+             var tags = Object.assign({}, survivor.tags);
 
-             if (data.length < this._minEntries) {
-               for (var i = 0; i < data.length; i++) {
-                 this.insert(data[i]);
-               }
+             if (survivor.geometry(graph) !== 'area') {
+               // ensure the feature persists as an area
+               tags.area = 'yes';
+             }
 
-               return this;
-             } // recursively build the tree with the given data from scratch using OMT algorithm
+             delete tags.type; // remove type=multipolygon
 
+             survivor = survivor.update({
+               tags: tags
+             });
+             graph = graph.replace(survivor);
+           }
 
-             var node = this._build(data.slice(), 0, data.length - 1, 0);
+           checkForSimpleMultipolygon();
+           return graph;
+         }; // Returns the number of nodes the resultant way is expected to have
 
-             if (!this.data.children.length) {
-               // save as is if tree is empty
-               this.data = node;
-             } else if (this.data.height === node.height) {
-               // split root if trees have the same height
-               this._splitRoot(this.data, node);
-             } else {
-               if (this.data.height < node.height) {
-                 // swap trees if inserted one is bigger
-                 var tmpNode = this.data;
-                 this.data = node;
-                 node = tmpNode;
-               } // insert the small tree into the large tree at appropriate level
 
+         action.resultingWayNodesLength = function (graph) {
+           return ids.reduce(function (count, id) {
+             return count + graph.entity(id).nodes.length;
+           }, 0) - ids.length - 1;
+         };
 
-               this._insert(node, this.data.height - node.height - 1, true);
-             }
+         action.disabled = function (graph) {
+           var geometries = groupEntitiesByGeometry(graph);
 
-             return this;
-           }
-         }, {
-           key: "insert",
-           value: function insert(item) {
-             if (item) this._insert(item, this.data.height - 1);
-             return this;
-           }
-         }, {
-           key: "clear",
-           value: function clear() {
-             this.data = createNode([]);
-             return this;
+           if (ids.length < 2 || ids.length !== geometries.line.length) {
+             return 'not_eligible';
            }
-         }, {
-           key: "remove",
-           value: function remove(item, equalsFn) {
-             if (!item) return this;
-             var node = this.data;
-             var bbox = this.toBBox(item);
-             var path = [];
-             var indexes = [];
-             var i, parent, goingUp; // depth-first iterative tree traversal
-
-             while (node || path.length) {
-               if (!node) {
-                 // go up
-                 node = path.pop();
-                 parent = path[path.length - 1];
-                 i = indexes.pop();
-                 goingUp = true;
-               }
 
-               if (node.leaf) {
-                 // check current node
-                 var index = findItem(item, node.children, equalsFn);
+           var joined = osmJoinWays(ids.map(graph.entity, graph), graph);
 
-                 if (index !== -1) {
-                   // item found, remove the item and condense tree upwards
-                   node.children.splice(index, 1);
-                   path.push(node);
+           if (joined.length > 1) {
+             return 'not_adjacent';
+           } // Loop through all combinations of path-pairs
+           // to check potential intersections between all pairs
 
-                   this._condense(path);
 
-                   return this;
-                 }
-               }
+           for (var i = 0; i < ids.length - 1; i++) {
+             for (var j = i + 1; j < ids.length; j++) {
+               var path1 = graph.childNodes(graph.entity(ids[i])).map(function (e) {
+                 return e.loc;
+               });
+               var path2 = graph.childNodes(graph.entity(ids[j])).map(function (e) {
+                 return e.loc;
+               });
+               var intersections = geoPathIntersections(path1, path2); // Check if intersections are just nodes lying on top of
+               // each other/the line, as opposed to crossing it
 
-               if (!goingUp && !node.leaf && contains(node, bbox)) {
-                 // go down
-                 path.push(node);
-                 indexes.push(i);
-                 i = 0;
-                 parent = node;
-                 node = node.children[0];
-               } else if (parent) {
-                 // go right
-                 i++;
-                 node = parent.children[i];
-                 goingUp = false;
-               } else node = null; // nothing found
+               var common = utilArrayIntersection(joined[0].nodes.map(function (n) {
+                 return n.loc.toString();
+               }), intersections.map(function (n) {
+                 return n.toString();
+               }));
 
+               if (common.length !== intersections.length) {
+                 return 'paths_intersect';
+               }
              }
-
-             return this;
-           }
-         }, {
-           key: "toBBox",
-           value: function toBBox(item) {
-             return item;
-           }
-         }, {
-           key: "compareMinX",
-           value: function compareMinX(a, b) {
-             return a.minX - b.minX;
            }
-         }, {
-           key: "compareMinY",
-           value: function compareMinY(a, b) {
-             return a.minY - b.minY;
-           }
-         }, {
-           key: "toJSON",
-           value: function toJSON() {
-             return this.data;
-           }
-         }, {
-           key: "fromJSON",
-           value: function fromJSON(data) {
-             this.data = data;
-             return this;
-           }
-         }, {
-           key: "_all",
-           value: function _all(node, result) {
-             var nodesToSearch = [];
 
-             while (node) {
-               if (node.leaf) result.push.apply(result, _toConsumableArray(node.children));else nodesToSearch.push.apply(nodesToSearch, _toConsumableArray(node.children));
-               node = nodesToSearch.pop();
+           var nodeIds = joined[0].nodes.map(function (n) {
+             return n.id;
+           }).slice(1, -1);
+           var relation;
+           var tags = {};
+           var 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] && osmIsInterestingTag(k) && tags[k] !== way.tags[k]) {
+                 conflicting = true;
+               }
              }
+           });
 
-             return result;
+           if (relation) {
+             return 'restriction';
            }
-         }, {
-           key: "_build",
-           value: function _build(items, left, right, height) {
-             var N = right - left + 1;
-             var M = this._maxEntries;
-             var node;
 
-             if (N <= M) {
-               // reached leaf level; return leaf
-               node = createNode(items.slice(left, right + 1));
-               calcBBox(node, this.toBBox);
-               return node;
-             }
+           if (conflicting) {
+             return 'conflicting_tags';
+           }
+         };
 
-             if (!height) {
-               // target height of the bulk-loaded tree
-               height = Math.ceil(Math.log(N) / Math.log(M)); // target number of root entries to maximize storage utilization
+         return action;
+       }
 
-               M = Math.ceil(N / Math.pow(M, height - 1));
-             }
+       function actionMerge(ids) {
+         function groupEntitiesByGeometry(graph) {
+           var entities = ids.map(function (id) {
+             return graph.entity(id);
+           });
+           return Object.assign({
+             point: [],
+             area: [],
+             line: [],
+             relation: []
+           }, utilArrayGroupBy(entities, function (entity) {
+             return entity.geometry(graph);
+           }));
+         }
 
-             node = createNode([]);
-             node.leaf = false;
-             node.height = height; // split the items into M mostly square tiles
+         var action = function action(graph) {
+           var geometries = groupEntitiesByGeometry(graph);
+           var target = geometries.area[0] || geometries.line[0];
+           var points = geometries.point;
+           points.forEach(function (point) {
+             target = target.mergeTags(point.tags);
+             graph = graph.replace(target);
+             graph.parentRelations(point).forEach(function (parent) {
+               graph = graph.replace(parent.replaceMember(point, target));
+             });
+             var nodes = utilArrayUniq(graph.childNodes(target));
+             var removeNode = point;
 
-             var N2 = Math.ceil(N / M);
-             var N1 = N2 * Math.ceil(Math.sqrt(M));
-             multiSelect(items, left, right, N1, this.compareMinX);
+             for (var i = 0; i < nodes.length; i++) {
+               var node = nodes[i];
 
-             for (var i = left; i <= right; i += N1) {
-               var right2 = Math.min(i + N1 - 1, right);
-               multiSelect(items, i, right2, N2, this.compareMinY);
+               if (graph.parentWays(node).length > 1 || graph.parentRelations(node).length || node.hasInterestingTags()) {
+                 continue;
+               } // Found an uninteresting child node on the target way.
+               // Move orig point into its place to preserve point's history. #3683
 
-               for (var j = i; j <= right2; j += N2) {
-                 var right3 = Math.min(j + N2 - 1, right2); // pack each entry recursively
 
-                 node.children.push(this._build(items, j, right3, height - 1));
-               }
+               graph = graph.replace(point.update({
+                 tags: {},
+                 loc: node.loc
+               }));
+               target = target.replaceNode(node.id, point.id);
+               graph = graph.replace(target);
+               removeNode = node;
+               break;
              }
 
-             calcBBox(node, this.toBBox);
-             return node;
-           }
-         }, {
-           key: "_chooseSubtree",
-           value: function _chooseSubtree(bbox, node, level, path) {
-             while (true) {
-               path.push(node);
-               if (node.leaf || path.length - 1 === level) break;
-               var minArea = Infinity;
-               var minEnlargement = Infinity;
-               var targetNode = void 0;
+             graph = graph.remove(removeNode);
+           });
 
-               for (var i = 0; i < node.children.length; i++) {
-                 var child = node.children[i];
-                 var area = bboxArea(child);
-                 var enlargement = enlargedArea(bbox, child) - area; // choose entry with the least area enlargement
+           if (target.tags.area === 'yes') {
+             var tags = Object.assign({}, target.tags); // shallow copy
 
-                 if (enlargement < minEnlargement) {
-                   minEnlargement = enlargement;
-                   minArea = area < minArea ? area : minArea;
-                   targetNode = child;
-                 } else if (enlargement === minEnlargement) {
-                   // otherwise choose one with the smallest area
-                   if (area < minArea) {
-                     minArea = area;
-                     targetNode = child;
-                   }
-                 }
-               }
+             delete tags.area;
 
-               node = targetNode || node.children[0];
+             if (osmTagSuggestingArea(tags)) {
+               // remove the `area` tag if area geometry is now implied - #3851
+               target = target.update({
+                 tags: tags
+               });
+               graph = graph.replace(target);
              }
-
-             return node;
            }
-         }, {
-           key: "_insert",
-           value: function _insert(item, level, isNode) {
-             var bbox = isNode ? item : this.toBBox(item);
-             var insertPath = []; // find the best node for accommodating the item, saving all nodes along the path too
 
-             var node = this._chooseSubtree(bbox, this.data, level, insertPath); // put the item into the node
+           return graph;
+         };
 
+         action.disabled = function (graph) {
+           var geometries = groupEntitiesByGeometry(graph);
 
-             node.children.push(item);
-             extend$1(node, bbox); // split on node overflow; propagate upwards if necessary
+           if (geometries.point.length === 0 || geometries.area.length + geometries.line.length !== 1 || geometries.relation.length !== 0) {
+             return 'not_eligible';
+           }
+         };
 
-             while (level >= 0) {
-               if (insertPath[level].children.length > this._maxEntries) {
-                 this._split(insertPath, level);
+         return action;
+       }
 
-                 level--;
-               } else break;
-             } // adjust bboxes along the insertion path
+       //
+       // 1. move all the nodes to a common location
+       // 2. `actionConnect` them
 
+       function actionMergeNodes(nodeIDs, loc) {
+         // If there is a single "interesting" node, use that as the location.
+         // Otherwise return the average location of all the nodes.
+         function chooseLoc(graph) {
+           if (!nodeIDs.length) return null;
+           var sum = [0, 0];
+           var interestingCount = 0;
+           var interestingLoc;
 
-             this._adjustParentBBoxes(bbox, insertPath, level);
-           } // split overflowed node into two
+           for (var i = 0; i < nodeIDs.length; i++) {
+             var node = graph.entity(nodeIDs[i]);
 
-         }, {
-           key: "_split",
-           value: function _split(insertPath, level) {
-             var node = insertPath[level];
-             var M = node.children.length;
-             var m = this._minEntries;
+             if (node.hasInterestingTags()) {
+               interestingLoc = ++interestingCount === 1 ? node.loc : null;
+             }
 
-             this._chooseSplitAxis(node, m, M);
+             sum = geoVecAdd(sum, node.loc);
+           }
 
-             var splitIndex = this._chooseSplitIndex(node, m, M);
+           return interestingLoc || geoVecScale(sum, 1 / nodeIDs.length);
+         }
 
-             var newNode = createNode(node.children.splice(splitIndex, node.children.length - splitIndex));
-             newNode.height = node.height;
-             newNode.leaf = node.leaf;
-             calcBBox(node, this.toBBox);
-             calcBBox(newNode, this.toBBox);
-             if (level) insertPath[level - 1].children.push(newNode);else this._splitRoot(node, newNode);
-           }
-         }, {
-           key: "_splitRoot",
-           value: function _splitRoot(node, newNode) {
-             // split root node
-             this.data = createNode([node, newNode]);
-             this.data.height = node.height + 1;
-             this.data.leaf = false;
-             calcBBox(this.data, this.toBBox);
+         var action = function action(graph) {
+           if (nodeIDs.length < 2) return graph;
+           var toLoc = loc;
+
+           if (!toLoc) {
+             toLoc = chooseLoc(graph);
            }
-         }, {
-           key: "_chooseSplitIndex",
-           value: function _chooseSplitIndex(node, m, M) {
-             var index;
-             var minOverlap = Infinity;
-             var minArea = Infinity;
 
-             for (var i = m; i <= M - m; i++) {
-               var bbox1 = distBBox(node, 0, i, this.toBBox);
-               var bbox2 = distBBox(node, i, M, this.toBBox);
-               var overlap = intersectionArea(bbox1, bbox2);
-               var area = bboxArea(bbox1) + bboxArea(bbox2); // choose distribution with minimum overlap
+           for (var i = 0; i < nodeIDs.length; i++) {
+             var node = graph.entity(nodeIDs[i]);
 
-               if (overlap < minOverlap) {
-                 minOverlap = overlap;
-                 index = i;
-                 minArea = area < minArea ? area : minArea;
-               } else if (overlap === minOverlap) {
-                 // otherwise choose distribution with minimum area
-                 if (area < minArea) {
-                   minArea = area;
-                   index = i;
-                 }
-               }
+             if (node.loc !== toLoc) {
+               graph = graph.replace(node.move(toLoc));
              }
+           }
 
-             return index || M - m;
-           } // sorts node children by the best axis for split
+           return actionConnect(nodeIDs)(graph);
+         };
 
-         }, {
-           key: "_chooseSplitAxis",
-           value: function _chooseSplitAxis(node, m, M) {
-             var compareMinX = node.leaf ? this.compareMinX : compareNodeMinX;
-             var compareMinY = node.leaf ? this.compareMinY : compareNodeMinY;
+         action.disabled = function (graph) {
+           if (nodeIDs.length < 2) return 'not_eligible';
 
-             var xMargin = this._allDistMargin(node, m, M, compareMinX);
+           for (var i = 0; i < nodeIDs.length; i++) {
+             var entity = graph.entity(nodeIDs[i]);
+             if (entity.type !== 'node') return 'not_eligible';
+           }
 
-             var yMargin = this._allDistMargin(node, m, M, compareMinY); // if total distributions margin value is minimal for x, sort by minX,
-             // otherwise it's already sorted by minY
+           return actionConnect(nodeIDs).disabled(graph);
+         };
 
+         return action;
+       }
 
-             if (xMargin < yMargin) node.children.sort(compareMinX);
-           } // total margin of all possible split distributions where each node is at least m full
+       function osmChangeset() {
+         if (!(this instanceof osmChangeset)) {
+           return new osmChangeset().initialize(arguments);
+         } else if (arguments.length) {
+           this.initialize(arguments);
+         }
+       }
+       osmEntity.changeset = osmChangeset;
+       osmChangeset.prototype = Object.create(osmEntity.prototype);
+       Object.assign(osmChangeset.prototype, {
+         type: 'changeset',
+         extent: function extent() {
+           return new geoExtent();
+         },
+         geometry: function geometry() {
+           return 'changeset';
+         },
+         asJXON: function asJXON() {
+           return {
+             osm: {
+               changeset: {
+                 tag: Object.keys(this.tags).map(function (k) {
+                   return {
+                     '@k': k,
+                     '@v': this.tags[k]
+                   };
+                 }, this),
+                 '@version': 0.6,
+                 '@generator': 'iD'
+               }
+             }
+           };
+         },
+         // Generate [osmChange](http://wiki.openstreetmap.org/wiki/OsmChange)
+         // XML. Returns a string.
+         osmChangeJXON: function osmChangeJXON(changes) {
+           var changeset_id = this.id;
 
-         }, {
-           key: "_allDistMargin",
-           value: function _allDistMargin(node, m, M, compare) {
-             node.children.sort(compare);
-             var toBBox = this.toBBox;
-             var leftBBox = distBBox(node, 0, m, toBBox);
-             var rightBBox = distBBox(node, M - m, M, toBBox);
-             var margin = bboxMargin(leftBBox) + bboxMargin(rightBBox);
+           function nest(x, order) {
+             var groups = {};
 
-             for (var i = m; i < M - m; i++) {
-               var child = node.children[i];
-               extend$1(leftBBox, node.leaf ? toBBox(child) : child);
-               margin += bboxMargin(leftBBox);
+             for (var i = 0; i < x.length; i++) {
+               var tagName = Object.keys(x[i])[0];
+               if (!groups[tagName]) groups[tagName] = [];
+               groups[tagName].push(x[i][tagName]);
              }
 
-             for (var _i = M - m - 1; _i >= m; _i--) {
-               var _child = node.children[_i];
-               extend$1(rightBBox, node.leaf ? toBBox(_child) : _child);
-               margin += bboxMargin(rightBBox);
-             }
+             var ordered = {};
+             order.forEach(function (o) {
+               if (groups[o]) ordered[o] = groups[o];
+             });
+             return ordered;
+           } // sort relations in a changeset by dependencies
 
-             return margin;
-           }
-         }, {
-           key: "_adjustParentBBoxes",
-           value: function _adjustParentBBoxes(bbox, path, level) {
-             // adjust bboxes along the given tree path
-             for (var i = level; i >= 0; i--) {
-               extend$1(path[i], bbox);
-             }
-           }
-         }, {
-           key: "_condense",
-           value: function _condense(path) {
-             // go through the path, removing empty nodes and updating bboxes
-             for (var i = path.length - 1, siblings; i >= 0; i--) {
-               if (path[i].children.length === 0) {
-                 if (i > 0) {
-                   siblings = path[i - 1].children;
-                   siblings.splice(siblings.indexOf(path[i]), 1);
-                 } else this.clear();
-               } else calcBBox(path[i], this.toBBox);
-             }
-           }
-         }]);
 
-         return RBush;
-       }();
+           function sort(changes) {
+             // find a referenced relation in the current changeset
+             function resolve(item) {
+               return relations.find(function (relation) {
+                 return item.keyAttributes.type === 'relation' && item.keyAttributes.ref === relation['@id'];
+               });
+             } // a new item is an item that has not been already processed
 
-       function findItem(item, items, equalsFn) {
-         if (!equalsFn) return items.indexOf(item);
 
-         for (var i = 0; i < items.length; i++) {
-           if (equalsFn(item, items[i])) return i;
-         }
+             function isNew(item) {
+               return !sorted[item['@id']] && !processing.find(function (proc) {
+                 return proc['@id'] === item['@id'];
+               });
+             }
 
-         return -1;
-       } // calculate node's bbox from bboxes of its children
+             var processing = [];
+             var sorted = {};
+             var relations = changes.relation;
+             if (!relations) return changes;
 
+             for (var i = 0; i < relations.length; i++) {
+               var relation = relations[i]; // skip relation if already sorted
 
-       function calcBBox(node, toBBox) {
-         distBBox(node, 0, node.children.length, toBBox, node);
-       } // min bounding rectangle of node children from k to p-1
+               if (!sorted[relation['@id']]) {
+                 processing.push(relation);
+               }
 
+               while (processing.length > 0) {
+                 var next = processing[0],
+                     deps = next.member.map(resolve).filter(Boolean).filter(isNew);
 
-       function distBBox(node, k, p, toBBox, destNode) {
-         if (!destNode) destNode = createNode(null);
-         destNode.minX = Infinity;
-         destNode.minY = Infinity;
-         destNode.maxX = -Infinity;
-         destNode.maxY = -Infinity;
+                 if (deps.length === 0) {
+                   sorted[next['@id']] = next;
+                   processing.shift();
+                 } else {
+                   processing = deps.concat(processing);
+                 }
+               }
+             }
 
-         for (var i = k; i < p; i++) {
-           var child = node.children[i];
-           extend$1(destNode, node.leaf ? toBBox(child) : child);
+             changes.relation = Object.values(sorted);
+             return changes;
+           }
+
+           function rep(entity) {
+             return entity.asJXON(changeset_id);
+           }
+
+           return {
+             osmChange: {
+               '@version': 0.6,
+               '@generator': 'iD',
+               'create': sort(nest(changes.created.map(rep), ['node', 'way', 'relation'])),
+               'modify': nest(changes.modified.map(rep), ['node', 'way', 'relation']),
+               'delete': Object.assign(nest(changes.deleted.map(rep), ['relation', 'way', 'node']), {
+                 '@if-unused': true
+               })
+             }
+           };
+         },
+         asGeoJSON: function asGeoJSON() {
+           return {};
          }
+       });
 
-         return destNode;
+       function osmNote() {
+         if (!(this instanceof osmNote)) {
+           return new osmNote().initialize(arguments);
+         } else if (arguments.length) {
+           this.initialize(arguments);
+         }
        }
 
-       function extend$1(a, b) {
-         a.minX = Math.min(a.minX, b.minX);
-         a.minY = Math.min(a.minY, b.minY);
-         a.maxX = Math.max(a.maxX, b.maxX);
-         a.maxY = Math.max(a.maxY, b.maxY);
-         return a;
-       }
+       osmNote.id = function () {
+         return osmNote.id.next--;
+       };
 
-       function compareNodeMinX(a, b) {
-         return a.minX - b.minX;
-       }
+       osmNote.id.next = -1;
+       Object.assign(osmNote.prototype, {
+         type: 'note',
+         initialize: function initialize(sources) {
+           for (var i = 0; i < sources.length; ++i) {
+             var source = sources[i];
 
-       function compareNodeMinY(a, b) {
-         return a.minY - b.minY;
-       }
+             for (var prop in source) {
+               if (Object.prototype.hasOwnProperty.call(source, prop)) {
+                 if (source[prop] === undefined) {
+                   delete this[prop];
+                 } else {
+                   this[prop] = source[prop];
+                 }
+               }
+             }
+           }
 
-       function bboxArea(a) {
-         return (a.maxX - a.minX) * (a.maxY - a.minY);
-       }
+           if (!this.id) {
+             this.id = osmNote.id().toString();
+           }
 
-       function bboxMargin(a) {
-         return a.maxX - a.minX + (a.maxY - a.minY);
-       }
+           return this;
+         },
+         extent: function extent() {
+           return new geoExtent(this.loc);
+         },
+         update: function update(attrs) {
+           return osmNote(this, attrs); // {v: 1 + (this.v || 0)}
+         },
+         isNew: function isNew() {
+           return this.id < 0;
+         },
+         move: function move(loc) {
+           return this.update({
+             loc: loc
+           });
+         }
+       });
 
-       function enlargedArea(a, b) {
-         return (Math.max(b.maxX, a.maxX) - Math.min(b.minX, a.minX)) * (Math.max(b.maxY, a.maxY) - Math.min(b.minY, a.minY));
+       function osmRelation() {
+         if (!(this instanceof osmRelation)) {
+           return new osmRelation().initialize(arguments);
+         } else if (arguments.length) {
+           this.initialize(arguments);
+         }
        }
+       osmEntity.relation = osmRelation;
+       osmRelation.prototype = Object.create(osmEntity.prototype);
 
-       function intersectionArea(a, b) {
-         var minX = Math.max(a.minX, b.minX);
-         var minY = Math.max(a.minY, b.minY);
-         var maxX = Math.min(a.maxX, b.maxX);
-         var maxY = Math.min(a.maxY, b.maxY);
-         return Math.max(0, maxX - minX) * Math.max(0, maxY - minY);
-       }
+       osmRelation.creationOrder = function (a, b) {
+         var aId = parseInt(osmEntity.id.toOSM(a.id), 10);
+         var bId = parseInt(osmEntity.id.toOSM(b.id), 10);
+         if (aId < 0 || bId < 0) return aId - bId;
+         return bId - aId;
+       };
 
-       function contains(a, b) {
-         return a.minX <= b.minX && a.minY <= b.minY && b.maxX <= a.maxX && b.maxY <= a.maxY;
-       }
+       Object.assign(osmRelation.prototype, {
+         type: 'relation',
+         members: [],
+         copy: function copy(resolver, copies) {
+           if (copies[this.id]) return copies[this.id];
+           var copy = osmEntity.prototype.copy.call(this, resolver, copies);
+           var members = this.members.map(function (member) {
+             return Object.assign({}, member, {
+               id: resolver.entity(member.id).copy(resolver, copies).id
+             });
+           });
+           copy = copy.update({
+             members: members
+           });
+           copies[this.id] = copy;
+           return copy;
+         },
+         extent: function extent(resolver, memo) {
+           return resolver["transient"](this, 'extent', function () {
+             if (memo && memo[this.id]) return geoExtent();
+             memo = memo || {};
+             memo[this.id] = true;
+             var extent = geoExtent();
 
-       function intersects(a, b) {
-         return b.minX <= a.maxX && b.minY <= a.maxY && b.maxX >= a.minX && b.maxY >= a.minY;
-       }
+             for (var i = 0; i < this.members.length; i++) {
+               var member = resolver.hasEntity(this.members[i].id);
 
-       function createNode(children) {
-         return {
-           children: children,
-           height: 1,
-           leaf: true,
-           minX: Infinity,
-           minY: Infinity,
-           maxX: -Infinity,
-           maxY: -Infinity
-         };
-       } // sort an array so that items come in groups of n unsorted items, with groups sorted between each other;
-       // combines selection algorithm with binary divide & conquer approach
+               if (member) {
+                 extent._extend(member.extent(resolver, memo));
+               }
+             }
 
+             return extent;
+           });
+         },
+         geometry: function geometry(graph) {
+           return graph["transient"](this, 'geometry', function () {
+             return this.isMultipolygon() ? 'area' : 'relation';
+           });
+         },
+         isDegenerate: function isDegenerate() {
+           return this.members.length === 0;
+         },
+         // Return an array of members, each extended with an 'index' property whose value
+         // is the member index.
+         indexedMembers: function indexedMembers() {
+           var result = new Array(this.members.length);
 
-       function multiSelect(arr, left, right, n, compare) {
-         var stack = [left, right];
+           for (var i = 0; i < this.members.length; i++) {
+             result[i] = Object.assign({}, this.members[i], {
+               index: i
+             });
+           }
 
-         while (stack.length) {
-           right = stack.pop();
-           left = stack.pop();
-           if (right - left <= n) continue;
-           var mid = left + Math.ceil((right - left) / n / 2) * n;
-           quickselect$1(arr, mid, left, right, compare);
-           stack.push(left, mid, mid, right);
-         }
-       }
+           return result;
+         },
+         // Return the first member with the given role. A copy of the member object
+         // is returned, extended with an 'index' property whose value is the member index.
+         memberByRole: function memberByRole(role) {
+           for (var i = 0; i < this.members.length; i++) {
+             if (this.members[i].role === role) {
+               return Object.assign({}, this.members[i], {
+                 index: i
+               });
+             }
+           }
+         },
+         // Same as memberByRole, but returns all members with the given role
+         membersByRole: function membersByRole(role) {
+           var result = [];
 
-       var tiler = utilTiler();
-       var dispatch$1 = dispatch('loaded');
-       var _tileZoom = 14;
-       var _krUrlRoot = 'https://www.keepright.at';
-       var _krData = {
-         errorTypes: {},
-         localizeStrings: {}
-       }; // This gets reassigned if reset
+           for (var i = 0; i < this.members.length; i++) {
+             if (this.members[i].role === role) {
+               result.push(Object.assign({}, this.members[i], {
+                 index: i
+               }));
+             }
+           }
 
-       var _cache;
+           return result;
+         },
+         // Return the first member with the given id. A copy of the member object
+         // is returned, extended with an 'index' property whose value is the member index.
+         memberById: function memberById(id) {
+           for (var i = 0; i < this.members.length; i++) {
+             if (this.members[i].id === id) {
+               return Object.assign({}, this.members[i], {
+                 index: i
+               });
+             }
+           }
+         },
+         // Return the first member with the given id and role. A copy of the member object
+         // is returned, extended with an 'index' property whose value is the member index.
+         memberByIdAndRole: function memberByIdAndRole(id, role) {
+           for (var i = 0; i < this.members.length; i++) {
+             if (this.members[i].id === id && this.members[i].role === role) {
+               return Object.assign({}, this.members[i], {
+                 index: i
+               });
+             }
+           }
+         },
+         addMember: function addMember(member, index) {
+           var members = this.members.slice();
+           members.splice(index === undefined ? members.length : index, 0, member);
+           return this.update({
+             members: members
+           });
+         },
+         updateMember: function updateMember(member, index) {
+           var members = this.members.slice();
+           members.splice(index, 1, Object.assign({}, members[index], member));
+           return this.update({
+             members: members
+           });
+         },
+         removeMember: function removeMember(index) {
+           var members = this.members.slice();
+           members.splice(index, 1);
+           return this.update({
+             members: members
+           });
+         },
+         removeMembersWithID: function removeMembersWithID(id) {
+           var members = this.members.filter(function (m) {
+             return m.id !== id;
+           });
+           return this.update({
+             members: members
+           });
+         },
+         moveMember: function moveMember(fromIndex, toIndex) {
+           var members = this.members.slice();
+           members.splice(toIndex, 0, members.splice(fromIndex, 1)[0]);
+           return this.update({
+             members: members
+           });
+         },
+         // Wherever a member appears with id `needle.id`, replace it with a member
+         // with id `replacement.id`, type `replacement.type`, and the original role,
+         // By default, adding a duplicate member (by id and role) is prevented.
+         // Return an updated relation.
+         replaceMember: function replaceMember(needle, replacement, keepDuplicates) {
+           if (!this.memberById(needle.id)) return this;
+           var members = [];
 
-       var _krRuleset = [// no 20 - multiple node on same spot - these are mostly boundaries overlapping roads
-       30, 40, 50, 60, 70, 90, 100, 110, 120, 130, 150, 160, 170, 180, 190, 191, 192, 193, 194, 195, 196, 197, 198, 200, 201, 202, 203, 204, 205, 206, 207, 208, 210, 220, 230, 231, 232, 270, 280, 281, 282, 283, 284, 285, 290, 291, 292, 293, 294, 295, 296, 297, 298, 300, 310, 311, 312, 313, 320, 350, 360, 370, 380, 390, 400, 401, 402, 410, 411, 412, 413];
+           for (var i = 0; i < this.members.length; i++) {
+             var member = this.members[i];
 
-       function abortRequest(controller) {
-         if (controller) {
-           controller.abort();
-         }
-       }
+             if (member.id !== needle.id) {
+               members.push(member);
+             } else if (keepDuplicates || !this.memberByIdAndRole(replacement.id, member.role)) {
+               members.push({
+                 id: replacement.id,
+                 type: replacement.type,
+                 role: member.role
+               });
+             }
+           }
 
-       function abortUnwantedRequests(cache, tiles) {
-         Object.keys(cache.inflightTile).forEach(function (k) {
-           var wanted = tiles.find(function (tile) {
-             return k === tile.id;
+           return this.update({
+             members: members
            });
+         },
+         asJXON: function asJXON(changeset_id) {
+           var r = {
+             relation: {
+               '@id': this.osmId(),
+               '@version': this.version || 0,
+               member: this.members.map(function (member) {
+                 return {
+                   keyAttributes: {
+                     type: member.type,
+                     role: member.role,
+                     ref: osmEntity.id.toOSM(member.id)
+                   }
+                 };
+               }, this),
+               tag: Object.keys(this.tags).map(function (k) {
+                 return {
+                   keyAttributes: {
+                     k: k,
+                     v: this.tags[k]
+                   }
+                 };
+               }, this)
+             }
+           };
 
-           if (!wanted) {
-             abortRequest(cache.inflightTile[k]);
-             delete cache.inflightTile[k];
+           if (changeset_id) {
+             r.relation['@changeset'] = changeset_id;
            }
-         });
-       }
-
-       function encodeIssueRtree(d) {
-         return {
-           minX: d.loc[0],
-           minY: d.loc[1],
-           maxX: d.loc[0],
-           maxY: d.loc[1],
-           data: d
-         };
-       } // Replace or remove QAItem from rtree
 
+           return r;
+         },
+         asGeoJSON: function asGeoJSON(resolver) {
+           return resolver["transient"](this, 'GeoJSON', function () {
+             if (this.isMultipolygon()) {
+               return {
+                 type: 'MultiPolygon',
+                 coordinates: this.multipolygon(resolver)
+               };
+             } else {
+               return {
+                 type: 'FeatureCollection',
+                 properties: this.tags,
+                 features: this.members.map(function (member) {
+                   return Object.assign({
+                     role: member.role
+                   }, resolver.entity(member.id).asGeoJSON(resolver));
+                 })
+               };
+             }
+           });
+         },
+         area: function area(resolver) {
+           return resolver["transient"](this, 'area', function () {
+             return d3_geoArea(this.asGeoJSON(resolver));
+           });
+         },
+         isMultipolygon: function isMultipolygon() {
+           return this.tags.type === 'multipolygon';
+         },
+         isComplete: function isComplete(resolver) {
+           for (var i = 0; i < this.members.length; i++) {
+             if (!resolver.hasEntity(this.members[i].id)) {
+               return false;
+             }
+           }
 
-       function updateRtree(item, replace) {
-         _cache.rtree.remove(item, function (a, b) {
-           return a.data.id === b.data.id;
-         });
+           return true;
+         },
+         hasFromViaTo: function hasFromViaTo() {
+           return this.members.some(function (m) {
+             return m.role === 'from';
+           }) && this.members.some(function (m) {
+             return m.role === 'via';
+           }) && this.members.some(function (m) {
+             return m.role === 'to';
+           });
+         },
+         isRestriction: function isRestriction() {
+           return !!(this.tags.type && this.tags.type.match(/^restriction:?/));
+         },
+         isValidRestriction: function isValidRestriction() {
+           if (!this.isRestriction()) return false;
+           var froms = this.members.filter(function (m) {
+             return m.role === 'from';
+           });
+           var vias = this.members.filter(function (m) {
+             return m.role === 'via';
+           });
+           var tos = this.members.filter(function (m) {
+             return m.role === 'to';
+           });
+           if (froms.length !== 1 && this.tags.restriction !== 'no_entry') return false;
+           if (froms.some(function (m) {
+             return m.type !== 'way';
+           })) return false;
+           if (tos.length !== 1 && this.tags.restriction !== 'no_exit') return false;
+           if (tos.some(function (m) {
+             return m.type !== 'way';
+           })) return false;
+           if (vias.length === 0) return false;
+           if (vias.length > 1 && vias.some(function (m) {
+             return m.type !== 'way';
+           })) return false;
+           return true;
+         },
+         // Returns an array [A0, ... An], each Ai being an array of node arrays [Nds0, ... Ndsm],
+         // where Nds0 is an outer ring and subsequent Ndsi's (if any i > 0) being inner rings.
+         //
+         // This corresponds to the structure needed for rendering a multipolygon path using a
+         // `evenodd` fill rule, as well as the structure of a GeoJSON MultiPolygon geometry.
+         //
+         // In the case of invalid geometries, this function will still return a result which
+         // includes the nodes of all way members, but some Nds may be unclosed and some inner
+         // rings not matched with the intended outer ring.
+         //
+         multipolygon: function multipolygon(resolver) {
+           var outers = this.members.filter(function (m) {
+             return 'outer' === (m.role || 'outer');
+           });
+           var inners = this.members.filter(function (m) {
+             return 'inner' === m.role;
+           });
+           outers = osmJoinWays(outers, resolver);
+           inners = osmJoinWays(inners, resolver);
 
-         if (replace) {
-           _cache.rtree.insert(item);
-         }
-       }
+           var sequenceToLineString = function sequenceToLineString(sequence) {
+             if (sequence.nodes.length > 2 && sequence.nodes[0] !== sequence.nodes[sequence.nodes.length - 1]) {
+               // close unclosed parts to ensure correct area rendering - #2945
+               sequence.nodes.push(sequence.nodes[0]);
+             }
 
-       function tokenReplacements(d) {
-         if (!(d instanceof QAItem)) return;
-         var htmlRegex = new RegExp(/<\/[a-z][\s\S]*>/);
-         var replacements = {};
-         var issueTemplate = _krData.errorTypes[d.whichType];
+             return sequence.nodes.map(function (node) {
+               return node.loc;
+             });
+           };
 
-         if (!issueTemplate) {
-           /* eslint-disable no-console */
-           console.log('No Template: ', d.whichType);
-           console.log('  ', d.description);
-           /* eslint-enable no-console */
+           outers = outers.map(sequenceToLineString);
+           inners = inners.map(sequenceToLineString);
+           var result = outers.map(function (o) {
+             // Heuristic for detecting counterclockwise winding order. Assumes
+             // that OpenStreetMap polygons are not hemisphere-spanning.
+             return [d3_geoArea({
+               type: 'Polygon',
+               coordinates: [o]
+             }) > 2 * Math.PI ? o.reverse() : o];
+           });
 
-           return;
-         } // some descriptions are just fixed text
+           function findOuter(inner) {
+             var o, outer;
 
+             for (o = 0; o < outers.length; o++) {
+               outer = outers[o];
 
-         if (!issueTemplate.regex) return; // regex pattern should match description with variable details captured
+               if (geoPolygonContainsPolygon(outer, inner)) {
+                 return o;
+               }
+             }
 
-         var errorRegex = new RegExp(issueTemplate.regex, 'i');
-         var errorMatch = errorRegex.exec(d.description);
+             for (o = 0; o < outers.length; o++) {
+               outer = outers[o];
 
-         if (!errorMatch) {
-           /* eslint-disable no-console */
-           console.log('Unmatched: ', d.whichType);
-           console.log('  ', d.description);
-           console.log('  ', errorRegex);
-           /* eslint-enable no-console */
+               if (geoPolygonIntersectsPolygon(outer, inner, false)) {
+                 return o;
+               }
+             }
+           }
 
-           return;
-         }
+           for (var i = 0; i < inners.length; i++) {
+             var inner = inners[i];
 
-         for (var i = 1; i < errorMatch.length; i++) {
-           // skip first
-           var capture = errorMatch[i];
-           var idType = void 0;
-           idType = 'IDs' in issueTemplate ? issueTemplate.IDs[i - 1] : '';
+             if (d3_geoArea({
+               type: 'Polygon',
+               coordinates: [inner]
+             }) < 2 * Math.PI) {
+               inner = inner.reverse();
+             }
 
-           if (idType && capture) {
-             // link IDs if present in the capture
-             capture = parseError(capture, idType);
-           } else if (htmlRegex.test(capture)) {
-             // escape any html in non-IDs
-             capture = '\\' + capture + '\\';
-           } else {
-             var compare = capture.toLowerCase();
+             var o = findOuter(inners[i]);
 
-             if (_krData.localizeStrings[compare]) {
-               // some replacement strings can be localized
-               capture = _t('QA.keepRight.error_parts.' + _krData.localizeStrings[compare]);
+             if (o !== undefined) {
+               result[o].push(inners[i]);
+             } else {
+               result.push([inners[i]]); // Invalid geometry
              }
            }
 
-           replacements['var' + i] = capture;
+           return result;
          }
+       });
 
-         return replacements;
-       }
+       var QAItem = /*#__PURE__*/function () {
+         function QAItem(loc, service, itemType, id, props) {
+           _classCallCheck$1(this, QAItem);
 
-       function parseError(capture, idType) {
-         var compare = capture.toLowerCase();
+           // Store required properties
+           this.loc = loc;
+           this.service = service.title;
+           this.itemType = itemType; // All issues must have an ID for selection, use generic if none specified
 
-         if (_krData.localizeStrings[compare]) {
-           // some replacement strings can be localized
-           capture = _t('QA.keepRight.error_parts.' + _krData.localizeStrings[compare]);
+           this.id = id ? id : "".concat(QAItem.id());
+           this.update(props); // Some QA services have marker icons to differentiate issues
+
+           if (service && typeof service.getIcon === 'function') {
+             this.icon = service.getIcon(itemType);
+           }
          }
 
-         switch (idType) {
-           // link a string like "this node"
-           case 'this':
-             capture = linkErrorObject(capture);
-             break;
+         _createClass$1(QAItem, [{
+           key: "update",
+           value: function update(props) {
+             var _this = this;
 
-           case 'url':
-             capture = linkURL(capture);
-             break;
-           // link an entity ID
+             // You can't override this initial information
+             var loc = this.loc,
+                 service = this.service,
+                 itemType = this.itemType,
+                 id = this.id;
+             Object.keys(props).forEach(function (prop) {
+               return _this[prop] = props[prop];
+             });
+             this.loc = loc;
+             this.service = service;
+             this.itemType = itemType;
+             this.id = id;
+             return this;
+           } // Generic handling for newly created QAItems
 
-           case 'n':
-           case 'w':
-           case 'r':
-             capture = linkEntity(idType + capture);
-             break;
-           // some errors have more complex ID lists/variance
+         }], [{
+           key: "id",
+           value: function id() {
+             return this.nextId--;
+           }
+         }]);
 
-           case '20':
-             capture = parse20(capture);
-             break;
+         return QAItem;
+       }();
+       QAItem.nextId = -1;
 
-           case '211':
-             capture = parse211(capture);
-             break;
+       //
+       // Optionally, split only the given ways, if multiple ways share
+       // the given node.
+       //
+       // This is the inverse of `iD.actionJoin`.
+       //
+       // For testing convenience, accepts an ID to assign to the new way.
+       // Normally, this will be undefined and the way will automatically
+       // be assigned a new ID.
+       //
+       // Reference:
+       //   https://github.com/systemed/potlatch2/blob/master/net/systemeD/halcyon/connection/actions/SplitWayAction.as
+       //
 
-           case '231':
-             capture = parse231(capture);
-             break;
+       function actionSplit(nodeIds, newWayIds) {
+         // accept single ID for backwards-compatiblity
+         if (typeof nodeIds === 'string') nodeIds = [nodeIds];
 
-           case '294':
-             capture = parse294(capture);
-             break;
+         var _wayIDs; // the strategy for picking which way will have a new version and which way is newly created
 
-           case '370':
-             capture = parse370(capture);
-             break;
-         }
 
-         return capture;
+         var _keepHistoryOn = 'longest'; // 'longest', 'first'
+         // The IDs of the ways actually created by running this action
 
-         function linkErrorObject(d) {
-           return "<a class=\"error_object_link\">".concat(d, "</a>");
-         }
+         var _createdWayIDs = [];
 
-         function linkEntity(d) {
-           return "<a class=\"error_entity_link\">".concat(d, "</a>");
-         }
+         function dist(graph, nA, nB) {
+           var locA = graph.entity(nA).loc;
+           var locB = graph.entity(nB).loc;
+           var epsilon = 1e-6;
+           return locA && locB ? geoSphericalDistance(locA, locB) : epsilon;
+         } // If the way is closed, we need to search for a partner node
+         // to split the way at.
+         //
+         // The following looks for a node that is both far away from
+         // the initial node in terms of way segment length and nearby
+         // in terms of beeline-distance. This assures that areas get
+         // split on the most "natural" points (independent of the number
+         // of nodes).
+         // For example: bone-shaped areas get split across their waist
+         // line, circles across the diameter.
 
-         function linkURL(d) {
-           return "<a class=\"kr_external_link\" target=\"_blank\" href=\"".concat(d, "\">").concat(d, "</a>");
-         } // arbitrary node list of form: #ID, #ID, #ID...
 
+         function splitArea(nodes, idxA, graph) {
+           var lengths = new Array(nodes.length);
+           var length;
+           var i;
+           var best = 0;
+           var idxB;
 
-         function parse211(capture) {
-           var newList = [];
-           var items = capture.split(', ');
-           items.forEach(function (item) {
-             // ID has # at the front
-             var id = linkEntity('n' + item.slice(1));
-             newList.push(id);
-           });
-           return newList.join(', ');
-         } // arbitrary way list of form: #ID(layer),#ID(layer),#ID(layer)...
+           function wrap(index) {
+             return utilWrap(index, nodes.length);
+           } // calculate lengths
+
+
+           length = 0;
 
+           for (i = wrap(idxA + 1); i !== idxA; i = wrap(i + 1)) {
+             length += dist(graph, nodes[i], nodes[wrap(i - 1)]);
+             lengths[i] = length;
+           }
 
-         function parse231(capture) {
-           var newList = []; // unfortunately 'layer' can itself contain commas, so we split on '),'
+           length = 0;
 
-           var items = capture.split('),');
-           items.forEach(function (item) {
-             var match = item.match(/\#(\d+)\((.+)\)?/);
+           for (i = wrap(idxA - 1); i !== idxA; i = wrap(i - 1)) {
+             length += dist(graph, nodes[i], nodes[wrap(i + 1)]);
 
-             if (match !== null && match.length > 2) {
-               newList.push(linkEntity('w' + match[1]) + ' ' + _t('QA.keepRight.errorTypes.231.layer', {
-                 layer: match[2]
-               }));
+             if (length < lengths[i]) {
+               lengths[i] = length;
              }
-           });
-           return newList.join(', ');
-         } // arbitrary node/relation list of form: from node #ID,to relation #ID,to node #ID...
-
-
-         function parse294(capture) {
-           var newList = [];
-           var items = capture.split(',');
-           items.forEach(function (item) {
-             // item of form "from/to node/relation #ID"
-             item = item.split(' '); // to/from role is more clear in quotes
+           } // determine best opposite node to split
 
-             var role = "\"".concat(item[0], "\""); // first letter of node/relation provides the type
 
-             var idType = item[1].slice(0, 1); // ID has # at the front
+           for (i = 0; i < nodes.length; i++) {
+             var cost = lengths[i] / dist(graph, nodes[idxA], nodes[i]);
 
-             var id = item[2].slice(1);
-             id = linkEntity(idType + id);
-             newList.push("".concat(role, " ").concat(item[1], " ").concat(id));
-           });
-           return newList.join(', ');
-         } // may or may not include the string "(including the name 'name')"
+             if (cost > best) {
+               idxB = i;
+               best = cost;
+             }
+           }
 
+           return idxB;
+         }
 
-         function parse370(capture) {
-           if (!capture) return '';
-           var match = capture.match(/\(including the name (\'.+\')\)/);
+         function totalLengthBetweenNodes(graph, nodes) {
+           var totalLength = 0;
 
-           if (match && match.length) {
-             return _t('QA.keepRight.errorTypes.370.including_the_name', {
-               name: match[1]
-             });
+           for (var i = 0; i < nodes.length - 1; i++) {
+             totalLength += dist(graph, nodes[i], nodes[i + 1]);
            }
 
-           return '';
-         } // arbitrary node list of form: #ID,#ID,#ID...
+           return totalLength;
+         }
 
+         function split(graph, nodeId, wayA, newWayId) {
+           var wayB = osmWay({
+             id: newWayId,
+             tags: wayA.tags
+           }); // `wayB` is the NEW way
 
-         function parse20(capture) {
-           var newList = [];
-           var items = capture.split(',');
-           items.forEach(function (item) {
-             // ID has # at the front
-             var id = linkEntity('n' + item.slice(1));
-             newList.push(id);
-           });
-           return newList.join(', ');
-         }
-       }
+           var origNodes = wayA.nodes.slice();
+           var nodesA;
+           var nodesB;
+           var isArea = wayA.isArea();
+           var isOuter = osmIsOldMultipolygonOuterMember(wayA, graph);
 
-       var serviceKeepRight = {
-         title: 'keepRight',
-         init: function init() {
-           _mainFileFetcher.get('keepRight').then(function (d) {
-             return _krData = d;
-           });
+           if (wayA.isClosed()) {
+             var nodes = wayA.nodes.slice(0, -1);
+             var idxA = nodes.indexOf(nodeId);
+             var idxB = splitArea(nodes, idxA, graph);
 
-           if (!_cache) {
-             this.reset();
+             if (idxB < idxA) {
+               nodesA = nodes.slice(idxA).concat(nodes.slice(0, idxB + 1));
+               nodesB = nodes.slice(idxB, idxA + 1);
+             } else {
+               nodesA = nodes.slice(idxA, idxB + 1);
+               nodesB = nodes.slice(idxB).concat(nodes.slice(0, idxA + 1));
+             }
+           } else {
+             var idx = wayA.nodes.indexOf(nodeId, 1);
+             nodesA = wayA.nodes.slice(0, idx + 1);
+             nodesB = wayA.nodes.slice(idx);
            }
 
-           this.event = utilRebind(this, dispatch$1, 'on');
-         },
-         reset: function reset() {
-           if (_cache) {
-             Object.values(_cache.inflightTile).forEach(abortRequest);
+           var lengthA = totalLengthBetweenNodes(graph, nodesA);
+           var lengthB = totalLengthBetweenNodes(graph, nodesB);
+
+           if (_keepHistoryOn === 'longest' && lengthB > lengthA) {
+             // keep the history on the longer way, regardless of the node count
+             wayA = wayA.update({
+               nodes: nodesB
+             });
+             wayB = wayB.update({
+               nodes: nodesA
+             });
+             var temp = lengthA;
+             lengthA = lengthB;
+             lengthB = temp;
+           } else {
+             wayA = wayA.update({
+               nodes: nodesA
+             });
+             wayB = wayB.update({
+               nodes: nodesB
+             });
            }
 
-           _cache = {
-             data: {},
-             loadedTile: {},
-             inflightTile: {},
-             inflightPost: {},
-             closed: {},
-             rtree: new RBush()
-           };
-         },
-         // KeepRight API:  http://osm.mueschelsoft.de/keepright/interfacing.php
-         loadIssues: function loadIssues(projection) {
-           var _this = this;
+           if (wayA.tags.step_count) {
+             // divide up the the step count proportionally between the two ways
+             var stepCount = parseFloat(wayA.tags.step_count);
 
-           var options = {
-             format: 'geojson',
-             ch: _krRuleset
-           }; // determine the needed tiles to cover the view
+             if (stepCount && // ensure a number
+             isFinite(stepCount) && // ensure positive
+             stepCount > 0 && // ensure integer
+             Math.round(stepCount) === stepCount) {
+               var tagsA = Object.assign({}, wayA.tags);
+               var tagsB = Object.assign({}, wayB.tags);
+               var ratioA = lengthA / (lengthA + lengthB);
+               var countA = Math.round(stepCount * ratioA);
+               tagsA.step_count = countA.toString();
+               tagsB.step_count = (stepCount - countA).toString();
+               wayA = wayA.update({
+                 tags: tagsA
+               });
+               wayB = wayB.update({
+                 tags: tagsB
+               });
+             }
+           }
 
-           var tiles = tiler.zoomExtent([_tileZoom, _tileZoom]).getTiles(projection); // abort inflight requests that are no longer needed
+           graph = graph.replace(wayA);
+           graph = graph.replace(wayB);
+           graph.parentRelations(wayA).forEach(function (relation) {
+             var member; // Turn restrictions - make sure:
+             // 1. Splitting a FROM/TO way - only `wayA` OR `wayB` remains in relation
+             //    (whichever one is connected to the VIA node/ways)
+             // 2. Splitting a VIA way - `wayB` remains in relation as a VIA way
 
-           abortUnwantedRequests(_cache, tiles); // issue new requests..
+             if (relation.hasFromViaTo()) {
+               var f = relation.memberByRole('from');
+               var v = relation.membersByRole('via');
+               var t = relation.memberByRole('to');
+               var i; // 1. split a FROM/TO
 
-           tiles.forEach(function (tile) {
-             if (_cache.loadedTile[tile.id] || _cache.inflightTile[tile.id]) return;
+               if (f.id === wayA.id || t.id === wayA.id) {
+                 var keepB = false;
 
-             var _tile$extent$rectangl = tile.extent.rectangle(),
-                 _tile$extent$rectangl2 = _slicedToArray(_tile$extent$rectangl, 4),
-                 left = _tile$extent$rectangl2[0],
-                 top = _tile$extent$rectangl2[1],
-                 right = _tile$extent$rectangl2[2],
-                 bottom = _tile$extent$rectangl2[3];
+                 if (v.length === 1 && v[0].type === 'node') {
+                   // check via node
+                   keepB = wayB.contains(v[0].id);
+                 } else {
+                   // check via way(s)
+                   for (i = 0; i < v.length; i++) {
+                     if (v[i].type === 'way') {
+                       var wayVia = graph.hasEntity(v[i].id);
 
-             var params = Object.assign({}, options, {
-               left: left,
-               bottom: bottom,
-               right: right,
-               top: top
-             });
-             var url = "".concat(_krUrlRoot, "/export.php?") + utilQsString(params);
-             var controller = new AbortController();
-             _cache.inflightTile[tile.id] = controller;
-             d3_json(url, {
-               signal: controller.signal
-             }).then(function (data) {
-               delete _cache.inflightTile[tile.id];
-               _cache.loadedTile[tile.id] = true;
+                       if (wayVia && utilArrayIntersection(wayB.nodes, wayVia.nodes).length) {
+                         keepB = true;
+                         break;
+                       }
+                     }
+                   }
+                 }
 
-               if (!data || !data.features || !data.features.length) {
-                 throw new Error('No Data');
-               }
+                 if (keepB) {
+                   relation = relation.replaceMember(wayA, wayB);
+                   graph = graph.replace(relation);
+                 } // 2. split a VIA
 
-               data.features.forEach(function (feature) {
-                 var _feature$properties = feature.properties,
-                     itemType = _feature$properties.error_type,
-                     id = _feature$properties.error_id,
-                     _feature$properties$c = _feature$properties.comment,
-                     comment = _feature$properties$c === void 0 ? null : _feature$properties$c,
-                     objectId = _feature$properties.object_id,
-                     objectType = _feature$properties.object_type,
-                     schema = _feature$properties.schema,
-                     title = _feature$properties.title;
-                 var loc = feature.geometry.coordinates,
-                     _feature$properties$d = feature.properties.description,
-                     description = _feature$properties$d === void 0 ? '' : _feature$properties$d; // if there is a parent, save its error type e.g.:
-                 //  Error 191 = "highway-highway"
-                 //  Error 190 = "intersections without junctions"  (parent)
+               } else {
+                 for (i = 0; i < v.length; i++) {
+                   if (v[i].type === 'way' && v[i].id === wayA.id) {
+                     member = {
+                       id: wayB.id,
+                       type: 'way',
+                       role: 'via'
+                     };
+                     graph = actionAddMember(relation.id, member, v[i].index + 1)(graph);
+                     break;
+                   }
+                 }
+               } // All other relations (Routes, Multipolygons, etc):
+               // 1. Both `wayA` and `wayB` remain in the relation
+               // 2. But must be inserted as a pair (see `actionAddMember` for details)
 
-                 var issueTemplate = _krData.errorTypes[itemType];
-                 var parentIssueType = (Math.floor(itemType / 10) * 10).toString(); // try to handle error type directly, fallback to parent error type.
+             } else {
+               if (relation === isOuter) {
+                 graph = graph.replace(relation.mergeTags(wayA.tags));
+                 graph = graph.replace(wayA.update({
+                   tags: {}
+                 }));
+                 graph = graph.replace(wayB.update({
+                   tags: {}
+                 }));
+               }
 
-                 var whichType = issueTemplate ? itemType : parentIssueType;
-                 var whichTemplate = _krData.errorTypes[whichType]; // Rewrite a few of the errors at this point..
-                 // This is done to make them easier to linkify and translate.
+               member = {
+                 id: wayB.id,
+                 type: 'way',
+                 role: relation.memberById(wayA.id).role
+               };
+               var insertPair = {
+                 originalID: wayA.id,
+                 insertedID: wayB.id,
+                 nodes: origNodes
+               };
+               graph = actionAddMember(relation.id, member, undefined, insertPair)(graph);
+             }
+           });
 
-                 switch (whichType) {
-                   case '170':
-                     description = "This feature has a FIXME tag: ".concat(description);
-                     break;
+           if (!isOuter && isArea) {
+             var multipolygon = osmRelation({
+               tags: Object.assign({}, wayA.tags, {
+                 type: 'multipolygon'
+               }),
+               members: [{
+                 id: wayA.id,
+                 role: 'outer',
+                 type: 'way'
+               }, {
+                 id: wayB.id,
+                 role: 'outer',
+                 type: 'way'
+               }]
+             });
+             graph = graph.replace(multipolygon);
+             graph = graph.replace(wayA.update({
+               tags: {}
+             }));
+             graph = graph.replace(wayB.update({
+               tags: {}
+             }));
+           }
 
-                   case '292':
-                   case '293':
-                     description = description.replace('A turn-', 'This turn-');
-                     break;
+           _createdWayIDs.push(wayB.id);
 
-                   case '294':
-                   case '295':
-                   case '296':
-                   case '297':
-                   case '298':
-                     description = "This turn-restriction~".concat(description);
-                     break;
+           return graph;
+         }
 
-                   case '300':
-                     description = 'This highway is missing a maxspeed tag';
-                     break;
+         var action = function action(graph) {
+           _createdWayIDs = [];
+           var newWayIndex = 0;
 
-                   case '411':
-                   case '412':
-                   case '413':
-                     description = "This feature~".concat(description);
-                     break;
-                 } // move markers slightly so it doesn't obscure the geometry,
-                 // then move markers away from other coincident markers
+           for (var i = 0; i < nodeIds.length; i++) {
+             var nodeId = nodeIds[i];
+             var candidates = action.waysForNode(nodeId, graph);
 
+             for (var j = 0; j < candidates.length; j++) {
+               graph = split(graph, nodeId, candidates[j], newWayIds && newWayIds[newWayIndex]);
+               newWayIndex += 1;
+             }
+           }
 
-                 var coincident = false;
+           return graph;
+         };
 
-                 do {
-                   // first time, move marker up. after that, move marker right.
-                   var delta = coincident ? [0.00001, 0] : [0, 0.00001];
-                   loc = geoVecAdd(loc, delta);
-                   var bbox = geoExtent(loc).bbox();
-                   coincident = _cache.rtree.search(bbox).length;
-                 } while (coincident);
+         action.getCreatedWayIDs = function () {
+           return _createdWayIDs;
+         };
 
-                 var d = new QAItem(loc, _this, itemType, id, {
-                   comment: comment,
-                   description: description,
-                   whichType: whichType,
-                   parentIssueType: parentIssueType,
-                   severity: whichTemplate.severity || 'error',
-                   objectId: objectId,
-                   objectType: objectType,
-                   schema: schema,
-                   title: title
-                 });
-                 d.replacements = tokenReplacements(d);
-                 _cache.data[id] = d;
+         action.waysForNode = function (nodeId, graph) {
+           var node = graph.entity(nodeId);
+           var splittableParents = graph.parentWays(node).filter(isSplittable);
 
-                 _cache.rtree.insert(encodeIssueRtree(d));
-               });
-               dispatch$1.call('loaded');
-             })["catch"](function () {
-               delete _cache.inflightTile[tile.id];
-               _cache.loadedTile[tile.id] = true;
+           if (!_wayIDs) {
+             // If the ways to split aren't specified, only split the lines.
+             // If there are no lines to split, split the areas.
+             var hasLine = splittableParents.some(function (parent) {
+               return parent.geometry(graph) === 'line';
              });
-           });
-         },
-         postUpdate: function postUpdate(d, callback) {
-           var _this2 = this;
 
-           if (_cache.inflightPost[d.id]) {
-             return callback({
-               message: 'Error update already inflight',
-               status: -2
-             }, d);
+             if (hasLine) {
+               return splittableParents.filter(function (parent) {
+                 return parent.geometry(graph) === 'line';
+               });
+             }
            }
 
-           var params = {
-             schema: d.schema,
-             id: d.id
-           };
+           return splittableParents;
 
-           if (d.newStatus) {
-             params.st = d.newStatus;
-           }
+           function isSplittable(parent) {
+             // If the ways to split are specified, ignore everything else.
+             if (_wayIDs && _wayIDs.indexOf(parent.id) === -1) return false; // We can fake splitting closed ways at their endpoints...
 
-           if (d.newComment !== undefined) {
-             params.co = d.newComment;
-           } // NOTE: This throws a CORS err, but it seems successful.
-           // We don't care too much about the response, so this is fine.
+             if (parent.isClosed()) return true; // otherwise, we can't split nodes at their endpoints.
 
+             for (var i = 1; i < parent.nodes.length - 1; i++) {
+               if (parent.nodes[i] === nodeId) return true;
+             }
 
-           var url = "".concat(_krUrlRoot, "/comment.php?") + utilQsString(params);
-           var controller = new AbortController();
-           _cache.inflightPost[d.id] = controller; // Since this is expected to throw an error just continue as if it worked
-           // (worst case scenario the request truly fails and issue will show up if iD restarts)
+             return false;
+           }
+         };
 
-           d3_json(url, {
-             signal: controller.signal
-           })["finally"](function () {
-             delete _cache.inflightPost[d.id];
+         action.ways = function (graph) {
+           return utilArrayUniq([].concat.apply([], nodeIds.map(function (nodeId) {
+             return action.waysForNode(nodeId, graph);
+           })));
+         };
 
-             if (d.newStatus === 'ignore') {
-               // ignore permanently (false positive)
-               _this2.removeItem(d);
-             } else if (d.newStatus === 'ignore_t') {
-               // ignore temporarily (error fixed)
-               _this2.removeItem(d);
+         action.disabled = function (graph) {
+           for (var i = 0; i < nodeIds.length; i++) {
+             var nodeId = nodeIds[i];
+             var candidates = action.waysForNode(nodeId, graph);
 
-               _cache.closed["".concat(d.schema, ":").concat(d.id)] = true;
-             } else {
-               d = _this2.replaceItem(d.update({
-                 comment: d.newComment,
-                 newComment: undefined,
-                 newState: undefined
-               }));
+             if (candidates.length === 0 || _wayIDs && _wayIDs.length !== candidates.length) {
+               return 'not_eligible';
              }
+           }
+         };
 
-             if (callback) callback(null, d);
-           });
-         },
-         // Get all cached QAItems covering the viewport
-         getItems: function getItems(projection) {
-           var viewport = projection.clipExtent();
-           var min = [viewport[0][0], viewport[1][1]];
-           var max = [viewport[1][0], viewport[0][1]];
-           var bbox = geoExtent(projection.invert(min), projection.invert(max)).bbox();
-           return _cache.rtree.search(bbox).map(function (d) {
-             return d.data;
-           });
-         },
-         // Get a QAItem from cache
-         // NOTE: Don't change method name until UI v3 is merged
-         getError: function getError(id) {
-           return _cache.data[id];
+         action.limitWays = function (val) {
+           if (!arguments.length) return _wayIDs;
+           _wayIDs = val;
+           return action;
+         };
+
+         action.keepHistoryOn = function (val) {
+           if (!arguments.length) return _keepHistoryOn;
+           _keepHistoryOn = val;
+           return action;
+         };
+
+         return action;
+       }
+
+       function coreGraph(other, mutable) {
+         if (!(this instanceof coreGraph)) return new coreGraph(other, mutable);
+
+         if (other instanceof coreGraph) {
+           var base = other.base();
+           this.entities = Object.assign(Object.create(base.entities), other.entities);
+           this._parentWays = Object.assign(Object.create(base.parentWays), other._parentWays);
+           this._parentRels = Object.assign(Object.create(base.parentRels), other._parentRels);
+         } else {
+           this.entities = Object.create({});
+           this._parentWays = Object.create({});
+           this._parentRels = Object.create({});
+           this.rebase(other || [], [this]);
+         }
+
+         this.transients = {};
+         this._childNodes = {};
+         this.frozen = !mutable;
+       }
+       coreGraph.prototype = {
+         hasEntity: function hasEntity(id) {
+           return this.entities[id];
          },
-         // Replace a single QAItem in the cache
-         replaceItem: function replaceItem(item) {
-           if (!(item instanceof QAItem) || !item.id) return;
-           _cache.data[item.id] = item;
-           updateRtree(encodeIssueRtree(item), true); // true = replace
+         entity: function entity(id) {
+           var entity = this.entities[id]; //https://github.com/openstreetmap/iD/issues/3973#issuecomment-307052376
+
+           if (!entity) {
+             entity = this.entities.__proto__[id]; // eslint-disable-line no-proto
+           }
+
+           if (!entity) {
+             throw new Error('entity ' + id + ' not found');
+           }
 
-           return item;
-         },
-         // Remove a single QAItem from the cache
-         removeItem: function removeItem(item) {
-           if (!(item instanceof QAItem) || !item.id) return;
-           delete _cache.data[item.id];
-           updateRtree(encodeIssueRtree(item), false); // false = remove
+           return entity;
          },
-         issueURL: function issueURL(item) {
-           return "".concat(_krUrlRoot, "/report_map.php?schema=").concat(item.schema, "&error=").concat(item.id);
+         geometry: function geometry(id) {
+           return this.entity(id).geometry(this);
          },
-         // Get an array of issues closed during this session.
-         // Used to populate `closed:keepright` changeset tag
-         getClosedIDs: function getClosedIDs() {
-           return Object.keys(_cache.closed).sort();
-         }
-       };
-
-       var tiler$1 = utilTiler();
-       var dispatch$2 = dispatch('loaded');
-       var _tileZoom$1 = 14;
-       var _impOsmUrls = {
-         ow: 'https://grab.community.improve-osm.org/directionOfFlowService',
-         mr: 'https://grab.community.improve-osm.org/missingGeoService',
-         tr: 'https://grab.community.improve-osm.org/turnRestrictionService'
-       };
-       var _impOsmData = {
-         icons: {}
-       }; // This gets reassigned if reset
-
-       var _cache$1;
+         "transient": function transient(entity, key, fn) {
+           var id = entity.id;
+           var transients = this.transients[id] || (this.transients[id] = {});
 
-       function abortRequest$1(i) {
-         Object.values(i).forEach(function (controller) {
-           if (controller) {
-             controller.abort();
+           if (transients[key] !== undefined) {
+             return transients[key];
            }
-         });
-       }
 
-       function abortUnwantedRequests$1(cache, tiles) {
-         Object.keys(cache.inflightTile).forEach(function (k) {
-           var wanted = tiles.find(function (tile) {
-             return k === tile.id;
-           });
+           transients[key] = fn.call(entity);
+           return transients[key];
+         },
+         parentWays: function parentWays(entity) {
+           var parents = this._parentWays[entity.id];
+           var result = [];
 
-           if (!wanted) {
-             abortRequest$1(cache.inflightTile[k]);
-             delete cache.inflightTile[k];
+           if (parents) {
+             parents.forEach(function (id) {
+               result.push(this.entity(id));
+             }, this);
            }
-         });
-       }
-
-       function encodeIssueRtree$1(d) {
-         return {
-           minX: d.loc[0],
-           minY: d.loc[1],
-           maxX: d.loc[0],
-           maxY: d.loc[1],
-           data: d
-         };
-       } // Replace or remove QAItem from rtree
 
+           return result;
+         },
+         isPoi: function isPoi(entity) {
+           var parents = this._parentWays[entity.id];
+           return !parents || parents.size === 0;
+         },
+         isShared: function isShared(entity) {
+           var parents = this._parentWays[entity.id];
+           return parents && parents.size > 1;
+         },
+         parentRelations: function parentRelations(entity) {
+           var parents = this._parentRels[entity.id];
+           var result = [];
 
-       function updateRtree$1(item, replace) {
-         _cache$1.rtree.remove(item, function (a, b) {
-           return a.data.id === b.data.id;
-         });
+           if (parents) {
+             parents.forEach(function (id) {
+               result.push(this.entity(id));
+             }, this);
+           }
 
-         if (replace) {
-           _cache$1.rtree.insert(item);
-         }
-       }
+           return result;
+         },
+         parentMultipolygons: function parentMultipolygons(entity) {
+           return this.parentRelations(entity).filter(function (relation) {
+             return relation.isMultipolygon();
+           });
+         },
+         childNodes: function childNodes(entity) {
+           if (this._childNodes[entity.id]) return this._childNodes[entity.id];
+           if (!entity.nodes) return [];
+           var nodes = [];
 
-       function linkErrorObject(d) {
-         return "<a class=\"error_object_link\">".concat(d, "</a>");
-       }
+           for (var i = 0; i < entity.nodes.length; i++) {
+             nodes[i] = this.entity(entity.nodes[i]);
+           }
+           this._childNodes[entity.id] = nodes;
+           return this._childNodes[entity.id];
+         },
+         base: function base() {
+           return {
+             'entities': Object.getPrototypeOf(this.entities),
+             'parentWays': Object.getPrototypeOf(this._parentWays),
+             'parentRels': Object.getPrototypeOf(this._parentRels)
+           };
+         },
+         // Unlike other graph methods, rebase mutates in place. This is because it
+         // is used only during the history operation that merges newly downloaded
+         // data into each state. To external consumers, it should appear as if the
+         // graph always contained the newly downloaded data.
+         rebase: function rebase(entities, stack, force) {
+           var base = this.base();
+           var i, j, k, id;
 
-       function linkEntity(d) {
-         return "<a class=\"error_entity_link\">".concat(d, "</a>");
-       }
+           for (i = 0; i < entities.length; i++) {
+             var entity = entities[i];
+             if (!entity.visible || !force && base.entities[entity.id]) continue; // Merging data into the base graph
 
-       function pointAverage(points) {
-         if (points.length) {
-           var sum = points.reduce(function (acc, point) {
-             return geoVecAdd(acc, [point.lon, point.lat]);
-           }, [0, 0]);
-           return geoVecScale(sum, 1 / points.length);
-         } else {
-           return [0, 0];
-         }
-       }
+             base.entities[entity.id] = entity;
 
-       function relativeBearing(p1, p2) {
-         var angle = Math.atan2(p2.lon - p1.lon, p2.lat - p1.lat);
+             this._updateCalculated(undefined, entity, base.parentWays, base.parentRels); // Restore provisionally-deleted nodes that are discovered to have an extant parent
 
-         if (angle < 0) {
-           angle += 2 * Math.PI;
-         } // Return degrees
 
+             if (entity.type === 'way') {
+               for (j = 0; j < entity.nodes.length; j++) {
+                 id = entity.nodes[j];
 
-         return angle * 180 / Math.PI;
-       } // Assuming range [0,360)
+                 for (k = 1; k < stack.length; k++) {
+                   var ents = stack[k].entities;
 
+                   if (ents.hasOwnProperty(id) && ents[id] === undefined) {
+                     delete ents[id];
+                   }
+                 }
+               }
+             }
+           }
 
-       function cardinalDirection(bearing) {
-         var dir = 45 * Math.round(bearing / 45);
-         var compass = {
-           0: 'north',
-           45: 'northeast',
-           90: 'east',
-           135: 'southeast',
-           180: 'south',
-           225: 'southwest',
-           270: 'west',
-           315: 'northwest',
-           360: 'north'
-         };
-         return _t("QA.improveOSM.directions.".concat(compass[dir]));
-       } // Errors shouldn't obscure each other
+           for (i = 0; i < stack.length; i++) {
+             stack[i]._updateRebased();
+           }
+         },
+         _updateRebased: function _updateRebased() {
+           var base = this.base();
+           Object.keys(this._parentWays).forEach(function (child) {
+             if (base.parentWays[child]) {
+               base.parentWays[child].forEach(function (id) {
+                 if (!this.entities.hasOwnProperty(id)) {
+                   this._parentWays[child].add(id);
+                 }
+               }, this);
+             }
+           }, this);
+           Object.keys(this._parentRels).forEach(function (child) {
+             if (base.parentRels[child]) {
+               base.parentRels[child].forEach(function (id) {
+                 if (!this.entities.hasOwnProperty(id)) {
+                   this._parentRels[child].add(id);
+                 }
+               }, this);
+             }
+           }, this);
+           this.transients = {}; // this._childNodes is not updated, under the assumption that
+           // ways are always downloaded with their child nodes.
+         },
+         // Updates calculated properties (parentWays, parentRels) for the specified change
+         _updateCalculated: function _updateCalculated(oldentity, entity, parentWays, parentRels) {
+           parentWays = parentWays || this._parentWays;
+           parentRels = parentRels || this._parentRels;
+           var type = entity && entity.type || oldentity && oldentity.type;
+           var removed, added, i;
 
+           if (type === 'way') {
+             // Update parentWays
+             if (oldentity && entity) {
+               removed = utilArrayDifference(oldentity.nodes, entity.nodes);
+               added = utilArrayDifference(entity.nodes, oldentity.nodes);
+             } else if (oldentity) {
+               removed = oldentity.nodes;
+               added = [];
+             } else if (entity) {
+               removed = [];
+               added = entity.nodes;
+             }
 
-       function preventCoincident(loc, bumpUp) {
-         var coincident = false;
+             for (i = 0; i < removed.length; i++) {
+               // make a copy of prototype property, store as own property, and update..
+               parentWays[removed[i]] = new Set(parentWays[removed[i]]);
+               parentWays[removed[i]]["delete"](oldentity.id);
+             }
 
-         do {
-           // first time, move marker up. after that, move marker right.
-           var delta = coincident ? [0.00001, 0] : bumpUp ? [0, 0.00001] : [0, 0];
-           loc = geoVecAdd(loc, delta);
-           var bbox = geoExtent(loc).bbox();
-           coincident = _cache$1.rtree.search(bbox).length;
-         } while (coincident);
+             for (i = 0; i < added.length; i++) {
+               // make a copy of prototype property, store as own property, and update..
+               parentWays[added[i]] = new Set(parentWays[added[i]]);
+               parentWays[added[i]].add(entity.id);
+             }
+           } else if (type === 'relation') {
+             // Update parentRels
+             // diff only on the IDs since the same entity can be a member multiple times with different roles
+             var oldentityMemberIDs = oldentity ? oldentity.members.map(function (m) {
+               return m.id;
+             }) : [];
+             var entityMemberIDs = entity ? entity.members.map(function (m) {
+               return m.id;
+             }) : [];
 
-         return loc;
-       }
+             if (oldentity && entity) {
+               removed = utilArrayDifference(oldentityMemberIDs, entityMemberIDs);
+               added = utilArrayDifference(entityMemberIDs, oldentityMemberIDs);
+             } else if (oldentity) {
+               removed = oldentityMemberIDs;
+               added = [];
+             } else if (entity) {
+               removed = [];
+               added = entityMemberIDs;
+             }
 
-       var serviceImproveOSM = {
-         title: 'improveOSM',
-         init: function init() {
-           _mainFileFetcher.get('qa_data').then(function (d) {
-             return _impOsmData = d.improveOSM;
-           });
+             for (i = 0; i < removed.length; i++) {
+               // make a copy of prototype property, store as own property, and update..
+               parentRels[removed[i]] = new Set(parentRels[removed[i]]);
+               parentRels[removed[i]]["delete"](oldentity.id);
+             }
 
-           if (!_cache$1) {
-             this.reset();
+             for (i = 0; i < added.length; i++) {
+               // make a copy of prototype property, store as own property, and update..
+               parentRels[added[i]] = new Set(parentRels[added[i]]);
+               parentRels[added[i]].add(entity.id);
+             }
            }
-
-           this.event = utilRebind(this, dispatch$2, 'on');
          },
-         reset: function reset() {
-           if (_cache$1) {
-             Object.values(_cache$1.inflightTile).forEach(abortRequest$1);
-           }
+         replace: function replace(entity) {
+           if (this.entities[entity.id] === entity) return this;
+           return this.update(function () {
+             this._updateCalculated(this.entities[entity.id], entity);
 
-           _cache$1 = {
-             data: {},
-             loadedTile: {},
-             inflightTile: {},
-             inflightPost: {},
-             closed: {},
-             rtree: new RBush()
-           };
+             this.entities[entity.id] = entity;
+           });
          },
-         loadIssues: function loadIssues(projection) {
-           var _this = this;
+         remove: function remove(entity) {
+           return this.update(function () {
+             this._updateCalculated(entity, undefined);
 
-           var options = {
-             client: 'iD',
-             status: 'OPEN',
-             zoom: '19' // Use a high zoom so that clusters aren't returned
+             this.entities[entity.id] = undefined;
+           });
+         },
+         revert: function revert(id) {
+           var baseEntity = this.base().entities[id];
+           var headEntity = this.entities[id];
+           if (headEntity === baseEntity) return this;
+           return this.update(function () {
+             this._updateCalculated(headEntity, baseEntity);
 
-           }; // determine the needed tiles to cover the view
+             delete this.entities[id];
+           });
+         },
+         update: function update() {
+           var graph = this.frozen ? coreGraph(this, true) : this;
 
-           var tiles = tiler$1.zoomExtent([_tileZoom$1, _tileZoom$1]).getTiles(projection); // abort inflight requests that are no longer needed
+           for (var i = 0; i < arguments.length; i++) {
+             arguments[i].call(graph, graph);
+           }
 
-           abortUnwantedRequests$1(_cache$1, tiles); // issue new requests..
+           if (this.frozen) graph.frozen = true;
+           return graph;
+         },
+         // Obliterates any existing entities
+         load: function load(entities) {
+           var base = this.base();
+           this.entities = Object.create(base.entities);
 
-           tiles.forEach(function (tile) {
-             if (_cache$1.loadedTile[tile.id] || _cache$1.inflightTile[tile.id]) return;
+           for (var i in entities) {
+             this.entities[i] = entities[i];
 
-             var _tile$extent$rectangl = tile.extent.rectangle(),
-                 _tile$extent$rectangl2 = _slicedToArray(_tile$extent$rectangl, 4),
-                 east = _tile$extent$rectangl2[0],
-                 north = _tile$extent$rectangl2[1],
-                 west = _tile$extent$rectangl2[2],
-                 south = _tile$extent$rectangl2[3];
+             this._updateCalculated(base.entities[i], this.entities[i]);
+           }
 
-             var params = Object.assign({}, options, {
-               east: east,
-               south: south,
-               west: west,
-               north: north
-             }); // 3 separate requests to store for each tile
+           return this;
+         }
+       };
 
-             var requests = {};
-             Object.keys(_impOsmUrls).forEach(function (k) {
-               // We exclude WATER from missing geometry as it doesn't seem useful
-               // We use most confident one-way and turn restrictions only, still have false positives
-               var kParams = Object.assign({}, params, k === 'mr' ? {
-                 type: 'PARKING,ROAD,BOTH,PATH'
-               } : {
-                 confidenceLevel: 'C1'
-               });
-               var url = "".concat(_impOsmUrls[k], "/search?") + utilQsString(kParams);
-               var controller = new AbortController();
-               requests[k] = controller;
-               d3_json(url, {
-                 signal: controller.signal
-               }).then(function (data) {
-                 delete _cache$1.inflightTile[tile.id][k];
+       function osmTurn(turn) {
+         if (!(this instanceof osmTurn)) {
+           return new osmTurn(turn);
+         }
 
-                 if (!Object.keys(_cache$1.inflightTile[tile.id]).length) {
-                   delete _cache$1.inflightTile[tile.id];
-                   _cache$1.loadedTile[tile.id] = true;
-                 } // Road segments at high zoom == oneways
+         Object.assign(this, turn);
+       }
+       function osmIntersection(graph, startVertexId, maxDistance) {
+         maxDistance = maxDistance || 30; // in meters
 
+         var vgraph = coreGraph(); // virtual graph
 
-                 if (data.roadSegments) {
-                   data.roadSegments.forEach(function (feature) {
-                     // Position error at the approximate middle of the segment
-                     var points = feature.points,
-                         wayId = feature.wayId,
-                         fromNodeId = feature.fromNodeId,
-                         toNodeId = feature.toNodeId;
-                     var itemId = "".concat(wayId).concat(fromNodeId).concat(toNodeId);
-                     var mid = points.length / 2;
-                     var loc; // Even number of points, find midpoint of the middle two
-                     // Odd number of points, use position of very middle point
+         var i, j, k;
 
-                     if (mid % 1 === 0) {
-                       loc = pointAverage([points[mid - 1], points[mid]]);
-                     } else {
-                       mid = points[Math.floor(mid)];
-                       loc = [mid.lon, mid.lat];
-                     } // One-ways can land on same segment in opposite direction
+         function memberOfRestriction(entity) {
+           return graph.parentRelations(entity).some(function (r) {
+             return r.isRestriction();
+           });
+         }
 
+         function isRoad(way) {
+           if (way.isArea() || way.isDegenerate()) return false;
+           var roads = {
+             'motorway': true,
+             'motorway_link': true,
+             'trunk': true,
+             'trunk_link': true,
+             'primary': true,
+             'primary_link': true,
+             'secondary': true,
+             'secondary_link': true,
+             'tertiary': true,
+             'tertiary_link': true,
+             'residential': true,
+             'unclassified': true,
+             'living_street': true,
+             'service': true,
+             'road': true,
+             'track': true
+           };
+           return roads[way.tags.highway];
+         }
 
-                     loc = preventCoincident(loc, false);
-                     var d = new QAItem(loc, _this, k, itemId, {
-                       issueKey: k,
-                       // used as a category
-                       identifier: {
-                         // used to post changes
-                         wayId: wayId,
-                         fromNodeId: fromNodeId,
-                         toNodeId: toNodeId
-                       },
-                       objectId: wayId,
-                       objectType: 'way'
-                     }); // Variables used in the description
+         var startNode = graph.entity(startVertexId);
+         var checkVertices = [startNode];
+         var checkWays;
+         var vertices = [];
+         var vertexIds = [];
+         var vertex;
+         var ways = [];
+         var wayIds = [];
+         var way;
+         var nodes = [];
+         var node;
+         var parents = [];
+         var parent; // `actions` will store whatever actions must be performed to satisfy
+         // preconditions for adding a turn restriction to this intersection.
+         //  - Remove any existing degenerate turn restrictions (missing from/to, etc)
+         //  - Reverse oneways so that they are drawn in the forward direction
+         //  - Split ways on key vertices
 
-                     d.replacements = {
-                       percentage: feature.percentOfTrips,
-                       num_trips: feature.numberOfTrips,
-                       highway: linkErrorObject(_t('QA.keepRight.error_parts.highway')),
-                       from_node: linkEntity('n' + feature.fromNodeId),
-                       to_node: linkEntity('n' + feature.toNodeId)
-                     };
-                     _cache$1.data[d.id] = d;
+         var actions = []; // STEP 1:  walk the graph outwards from starting vertex to search
+         //  for more key vertices and ways to include in the intersection..
 
-                     _cache$1.rtree.insert(encodeIssueRtree$1(d));
-                   });
-                 } // Tiles at high zoom == missing roads
+         while (checkVertices.length) {
+           vertex = checkVertices.pop(); // check this vertex for parent ways that are roads
 
+           checkWays = graph.parentWays(vertex);
+           var hasWays = false;
 
-                 if (data.tiles) {
-                   data.tiles.forEach(function (feature) {
-                     var type = feature.type,
-                         x = feature.x,
-                         y = feature.y,
-                         numberOfTrips = feature.numberOfTrips;
-                     var geoType = type.toLowerCase();
-                     var itemId = "".concat(geoType).concat(x).concat(y).concat(numberOfTrips); // Average of recorded points should land on the missing geometry
-                     // Missing geometry could happen to land on another error
+           for (i = 0; i < checkWays.length; i++) {
+             way = checkWays[i];
+             if (!isRoad(way) && !memberOfRestriction(way)) continue;
+             ways.push(way); // it's a road, or it's already in a turn restriction
 
-                     var loc = pointAverage(feature.points);
-                     loc = preventCoincident(loc, false);
-                     var d = new QAItem(loc, _this, "".concat(k, "-").concat(geoType), itemId, {
-                       issueKey: k,
-                       identifier: {
-                         x: x,
-                         y: y
-                       }
-                     });
-                     d.replacements = {
-                       num_trips: numberOfTrips,
-                       geometry_type: _t("QA.improveOSM.geometry_types.".concat(geoType))
-                     }; // -1 trips indicates data came from a 3rd party
+             hasWays = true; // check the way's children for more key vertices
 
-                     if (numberOfTrips === -1) {
-                       d.desc = _t('QA.improveOSM.error_types.mr.description_alt', d.replacements);
-                     }
+             nodes = utilArrayUniq(graph.childNodes(way));
 
-                     _cache$1.data[d.id] = d;
+             for (j = 0; j < nodes.length; j++) {
+               node = nodes[j];
+               if (node === vertex) continue; // same thing
 
-                     _cache$1.rtree.insert(encodeIssueRtree$1(d));
-                   });
-                 } // Entities at high zoom == turn restrictions
+               if (vertices.indexOf(node) !== -1) continue; // seen it already
 
+               if (geoSphericalDistance(node.loc, startNode.loc) > maxDistance) continue; // too far from start
+               // a key vertex will have parents that are also roads
 
-                 if (data.entities) {
-                   data.entities.forEach(function (feature) {
-                     var point = feature.point,
-                         id = feature.id,
-                         segments = feature.segments,
-                         numberOfPasses = feature.numberOfPasses,
-                         turnType = feature.turnType;
-                     var itemId = "".concat(id.replace(/[,:+#]/g, '_')); // Turn restrictions could be missing at same junction
-                     // We also want to bump the error up so node is accessible
+               var hasParents = false;
+               parents = graph.parentWays(node);
 
-                     var loc = preventCoincident([point.lon, point.lat], true); // Elements are presented in a strange way
+               for (k = 0; k < parents.length; k++) {
+                 parent = parents[k];
+                 if (parent === way) continue; // same thing
 
-                     var ids = id.split(',');
-                     var from_way = ids[0];
-                     var via_node = ids[3];
-                     var to_way = ids[2].split(':')[1];
-                     var d = new QAItem(loc, _this, k, itemId, {
-                       issueKey: k,
-                       identifier: id,
-                       objectId: via_node,
-                       objectType: 'node'
-                     }); // Travel direction along from_way clarifies the turn restriction
+                 if (ways.indexOf(parent) !== -1) continue; // seen it already
 
-                     var _segments$0$points = _slicedToArray(segments[0].points, 2),
-                         p1 = _segments$0$points[0],
-                         p2 = _segments$0$points[1];
+                 if (!isRoad(parent)) continue; // not a road
 
-                     var dir_of_travel = cardinalDirection(relativeBearing(p1, p2)); // Variables used in the description
+                 hasParents = true;
+                 break;
+               }
 
-                     d.replacements = {
-                       num_passed: numberOfPasses,
-                       num_trips: segments[0].numberOfTrips,
-                       turn_restriction: turnType.toLowerCase(),
-                       from_way: linkEntity('w' + from_way),
-                       to_way: linkEntity('w' + to_way),
-                       travel_direction: dir_of_travel,
-                       junction: linkErrorObject(_t('QA.keepRight.error_parts.this_node'))
-                     };
-                     _cache$1.data[d.id] = d;
+               if (hasParents) {
+                 checkVertices.push(node);
+               }
+             }
+           }
 
-                     _cache$1.rtree.insert(encodeIssueRtree$1(d));
+           if (hasWays) {
+             vertices.push(vertex);
+           }
+         }
 
-                     dispatch$2.call('loaded');
-                   });
-                 }
-               })["catch"](function () {
-                 delete _cache$1.inflightTile[tile.id][k];
+         vertices = utilArrayUniq(vertices);
+         ways = utilArrayUniq(ways); // STEP 2:  Build a virtual graph containing only the entities in the intersection..
+         // Everything done after this step should act on the virtual graph
+         // Any actions that must be performed later to the main graph go in `actions` array
 
-                 if (!Object.keys(_cache$1.inflightTile[tile.id]).length) {
-                   delete _cache$1.inflightTile[tile.id];
-                   _cache$1.loadedTile[tile.id] = true;
-                 }
-               });
-             });
-             _cache$1.inflightTile[tile.id] = requests;
+         ways.forEach(function (way) {
+           graph.childNodes(way).forEach(function (node) {
+             vgraph = vgraph.replace(node);
            });
-         },
-         getComments: function getComments(item) {
-           var _this2 = this;
+           vgraph = vgraph.replace(way);
+           graph.parentRelations(way).forEach(function (relation) {
+             if (relation.isRestriction()) {
+               if (relation.isValidRestriction(graph)) {
+                 vgraph = vgraph.replace(relation);
+               } else if (relation.isComplete(graph)) {
+                 actions.push(actionDeleteRelation(relation.id));
+               }
+             }
+           });
+         }); // STEP 3:  Force all oneways to be drawn in the forward direction
 
-           // If comments already retrieved no need to do so again
-           if (item.comments) {
-             return Promise.resolve(item);
+         ways.forEach(function (w) {
+           var way = vgraph.entity(w.id);
+
+           if (way.tags.oneway === '-1') {
+             var action = actionReverse(way.id, {
+               reverseOneway: true
+             });
+             actions.push(action);
+             vgraph = action(vgraph);
            }
+         }); // STEP 4:  Split ways on key vertices
 
-           var key = item.issueKey;
-           var qParams = {};
+         var origCount = osmEntity.id.next.way;
+         vertices.forEach(function (v) {
+           // This is an odd way to do it, but we need to find all the ways that
+           // will be split here, then split them one at a time to ensure that these
+           // actions can be replayed on the main graph exactly in the same order.
+           // (It is unintuitive, but the order of ways returned from graph.parentWays()
+           // is arbitrary, depending on how the main graph and vgraph were built)
+           var splitAll = actionSplit([v.id]).keepHistoryOn('first');
 
-           if (key === 'ow') {
-             qParams = item.identifier;
-           } else if (key === 'mr') {
-             qParams.tileX = item.identifier.x;
-             qParams.tileY = item.identifier.y;
-           } else if (key === 'tr') {
-             qParams.targetId = item.identifier;
+           if (!splitAll.disabled(vgraph)) {
+             splitAll.ways(vgraph).forEach(function (way) {
+               var splitOne = actionSplit([v.id]).limitWays([way.id]).keepHistoryOn('first');
+               actions.push(splitOne);
+               vgraph = splitOne(vgraph);
+             });
            }
+         }); // In here is where we should also split the intersection at nearby junction.
+         //   for https://github.com/mapbox/iD-internal/issues/31
+         // nearbyVertices.forEach(function(v) {
+         // });
+         // Reasons why we reset the way id count here:
+         //  1. Continuity with way ids created by the splits so that we can replay
+         //     these actions later if the user decides to create a turn restriction
+         //  2. Avoids churning way ids just by hovering over a vertex
+         //     and displaying the turn restriction editor
 
-           var url = "".concat(_impOsmUrls[key], "/retrieveComments?") + utilQsString(qParams);
+         osmEntity.id.next.way = origCount; // STEP 5:  Update arrays to point to vgraph entities
 
-           var cacheComments = function cacheComments(data) {
-             // Assign directly for immediate use afterwards
-             // comments are served newest to oldest
-             item.comments = data.comments ? data.comments.reverse() : [];
+         vertexIds = vertices.map(function (v) {
+           return v.id;
+         });
+         vertices = [];
+         ways = [];
+         vertexIds.forEach(function (id) {
+           var vertex = vgraph.entity(id);
+           var parents = vgraph.parentWays(vertex);
+           vertices.push(vertex);
+           ways = ways.concat(parents);
+         });
+         vertices = utilArrayUniq(vertices);
+         ways = utilArrayUniq(ways);
+         vertexIds = vertices.map(function (v) {
+           return v.id;
+         });
+         wayIds = ways.map(function (w) {
+           return w.id;
+         }); // STEP 6:  Update the ways with some metadata that will be useful for
+         // walking the intersection graph later and rendering turn arrows.
 
-             _this2.replaceItem(item);
-           };
+         function withMetadata(way, vertexIds) {
+           var __oneWay = way.isOneWay(); // which affixes are key vertices?
 
-           return d3_json(url).then(cacheComments).then(function () {
-             return item;
-           });
-         },
-         postUpdate: function postUpdate(d, callback) {
-           if (!serviceOsm.authenticated()) {
-             // Username required in payload
-             return callback({
-               message: 'Not Authenticated',
-               status: -3
-             }, d);
-           }
 
-           if (_cache$1.inflightPost[d.id]) {
-             return callback({
-               message: 'Error update already inflight',
-               status: -2
-             }, d);
-           } // Payload can only be sent once username is established
+           var __first = vertexIds.indexOf(way.first()) !== -1;
 
+           var __last = vertexIds.indexOf(way.last()) !== -1; // what roles is this way eligible for?
 
-           serviceOsm.userDetails(sendPayload.bind(this));
 
-           function sendPayload(err, user) {
-             var _this3 = this;
+           var __via = __first && __last;
 
-             if (err) {
-               return callback(err, d);
-             }
+           var __from = __first && !__oneWay || __last;
 
-             var key = d.issueKey;
-             var url = "".concat(_impOsmUrls[key], "/comment");
-             var payload = {
-               username: user.display_name,
-               targetIds: [d.identifier]
-             };
+           var __to = __first || __last && !__oneWay;
 
-             if (d.newStatus) {
-               payload.status = d.newStatus;
-               payload.text = 'status changed';
-             } // Comment take place of default text
+           return way.update({
+             __first: __first,
+             __last: __last,
+             __from: __from,
+             __via: __via,
+             __to: __to,
+             __oneWay: __oneWay
+           });
+         }
 
+         ways = [];
+         wayIds.forEach(function (id) {
+           var way = withMetadata(vgraph.entity(id), vertexIds);
+           vgraph = vgraph.replace(way);
+           ways.push(way);
+         }); // STEP 7:  Simplify - This is an iterative process where we:
+         //  1. Find trivial vertices with only 2 parents
+         //  2. trim off the leaf way from those vertices and remove from vgraph
 
-             if (d.newComment) {
-               payload.text = d.newComment;
-             }
+         var keepGoing;
+         var removeWayIds = [];
+         var removeVertexIds = [];
 
-             var controller = new AbortController();
-             _cache$1.inflightPost[d.id] = controller;
-             var options = {
-               method: 'POST',
-               signal: controller.signal,
-               body: JSON.stringify(payload)
-             };
-             d3_json(url, options).then(function () {
-               delete _cache$1.inflightPost[d.id]; // Just a comment, update error in cache
+         do {
+           keepGoing = false;
+           checkVertices = vertexIds.slice();
 
-               if (!d.newStatus) {
-                 var now = new Date();
-                 var comments = d.comments ? d.comments : [];
-                 comments.push({
-                   username: payload.username,
-                   text: payload.text,
-                   timestamp: now.getTime() / 1000
-                 });
+           for (i = 0; i < checkVertices.length; i++) {
+             var vertexId = checkVertices[i];
+             vertex = vgraph.hasEntity(vertexId);
 
-                 _this3.replaceItem(d.update({
-                   comments: comments,
-                   newComment: undefined
-                 }));
-               } else {
-                 _this3.removeItem(d);
+             if (!vertex) {
+               if (vertexIds.indexOf(vertexId) !== -1) {
+                 vertexIds.splice(vertexIds.indexOf(vertexId), 1); // stop checking this one
+               }
 
-                 if (d.newStatus === 'SOLVED') {
-                   // Keep track of the number of issues closed per type to tag the changeset
-                   if (!(d.issueKey in _cache$1.closed)) {
-                     _cache$1.closed[d.issueKey] = 0;
-                   }
+               removeVertexIds.push(vertexId);
+               continue;
+             }
 
-                   _cache$1.closed[d.issueKey] += 1;
-                 }
+             parents = vgraph.parentWays(vertex);
+
+             if (parents.length < 3) {
+               if (vertexIds.indexOf(vertexId) !== -1) {
+                 vertexIds.splice(vertexIds.indexOf(vertexId), 1); // stop checking this one
                }
+             }
 
-               if (callback) callback(null, d);
-             })["catch"](function (err) {
-               delete _cache$1.inflightPost[d.id];
-               if (callback) callback(err.message);
-             });
-           }
-         },
-         // Get all cached QAItems covering the viewport
-         getItems: function getItems(projection) {
-           var viewport = projection.clipExtent();
-           var min = [viewport[0][0], viewport[1][1]];
-           var max = [viewport[1][0], viewport[0][1]];
-           var bbox = geoExtent(projection.invert(min), projection.invert(max)).bbox();
-           return _cache$1.rtree.search(bbox).map(function (d) {
-             return d.data;
-           });
-         },
-         // Get a QAItem from cache
-         // NOTE: Don't change method name until UI v3 is merged
-         getError: function getError(id) {
-           return _cache$1.data[id];
-         },
-         // get the name of the icon to display for this item
-         getIcon: function getIcon(itemType) {
-           return _impOsmData.icons[itemType];
-         },
-         // Replace a single QAItem in the cache
-         replaceItem: function replaceItem(issue) {
-           if (!(issue instanceof QAItem) || !issue.id) return;
-           _cache$1.data[issue.id] = issue;
-           updateRtree$1(encodeIssueRtree$1(issue), true); // true = replace
+             if (parents.length === 2) {
+               // vertex with 2 parents is trivial
+               var a = parents[0];
+               var b = parents[1];
+               var aIsLeaf = a && !a.__via;
+               var bIsLeaf = b && !b.__via;
+               var leaf, survivor;
 
-           return issue;
-         },
-         // Remove a single QAItem from the cache
-         removeItem: function removeItem(issue) {
-           if (!(issue instanceof QAItem) || !issue.id) return;
-           delete _cache$1.data[issue.id];
-           updateRtree$1(encodeIssueRtree$1(issue), false); // false = remove
-         },
-         // Used to populate `closed:improveosm:*` changeset tags
-         getClosedCounts: function getClosedCounts() {
-           return _cache$1.closed;
-         }
-       };
+               if (aIsLeaf && !bIsLeaf) {
+                 leaf = a;
+                 survivor = b;
+               } else if (!aIsLeaf && bIsLeaf) {
+                 leaf = b;
+                 survivor = a;
+               }
 
-       var quot = /"/g;
+               if (leaf && survivor) {
+                 survivor = withMetadata(survivor, vertexIds); // update survivor way
 
-       // B.2.3.2.1 CreateHTML(string, tag, attribute, value)
-       // https://tc39.es/ecma262/#sec-createhtml
-       var createHtml = function (string, tag, attribute, value) {
-         var S = String(requireObjectCoercible(string));
-         var p1 = '<' + tag;
-         if (attribute !== '') p1 += ' ' + attribute + '="' + String(value).replace(quot, '&quot;') + '"';
-         return p1 + '>' + S + '</' + tag + '>';
-       };
+                 vgraph = vgraph.replace(survivor).remove(leaf); // update graph
 
-       // check the existence of a method, lowercase
-       // of a tag and escaping quotes in arguments
-       var stringHtmlForced = function (METHOD_NAME) {
-         return fails(function () {
-           var test = ''[METHOD_NAME]('"');
-           return test !== test.toLowerCase() || test.split('"').length > 3;
-         });
-       };
+                 removeWayIds.push(leaf.id);
+                 keepGoing = true;
+               }
+             }
 
-       // `String.prototype.link` method
-       // https://tc39.es/ecma262/#sec-string.prototype.link
-       _export({ target: 'String', proto: true, forced: stringHtmlForced('link') }, {
-         link: function link(url) {
-           return createHtml(this, 'a', 'href', url);
-         }
-       });
+             parents = vgraph.parentWays(vertex);
+
+             if (parents.length < 2) {
+               // vertex is no longer a key vertex
+               if (vertexIds.indexOf(vertexId) !== -1) {
+                 vertexIds.splice(vertexIds.indexOf(vertexId), 1); // stop checking this one
+               }
+
+               removeVertexIds.push(vertexId);
+               keepGoing = true;
+             }
 
-       var $trimEnd = stringTrim.end;
+             if (parents.length < 1) {
+               // vertex is no longer attached to anything
+               vgraph = vgraph.remove(vertex);
+             }
+           }
+         } while (keepGoing);
 
+         vertices = vertices.filter(function (vertex) {
+           return removeVertexIds.indexOf(vertex.id) === -1;
+         }).map(function (vertex) {
+           return vgraph.entity(vertex.id);
+         });
+         ways = ways.filter(function (way) {
+           return removeWayIds.indexOf(way.id) === -1;
+         }).map(function (way) {
+           return vgraph.entity(way.id);
+         }); // OK!  Here is our intersection..
 
-       var FORCED$e = stringTrimForced('trimEnd');
+         var intersection = {
+           graph: vgraph,
+           actions: actions,
+           vertices: vertices,
+           ways: ways
+         }; // Get all the valid turns through this intersection given a starting way id.
+         // This operates on the virtual graph for everything.
+         //
+         // Basically, walk through all possible paths from starting way,
+         //   honoring the existing turn restrictions as we go (watch out for loops!)
+         //
+         // For each path found, generate and return a `osmTurn` datastructure.
+         //
 
-       var trimEnd = FORCED$e ? function trimEnd() {
-         return $trimEnd(this);
-       } : ''.trimEnd;
+         intersection.turns = function (fromWayId, maxViaWay) {
+           if (!fromWayId) return [];
+           if (!maxViaWay) maxViaWay = 0;
+           var vgraph = intersection.graph;
+           var keyVertexIds = intersection.vertices.map(function (v) {
+             return v.id;
+           });
+           var start = vgraph.entity(fromWayId);
+           if (!start || !(start.__from || start.__via)) return []; // maxViaWay=0   from-*-to              (0 vias)
+           // maxViaWay=1   from-*-via-*-to        (1 via max)
+           // maxViaWay=2   from-*-via-*-via-*-to  (2 vias max)
 
-       // `String.prototype.{ trimEnd, trimRight }` methods
-       // https://tc39.es/ecma262/#sec-string.prototype.trimend
-       // https://tc39.es/ecma262/#String.prototype.trimright
-       _export({ target: 'String', proto: true, forced: FORCED$e }, {
-         trimEnd: trimEnd,
-         trimRight: trimEnd
-       });
+           var maxPathLength = maxViaWay * 2 + 3;
+           var turns = [];
+           step(start);
+           return turns; // traverse the intersection graph and find all the valid paths
 
-       var defaults = createCommonjsModule(function (module) {
-         function getDefaults() {
-           return {
-             baseUrl: null,
-             breaks: false,
-             gfm: true,
-             headerIds: true,
-             headerPrefix: '',
-             highlight: null,
-             langPrefix: 'language-',
-             mangle: true,
-             pedantic: false,
-             renderer: null,
-             sanitize: false,
-             sanitizer: null,
-             silent: false,
-             smartLists: false,
-             smartypants: false,
-             tokenizer: null,
-             walkTokens: null,
-             xhtml: false
-           };
-         }
+           function step(entity, currPath, currRestrictions, matchedRestriction) {
+             currPath = (currPath || []).slice(); // shallow copy
 
-         function changeDefaults(newDefaults) {
-           module.exports.defaults = newDefaults;
-         }
+             if (currPath.length >= maxPathLength) return;
+             currPath.push(entity.id);
+             currRestrictions = (currRestrictions || []).slice(); // shallow copy
 
-         module.exports = {
-           defaults: getDefaults(),
-           getDefaults: getDefaults,
-           changeDefaults: changeDefaults
-         };
-       });
+             var i, j;
 
-       /**
-        * Helpers
-        */
-       var escapeTest = /[&<>"']/;
-       var escapeReplace = /[&<>"']/g;
-       var escapeTestNoEncode = /[<>"']|&(?!#?\w+;)/;
-       var escapeReplaceNoEncode = /[<>"']|&(?!#?\w+;)/g;
-       var escapeReplacements = {
-         '&': '&amp;',
-         '<': '&lt;',
-         '>': '&gt;',
-         '"': '&quot;',
-         "'": '&#39;'
-       };
+             if (entity.type === 'node') {
+               var parents = vgraph.parentWays(entity);
+               var nextWays = []; // which ways can we step into?
 
-       var getEscapeReplacement = function getEscapeReplacement(ch) {
-         return escapeReplacements[ch];
-       };
+               for (i = 0; i < parents.length; i++) {
+                 var way = parents[i]; // if next way is a oneway incoming to this vertex, skip
 
-       function escape$1(html, encode) {
-         if (encode) {
-           if (escapeTest.test(html)) {
-             return html.replace(escapeReplace, getEscapeReplacement);
-           }
-         } else {
-           if (escapeTestNoEncode.test(html)) {
-             return html.replace(escapeReplaceNoEncode, getEscapeReplacement);
-           }
-         }
+                 if (way.__oneWay && way.nodes[0] !== entity.id) continue; // if we have seen it before (allowing for an initial u-turn), skip
 
-         return html;
-       }
+                 if (currPath.indexOf(way.id) !== -1 && currPath.length >= 3) continue; // Check all "current" restrictions (where we've already walked the `FROM`)
 
-       var unescapeTest = /&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/ig;
+                 var restrict = null;
 
-       function unescape$1(html) {
-         // explicitly match decimal, hex, and named HTML entities
-         return html.replace(unescapeTest, function (_, n) {
-           n = n.toLowerCase();
-           if (n === 'colon') return ':';
+                 for (j = 0; j < currRestrictions.length; j++) {
+                   var restriction = currRestrictions[j];
+                   var f = restriction.memberByRole('from');
+                   var v = restriction.membersByRole('via');
+                   var t = restriction.memberByRole('to');
+                   var isOnly = /^only_/.test(restriction.tags.restriction); // Does the current path match this turn restriction?
 
-           if (n.charAt(0) === '#') {
-             return n.charAt(1) === 'x' ? String.fromCharCode(parseInt(n.substring(2), 16)) : String.fromCharCode(+n.substring(1));
-           }
+                   var matchesFrom = f.id === fromWayId;
+                   var matchesViaTo = false;
+                   var isAlongOnlyPath = false;
 
-           return '';
-         });
-       }
+                   if (t.id === way.id) {
+                     // match TO
+                     if (v.length === 1 && v[0].type === 'node') {
+                       // match VIA node
+                       matchesViaTo = v[0].id === entity.id && (matchesFrom && currPath.length === 2 || !matchesFrom && currPath.length > 2);
+                     } else {
+                       // match all VIA ways
+                       var pathVias = [];
 
-       var caret = /(^|[^\[])\^/g;
+                       for (k = 2; k < currPath.length; k += 2) {
+                         // k = 2 skips FROM
+                         pathVias.push(currPath[k]); // (path goes way-node-way...)
+                       }
 
-       function edit(regex, opt) {
-         regex = regex.source || regex;
-         opt = opt || '';
-         var obj = {
-           replace: function replace(name, val) {
-             val = val.source || val;
-             val = val.replace(caret, '$1');
-             regex = regex.replace(name, val);
-             return obj;
-           },
-           getRegex: function getRegex() {
-             return new RegExp(regex, opt);
-           }
-         };
-         return obj;
-       }
+                       var restrictionVias = [];
 
-       var nonWordAndColonTest = /[^\w:]/g;
-       var originIndependentUrl = /^$|^[a-z][a-z0-9+.-]*:|^[?#]/i;
+                       for (k = 0; k < v.length; k++) {
+                         if (v[k].type === 'way') {
+                           restrictionVias.push(v[k].id);
+                         }
+                       }
 
-       function cleanUrl(sanitize, base, href) {
-         if (sanitize) {
-           var prot;
+                       var diff = utilArrayDifference(pathVias, restrictionVias);
+                       matchesViaTo = !diff.length;
+                     }
+                   } else if (isOnly) {
+                     for (k = 0; k < v.length; k++) {
+                       // way doesn't match TO, but is one of the via ways along the path of an "only"
+                       if (v[k].type === 'way' && v[k].id === way.id) {
+                         isAlongOnlyPath = true;
+                         break;
+                       }
+                     }
+                   }
 
-           try {
-             prot = decodeURIComponent(unescape$1(href)).replace(nonWordAndColonTest, '').toLowerCase();
-           } catch (e) {
-             return null;
-           }
+                   if (matchesViaTo) {
+                     if (isOnly) {
+                       restrict = {
+                         id: restriction.id,
+                         direct: matchesFrom,
+                         from: f.id,
+                         only: true,
+                         end: true
+                       };
+                     } else {
+                       restrict = {
+                         id: restriction.id,
+                         direct: matchesFrom,
+                         from: f.id,
+                         no: true,
+                         end: true
+                       };
+                     }
+                   } else {
+                     // indirect - caused by a different nearby restriction
+                     if (isAlongOnlyPath) {
+                       restrict = {
+                         id: restriction.id,
+                         direct: false,
+                         from: f.id,
+                         only: true,
+                         end: false
+                       };
+                     } else if (isOnly) {
+                       restrict = {
+                         id: restriction.id,
+                         direct: false,
+                         from: f.id,
+                         no: true,
+                         end: true
+                       };
+                     }
+                   } // stop looking if we find a "direct" restriction (matching FROM, VIA, TO)
 
-           if (prot.indexOf('javascript:') === 0 || prot.indexOf('vbscript:') === 0 || prot.indexOf('data:') === 0) {
-             return null;
-           }
-         }
 
-         if (base && !originIndependentUrl.test(href)) {
-           href = resolveUrl(base, href);
-         }
+                   if (restrict && restrict.direct) break;
+                 }
 
-         try {
-           href = encodeURI(href).replace(/%25/g, '%');
-         } catch (e) {
-           return null;
-         }
+                 nextWays.push({
+                   way: way,
+                   restrict: restrict
+                 });
+               }
 
-         return href;
-       }
+               nextWays.forEach(function (nextWay) {
+                 step(nextWay.way, currPath, currRestrictions, nextWay.restrict);
+               });
+             } else {
+               // entity.type === 'way'
+               if (currPath.length >= 3) {
+                 // this is a "complete" path..
+                 var turnPath = currPath.slice(); // shallow copy
+                 // an indirect restriction - only include the partial path (starting at FROM)
 
-       var baseUrls = {};
-       var justDomain = /^[^:]+:\/*[^/]*$/;
-       var protocol = /^([^:]+:)[\s\S]*$/;
-       var domain = /^([^:]+:\/*[^/]*)[\s\S]*$/;
+                 if (matchedRestriction && matchedRestriction.direct === false) {
+                   for (i = 0; i < turnPath.length; i++) {
+                     if (turnPath[i] === matchedRestriction.from) {
+                       turnPath = turnPath.slice(i);
+                       break;
+                     }
+                   }
+                 }
 
-       function resolveUrl(base, href) {
-         if (!baseUrls[' ' + base]) {
-           // we can ignore everything in base after the last slash of its path component,
-           // but we might need to add _that_
-           // https://tools.ietf.org/html/rfc3986#section-3
-           if (justDomain.test(base)) {
-             baseUrls[' ' + base] = base + '/';
-           } else {
-             baseUrls[' ' + base] = rtrim$1(base, '/', true);
-           }
-         }
+                 var turn = pathToTurn(turnPath);
 
-         base = baseUrls[' ' + base];
-         var relativeBase = base.indexOf(':') === -1;
+                 if (turn) {
+                   if (matchedRestriction) {
+                     turn.restrictionID = matchedRestriction.id;
+                     turn.no = matchedRestriction.no;
+                     turn.only = matchedRestriction.only;
+                     turn.direct = matchedRestriction.direct;
+                   }
 
-         if (href.substring(0, 2) === '//') {
-           if (relativeBase) {
-             return href;
-           }
+                   turns.push(osmTurn(turn));
+                 }
 
-           return base.replace(protocol, '$1') + href;
-         } else if (href.charAt(0) === '/') {
-           if (relativeBase) {
-             return href;
-           }
+                 if (currPath[0] === currPath[2]) return; // if we made a u-turn - stop here
+               }
 
-           return base.replace(domain, '$1') + href;
-         } else {
-           return base + href;
-         }
-       }
+               if (matchedRestriction && matchedRestriction.end) return; // don't advance any further
+               // which nodes can we step into?
 
-       var noopTest = {
-         exec: function noopTest() {}
-       };
+               var n1 = vgraph.entity(entity.first());
+               var n2 = vgraph.entity(entity.last());
+               var dist = geoSphericalDistance(n1.loc, n2.loc);
+               var nextNodes = [];
 
-       function merge$1(obj) {
-         var i = 1,
-             target,
-             key;
+               if (currPath.length > 1) {
+                 if (dist > maxDistance) return; // the next node is too far
 
-         for (; i < arguments.length; i++) {
-           target = arguments[i];
+                 if (!entity.__via) return; // this way is a leaf / can't be a via
+               }
 
-           for (key in target) {
-             if (Object.prototype.hasOwnProperty.call(target, key)) {
-               obj[key] = target[key];
-             }
-           }
-         }
+               if (!entity.__oneWay && // bidirectional..
+               keyVertexIds.indexOf(n1.id) !== -1 && // key vertex..
+               currPath.indexOf(n1.id) === -1) {
+                 // haven't seen it yet..
+                 nextNodes.push(n1); // can advance to first node
+               }
 
-         return obj;
-       }
+               if (keyVertexIds.indexOf(n2.id) !== -1 && // key vertex..
+               currPath.indexOf(n2.id) === -1) {
+                 // haven't seen it yet..
+                 nextNodes.push(n2); // can advance to last node
+               }
 
-       function splitCells(tableRow, count) {
-         // ensure that every cell-delimiting pipe has a space
-         // before it to distinguish it from an escaped pipe
-         var row = tableRow.replace(/\|/g, function (match, offset, str) {
-           var escaped = false,
-               curr = offset;
+               nextNodes.forEach(function (nextNode) {
+                 // gather restrictions FROM this way
+                 var fromRestrictions = vgraph.parentRelations(entity).filter(function (r) {
+                   if (!r.isRestriction()) return false;
+                   var f = r.memberByRole('from');
+                   if (!f || f.id !== entity.id) return false;
+                   var isOnly = /^only_/.test(r.tags.restriction);
+                   if (!isOnly) return true; // `only_` restrictions only matter along the direction of the VIA - #4849
 
-           while (--curr >= 0 && str[curr] === '\\') {
-             escaped = !escaped;
-           }
+                   var isOnlyVia = false;
+                   var v = r.membersByRole('via');
 
-           if (escaped) {
-             // odd number of slashes means | is escaped
-             // so we leave it alone
-             return '|';
-           } else {
-             // add space before unescaped |
-             return ' |';
-           }
-         }),
-             cells = row.split(/ \|/);
-         var i = 0;
+                   if (v.length === 1 && v[0].type === 'node') {
+                     // via node
+                     isOnlyVia = v[0].id === nextNode.id;
+                   } else {
+                     // via way(s)
+                     for (var i = 0; i < v.length; i++) {
+                       if (v[i].type !== 'way') continue;
+                       var viaWay = vgraph.entity(v[i].id);
 
-         if (cells.length > count) {
-           cells.splice(count);
-         } else {
-           while (cells.length < count) {
-             cells.push('');
-           }
-         }
+                       if (viaWay.first() === nextNode.id || viaWay.last() === nextNode.id) {
+                         isOnlyVia = true;
+                         break;
+                       }
+                     }
+                   }
 
-         for (; i < cells.length; i++) {
-           // leading or trailing whitespace is ignored per the gfm spec
-           cells[i] = cells[i].trim().replace(/\\\|/g, '|');
-         }
+                   return isOnlyVia;
+                 });
+                 step(nextNode, currPath, currRestrictions.concat(fromRestrictions), false);
+               });
+             }
+           } // assumes path is alternating way-node-way of odd length
 
-         return cells;
-       } // Remove trailing 'c's. Equivalent to str.replace(/c*$/, '').
-       // /c*$/ is vulnerable to REDOS.
-       // invert: Remove suffix of non-c chars instead. Default falsey.
 
+           function pathToTurn(path) {
+             if (path.length < 3) return;
+             var fromWayId, fromNodeId, fromVertexId;
+             var toWayId, toNodeId, toVertexId;
+             var viaWayIds, viaNodeId, isUturn;
+             fromWayId = path[0];
+             toWayId = path[path.length - 1];
 
-       function rtrim$1(str, c, invert) {
-         var l = str.length;
+             if (path.length === 3 && fromWayId === toWayId) {
+               // u turn
+               var way = vgraph.entity(fromWayId);
+               if (way.__oneWay) return null;
+               isUturn = true;
+               viaNodeId = fromVertexId = toVertexId = path[1];
+               fromNodeId = toNodeId = adjacentNode(fromWayId, viaNodeId);
+             } else {
+               isUturn = false;
+               fromVertexId = path[1];
+               fromNodeId = adjacentNode(fromWayId, fromVertexId);
+               toVertexId = path[path.length - 2];
+               toNodeId = adjacentNode(toWayId, toVertexId);
 
-         if (l === 0) {
-           return '';
-         } // Length of suffix matching the invert condition.
+               if (path.length === 3) {
+                 viaNodeId = path[1];
+               } else {
+                 viaWayIds = path.filter(function (entityId) {
+                   return entityId[0] === 'w';
+                 });
+                 viaWayIds = viaWayIds.slice(1, viaWayIds.length - 1); // remove first, last
+               }
+             }
 
+             return {
+               key: path.join('_'),
+               path: path,
+               from: {
+                 node: fromNodeId,
+                 way: fromWayId,
+                 vertex: fromVertexId
+               },
+               via: {
+                 node: viaNodeId,
+                 ways: viaWayIds
+               },
+               to: {
+                 node: toNodeId,
+                 way: toWayId,
+                 vertex: toVertexId
+               },
+               u: isUturn
+             };
 
-         var suffLen = 0; // Step left until we fail to match the invert condition.
+             function adjacentNode(wayId, affixId) {
+               var nodes = vgraph.entity(wayId).nodes;
+               return affixId === nodes[0] ? nodes[1] : nodes[nodes.length - 2];
+             }
+           }
+         };
 
-         while (suffLen < l) {
-           var currChar = str.charAt(l - suffLen - 1);
+         return intersection;
+       }
+       function osmInferRestriction(graph, turn, projection) {
+         var fromWay = graph.entity(turn.from.way);
+         var fromNode = graph.entity(turn.from.node);
+         var fromVertex = graph.entity(turn.from.vertex);
+         var toWay = graph.entity(turn.to.way);
+         var toNode = graph.entity(turn.to.node);
+         var toVertex = graph.entity(turn.to.vertex);
+         var fromOneWay = fromWay.tags.oneway === 'yes';
+         var toOneWay = toWay.tags.oneway === 'yes';
+         var angle = (geoAngle(fromVertex, fromNode, projection) - geoAngle(toVertex, toNode, projection)) * 180 / Math.PI;
 
-           if (currChar === c && !invert) {
-             suffLen++;
-           } else if (currChar !== c && invert) {
-             suffLen++;
-           } else {
-             break;
-           }
+         while (angle < 0) {
+           angle += 360;
          }
 
-         return str.substr(0, l - suffLen);
-       }
+         if (fromNode === toNode) {
+           return 'no_u_turn';
+         }
 
-       function findClosingBracket(str, b) {
-         if (str.indexOf(b[1]) === -1) {
-           return -1;
+         if ((angle < 23 || angle > 336) && fromOneWay && toOneWay) {
+           return 'no_u_turn'; // wider tolerance for u-turn if both ways are oneway
          }
 
-         var l = str.length;
-         var level = 0,
-             i = 0;
+         if ((angle < 40 || angle > 319) && fromOneWay && toOneWay && turn.from.vertex !== turn.to.vertex) {
+           return 'no_u_turn'; // even wider tolerance for u-turn if there is a via way (from !== to)
+         }
 
-         for (; i < l; i++) {
-           if (str[i] === '\\') {
-             i++;
-           } else if (str[i] === b[0]) {
-             level++;
-           } else if (str[i] === b[1]) {
-             level--;
+         if (angle < 158) {
+           return 'no_right_turn';
+         }
 
-             if (level < 0) {
-               return i;
-             }
-           }
+         if (angle > 202) {
+           return 'no_left_turn';
          }
 
-         return -1;
+         return 'no_straight_on';
        }
 
-       function checkSanitizeDeprecation(opt) {
-         if (opt && opt.sanitize && !opt.silent) {
-           console.warn('marked(): sanitize and sanitizer parameters are deprecated since version 0.7.0, should not be used and will be removed in the future. Read more here: https://marked.js.org/#/USING_ADVANCED.md#options');
+       function actionMergePolygon(ids, newRelationId) {
+         function groupEntities(graph) {
+           var entities = ids.map(function (id) {
+             return graph.entity(id);
+           });
+           var geometryGroups = utilArrayGroupBy(entities, function (entity) {
+             if (entity.type === 'way' && entity.isClosed()) {
+               return 'closedWay';
+             } else if (entity.type === 'relation' && entity.isMultipolygon()) {
+               return 'multipolygon';
+             } else {
+               return 'other';
+             }
+           });
+           return Object.assign({
+             closedWay: [],
+             multipolygon: [],
+             other: []
+           }, geometryGroups);
          }
-       } // copied from https://stackoverflow.com/a/5450113/806777
 
+         var action = function action(graph) {
+           var entities = groupEntities(graph); // An array representing all the polygons that are part of the multipolygon.
+           //
+           // Each element is itself an array of objects with an id property, and has a
+           // locs property which is an array of the locations forming the polygon.
 
-       function repeatString(pattern, count) {
-         if (count < 1) {
-           return '';
-         }
+           var polygons = entities.multipolygon.reduce(function (polygons, m) {
+             return polygons.concat(osmJoinWays(m.members, graph));
+           }, []).concat(entities.closedWay.map(function (d) {
+             var member = [{
+               id: d.id
+             }];
+             member.nodes = graph.childNodes(d);
+             return member;
+           })); // contained is an array of arrays of boolean values,
+           // where contained[j][k] is true iff the jth way is
+           // contained by the kth way.
 
-         var result = '';
+           var contained = polygons.map(function (w, i) {
+             return polygons.map(function (d, n) {
+               if (i === n) return null;
+               return geoPolygonContainsPolygon(d.nodes.map(function (n) {
+                 return n.loc;
+               }), w.nodes.map(function (n) {
+                 return n.loc;
+               }));
+             });
+           }); // Sort all polygons as either outer or inner ways
 
-         while (count > 1) {
-           if (count & 1) {
-             result += pattern;
-           }
+           var members = [];
+           var outer = true;
 
-           count >>= 1;
-           pattern += pattern;
-         }
+           while (polygons.length) {
+             extractUncontained(polygons);
+             polygons = polygons.filter(isContained);
+             contained = contained.filter(isContained).map(filterContained);
+           }
 
-         return result + pattern;
-       }
+           function isContained(d, i) {
+             return contained[i].some(function (val) {
+               return val;
+             });
+           }
 
-       var helpers = {
-         escape: escape$1,
-         unescape: unescape$1,
-         edit: edit,
-         cleanUrl: cleanUrl,
-         resolveUrl: resolveUrl,
-         noopTest: noopTest,
-         merge: merge$1,
-         splitCells: splitCells,
-         rtrim: rtrim$1,
-         findClosingBracket: findClosingBracket,
-         checkSanitizeDeprecation: checkSanitizeDeprecation,
-         repeatString: repeatString
-       };
+           function filterContained(d) {
+             return d.filter(isContained);
+           }
 
-       var defaults$1 = defaults.defaults;
-       var rtrim$2 = helpers.rtrim,
-           splitCells$1 = helpers.splitCells,
-           _escape = helpers.escape,
-           findClosingBracket$1 = helpers.findClosingBracket;
+           function extractUncontained(polygons) {
+             polygons.forEach(function (d, i) {
+               if (!isContained(d, i)) {
+                 d.forEach(function (member) {
+                   members.push({
+                     type: 'way',
+                     id: member.id,
+                     role: outer ? 'outer' : 'inner'
+                   });
+                 });
+               }
+             });
+             outer = !outer;
+           } // Move all tags to one relation
 
-       function outputLink(cap, link, raw) {
-         var href = link.href;
-         var title = link.title ? _escape(link.title) : null;
-         var text = cap[1].replace(/\\([\[\]])/g, '$1');
 
-         if (cap[0].charAt(0) !== '!') {
-           return {
-             type: 'link',
-             raw: raw,
-             href: href,
-             title: title,
-             text: text
-           };
-         } else {
-           return {
-             type: 'image',
-             raw: raw,
-             href: href,
-             title: title,
-             text: _escape(text)
-           };
-         }
-       }
+           var relation = entities.multipolygon[0] || osmRelation({
+             id: newRelationId,
+             tags: {
+               type: 'multipolygon'
+             }
+           });
+           entities.multipolygon.slice(1).forEach(function (m) {
+             relation = relation.mergeTags(m.tags);
+             graph = graph.remove(m);
+           });
+           entities.closedWay.forEach(function (way) {
+             function isThisOuter(m) {
+               return m.id === way.id && m.role !== 'inner';
+             }
 
-       function indentCodeCompensation(raw, text) {
-         var matchIndentToCode = raw.match(/^(\s+)(?:```)/);
+             if (members.some(isThisOuter)) {
+               relation = relation.mergeTags(way.tags);
+               graph = graph.replace(way.update({
+                 tags: {}
+               }));
+             }
+           });
+           return graph.replace(relation.update({
+             members: members,
+             tags: utilObjectOmit(relation.tags, ['area'])
+           }));
+         };
 
-         if (matchIndentToCode === null) {
-           return text;
-         }
+         action.disabled = function (graph) {
+           var entities = groupEntities(graph);
 
-         var indentToCode = matchIndentToCode[1];
-         return text.split('\n').map(function (node) {
-           var matchIndentInNode = node.match(/^\s+/);
+           if (entities.other.length > 0 || entities.closedWay.length + entities.multipolygon.length < 2) {
+             return 'not_eligible';
+           }
 
-           if (matchIndentInNode === null) {
-             return node;
+           if (!entities.multipolygon.every(function (r) {
+             return r.isComplete(graph);
+           })) {
+             return 'incomplete_relation';
            }
 
-           var _matchIndentInNode = _slicedToArray(matchIndentInNode, 1),
-               indentInNode = _matchIndentInNode[0];
+           if (!entities.multipolygon.length) {
+             var sharedMultipolygons = [];
+             entities.closedWay.forEach(function (way, i) {
+               if (i === 0) {
+                 sharedMultipolygons = graph.parentMultipolygons(way);
+               } else {
+                 sharedMultipolygons = utilArrayIntersection(sharedMultipolygons, graph.parentMultipolygons(way));
+               }
+             });
+             sharedMultipolygons = sharedMultipolygons.filter(function (relation) {
+               return relation.members.length === entities.closedWay.length;
+             });
 
-           if (indentInNode.length >= indentToCode.length) {
-             return node.slice(indentToCode.length);
+             if (sharedMultipolygons.length) {
+               // don't create a new multipolygon if it'd be redundant
+               return 'not_eligible';
+             }
+           } else if (entities.closedWay.some(function (way) {
+             return utilArrayIntersection(graph.parentMultipolygons(way), entities.multipolygon).length;
+           })) {
+             // don't add a way to a multipolygon again if it's already a member
+             return 'not_eligible';
            }
+         };
 
-           return node;
-         }).join('\n');
+         return action;
        }
-       /**
-        * Tokenizer
-        */
 
+       var FORCED$2 = descriptors && fails(function () {
+         // eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe
+         return Object.getOwnPropertyDescriptor(RegExp.prototype, 'flags').get.call({ dotAll: true, sticky: true }) !== 'sy';
+       });
 
-       var Tokenizer_1 = /*#__PURE__*/function () {
-         function Tokenizer(options) {
-           _classCallCheck(this, Tokenizer);
+       // `RegExp.prototype.flags` getter
+       // https://tc39.es/ecma262/#sec-get-regexp.prototype.flags
+       if (FORCED$2) objectDefineProperty.f(RegExp.prototype, 'flags', {
+         configurable: true,
+         get: regexpFlags
+       });
 
-           this.options = options || defaults$1;
-         }
+       var fastDeepEqual = function equal(a, b) {
+         if (a === b) return true;
 
-         _createClass(Tokenizer, [{
-           key: "space",
-           value: function space(src) {
-             var cap = this.rules.block.newline.exec(src);
+         if (a && b && _typeof(a) == 'object' && _typeof(b) == 'object') {
+           if (a.constructor !== b.constructor) return false;
+           var length, i, keys;
 
-             if (cap) {
-               if (cap[0].length > 1) {
-                 return {
-                   type: 'space',
-                   raw: cap[0]
-                 };
-               }
+           if (Array.isArray(a)) {
+             length = a.length;
+             if (length != b.length) return false;
 
-               return {
-                 raw: '\n'
-               };
+             for (i = length; i-- !== 0;) {
+               if (!equal(a[i], b[i])) return false;
              }
-           }
-         }, {
-           key: "code",
-           value: function code(src, tokens) {
-             var cap = this.rules.block.code.exec(src);
 
-             if (cap) {
-               var lastToken = tokens[tokens.length - 1]; // An indented code block cannot interrupt a paragraph.
+             return true;
+           }
 
-               if (lastToken && lastToken.type === 'paragraph') {
-                 return {
-                   raw: cap[0],
-                   text: cap[0].trimRight()
-                 };
-               }
+           if (a.constructor === RegExp) return a.source === b.source && a.flags === b.flags;
+           if (a.valueOf !== Object.prototype.valueOf) return a.valueOf() === b.valueOf();
+           if (a.toString !== Object.prototype.toString) return a.toString() === b.toString();
+           keys = Object.keys(a);
+           length = keys.length;
+           if (length !== Object.keys(b).length) return false;
 
-               var text = cap[0].replace(/^ {1,4}/gm, '');
-               return {
-                 type: 'code',
-                 raw: cap[0],
-                 codeBlockStyle: 'indented',
-                 text: !this.options.pedantic ? rtrim$2(text, '\n') : text
-               };
-             }
+           for (i = length; i-- !== 0;) {
+             if (!Object.prototype.hasOwnProperty.call(b, keys[i])) return false;
            }
-         }, {
-           key: "fences",
-           value: function fences(src) {
-             var cap = this.rules.block.fences.exec(src);
 
-             if (cap) {
-               var raw = cap[0];
-               var text = indentCodeCompensation(raw, cap[3] || '');
-               return {
-                 type: 'code',
-                 raw: raw,
-                 lang: cap[2] ? cap[2].trim() : cap[2],
-                 text: text
-               };
-             }
+           for (i = length; i-- !== 0;) {
+             var key = keys[i];
+             if (!equal(a[key], b[key])) return false;
            }
-         }, {
-           key: "heading",
-           value: function heading(src) {
-             var cap = this.rules.block.heading.exec(src);
 
-             if (cap) {
-               var text = cap[2].trim(); // remove trailing #s
+           return true;
+         } // true if both NaN, false otherwise
 
-               if (/#$/.test(text)) {
-                 var trimmed = rtrim$2(text, '#');
 
-                 if (this.options.pedantic) {
-                   text = trimmed.trim();
-                 } else if (!trimmed || / $/.test(trimmed)) {
-                   // CommonMark requires space before trailing #s
-                   text = trimmed.trim();
-                 }
-               }
+         return a !== a && b !== b;
+       };
 
-               return {
-                 type: 'heading',
-                 raw: cap[0],
-                 depth: cap[1].length,
-                 text: text
-               };
-             }
-           }
-         }, {
-           key: "nptable",
-           value: function nptable(src) {
-             var cap = this.rules.block.nptable.exec(src);
+       // J. W. Hunt and M. D. McIlroy, An algorithm for differential buffer
+       // comparison, Bell Telephone Laboratories CSTR #41 (1976)
+       // http://www.cs.dartmouth.edu/~doug/
+       // https://en.wikipedia.org/wiki/Longest_common_subsequence_problem
+       //
+       // Expects two arrays, finds longest common sequence
 
-             if (cap) {
-               var item = {
-                 type: 'table',
-                 header: splitCells$1(cap[1].replace(/^ *| *\| *$/g, '')),
-                 align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
-                 cells: cap[3] ? cap[3].replace(/\n$/, '').split('\n') : [],
-                 raw: cap[0]
-               };
+       function LCS(buffer1, buffer2) {
+         var equivalenceClasses = {};
 
-               if (item.header.length === item.align.length) {
-                 var l = item.align.length;
-                 var i;
+         for (var j = 0; j < buffer2.length; j++) {
+           var item = buffer2[j];
 
-                 for (i = 0; i < l; i++) {
-                   if (/^ *-+: *$/.test(item.align[i])) {
-                     item.align[i] = 'right';
-                   } else if (/^ *:-+: *$/.test(item.align[i])) {
-                     item.align[i] = 'center';
-                   } else if (/^ *:-+ *$/.test(item.align[i])) {
-                     item.align[i] = 'left';
-                   } else {
-                     item.align[i] = null;
-                   }
-                 }
+           if (equivalenceClasses[item]) {
+             equivalenceClasses[item].push(j);
+           } else {
+             equivalenceClasses[item] = [j];
+           }
+         }
 
-                 l = item.cells.length;
+         var NULLRESULT = {
+           buffer1index: -1,
+           buffer2index: -1,
+           chain: null
+         };
+         var candidates = [NULLRESULT];
 
-                 for (i = 0; i < l; i++) {
-                   item.cells[i] = splitCells$1(item.cells[i], item.header.length);
-                 }
+         for (var i = 0; i < buffer1.length; i++) {
+           var _item = buffer1[i];
+           var buffer2indices = equivalenceClasses[_item] || [];
+           var r = 0;
+           var c = candidates[0];
 
-                 return item;
+           for (var jx = 0; jx < buffer2indices.length; jx++) {
+             var _j = buffer2indices[jx];
+             var s = void 0;
+
+             for (s = r; s < candidates.length; s++) {
+               if (candidates[s].buffer2index < _j && (s === candidates.length - 1 || candidates[s + 1].buffer2index > _j)) {
+                 break;
                }
              }
-           }
-         }, {
-           key: "hr",
-           value: function hr(src) {
-             var cap = this.rules.block.hr.exec(src);
 
-             if (cap) {
-               return {
-                 type: 'hr',
-                 raw: cap[0]
+             if (s < candidates.length) {
+               var newCandidate = {
+                 buffer1index: i,
+                 buffer2index: _j,
+                 chain: candidates[s]
                };
-             }
-           }
-         }, {
-           key: "blockquote",
-           value: function blockquote(src) {
-             var cap = this.rules.block.blockquote.exec(src);
 
-             if (cap) {
-               var text = cap[0].replace(/^ *> ?/gm, '');
-               return {
-                 type: 'blockquote',
-                 raw: cap[0],
-                 text: text
-               };
+               if (r === candidates.length) {
+                 candidates.push(c);
+               } else {
+                 candidates[r] = c;
+               }
+
+               r = s + 1;
+               c = newCandidate;
+
+               if (r === candidates.length) {
+                 break; // no point in examining further (j)s
+               }
              }
            }
-         }, {
-           key: "list",
-           value: function list(src) {
-             var cap = this.rules.block.list.exec(src);
 
-             if (cap) {
-               var raw = cap[0];
-               var bull = cap[2];
-               var isordered = bull.length > 1;
-               var list = {
-                 type: 'list',
-                 raw: raw,
-                 ordered: isordered,
-                 start: isordered ? +bull.slice(0, -1) : '',
-                 loose: false,
-                 items: []
-               }; // Get each top-level item.
+           candidates[r] = c;
+         } // At this point, we know the LCS: it's in the reverse of the
+         // linked-list through .chain of candidates[candidates.length - 1].
 
-               var itemMatch = cap[0].match(this.rules.block.item);
-               var next = false,
-                   item,
-                   space,
-                   bcurr,
-                   bnext,
-                   addBack,
-                   loose,
-                   istask,
-                   ischecked;
-               var l = itemMatch.length;
-               bcurr = this.rules.block.listItemStart.exec(itemMatch[0]);
 
-               for (var i = 0; i < l; i++) {
-                 item = itemMatch[i];
-                 raw = item; // Determine whether the next list item belongs here.
-                 // Backpedal if it does not belong in this list.
+         return candidates[candidates.length - 1];
+       } // We apply the LCS to build a 'comm'-style picture of the
+       // offsets and lengths of mismatched chunks in the input
+       // buffers. This is used by diff3MergeRegions.
 
-                 if (i !== l - 1) {
-                   bnext = this.rules.block.listItemStart.exec(itemMatch[i + 1]);
 
-                   if (!this.options.pedantic ? bnext[1].length > bcurr[0].length || bnext[1].length > 3 : bnext[1].length > bcurr[1].length) {
-                     // nested list
-                     itemMatch.splice(i, 2, itemMatch[i] + '\n' + itemMatch[i + 1]);
-                     i--;
-                     l--;
-                     continue;
-                   } else {
-                     if ( // different bullet style
-                     !this.options.pedantic || this.options.smartLists ? bnext[2][bnext[2].length - 1] !== bull[bull.length - 1] : isordered === (bnext[2].length === 1)) {
-                       addBack = itemMatch.slice(i + 1).join('\n');
-                       list.raw = list.raw.substring(0, list.raw.length - addBack.length);
-                       i = l - 1;
-                     }
-                   }
+       function diffIndices(buffer1, buffer2) {
+         var lcs = LCS(buffer1, buffer2);
+         var result = [];
+         var tail1 = buffer1.length;
+         var tail2 = buffer2.length;
 
-                   bcurr = bnext;
-                 } // Remove the list item's bullet
-                 // so it is seen as the next token.
+         for (var candidate = lcs; candidate !== null; candidate = candidate.chain) {
+           var mismatchLength1 = tail1 - candidate.buffer1index - 1;
+           var mismatchLength2 = tail2 - candidate.buffer2index - 1;
+           tail1 = candidate.buffer1index;
+           tail2 = candidate.buffer2index;
 
+           if (mismatchLength1 || mismatchLength2) {
+             result.push({
+               buffer1: [tail1 + 1, mismatchLength1],
+               buffer1Content: buffer1.slice(tail1 + 1, tail1 + 1 + mismatchLength1),
+               buffer2: [tail2 + 1, mismatchLength2],
+               buffer2Content: buffer2.slice(tail2 + 1, tail2 + 1 + mismatchLength2)
+             });
+           }
+         }
 
-                 space = item.length;
-                 item = item.replace(/^ *([*+-]|\d+[.)]) ?/, ''); // Outdent whatever the
-                 // list item contains. Hacky.
+         result.reverse();
+         return result;
+       } // We apply the LCS to build a JSON representation of a
+       // independently derived from O, returns a fairly complicated
+       // internal representation of merge decisions it's taken. The
+       // interested reader may wish to consult
+       //
+       // Sanjeev Khanna, Keshav Kunal, and Benjamin C. Pierce.
+       // 'A Formal Investigation of ' In Arvind and Prasad,
+       // editors, Foundations of Software Technology and Theoretical
+       // Computer Science (FSTTCS), December 2007.
+       //
+       // (http://www.cis.upenn.edu/~bcpierce/papers/diff3-short.pdf)
+       //
 
-                 if (~item.indexOf('\n ')) {
-                   space -= item.length;
-                   item = !this.options.pedantic ? item.replace(new RegExp('^ {1,' + space + '}', 'gm'), '') : item.replace(/^ {1,4}/gm, '');
-                 } // Determine whether item is loose or not.
-                 // Use: /(^|\n)(?! )[^\n]+\n\n(?!\s*$)/
-                 // for discount behavior.
 
+       function diff3MergeRegions(a, o, b) {
+         // "hunks" are array subsets where `a` or `b` are different from `o`
+         // https://www.gnu.org/software/diffutils/manual/html_node/diff3-Hunks.html
+         var hunks = [];
 
-                 loose = next || /\n\n(?!\s*$)/.test(item);
+         function addHunk(h, ab) {
+           hunks.push({
+             ab: ab,
+             oStart: h.buffer1[0],
+             oLength: h.buffer1[1],
+             // length of o to remove
+             abStart: h.buffer2[0],
+             abLength: h.buffer2[1] // length of a/b to insert
+             // abContent: (ab === 'a' ? a : b).slice(h.buffer2[0], h.buffer2[0] + h.buffer2[1])
 
-                 if (i !== l - 1) {
-                   next = item.charAt(item.length - 1) === '\n';
-                   if (!loose) loose = next;
-                 }
+           });
+         }
 
-                 if (loose) {
-                   list.loose = true;
-                 } // Check for task list items
+         diffIndices(o, a).forEach(function (item) {
+           return addHunk(item, 'a');
+         });
+         diffIndices(o, b).forEach(function (item) {
+           return addHunk(item, 'b');
+         });
+         hunks.sort(function (x, y) {
+           return x.oStart - y.oStart;
+         });
+         var results = [];
+         var currOffset = 0;
 
+         function advanceTo(endOffset) {
+           if (endOffset > currOffset) {
+             results.push({
+               stable: true,
+               buffer: 'o',
+               bufferStart: currOffset,
+               bufferLength: endOffset - currOffset,
+               bufferContent: o.slice(currOffset, endOffset)
+             });
+             currOffset = endOffset;
+           }
+         }
 
-                 if (this.options.gfm) {
-                   istask = /^\[[ xX]\] /.test(item);
-                   ischecked = undefined;
+         while (hunks.length) {
+           var hunk = hunks.shift();
+           var regionStart = hunk.oStart;
+           var regionEnd = hunk.oStart + hunk.oLength;
+           var regionHunks = [hunk];
+           advanceTo(regionStart); // Try to pull next overlapping hunk into this region
 
-                   if (istask) {
-                     ischecked = item[1] !== ' ';
-                     item = item.replace(/^\[[ xX]\] +/, '');
-                   }
-                 }
+           while (hunks.length) {
+             var nextHunk = hunks[0];
+             var nextHunkStart = nextHunk.oStart;
+             if (nextHunkStart > regionEnd) break; // no overlap
 
-                 list.items.push({
-                   type: 'list_item',
-                   raw: raw,
-                   task: istask,
-                   checked: ischecked,
-                   loose: loose,
-                   text: item
-                 });
-               }
+             regionEnd = Math.max(regionEnd, nextHunkStart + nextHunk.oLength);
+             regionHunks.push(hunks.shift());
+           }
 
-               return list;
+           if (regionHunks.length === 1) {
+             // Only one hunk touches this region, meaning that there is no conflict here.
+             // Either `a` or `b` is inserting into a region of `o` unchanged by the other.
+             if (hunk.abLength > 0) {
+               var buffer = hunk.ab === 'a' ? a : b;
+               results.push({
+                 stable: true,
+                 buffer: hunk.ab,
+                 bufferStart: hunk.abStart,
+                 bufferLength: hunk.abLength,
+                 bufferContent: buffer.slice(hunk.abStart, hunk.abStart + hunk.abLength)
+               });
              }
-           }
-         }, {
-           key: "html",
-           value: function html(src) {
-             var cap = this.rules.block.html.exec(src);
+           } else {
+             // A true a/b conflict. Determine the bounds involved from `a`, `o`, and `b`.
+             // Effectively merge all the `a` hunks into one giant hunk, then do the
+             // same for the `b` hunks; then, correct for skew in the regions of `o`
+             // that each side changed, and report appropriate spans for the three sides.
+             var bounds = {
+               a: [a.length, -1, o.length, -1],
+               b: [b.length, -1, o.length, -1]
+             };
 
-             if (cap) {
-               return {
-                 type: this.options.sanitize ? 'paragraph' : 'html',
-                 raw: cap[0],
-                 pre: !this.options.sanitizer && (cap[1] === 'pre' || cap[1] === 'script' || cap[1] === 'style'),
-                 text: this.options.sanitize ? this.options.sanitizer ? this.options.sanitizer(cap[0]) : _escape(cap[0]) : cap[0]
-               };
+             while (regionHunks.length) {
+               hunk = regionHunks.shift();
+               var oStart = hunk.oStart;
+               var oEnd = oStart + hunk.oLength;
+               var abStart = hunk.abStart;
+               var abEnd = abStart + hunk.abLength;
+               var _b = bounds[hunk.ab];
+               _b[0] = Math.min(abStart, _b[0]);
+               _b[1] = Math.max(abEnd, _b[1]);
+               _b[2] = Math.min(oStart, _b[2]);
+               _b[3] = Math.max(oEnd, _b[3]);
              }
+
+             var aStart = bounds.a[0] + (regionStart - bounds.a[2]);
+             var aEnd = bounds.a[1] + (regionEnd - bounds.a[3]);
+             var bStart = bounds.b[0] + (regionStart - bounds.b[2]);
+             var bEnd = bounds.b[1] + (regionEnd - bounds.b[3]);
+             var result = {
+               stable: false,
+               aStart: aStart,
+               aLength: aEnd - aStart,
+               aContent: a.slice(aStart, aEnd),
+               oStart: regionStart,
+               oLength: regionEnd - regionStart,
+               oContent: o.slice(regionStart, regionEnd),
+               bStart: bStart,
+               bLength: bEnd - bStart,
+               bContent: b.slice(bStart, bEnd)
+             };
+             results.push(result);
            }
-         }, {
-           key: "def",
-           value: function def(src) {
-             var cap = this.rules.block.def.exec(src);
 
-             if (cap) {
-               if (cap[3]) cap[3] = cap[3].substring(1, cap[3].length - 1);
-               var tag = cap[1].toLowerCase().replace(/\s+/g, ' ');
-               return {
-                 tag: tag,
-                 raw: cap[0],
-                 href: cap[2],
-                 title: cap[3]
-               };
-             }
+           currOffset = regionEnd;
+         }
+
+         advanceTo(o.length);
+         return results;
+       } // Applies the output of diff3MergeRegions to actually
+       // construct the merged buffer; the returned result alternates
+       // between 'ok' and 'conflict' blocks.
+       // A "false conflict" is where `a` and `b` both change the same from `o`
+
+
+       function diff3Merge(a, o, b, options) {
+         var defaults = {
+           excludeFalseConflicts: true,
+           stringSeparator: /\s+/
+         };
+         options = Object.assign(defaults, options);
+         var aString = typeof a === 'string';
+         var oString = typeof o === 'string';
+         var bString = typeof b === 'string';
+         if (aString) a = a.split(options.stringSeparator);
+         if (oString) o = o.split(options.stringSeparator);
+         if (bString) b = b.split(options.stringSeparator);
+         var results = [];
+         var regions = diff3MergeRegions(a, o, b);
+         var okBuffer = [];
+
+         function flushOk() {
+           if (okBuffer.length) {
+             results.push({
+               ok: okBuffer
+             });
            }
-         }, {
-           key: "table",
-           value: function table(src) {
-             var cap = this.rules.block.table.exec(src);
 
-             if (cap) {
-               var item = {
-                 type: 'table',
-                 header: splitCells$1(cap[1].replace(/^ *| *\| *$/g, '')),
-                 align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
-                 cells: cap[3] ? cap[3].replace(/\n$/, '').split('\n') : []
-               };
+           okBuffer = [];
+         }
 
-               if (item.header.length === item.align.length) {
-                 item.raw = cap[0];
-                 var l = item.align.length;
-                 var i;
+         function isFalseConflict(a, b) {
+           if (a.length !== b.length) return false;
 
-                 for (i = 0; i < l; i++) {
-                   if (/^ *-+: *$/.test(item.align[i])) {
-                     item.align[i] = 'right';
-                   } else if (/^ *:-+: *$/.test(item.align[i])) {
-                     item.align[i] = 'center';
-                   } else if (/^ *:-+ *$/.test(item.align[i])) {
-                     item.align[i] = 'left';
-                   } else {
-                     item.align[i] = null;
-                   }
-                 }
+           for (var i = 0; i < a.length; i++) {
+             if (a[i] !== b[i]) return false;
+           }
 
-                 l = item.cells.length;
+           return true;
+         }
 
-                 for (i = 0; i < l; i++) {
-                   item.cells[i] = splitCells$1(item.cells[i].replace(/^ *\| *| *\| *$/g, ''), item.header.length);
-                 }
+         regions.forEach(function (region) {
+           if (region.stable) {
+             var _okBuffer;
 
-                 return item;
-               }
-             }
-           }
-         }, {
-           key: "lheading",
-           value: function lheading(src) {
-             var cap = this.rules.block.lheading.exec(src);
+             (_okBuffer = okBuffer).push.apply(_okBuffer, _toConsumableArray(region.bufferContent));
+           } else {
+             if (options.excludeFalseConflicts && isFalseConflict(region.aContent, region.bContent)) {
+               var _okBuffer2;
 
-             if (cap) {
-               return {
-                 type: 'heading',
-                 raw: cap[0],
-                 depth: cap[2].charAt(0) === '=' ? 1 : 2,
-                 text: cap[1]
-               };
+               (_okBuffer2 = okBuffer).push.apply(_okBuffer2, _toConsumableArray(region.aContent));
+             } else {
+               flushOk();
+               results.push({
+                 conflict: {
+                   a: region.aContent,
+                   aIndex: region.aStart,
+                   o: region.oContent,
+                   oIndex: region.oStart,
+                   b: region.bContent,
+                   bIndex: region.bStart
+                 }
+               });
              }
            }
-         }, {
-           key: "paragraph",
-           value: function paragraph(src) {
-             var cap = this.rules.block.paragraph.exec(src);
+         });
+         flushOk();
+         return results;
+       }
 
-             if (cap) {
-               return {
-                 type: 'paragraph',
-                 raw: cap[0],
-                 text: cap[1].charAt(cap[1].length - 1) === '\n' ? cap[1].slice(0, -1) : cap[1]
-               };
-             }
-           }
-         }, {
-           key: "text",
-           value: function text(src, tokens) {
-             var cap = this.rules.block.text.exec(src);
+       function actionMergeRemoteChanges(id, localGraph, remoteGraph, discardTags, formatUser) {
+         discardTags = discardTags || {};
+         var _option = 'safe'; // 'safe', 'force_local', 'force_remote'
 
-             if (cap) {
-               var lastToken = tokens[tokens.length - 1];
+         var _conflicts = [];
 
-               if (lastToken && lastToken.type === 'text') {
-                 return {
-                   raw: cap[0],
-                   text: cap[0]
-                 };
-               }
+         function user(d) {
+           return typeof formatUser === 'function' ? formatUser(d) : d;
+         }
 
-               return {
-                 type: 'text',
-                 raw: cap[0],
-                 text: cap[0]
-               };
-             }
+         function mergeLocation(remote, target) {
+           function pointEqual(a, b) {
+             var epsilon = 1e-6;
+             return Math.abs(a[0] - b[0]) < epsilon && Math.abs(a[1] - b[1]) < epsilon;
            }
-         }, {
-           key: "escape",
-           value: function escape(src) {
-             var cap = this.rules.inline.escape.exec(src);
 
-             if (cap) {
-               return {
-                 type: 'escape',
-                 raw: cap[0],
-                 text: _escape(cap[1])
-               };
-             }
+           if (_option === 'force_local' || pointEqual(target.loc, remote.loc)) {
+             return target;
            }
-         }, {
-           key: "tag",
-           value: function tag(src, inLink, inRawBlock) {
-             var cap = this.rules.inline.tag.exec(src);
-
-             if (cap) {
-               if (!inLink && /^<a /i.test(cap[0])) {
-                 inLink = true;
-               } else if (inLink && /^<\/a>/i.test(cap[0])) {
-                 inLink = false;
-               }
-
-               if (!inRawBlock && /^<(pre|code|kbd|script)(\s|>)/i.test(cap[0])) {
-                 inRawBlock = true;
-               } else if (inRawBlock && /^<\/(pre|code|kbd|script)(\s|>)/i.test(cap[0])) {
-                 inRawBlock = false;
-               }
 
-               return {
-                 type: this.options.sanitize ? 'text' : 'html',
-                 raw: cap[0],
-                 inLink: inLink,
-                 inRawBlock: inRawBlock,
-                 text: this.options.sanitize ? this.options.sanitizer ? this.options.sanitizer(cap[0]) : _escape(cap[0]) : cap[0]
-               };
-             }
+           if (_option === 'force_remote') {
+             return target.update({
+               loc: remote.loc
+             });
            }
-         }, {
-           key: "link",
-           value: function link(src) {
-             var cap = this.rules.inline.link.exec(src);
-
-             if (cap) {
-               var trimmedUrl = cap[2].trim();
 
-               if (!this.options.pedantic && /^</.test(trimmedUrl)) {
-                 // commonmark requires matching angle brackets
-                 if (!/>$/.test(trimmedUrl)) {
-                   return;
-                 } // ending angle bracket cannot be escaped
+           _conflicts.push(_t('merge_remote_changes.conflict.location', {
+             user: user(remote.user)
+           }));
 
+           return target;
+         }
 
-                 var rtrimSlash = rtrim$2(trimmedUrl.slice(0, -1), '\\');
+         function mergeNodes(base, remote, target) {
+           if (_option === 'force_local' || fastDeepEqual(target.nodes, remote.nodes)) {
+             return target;
+           }
 
-                 if ((trimmedUrl.length - rtrimSlash.length) % 2 === 0) {
-                   return;
-                 }
-               } else {
-                 // find closing parenthesis
-                 var lastParenIndex = findClosingBracket$1(cap[2], '()');
+           if (_option === 'force_remote') {
+             return target.update({
+               nodes: remote.nodes
+             });
+           }
 
-                 if (lastParenIndex > -1) {
-                   var start = cap[0].indexOf('!') === 0 ? 5 : 4;
-                   var linkLen = start + cap[1].length + lastParenIndex;
-                   cap[2] = cap[2].substring(0, lastParenIndex);
-                   cap[0] = cap[0].substring(0, linkLen).trim();
-                   cap[3] = '';
-                 }
-               }
+           var ccount = _conflicts.length;
+           var o = base.nodes || [];
+           var a = target.nodes || [];
+           var b = remote.nodes || [];
+           var nodes = [];
+           var hunks = diff3Merge(a, o, b, {
+             excludeFalseConflicts: true
+           });
 
-               var href = cap[2];
-               var title = '';
+           for (var i = 0; i < hunks.length; i++) {
+             var hunk = hunks[i];
 
-               if (this.options.pedantic) {
-                 // split pedantic href and title
-                 var link = /^([^'"]*[^\s])\s+(['"])(.*)\2/.exec(href);
+             if (hunk.ok) {
+               nodes.push.apply(nodes, hunk.ok);
+             } else {
+               // for all conflicts, we can assume c.a !== c.b
+               // because `diff3Merge` called with `true` option to exclude false conflicts..
+               var c = hunk.conflict;
 
-                 if (link) {
-                   href = link[1];
-                   title = link[3];
-                 }
+               if (fastDeepEqual(c.o, c.a)) {
+                 // only changed remotely
+                 nodes.push.apply(nodes, c.b);
+               } else if (fastDeepEqual(c.o, c.b)) {
+                 // only changed locally
+                 nodes.push.apply(nodes, c.a);
                } else {
-                 title = cap[3] ? cap[3].slice(1, -1) : '';
-               }
-
-               href = href.trim();
+                 // changed both locally and remotely
+                 _conflicts.push(_t('merge_remote_changes.conflict.nodelist', {
+                   user: user(remote.user)
+                 }));
 
-               if (/^</.test(href)) {
-                 if (this.options.pedantic && !/>$/.test(trimmedUrl)) {
-                   // pedantic allows starting angle bracket without ending angle bracket
-                   href = href.slice(1);
-                 } else {
-                   href = href.slice(1, -1);
-                 }
+                 break;
                }
-
-               return outputLink(cap, {
-                 href: href ? href.replace(this.rules.inline._escapes, '$1') : href,
-                 title: title ? title.replace(this.rules.inline._escapes, '$1') : title
-               }, cap[0]);
              }
            }
-         }, {
-           key: "reflink",
-           value: function reflink(src, links) {
-             var cap;
-
-             if ((cap = this.rules.inline.reflink.exec(src)) || (cap = this.rules.inline.nolink.exec(src))) {
-               var link = (cap[2] || cap[1]).replace(/\s+/g, ' ');
-               link = links[link.toLowerCase()];
 
-               if (!link || !link.href) {
-                 var text = cap[0].charAt(0);
-                 return {
-                   type: 'text',
-                   raw: text,
-                   text: text
-                 };
-               }
+           return _conflicts.length === ccount ? target.update({
+             nodes: nodes
+           }) : target;
+         }
 
-               return outputLink(cap, link, cap[0]);
-             }
+         function mergeChildren(targetWay, children, updates, graph) {
+           function isUsed(node, targetWay) {
+             var hasInterestingParent = graph.parentWays(node).some(function (way) {
+               return way.id !== targetWay.id;
+             });
+             return node.hasInterestingTags() || hasInterestingParent || graph.parentRelations(node).length > 0;
            }
-         }, {
-           key: "strong",
-           value: function strong(src, maskedSrc) {
-             var prevChar = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : '';
-             var match = this.rules.inline.strong.start.exec(src);
 
-             if (match && (!match[1] || match[1] && (prevChar === '' || this.rules.inline.punctuation.exec(prevChar)))) {
-               maskedSrc = maskedSrc.slice(-1 * src.length);
-               var endReg = match[0] === '**' ? this.rules.inline.strong.endAst : this.rules.inline.strong.endUnd;
-               endReg.lastIndex = 0;
-               var cap;
+           var ccount = _conflicts.length;
 
-               while ((match = endReg.exec(maskedSrc)) != null) {
-                 cap = this.rules.inline.strong.middle.exec(maskedSrc.slice(0, match.index + 3));
+           for (var i = 0; i < children.length; i++) {
+             var id = children[i];
+             var node = graph.hasEntity(id); // remove unused childNodes..
 
-                 if (cap) {
-                   return {
-                     type: 'strong',
-                     raw: src.slice(0, cap[0].length),
-                     text: src.slice(2, cap[0].length - 2)
-                   };
-                 }
+             if (targetWay.nodes.indexOf(id) === -1) {
+               if (node && !isUsed(node, targetWay)) {
+                 updates.removeIds.push(id);
                }
-             }
-           }
-         }, {
-           key: "em",
-           value: function em(src, maskedSrc) {
-             var prevChar = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : '';
-             var match = this.rules.inline.em.start.exec(src);
 
-             if (match && (!match[1] || match[1] && (prevChar === '' || this.rules.inline.punctuation.exec(prevChar)))) {
-               maskedSrc = maskedSrc.slice(-1 * src.length);
-               var endReg = match[0] === '*' ? this.rules.inline.em.endAst : this.rules.inline.em.endUnd;
-               endReg.lastIndex = 0;
-               var cap;
+               continue;
+             } // restore used childNodes..
 
-               while ((match = endReg.exec(maskedSrc)) != null) {
-                 cap = this.rules.inline.em.middle.exec(maskedSrc.slice(0, match.index + 2));
 
-                 if (cap) {
-                   return {
-                     type: 'em',
-                     raw: src.slice(0, cap[0].length),
-                     text: src.slice(1, cap[0].length - 1)
-                   };
-                 }
+             var local = localGraph.hasEntity(id);
+             var remote = remoteGraph.hasEntity(id);
+             var target;
+
+             if (_option === 'force_remote' && remote && remote.visible) {
+               updates.replacements.push(remote);
+             } else if (_option === 'force_local' && local) {
+               target = osmEntity(local);
+
+               if (remote) {
+                 target = target.update({
+                   version: remote.version
+                 });
                }
-             }
-           }
-         }, {
-           key: "codespan",
-           value: function codespan(src) {
-             var cap = this.rules.inline.code.exec(src);
 
-             if (cap) {
-               var text = cap[2].replace(/\n/g, ' ');
-               var hasNonSpaceChars = /[^ ]/.test(text);
-               var hasSpaceCharsOnBothEnds = /^ /.test(text) && / $/.test(text);
+               updates.replacements.push(target);
+             } else if (_option === 'safe' && local && remote && local.version !== remote.version) {
+               target = osmEntity(local, {
+                 version: remote.version
+               });
 
-               if (hasNonSpaceChars && hasSpaceCharsOnBothEnds) {
-                 text = text.substring(1, text.length - 1);
+               if (remote.visible) {
+                 target = mergeLocation(remote, target);
+               } else {
+                 _conflicts.push(_t('merge_remote_changes.conflict.deleted', {
+                   user: user(remote.user)
+                 }));
                }
 
-               text = _escape(text, true);
-               return {
-                 type: 'codespan',
-                 raw: cap[0],
-                 text: text
-               };
+               if (_conflicts.length !== ccount) break;
+               updates.replacements.push(target);
              }
            }
-         }, {
-           key: "br",
-           value: function br(src) {
-             var cap = this.rules.inline.br.exec(src);
 
-             if (cap) {
-               return {
-                 type: 'br',
-                 raw: cap[0]
-               };
-             }
+           return targetWay;
+         }
+
+         function updateChildren(updates, graph) {
+           for (var i = 0; i < updates.replacements.length; i++) {
+             graph = graph.replace(updates.replacements[i]);
            }
-         }, {
-           key: "del",
-           value: function del(src) {
-             var cap = this.rules.inline.del.exec(src);
 
-             if (cap) {
-               return {
-                 type: 'del',
-                 raw: cap[0],
-                 text: cap[2]
-               };
-             }
+           if (updates.removeIds.length) {
+             graph = actionDeleteMultiple(updates.removeIds)(graph);
            }
-         }, {
-           key: "autolink",
-           value: function autolink(src, mangle) {
-             var cap = this.rules.inline.autolink.exec(src);
 
-             if (cap) {
-               var text, href;
+           return graph;
+         }
 
-               if (cap[2] === '@') {
-                 text = _escape(this.options.mangle ? mangle(cap[1]) : cap[1]);
-                 href = 'mailto:' + text;
-               } else {
-                 text = _escape(cap[1]);
-                 href = text;
-               }
+         function mergeMembers(remote, target) {
+           if (_option === 'force_local' || fastDeepEqual(target.members, remote.members)) {
+             return target;
+           }
 
-               return {
-                 type: 'link',
-                 raw: cap[0],
-                 text: text,
-                 href: href,
-                 tokens: [{
-                   type: 'text',
-                   raw: text,
-                   text: text
-                 }]
-               };
-             }
+           if (_option === 'force_remote') {
+             return target.update({
+               members: remote.members
+             });
            }
-         }, {
-           key: "url",
-           value: function url(src, mangle) {
-             var cap;
 
-             if (cap = this.rules.inline.url.exec(src)) {
-               var text, href;
+           _conflicts.push(_t('merge_remote_changes.conflict.memberlist', {
+             user: user(remote.user)
+           }));
 
-               if (cap[2] === '@') {
-                 text = _escape(this.options.mangle ? mangle(cap[0]) : cap[0]);
-                 href = 'mailto:' + text;
-               } else {
-                 // do extended autolink path validation
-                 var prevCapZero;
+           return target;
+         }
 
-                 do {
-                   prevCapZero = cap[0];
-                   cap[0] = this.rules.inline._backpedal.exec(cap[0])[0];
-                 } while (prevCapZero !== cap[0]);
+         function mergeTags(base, remote, target) {
+           if (_option === 'force_local' || fastDeepEqual(target.tags, remote.tags)) {
+             return target;
+           }
 
-                 text = _escape(cap[0]);
+           if (_option === 'force_remote') {
+             return target.update({
+               tags: remote.tags
+             });
+           }
 
-                 if (cap[1] === 'www.') {
-                   href = 'http://' + text;
-                 } else {
-                   href = text;
-                 }
-               }
+           var ccount = _conflicts.length;
+           var o = base.tags || {};
+           var a = target.tags || {};
+           var b = remote.tags || {};
+           var keys = utilArrayUnion(utilArrayUnion(Object.keys(o), Object.keys(a)), Object.keys(b)).filter(function (k) {
+             return !discardTags[k];
+           });
+           var tags = Object.assign({}, a); // shallow copy
 
-               return {
-                 type: 'link',
-                 raw: cap[0],
-                 text: text,
-                 href: href,
-                 tokens: [{
-                   type: 'text',
-                   raw: text,
-                   text: text
-                 }]
-               };
-             }
-           }
-         }, {
-           key: "inlineText",
-           value: function inlineText(src, inRawBlock, smartypants) {
-             var cap = this.rules.inline.text.exec(src);
+           var changed = false;
 
-             if (cap) {
-               var text;
+           for (var i = 0; i < keys.length; i++) {
+             var k = keys[i];
 
-               if (inRawBlock) {
-                 text = this.options.sanitize ? this.options.sanitizer ? this.options.sanitizer(cap[0]) : _escape(cap[0]) : cap[0];
+             if (o[k] !== b[k] && a[k] !== b[k]) {
+               // changed remotely..
+               if (o[k] !== a[k]) {
+                 // changed locally..
+                 _conflicts.push(_t('merge_remote_changes.conflict.tags', {
+                   tag: k,
+                   local: a[k],
+                   remote: b[k],
+                   user: user(remote.user)
+                 }));
                } else {
-                 text = _escape(this.options.smartypants ? smartypants(cap[0]) : cap[0]);
+                 // unchanged locally, accept remote change..
+                 if (b.hasOwnProperty(k)) {
+                   tags[k] = b[k];
+                 } else {
+                   delete tags[k];
+                 }
+
+                 changed = true;
                }
-
-               return {
-                 type: 'text',
-                 raw: cap[0],
-                 text: text
-               };
              }
            }
-         }]);
-
-         return Tokenizer;
-       }();
-
-       var noopTest$1 = helpers.noopTest,
-           edit$1 = helpers.edit,
-           merge$2 = helpers.merge;
-       /**
-        * Block-Level Grammar
-        */
 
-       var block = {
-         newline: /^(?: *(?:\n|$))+/,
-         code: /^( {4}[^\n]+(?:\n(?: *(?:\n|$))*)?)+/,
-         fences: /^ {0,3}(`{3,}(?=[^`\n]*\n)|~{3,})([^\n]*)\n(?:|([\s\S]*?)\n)(?: {0,3}\1[~`]* *(?:\n+|$)|$)/,
-         hr: /^ {0,3}((?:- *){3,}|(?:_ *){3,}|(?:\* *){3,})(?:\n+|$)/,
-         heading: /^ {0,3}(#{1,6})(?=\s|$)(.*)(?:\n+|$)/,
-         blockquote: /^( {0,3}> ?(paragraph|[^\n]*)(?:\n|$))+/,
-         list: /^( {0,3})(bull) [\s\S]+?(?:hr|def|\n{2,}(?! )(?! {0,3}bull )\n*|\s*$)/,
-         html: '^ {0,3}(?:' // optional indentation
-         + '<(script|pre|style)[\\s>][\\s\\S]*?(?:</\\1>[^\\n]*\\n+|$)' // (1)
-         + '|comment[^\\n]*(\\n+|$)' // (2)
-         + '|<\\?[\\s\\S]*?(?:\\?>\\n*|$)' // (3)
-         + '|<![A-Z][\\s\\S]*?(?:>\\n*|$)' // (4)
-         + '|<!\\[CDATA\\[[\\s\\S]*?(?:\\]\\]>\\n*|$)' // (5)
-         + '|</?(tag)(?: +|\\n|/?>)[\\s\\S]*?(?:\\n{2,}|$)' // (6)
-         + '|<(?!script|pre|style)([a-z][\\w-]*)(?:attribute)*? */?>(?=[ \\t]*(?:\\n|$))[\\s\\S]*?(?:\\n{2,}|$)' // (7) open tag
-         + '|</(?!script|pre|style)[a-z][\\w-]*\\s*>(?=[ \\t]*(?:\\n|$))[\\s\\S]*?(?:\\n{2,}|$)' // (7) closing tag
-         + ')',
-         def: /^ {0,3}\[(label)\]: *\n? *<?([^\s>]+)>?(?:(?: +\n? *| *\n *)(title))? *(?:\n+|$)/,
-         nptable: noopTest$1,
-         table: noopTest$1,
-         lheading: /^([^\n]+)\n {0,3}(=+|-+) *(?:\n+|$)/,
-         // regex template, placeholders will be replaced according to different paragraph
-         // interruption rules of commonmark and the original markdown spec:
-         _paragraph: /^([^\n]+(?:\n(?!hr|heading|lheading|blockquote|fences|list|html| +\n)[^\n]+)*)/,
-         text: /^[^\n]+/
-       };
-       block._label = /(?!\s*\])(?:\\[\[\]]|[^\[\]])+/;
-       block._title = /(?:"(?:\\"?|[^"\\])*"|'[^'\n]*(?:\n[^'\n]+)*\n?'|\([^()]*\))/;
-       block.def = edit$1(block.def).replace('label', block._label).replace('title', block._title).getRegex();
-       block.bullet = /(?:[*+-]|\d{1,9}[.)])/;
-       block.item = /^( *)(bull) ?[^\n]*(?:\n(?! *bull ?)[^\n]*)*/;
-       block.item = edit$1(block.item, 'gm').replace(/bull/g, block.bullet).getRegex();
-       block.listItemStart = edit$1(/^( *)(bull)/).replace('bull', block.bullet).getRegex();
-       block.list = edit$1(block.list).replace(/bull/g, block.bullet).replace('hr', '\\n+(?=\\1?(?:(?:- *){3,}|(?:_ *){3,}|(?:\\* *){3,})(?:\\n+|$))').replace('def', '\\n+(?=' + block.def.source + ')').getRegex();
-       block._tag = 'address|article|aside|base|basefont|blockquote|body|caption' + '|center|col|colgroup|dd|details|dialog|dir|div|dl|dt|fieldset|figcaption' + '|figure|footer|form|frame|frameset|h[1-6]|head|header|hr|html|iframe' + '|legend|li|link|main|menu|menuitem|meta|nav|noframes|ol|optgroup|option' + '|p|param|section|source|summary|table|tbody|td|tfoot|th|thead|title|tr' + '|track|ul';
-       block._comment = /<!--(?!-?>)[\s\S]*?(?:-->|$)/;
-       block.html = edit$1(block.html, 'i').replace('comment', block._comment).replace('tag', block._tag).replace('attribute', / +[a-zA-Z:_][\w.:-]*(?: *= *"[^"\n]*"| *= *'[^'\n]*'| *= *[^\s"'=<>`]+)?/).getRegex();
-       block.paragraph = edit$1(block._paragraph).replace('hr', block.hr).replace('heading', ' {0,3}#{1,6} ').replace('|lheading', '') // setex headings don't interrupt commonmark paragraphs
-       .replace('blockquote', ' {0,3}>').replace('fences', ' {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n').replace('list', ' {0,3}(?:[*+-]|1[.)]) ') // only lists starting from 1 can interrupt
-       .replace('html', '</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|!--)').replace('tag', block._tag) // pars can be interrupted by type (6) html blocks
-       .getRegex();
-       block.blockquote = edit$1(block.blockquote).replace('paragraph', block.paragraph).getRegex();
-       /**
-        * Normal Block Grammar
-        */
+           return changed && _conflicts.length === ccount ? target.update({
+             tags: tags
+           }) : target;
+         } //  `graph.base()` is the common ancestor of the two graphs.
+         //  `localGraph` contains user's edits up to saving
+         //  `remoteGraph` contains remote edits to modified nodes
+         //  `graph` must be a descendent of `localGraph` and may include
+         //      some conflict resolution actions performed on it.
+         //
+         //                  --- ... --- `localGraph` -- ... -- `graph`
+         //                 /
+         //  `graph.base()` --- ... --- `remoteGraph`
+         //
 
-       block.normal = merge$2({}, block);
-       /**
-        * GFM Block Grammar
-        */
 
-       block.gfm = merge$2({}, block.normal, {
-         nptable: '^ *([^|\\n ].*\\|.*)\\n' // Header
-         + ' {0,3}([-:]+ *\\|[-| :]*)' // Align
-         + '(?:\\n((?:(?!\\n|hr|heading|blockquote|code|fences|list|html).*(?:\\n|$))*)\\n*|$)',
-         // Cells
-         table: '^ *\\|(.+)\\n' // Header
-         + ' {0,3}\\|?( *[-:]+[-| :]*)' // Align
-         + '(?:\\n *((?:(?!\\n|hr|heading|blockquote|code|fences|list|html).*(?:\\n|$))*)\\n*|$)' // Cells
+         var action = function action(graph) {
+           var updates = {
+             replacements: [],
+             removeIds: []
+           };
+           var base = graph.base().entities[id];
+           var local = localGraph.entity(id);
+           var remote = remoteGraph.entity(id);
+           var target = osmEntity(local, {
+             version: remote.version
+           }); // delete/undelete
 
-       });
-       block.gfm.nptable = edit$1(block.gfm.nptable).replace('hr', block.hr).replace('heading', ' {0,3}#{1,6} ').replace('blockquote', ' {0,3}>').replace('code', ' {4}[^\\n]').replace('fences', ' {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n').replace('list', ' {0,3}(?:[*+-]|1[.)]) ') // only lists starting from 1 can interrupt
-       .replace('html', '</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|!--)').replace('tag', block._tag) // tables can be interrupted by type (6) html blocks
-       .getRegex();
-       block.gfm.table = edit$1(block.gfm.table).replace('hr', block.hr).replace('heading', ' {0,3}#{1,6} ').replace('blockquote', ' {0,3}>').replace('code', ' {4}[^\\n]').replace('fences', ' {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n').replace('list', ' {0,3}(?:[*+-]|1[.)]) ') // only lists starting from 1 can interrupt
-       .replace('html', '</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|!--)').replace('tag', block._tag) // tables can be interrupted by type (6) html blocks
-       .getRegex();
-       /**
-        * Pedantic grammar (original John Gruber's loose markdown specification)
-        */
+           if (!remote.visible) {
+             if (_option === 'force_remote') {
+               return actionDeleteMultiple([id])(graph);
+             } else if (_option === 'force_local') {
+               if (target.type === 'way') {
+                 target = mergeChildren(target, utilArrayUniq(local.nodes), updates, graph);
+                 graph = updateChildren(updates, graph);
+               }
 
-       block.pedantic = merge$2({}, block.normal, {
-         html: edit$1('^ *(?:comment *(?:\\n|\\s*$)' + '|<(tag)[\\s\\S]+?</\\1> *(?:\\n{2,}|\\s*$)' // closed tag
-         + '|<tag(?:"[^"]*"|\'[^\']*\'|\\s[^\'"/>\\s]*)*?/?> *(?:\\n{2,}|\\s*$))').replace('comment', block._comment).replace(/tag/g, '(?!(?:' + 'a|em|strong|small|s|cite|q|dfn|abbr|data|time|code|var|samp|kbd|sub' + '|sup|i|b|u|mark|ruby|rt|rp|bdi|bdo|span|br|wbr|ins|del|img)' + '\\b)\\w+(?!:|[^\\w\\s@]*@)\\b').getRegex(),
-         def: /^ *\[([^\]]+)\]: *<?([^\s>]+)>?(?: +(["(][^\n]+[")]))? *(?:\n+|$)/,
-         heading: /^(#{1,6})(.*)(?:\n+|$)/,
-         fences: noopTest$1,
-         // fences not supported
-         paragraph: edit$1(block.normal._paragraph).replace('hr', block.hr).replace('heading', ' *#{1,6} *[^\n]').replace('lheading', block.lheading).replace('blockquote', ' {0,3}>').replace('|fences', '').replace('|list', '').replace('|html', '').getRegex()
-       });
-       /**
-        * Inline-Level Grammar
-        */
+               return graph.replace(target);
+             } else {
+               _conflicts.push(_t('merge_remote_changes.conflict.deleted', {
+                 user: user(remote.user)
+               }));
 
-       var inline = {
-         escape: /^\\([!"#$%&'()*+,\-./:;<=>?@\[\]\\^_`{|}~])/,
-         autolink: /^<(scheme:[^\s\x00-\x1f<>]*|email)>/,
-         url: noopTest$1,
-         tag: '^comment' + '|^</[a-zA-Z][\\w:-]*\\s*>' // self-closing tag
-         + '|^<[a-zA-Z][\\w-]*(?:attribute)*?\\s*/?>' // open tag
-         + '|^<\\?[\\s\\S]*?\\?>' // processing instruction, e.g. <?php ?>
-         + '|^<![a-zA-Z]+\\s[\\s\\S]*?>' // declaration, e.g. <!DOCTYPE html>
-         + '|^<!\\[CDATA\\[[\\s\\S]*?\\]\\]>',
-         // CDATA section
-         link: /^!?\[(label)\]\(\s*(href)(?:\s+(title))?\s*\)/,
-         reflink: /^!?\[(label)\]\[(?!\s*\])((?:\\[\[\]]?|[^\[\]\\])+)\]/,
-         nolink: /^!?\[(?!\s*\])((?:\[[^\[\]]*\]|\\[\[\]]|[^\[\]])*)\](?:\[\])?/,
-         reflinkSearch: 'reflink|nolink(?!\\()',
-         strong: {
-           start: /^(?:(\*\*(?=[*punctuation]))|\*\*)(?![\s])|__/,
-           // (1) returns if starts w/ punctuation
-           middle: /^\*\*(?:(?:(?!overlapSkip)(?:[^*]|\\\*)|overlapSkip)|\*(?:(?!overlapSkip)(?:[^*]|\\\*)|overlapSkip)*?\*)+?\*\*$|^__(?![\s])((?:(?:(?!overlapSkip)(?:[^_]|\\_)|overlapSkip)|_(?:(?!overlapSkip)(?:[^_]|\\_)|overlapSkip)*?_)+?)__$/,
-           endAst: /[^punctuation\s]\*\*(?!\*)|[punctuation]\*\*(?!\*)(?:(?=[punctuation_\s]|$))/,
-           // last char can't be punct, or final * must also be followed by punct (or endline)
-           endUnd: /[^\s]__(?!_)(?:(?=[punctuation*\s])|$)/ // last char can't be a space, and final _ must preceed punct or \s (or endline)
+               return graph; // do nothing
+             }
+           } // merge
 
-         },
-         em: {
-           start: /^(?:(\*(?=[punctuation]))|\*)(?![*\s])|_/,
-           // (1) returns if starts w/ punctuation
-           middle: /^\*(?:(?:(?!overlapSkip)(?:[^*]|\\\*)|overlapSkip)|\*(?:(?!overlapSkip)(?:[^*]|\\\*)|overlapSkip)*?\*)+?\*$|^_(?![_\s])(?:(?:(?!overlapSkip)(?:[^_]|\\_)|overlapSkip)|_(?:(?!overlapSkip)(?:[^_]|\\_)|overlapSkip)*?_)+?_$/,
-           endAst: /[^punctuation\s]\*(?!\*)|[punctuation]\*(?!\*)(?:(?=[punctuation_\s]|$))/,
-           // last char can't be punct, or final * must also be followed by punct (or endline)
-           endUnd: /[^\s]_(?!_)(?:(?=[punctuation*\s])|$)/ // last char can't be a space, and final _ must preceed punct or \s (or endline)
 
-         },
-         code: /^(`+)([^`]|[^`][\s\S]*?[^`])\1(?!`)/,
-         br: /^( {2,}|\\)\n(?!\s*$)/,
-         del: noopTest$1,
-         text: /^(`+|[^`])(?:(?= {2,}\n)|[\s\S]*?(?:(?=[\\<!\[`*]|\b_|$)|[^ ](?= {2,}\n)))/,
-         punctuation: /^([\s*punctuation])/
-       }; // list of punctuation marks from common mark spec
-       // without * and _ to workaround cases with double emphasis
-
-       inline._punctuation = '!"#$%&\'()+\\-.,/:;<=>?@\\[\\]`^{|}~';
-       inline.punctuation = edit$1(inline.punctuation).replace(/punctuation/g, inline._punctuation).getRegex(); // sequences em should skip over [title](link), `code`, <html>
-
-       inline._blockSkip = '\\[[^\\]]*?\\]\\([^\\)]*?\\)|`[^`]*?`|<[^>]*?>';
-       inline._overlapSkip = '__[^_]*?__|\\*\\*\\[^\\*\\]*?\\*\\*';
-       inline._comment = edit$1(block._comment).replace('(?:-->|$)', '-->').getRegex();
-       inline.em.start = edit$1(inline.em.start).replace(/punctuation/g, inline._punctuation).getRegex();
-       inline.em.middle = edit$1(inline.em.middle).replace(/punctuation/g, inline._punctuation).replace(/overlapSkip/g, inline._overlapSkip).getRegex();
-       inline.em.endAst = edit$1(inline.em.endAst, 'g').replace(/punctuation/g, inline._punctuation).getRegex();
-       inline.em.endUnd = edit$1(inline.em.endUnd, 'g').replace(/punctuation/g, inline._punctuation).getRegex();
-       inline.strong.start = edit$1(inline.strong.start).replace(/punctuation/g, inline._punctuation).getRegex();
-       inline.strong.middle = edit$1(inline.strong.middle).replace(/punctuation/g, inline._punctuation).replace(/overlapSkip/g, inline._overlapSkip).getRegex();
-       inline.strong.endAst = edit$1(inline.strong.endAst, 'g').replace(/punctuation/g, inline._punctuation).getRegex();
-       inline.strong.endUnd = edit$1(inline.strong.endUnd, 'g').replace(/punctuation/g, inline._punctuation).getRegex();
-       inline.blockSkip = edit$1(inline._blockSkip, 'g').getRegex();
-       inline.overlapSkip = edit$1(inline._overlapSkip, 'g').getRegex();
-       inline._escapes = /\\([!"#$%&'()*+,\-./:;<=>?@\[\]\\^_`{|}~])/g;
-       inline._scheme = /[a-zA-Z][a-zA-Z0-9+.-]{1,31}/;
-       inline._email = /[a-zA-Z0-9.!#$%&'*+/=?^_`{|}~-]+(@)[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)+(?![-_])/;
-       inline.autolink = edit$1(inline.autolink).replace('scheme', inline._scheme).replace('email', inline._email).getRegex();
-       inline._attribute = /\s+[a-zA-Z:_][\w.:-]*(?:\s*=\s*"[^"]*"|\s*=\s*'[^']*'|\s*=\s*[^\s"'=<>`]+)?/;
-       inline.tag = edit$1(inline.tag).replace('comment', inline._comment).replace('attribute', inline._attribute).getRegex();
-       inline._label = /(?:\[(?:\\.|[^\[\]\\])*\]|\\.|`[^`]*`|[^\[\]\\`])*?/;
-       inline._href = /<(?:\\.|[^\n<>\\])+>|[^\s\x00-\x1f]*/;
-       inline._title = /"(?:\\"?|[^"\\])*"|'(?:\\'?|[^'\\])*'|\((?:\\\)?|[^)\\])*\)/;
-       inline.link = edit$1(inline.link).replace('label', inline._label).replace('href', inline._href).replace('title', inline._title).getRegex();
-       inline.reflink = edit$1(inline.reflink).replace('label', inline._label).getRegex();
-       inline.reflinkSearch = edit$1(inline.reflinkSearch, 'g').replace('reflink', inline.reflink).replace('nolink', inline.nolink).getRegex();
-       /**
-        * Normal Inline Grammar
-        */
+           if (target.type === 'node') {
+             target = mergeLocation(remote, target);
+           } else if (target.type === 'way') {
+             // pull in any child nodes that may not be present locally..
+             graph.rebase(remoteGraph.childNodes(remote), [graph], false);
+             target = mergeNodes(base, remote, target);
+             target = mergeChildren(target, utilArrayUnion(local.nodes, remote.nodes), updates, graph);
+           } else if (target.type === 'relation') {
+             target = mergeMembers(remote, target);
+           }
 
-       inline.normal = merge$2({}, inline);
-       /**
-        * Pedantic Inline Grammar
-        */
+           target = mergeTags(base, remote, target);
 
-       inline.pedantic = merge$2({}, inline.normal, {
-         strong: {
-           start: /^__|\*\*/,
-           middle: /^__(?=\S)([\s\S]*?\S)__(?!_)|^\*\*(?=\S)([\s\S]*?\S)\*\*(?!\*)/,
-           endAst: /\*\*(?!\*)/g,
-           endUnd: /__(?!_)/g
-         },
-         em: {
-           start: /^_|\*/,
-           middle: /^()\*(?=\S)([\s\S]*?\S)\*(?!\*)|^_(?=\S)([\s\S]*?\S)_(?!_)/,
-           endAst: /\*(?!\*)/g,
-           endUnd: /_(?!_)/g
-         },
-         link: edit$1(/^!?\[(label)\]\((.*?)\)/).replace('label', inline._label).getRegex(),
-         reflink: edit$1(/^!?\[(label)\]\s*\[([^\]]*)\]/).replace('label', inline._label).getRegex()
-       });
-       /**
-        * GFM Inline Grammar
-        */
+           if (!_conflicts.length) {
+             graph = updateChildren(updates, graph).replace(target);
+           }
 
-       inline.gfm = merge$2({}, inline.normal, {
-         escape: edit$1(inline.escape).replace('])', '~|])').getRegex(),
-         _extended_email: /[A-Za-z0-9._+-]+(@)[a-zA-Z0-9-_]+(?:\.[a-zA-Z0-9-_]*[a-zA-Z0-9])+(?![-_])/,
-         url: /^((?:ftp|https?):\/\/|www\.)(?:[a-zA-Z0-9\-]+\.?)+[^\s<]*|^email/,
-         _backpedal: /(?:[^?!.,:;*_~()&]+|\([^)]*\)|&(?![a-zA-Z0-9]+;$)|[?!.,:;*_~)]+(?!$))+/,
-         del: /^(~~?)(?=[^\s~])([\s\S]*?[^\s~])\1(?=[^~]|$)/,
-         text: /^([`~]+|[^`~])(?:(?= {2,}\n)|[\s\S]*?(?:(?=[\\<!\[`*~]|\b_|https?:\/\/|ftp:\/\/|www\.|$)|[^ ](?= {2,}\n)|[^a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-](?=[a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-]+@))|(?=[a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-]+@))/
-       });
-       inline.gfm.url = edit$1(inline.gfm.url, 'i').replace('email', inline.gfm._extended_email).getRegex();
-       /**
-        * GFM + Line Breaks Inline Grammar
-        */
+           return graph;
+         };
 
-       inline.breaks = merge$2({}, inline.gfm, {
-         br: edit$1(inline.br).replace('{2,}', '*').getRegex(),
-         text: edit$1(inline.gfm.text).replace('\\b_', '\\b_| {2,}\\n').replace(/\{2,\}/g, '*').getRegex()
-       });
-       var rules = {
-         block: block,
-         inline: inline
-       };
+         action.withOption = function (opt) {
+           _option = opt;
+           return action;
+         };
 
-       var defaults$2 = defaults.defaults;
-       var block$1 = rules.block,
-           inline$1 = rules.inline;
-       var repeatString$1 = helpers.repeatString;
-       /**
-        * smartypants text replacement
-        */
+         action.conflicts = function () {
+           return _conflicts;
+         };
 
-       function smartypants(text) {
-         return text // em-dashes
-         .replace(/---/g, "\u2014") // en-dashes
-         .replace(/--/g, "\u2013") // opening singles
-         .replace(/(^|[-\u2014/(\[{"\s])'/g, "$1\u2018") // closing singles & apostrophes
-         .replace(/'/g, "\u2019") // opening doubles
-         .replace(/(^|[-\u2014/(\[{\u2018\s])"/g, "$1\u201C") // closing doubles
-         .replace(/"/g, "\u201D") // ellipses
-         .replace(/\.{3}/g, "\u2026");
+         return action;
        }
-       /**
-        * mangle email addresses
-        */
 
+       // https://github.com/openstreetmap/potlatch2/blob/master/net/systemeD/halcyon/connection/actions/MoveNodeAction.as
 
-       function mangle(text) {
-         var out = '',
-             i,
-             ch;
-         var l = text.length;
+       function actionMove(moveIDs, tryDelta, projection, cache) {
+         var _delta = tryDelta;
 
-         for (i = 0; i < l; i++) {
-           ch = text.charCodeAt(i);
+         function setupCache(graph) {
+           function canMove(nodeID) {
+             // Allow movement of any node that is in the selectedIDs list..
+             if (moveIDs.indexOf(nodeID) !== -1) return true; // Allow movement of a vertex where 2 ways meet..
 
-           if (Math.random() > 0.5) {
-             ch = 'x' + ch.toString(16);
+             var parents = graph.parentWays(graph.entity(nodeID));
+             if (parents.length < 3) return true; // Restrict movement of a vertex where >2 ways meet, unless all parentWays are moving too..
+
+             var parentsMoving = parents.every(function (way) {
+               return cache.moving[way.id];
+             });
+             if (!parentsMoving) delete cache.moving[nodeID];
+             return parentsMoving;
            }
 
-           out += '&#' + ch + ';';
-         }
+           function cacheEntities(ids) {
+             for (var i = 0; i < ids.length; i++) {
+               var id = ids[i];
+               if (cache.moving[id]) continue;
+               cache.moving[id] = true;
+               var entity = graph.hasEntity(id);
+               if (!entity) continue;
 
-         return out;
-       }
-       /**
-        * Block Lexer
-        */
+               if (entity.type === 'node') {
+                 cache.nodes.push(id);
+                 cache.startLoc[id] = entity.loc;
+               } else if (entity.type === 'way') {
+                 cache.ways.push(id);
+                 cacheEntities(entity.nodes);
+               } else {
+                 cacheEntities(entity.members.map(function (member) {
+                   return member.id;
+                 }));
+               }
+             }
+           }
 
+           function cacheIntersections(ids) {
+             function isEndpoint(way, id) {
+               return !way.isClosed() && !!way.affix(id);
+             }
 
-       var Lexer_1 = /*#__PURE__*/function () {
-         function Lexer(options) {
-           _classCallCheck(this, Lexer);
+             for (var i = 0; i < ids.length; i++) {
+               var id = ids[i]; // consider only intersections with 1 moved and 1 unmoved way.
 
-           this.tokens = [];
-           this.tokens.links = Object.create(null);
-           this.options = options || defaults$2;
-           this.options.tokenizer = this.options.tokenizer || new Tokenizer_1();
-           this.tokenizer = this.options.tokenizer;
-           this.tokenizer.options = this.options;
-           var rules = {
-             block: block$1.normal,
-             inline: inline$1.normal
-           };
+               var childNodes = graph.childNodes(graph.entity(id));
 
-           if (this.options.pedantic) {
-             rules.block = block$1.pedantic;
-             rules.inline = inline$1.pedantic;
-           } else if (this.options.gfm) {
-             rules.block = block$1.gfm;
+               for (var j = 0; j < childNodes.length; j++) {
+                 var node = childNodes[j];
+                 var parents = graph.parentWays(node);
+                 if (parents.length !== 2) continue;
+                 var moved = graph.entity(id);
+                 var unmoved = null;
 
-             if (this.options.breaks) {
-               rules.inline = inline$1.breaks;
-             } else {
-               rules.inline = inline$1.gfm;
-             }
-           }
+                 for (var k = 0; k < parents.length; k++) {
+                   var way = parents[k];
 
-           this.tokenizer.rules = rules;
-         }
-         /**
-          * Expose Rules
-          */
+                   if (!cache.moving[way.id]) {
+                     unmoved = way;
+                     break;
+                   }
+                 }
 
+                 if (!unmoved) continue; // exclude ways that are overly connected..
 
-         _createClass(Lexer, [{
-           key: "lex",
-           value:
-           /**
-            * Preprocessing
-            */
-           function lex(src) {
-             src = src.replace(/\r\n|\r/g, '\n').replace(/\t/g, '    ');
-             this.blockTokens(src, this.tokens, true);
-             this.inline(this.tokens);
-             return this.tokens;
+                 if (utilArrayIntersection(moved.nodes, unmoved.nodes).length > 2) continue;
+                 if (moved.isArea() || unmoved.isArea()) continue;
+                 cache.intersections.push({
+                   nodeId: node.id,
+                   movedId: moved.id,
+                   unmovedId: unmoved.id,
+                   movedIsEP: isEndpoint(moved, node.id),
+                   unmovedIsEP: isEndpoint(unmoved, node.id)
+                 });
+               }
+             }
            }
-           /**
-            * Lexing
-            */
-
-         }, {
-           key: "blockTokens",
-           value: function blockTokens(src) {
-             var tokens = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : [];
-             var top = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : true;
 
-             if (this.options.pedantic) {
-               src = src.replace(/^ +$/gm, '');
-             }
+           if (!cache) {
+             cache = {};
+           }
 
-             var token, i, l, lastToken;
+           if (!cache.ok) {
+             cache.moving = {};
+             cache.intersections = [];
+             cache.replacedVertex = {};
+             cache.startLoc = {};
+             cache.nodes = [];
+             cache.ways = [];
+             cacheEntities(moveIDs);
+             cacheIntersections(cache.ways);
+             cache.nodes = cache.nodes.filter(canMove);
+             cache.ok = true;
+           }
+         } // Place a vertex where the moved vertex used to be, to preserve way shape..
+         //
+         //  Start:
+         //      b ---- e
+         //     / \
+         //    /   \
+         //   /     \
+         //  a       c
+         //
+         //      *               node '*' added to preserve shape
+         //     / \
+         //    /   b ---- e      way `b,e` moved here:
+         //   /     \
+         //  a       c
+         //
+         //
 
-             while (src) {
-               // newline
-               if (token = this.tokenizer.space(src)) {
-                 src = src.substring(token.raw.length);
 
-                 if (token.type) {
-                   tokens.push(token);
-                 }
+         function replaceMovedVertex(nodeId, wayId, graph, delta) {
+           var way = graph.entity(wayId);
+           var moved = graph.entity(nodeId);
+           var movedIndex = way.nodes.indexOf(nodeId);
+           var len, prevIndex, nextIndex;
 
-                 continue;
-               } // code
+           if (way.isClosed()) {
+             len = way.nodes.length - 1;
+             prevIndex = (movedIndex + len - 1) % len;
+             nextIndex = (movedIndex + len + 1) % len;
+           } else {
+             len = way.nodes.length;
+             prevIndex = movedIndex - 1;
+             nextIndex = movedIndex + 1;
+           }
 
+           var prev = graph.hasEntity(way.nodes[prevIndex]);
+           var next = graph.hasEntity(way.nodes[nextIndex]); // Don't add orig vertex at endpoint..
 
-               if (token = this.tokenizer.code(src, tokens)) {
-                 src = src.substring(token.raw.length);
+           if (!prev || !next) return graph;
+           var key = wayId + '_' + nodeId;
+           var orig = cache.replacedVertex[key];
 
-                 if (token.type) {
-                   tokens.push(token);
-                 } else {
-                   lastToken = tokens[tokens.length - 1];
-                   lastToken.raw += '\n' + token.raw;
-                   lastToken.text += '\n' + token.text;
-                 }
+           if (!orig) {
+             orig = osmNode();
+             cache.replacedVertex[key] = orig;
+             cache.startLoc[orig.id] = cache.startLoc[nodeId];
+           }
 
-                 continue;
-               } // fences
+           var start, end;
 
+           if (delta) {
+             start = projection(cache.startLoc[nodeId]);
+             end = projection.invert(geoVecAdd(start, delta));
+           } else {
+             end = cache.startLoc[nodeId];
+           }
 
-               if (token = this.tokenizer.fences(src)) {
-                 src = src.substring(token.raw.length);
-                 tokens.push(token);
-                 continue;
-               } // heading
+           orig = orig.move(end);
+           var angle = Math.abs(geoAngle(orig, prev, projection) - geoAngle(orig, next, projection)) * 180 / Math.PI; // Don't add orig vertex if it would just make a straight line..
 
+           if (angle > 175 && angle < 185) return graph; // moving forward or backward along way?
 
-               if (token = this.tokenizer.heading(src)) {
-                 src = src.substring(token.raw.length);
-                 tokens.push(token);
-                 continue;
-               } // table no leading pipe (gfm)
+           var p1 = [prev.loc, orig.loc, moved.loc, next.loc].map(projection);
+           var p2 = [prev.loc, moved.loc, orig.loc, next.loc].map(projection);
+           var d1 = geoPathLength(p1);
+           var d2 = geoPathLength(p2);
+           var insertAt = d1 <= d2 ? movedIndex : nextIndex; // moving around closed loop?
 
+           if (way.isClosed() && insertAt === 0) insertAt = len;
+           way = way.addNode(orig.id, insertAt);
+           return graph.replace(orig).replace(way);
+         } // Remove duplicate vertex that might have been added by
+         // replaceMovedVertex.  This is done after the unzorro checks.
 
-               if (token = this.tokenizer.nptable(src)) {
-                 src = src.substring(token.raw.length);
-                 tokens.push(token);
-                 continue;
-               } // hr
 
+         function removeDuplicateVertices(wayId, graph) {
+           var way = graph.entity(wayId);
+           var epsilon = 1e-6;
+           var prev, curr;
 
-               if (token = this.tokenizer.hr(src)) {
-                 src = src.substring(token.raw.length);
-                 tokens.push(token);
-                 continue;
-               } // blockquote
+           function isInteresting(node, graph) {
+             return graph.parentWays(node).length > 1 || graph.parentRelations(node).length || node.hasInterestingTags();
+           }
 
+           for (var i = 0; i < way.nodes.length; i++) {
+             curr = graph.entity(way.nodes[i]);
 
-               if (token = this.tokenizer.blockquote(src)) {
-                 src = src.substring(token.raw.length);
-                 token.tokens = this.blockTokens(token.text, [], top);
-                 tokens.push(token);
-                 continue;
-               } // list
+             if (prev && curr && geoVecEqual(prev.loc, curr.loc, epsilon)) {
+               if (!isInteresting(prev, graph)) {
+                 way = way.removeNode(prev.id);
+                 graph = graph.replace(way).remove(prev);
+               } else if (!isInteresting(curr, graph)) {
+                 way = way.removeNode(curr.id);
+                 graph = graph.replace(way).remove(curr);
+               }
+             }
 
+             prev = curr;
+           }
 
-               if (token = this.tokenizer.list(src)) {
-                 src = src.substring(token.raw.length);
-                 l = token.items.length;
+           return graph;
+         } // Reorder nodes around intersections that have moved..
+         //
+         //  Start:                way1.nodes: b,e         (moving)
+         //  a - b - c ----- d     way2.nodes: a,b,c,d     (static)
+         //      |                 vertex: b
+         //      e                 isEP1: true,  isEP2, false
+         //
+         //  way1 `b,e` moved here:
+         //  a ----- c = b - d
+         //              |
+         //              e
+         //
+         //  reorder nodes         way1.nodes: b,e
+         //  a ----- c - b - d     way2.nodes: a,c,b,d
+         //              |
+         //              e
+         //
 
-                 for (i = 0; i < l; i++) {
-                   token.items[i].tokens = this.blockTokens(token.items[i].text, [], false);
-                 }
 
-                 tokens.push(token);
-                 continue;
-               } // html
+         function unZorroIntersection(intersection, graph) {
+           var vertex = graph.entity(intersection.nodeId);
+           var way1 = graph.entity(intersection.movedId);
+           var way2 = graph.entity(intersection.unmovedId);
+           var isEP1 = intersection.movedIsEP;
+           var isEP2 = intersection.unmovedIsEP; // don't move the vertex if it is the endpoint of both ways.
 
+           if (isEP1 && isEP2) return graph;
+           var nodes1 = graph.childNodes(way1).filter(function (n) {
+             return n !== vertex;
+           });
+           var nodes2 = graph.childNodes(way2).filter(function (n) {
+             return n !== vertex;
+           });
+           if (way1.isClosed() && way1.first() === vertex.id) nodes1.push(nodes1[0]);
+           if (way2.isClosed() && way2.first() === vertex.id) nodes2.push(nodes2[0]);
+           var edge1 = !isEP1 && geoChooseEdge(nodes1, projection(vertex.loc), projection);
+           var edge2 = !isEP2 && geoChooseEdge(nodes2, projection(vertex.loc), projection);
+           var loc; // snap vertex to nearest edge (or some point between them)..
 
-               if (token = this.tokenizer.html(src)) {
-                 src = src.substring(token.raw.length);
-                 tokens.push(token);
-                 continue;
-               } // def
+           if (!isEP1 && !isEP2) {
+             var epsilon = 1e-6,
+                 maxIter = 10;
 
+             for (var i = 0; i < maxIter; i++) {
+               loc = geoVecInterp(edge1.loc, edge2.loc, 0.5);
+               edge1 = geoChooseEdge(nodes1, projection(loc), projection);
+               edge2 = geoChooseEdge(nodes2, projection(loc), projection);
+               if (Math.abs(edge1.distance - edge2.distance) < epsilon) break;
+             }
+           } else if (!isEP1) {
+             loc = edge1.loc;
+           } else {
+             loc = edge2.loc;
+           }
 
-               if (top && (token = this.tokenizer.def(src))) {
-                 src = src.substring(token.raw.length);
+           graph = graph.replace(vertex.move(loc)); // if zorro happened, reorder nodes..
 
-                 if (!this.tokens.links[token.tag]) {
-                   this.tokens.links[token.tag] = {
-                     href: token.href,
-                     title: token.title
-                   };
-                 }
+           if (!isEP1 && edge1.index !== way1.nodes.indexOf(vertex.id)) {
+             way1 = way1.removeNode(vertex.id).addNode(vertex.id, edge1.index);
+             graph = graph.replace(way1);
+           }
 
-                 continue;
-               } // table (gfm)
+           if (!isEP2 && edge2.index !== way2.nodes.indexOf(vertex.id)) {
+             way2 = way2.removeNode(vertex.id).addNode(vertex.id, edge2.index);
+             graph = graph.replace(way2);
+           }
 
+           return graph;
+         }
 
-               if (token = this.tokenizer.table(src)) {
-                 src = src.substring(token.raw.length);
-                 tokens.push(token);
-                 continue;
-               } // lheading
+         function cleanupIntersections(graph) {
+           for (var i = 0; i < cache.intersections.length; i++) {
+             var obj = cache.intersections[i];
+             graph = replaceMovedVertex(obj.nodeId, obj.movedId, graph, _delta);
+             graph = replaceMovedVertex(obj.nodeId, obj.unmovedId, graph, null);
+             graph = unZorroIntersection(obj, graph);
+             graph = removeDuplicateVertices(obj.movedId, graph);
+             graph = removeDuplicateVertices(obj.unmovedId, graph);
+           }
 
+           return graph;
+         } // check if moving way endpoint can cross an unmoved way, if so limit delta..
 
-               if (token = this.tokenizer.lheading(src)) {
-                 src = src.substring(token.raw.length);
-                 tokens.push(token);
-                 continue;
-               } // top-level paragraph
 
+         function limitDelta(graph) {
+           function moveNode(loc) {
+             return geoVecAdd(projection(loc), _delta);
+           }
 
-               if (top && (token = this.tokenizer.paragraph(src))) {
-                 src = src.substring(token.raw.length);
-                 tokens.push(token);
-                 continue;
-               } // text
+           for (var i = 0; i < cache.intersections.length; i++) {
+             var obj = cache.intersections[i]; // Don't limit movement if this is vertex joins 2 endpoints..
 
+             if (obj.movedIsEP && obj.unmovedIsEP) continue; // Don't limit movement if this vertex is not an endpoint anyway..
 
-               if (token = this.tokenizer.text(src, tokens)) {
-                 src = src.substring(token.raw.length);
+             if (!obj.movedIsEP) continue;
+             var node = graph.entity(obj.nodeId);
+             var start = projection(node.loc);
+             var end = geoVecAdd(start, _delta);
+             var movedNodes = graph.childNodes(graph.entity(obj.movedId));
+             var movedPath = movedNodes.map(function (n) {
+               return moveNode(n.loc);
+             });
+             var unmovedNodes = graph.childNodes(graph.entity(obj.unmovedId));
+             var unmovedPath = unmovedNodes.map(function (n) {
+               return projection(n.loc);
+             });
+             var hits = geoPathIntersections(movedPath, unmovedPath);
 
-                 if (token.type) {
-                   tokens.push(token);
-                 } else {
-                   lastToken = tokens[tokens.length - 1];
-                   lastToken.raw += '\n' + token.raw;
-                   lastToken.text += '\n' + token.text;
-                 }
+             for (var j = 0; i < hits.length; i++) {
+               if (geoVecEqual(hits[j], end)) continue;
+               var edge = geoChooseEdge(unmovedNodes, end, projection);
+               _delta = geoVecSubtract(projection(edge.loc), start);
+             }
+           }
+         }
 
-                 continue;
-               }
+         var action = function action(graph) {
+           if (_delta[0] === 0 && _delta[1] === 0) return graph;
+           setupCache(graph);
 
-               if (src) {
-                 var errMsg = 'Infinite loop on byte: ' + src.charCodeAt(0);
+           if (cache.intersections.length) {
+             limitDelta(graph);
+           }
 
-                 if (this.options.silent) {
-                   console.error(errMsg);
-                   break;
-                 } else {
-                   throw new Error(errMsg);
-                 }
-               }
-             }
+           for (var i = 0; i < cache.nodes.length; i++) {
+             var node = graph.entity(cache.nodes[i]);
+             var start = projection(node.loc);
+             var end = geoVecAdd(start, _delta);
+             graph = graph.replace(node.move(projection.invert(end)));
+           }
 
-             return tokens;
+           if (cache.intersections.length) {
+             graph = cleanupIntersections(graph);
            }
-         }, {
-           key: "inline",
-           value: function inline(tokens) {
-             var i, j, k, l2, row, token;
-             var l = tokens.length;
 
-             for (i = 0; i < l; i++) {
-               token = tokens[i];
+           return graph;
+         };
 
-               switch (token.type) {
-                 case 'paragraph':
-                 case 'text':
-                 case 'heading':
-                   {
-                     token.tokens = [];
-                     this.inlineTokens(token.text, token.tokens);
-                     break;
-                   }
+         action.delta = function () {
+           return _delta;
+         };
 
-                 case 'table':
-                   {
-                     token.tokens = {
-                       header: [],
-                       cells: []
-                     }; // header
+         return action;
+       }
 
-                     l2 = token.header.length;
+       function actionMoveMember(relationId, fromIndex, toIndex) {
+         return function (graph) {
+           return graph.replace(graph.entity(relationId).moveMember(fromIndex, toIndex));
+         };
+       }
 
-                     for (j = 0; j < l2; j++) {
-                       token.tokens.header[j] = [];
-                       this.inlineTokens(token.header[j], token.tokens.header[j]);
-                     } // cells
+       function actionMoveNode(nodeID, toLoc) {
+         var action = function action(graph, t) {
+           if (t === null || !isFinite(t)) t = 1;
+           t = Math.min(Math.max(+t, 0), 1);
+           var node = graph.entity(nodeID);
+           return graph.replace(node.move(geoVecInterp(node.loc, toLoc, t)));
+         };
 
+         action.transitionable = true;
+         return action;
+       }
 
-                     l2 = token.cells.length;
+       function actionNoop() {
+         return function (graph) {
+           return graph;
+         };
+       }
 
-                     for (j = 0; j < l2; j++) {
-                       row = token.cells[j];
-                       token.tokens.cells[j] = [];
+       function actionOrthogonalize(wayID, projection, vertexID, degThresh, ep) {
+         var epsilon = ep || 1e-4;
+         var threshold = degThresh || 13; // degrees within right or straight to alter
+         // We test normalized dot products so we can compare as cos(angle)
 
-                       for (k = 0; k < row.length; k++) {
-                         token.tokens.cells[j][k] = [];
-                         this.inlineTokens(row[k], token.tokens.cells[j][k]);
-                       }
-                     }
+         var lowerThreshold = Math.cos((90 - threshold) * Math.PI / 180);
+         var upperThreshold = Math.cos(threshold * Math.PI / 180);
 
-                     break;
-                   }
+         var action = function action(graph, t) {
+           if (t === null || !isFinite(t)) t = 1;
+           t = Math.min(Math.max(+t, 0), 1);
+           var way = graph.entity(wayID);
+           way = way.removeNode(''); // sanity check - remove any consecutive duplicates
 
-                 case 'blockquote':
-                   {
-                     this.inline(token.tokens);
-                     break;
-                   }
+           if (way.tags.nonsquare) {
+             var tags = Object.assign({}, way.tags); // since we're squaring, remove indication that this is physically unsquare
 
-                 case 'list':
-                   {
-                     l2 = token.items.length;
+             delete tags.nonsquare;
+             way = way.update({
+               tags: tags
+             });
+           }
 
-                     for (j = 0; j < l2; j++) {
-                       this.inline(token.items[j].tokens);
-                     }
+           graph = graph.replace(way);
+           var isClosed = way.isClosed();
+           var nodes = graph.childNodes(way).slice(); // shallow copy
 
-                     break;
-                   }
-               }
-             }
+           if (isClosed) nodes.pop();
 
-             return tokens;
-           }
-           /**
-            * Lexing/Compiling
-            */
+           if (vertexID !== undefined) {
+             nodes = nodeSubset(nodes, vertexID, isClosed);
+             if (nodes.length !== 3) return graph;
+           } // note: all geometry functions here use the unclosed node/point/coord list
 
-         }, {
-           key: "inlineTokens",
-           value: function inlineTokens(src) {
-             var tokens = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : [];
-             var inLink = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false;
-             var inRawBlock = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : false;
-             var token; // String with links masked to avoid interference with em and strong
 
-             var maskedSrc = src;
-             var match;
-             var keepPrevChar, prevChar; // Mask out reflinks
+           var nodeCount = {};
+           var points = [];
+           var corner = {
+             i: 0,
+             dotp: 1
+           };
+           var node, point, loc, score, motions, i, j;
 
-             if (this.tokens.links) {
-               var links = Object.keys(this.tokens.links);
+           for (i = 0; i < nodes.length; i++) {
+             node = nodes[i];
+             nodeCount[node.id] = (nodeCount[node.id] || 0) + 1;
+             points.push({
+               id: node.id,
+               coord: projection(node.loc)
+             });
+           }
 
-               if (links.length > 0) {
-                 while ((match = this.tokenizer.rules.inline.reflinkSearch.exec(maskedSrc)) != null) {
-                   if (links.includes(match[0].slice(match[0].lastIndexOf('[') + 1, -1))) {
-                     maskedSrc = maskedSrc.slice(0, match.index) + '[' + repeatString$1('a', match[0].length - 2) + ']' + maskedSrc.slice(this.tokenizer.rules.inline.reflinkSearch.lastIndex);
-                   }
-                 }
+           if (points.length === 3) {
+             // move only one vertex for right triangle
+             for (i = 0; i < 1000; i++) {
+               motions = points.map(calcMotion);
+               points[corner.i].coord = geoVecAdd(points[corner.i].coord, motions[corner.i]);
+               score = corner.dotp;
+
+               if (score < epsilon) {
+                 break;
                }
-             } // Mask out other blocks
+             }
 
+             node = graph.entity(nodes[corner.i].id);
+             loc = projection.invert(points[corner.i].coord);
+             graph = graph.replace(node.move(geoVecInterp(node.loc, loc, t)));
+           } else {
+             var straights = [];
+             var simplified = []; // Remove points from nearly straight sections..
+             // This produces a simplified shape to orthogonalize
 
-             while ((match = this.tokenizer.rules.inline.blockSkip.exec(maskedSrc)) != null) {
-               maskedSrc = maskedSrc.slice(0, match.index) + '[' + repeatString$1('a', match[0].length - 2) + ']' + maskedSrc.slice(this.tokenizer.rules.inline.blockSkip.lastIndex);
-             }
+             for (i = 0; i < points.length; i++) {
+               point = points[i];
+               var dotp = 0;
 
-             while (src) {
-               if (!keepPrevChar) {
-                 prevChar = '';
+               if (isClosed || i > 0 && i < points.length - 1) {
+                 var a = points[(i - 1 + points.length) % points.length];
+                 var b = points[(i + 1) % points.length];
+                 dotp = Math.abs(geoOrthoNormalizedDotProduct(a.coord, b.coord, point.coord));
                }
 
-               keepPrevChar = false; // escape
-
-               if (token = this.tokenizer.escape(src)) {
-                 src = src.substring(token.raw.length);
-                 tokens.push(token);
-                 continue;
-               } // tag
+               if (dotp > upperThreshold) {
+                 straights.push(point);
+               } else {
+                 simplified.push(point);
+               }
+             } // Orthogonalize the simplified shape
 
 
-               if (token = this.tokenizer.tag(src, inLink, inRawBlock)) {
-                 src = src.substring(token.raw.length);
-                 inLink = token.inLink;
-                 inRawBlock = token.inRawBlock;
-                 tokens.push(token);
-                 continue;
-               } // link
+             var bestPoints = clonePoints(simplified);
+             var originalPoints = clonePoints(simplified);
+             score = Infinity;
 
+             for (i = 0; i < 1000; i++) {
+               motions = simplified.map(calcMotion);
 
-               if (token = this.tokenizer.link(src)) {
-                 src = src.substring(token.raw.length);
+               for (j = 0; j < motions.length; j++) {
+                 simplified[j].coord = geoVecAdd(simplified[j].coord, motions[j]);
+               }
 
-                 if (token.type === 'link') {
-                   token.tokens = this.inlineTokens(token.text, [], true, inRawBlock);
-                 }
+               var newScore = geoOrthoCalcScore(simplified, isClosed, epsilon, threshold);
 
-                 tokens.push(token);
-                 continue;
-               } // reflink, nolink
+               if (newScore < score) {
+                 bestPoints = clonePoints(simplified);
+                 score = newScore;
+               }
 
+               if (score < epsilon) {
+                 break;
+               }
+             }
 
-               if (token = this.tokenizer.reflink(src, this.tokens.links)) {
-                 src = src.substring(token.raw.length);
+             var bestCoords = bestPoints.map(function (p) {
+               return p.coord;
+             });
+             if (isClosed) bestCoords.push(bestCoords[0]); // move the nodes that should move
 
-                 if (token.type === 'link') {
-                   token.tokens = this.inlineTokens(token.text, [], true, inRawBlock);
-                 }
+             for (i = 0; i < bestPoints.length; i++) {
+               point = bestPoints[i];
 
-                 tokens.push(token);
-                 continue;
-               } // strong
+               if (!geoVecEqual(originalPoints[i].coord, point.coord)) {
+                 node = graph.entity(point.id);
+                 loc = projection.invert(point.coord);
+                 graph = graph.replace(node.move(geoVecInterp(node.loc, loc, t)));
+               }
+             } // move the nodes along straight segments
 
 
-               if (token = this.tokenizer.strong(src, maskedSrc, prevChar)) {
-                 src = src.substring(token.raw.length);
-                 token.tokens = this.inlineTokens(token.text, [], inLink, inRawBlock);
-                 tokens.push(token);
-                 continue;
-               } // em
+             for (i = 0; i < straights.length; i++) {
+               point = straights[i];
+               if (nodeCount[point.id] > 1) continue; // skip self-intersections
 
+               node = graph.entity(point.id);
 
-               if (token = this.tokenizer.em(src, maskedSrc, prevChar)) {
-                 src = src.substring(token.raw.length);
-                 token.tokens = this.inlineTokens(token.text, [], inLink, inRawBlock);
-                 tokens.push(token);
-                 continue;
-               } // code
+               if (t === 1 && graph.parentWays(node).length === 1 && graph.parentRelations(node).length === 0 && !node.hasInterestingTags()) {
+                 // remove uninteresting points..
+                 graph = actionDeleteNode(node.id)(graph);
+               } else {
+                 // move interesting points to the nearest edge..
+                 var choice = geoVecProject(point.coord, bestCoords);
 
+                 if (choice) {
+                   loc = projection.invert(choice.target);
+                   graph = graph.replace(node.move(geoVecInterp(node.loc, loc, t)));
+                 }
+               }
+             }
+           }
 
-               if (token = this.tokenizer.codespan(src)) {
-                 src = src.substring(token.raw.length);
-                 tokens.push(token);
-                 continue;
-               } // br
+           return graph;
 
+           function clonePoints(array) {
+             return array.map(function (p) {
+               return {
+                 id: p.id,
+                 coord: [p.coord[0], p.coord[1]]
+               };
+             });
+           }
 
-               if (token = this.tokenizer.br(src)) {
-                 src = src.substring(token.raw.length);
-                 tokens.push(token);
-                 continue;
-               } // del (gfm)
+           function calcMotion(point, i, array) {
+             // don't try to move the endpoints of a non-closed way.
+             if (!isClosed && (i === 0 || i === array.length - 1)) return [0, 0]; // don't try to move a node that appears more than once (self intersection)
 
+             if (nodeCount[array[i].id] > 1) return [0, 0];
+             var a = array[(i - 1 + array.length) % array.length].coord;
+             var origin = point.coord;
+             var b = array[(i + 1) % array.length].coord;
+             var p = geoVecSubtract(a, origin);
+             var q = geoVecSubtract(b, origin);
+             var scale = 2 * Math.min(geoVecLength(p), geoVecLength(q));
+             p = geoVecNormalize(p);
+             q = geoVecNormalize(q);
+             var dotp = p[0] * q[0] + p[1] * q[1];
+             var val = Math.abs(dotp);
 
-               if (token = this.tokenizer.del(src)) {
-                 src = src.substring(token.raw.length);
-                 token.tokens = this.inlineTokens(token.text, [], inLink, inRawBlock);
-                 tokens.push(token);
-                 continue;
-               } // autolink
+             if (val < lowerThreshold) {
+               // nearly orthogonal
+               corner.i = i;
+               corner.dotp = val;
+               var vec = geoVecNormalize(geoVecAdd(p, q));
+               return geoVecScale(vec, 0.1 * dotp * scale);
+             }
 
+             return [0, 0]; // do nothing
+           }
+         }; // if we are only orthogonalizing one vertex,
+         // get that vertex and the previous and next
 
-               if (token = this.tokenizer.autolink(src, mangle)) {
-                 src = src.substring(token.raw.length);
-                 tokens.push(token);
-                 continue;
-               } // url (gfm)
 
+         function nodeSubset(nodes, vertexID, isClosed) {
+           var first = isClosed ? 0 : 1;
+           var last = isClosed ? nodes.length : nodes.length - 1;
 
-               if (!inLink && (token = this.tokenizer.url(src, mangle))) {
-                 src = src.substring(token.raw.length);
-                 tokens.push(token);
-                 continue;
-               } // text
+           for (var i = first; i < last; i++) {
+             if (nodes[i].id === vertexID) {
+               return [nodes[(i - 1 + nodes.length) % nodes.length], nodes[i], nodes[(i + 1) % nodes.length]];
+             }
+           }
 
+           return [];
+         }
 
-               if (token = this.tokenizer.inlineText(src, inRawBlock, smartypants)) {
-                 src = src.substring(token.raw.length);
-                 prevChar = token.raw.slice(-1);
-                 keepPrevChar = true;
-                 tokens.push(token);
-                 continue;
-               }
+         action.disabled = function (graph) {
+           var way = graph.entity(wayID);
+           way = way.removeNode(''); // sanity check - remove any consecutive duplicates
 
-               if (src) {
-                 var errMsg = 'Infinite loop on byte: ' + src.charCodeAt(0);
+           graph = graph.replace(way);
+           var isClosed = way.isClosed();
+           var nodes = graph.childNodes(way).slice(); // shallow copy
 
-                 if (this.options.silent) {
-                   console.error(errMsg);
-                   break;
-                 } else {
-                   throw new Error(errMsg);
-                 }
-               }
-             }
+           if (isClosed) nodes.pop();
+           var allowStraightAngles = false;
 
-             return tokens;
-           }
-         }], [{
-           key: "rules",
-           get: function get() {
-             return {
-               block: block$1,
-               inline: inline$1
-             };
+           if (vertexID !== undefined) {
+             allowStraightAngles = true;
+             nodes = nodeSubset(nodes, vertexID, isClosed);
+             if (nodes.length !== 3) return 'end_vertex';
            }
-           /**
-            * Static Lex Method
-            */
 
-         }, {
-           key: "lex",
-           value: function lex(src, options) {
-             var lexer = new Lexer(options);
-             return lexer.lex(src);
-           }
-           /**
-            * Static Lex Inline Method
-            */
+           var coords = nodes.map(function (n) {
+             return projection(n.loc);
+           });
+           var score = geoOrthoCanOrthogonalize(coords, isClosed, epsilon, threshold, allowStraightAngles);
 
-         }, {
-           key: "lexInline",
-           value: function lexInline(src, options) {
-             var lexer = new Lexer(options);
-             return lexer.inlineTokens(src);
+           if (score === null) {
+             return 'not_squarish';
+           } else if (score === 0) {
+             return 'square_enough';
+           } else {
+             return false;
            }
-         }]);
-
-         return Lexer;
-       }();
+         };
 
-       var defaults$3 = defaults.defaults;
-       var cleanUrl$1 = helpers.cleanUrl,
-           escape$2 = helpers.escape;
-       /**
-        * Renderer
-        */
+         action.transitionable = true;
+         return action;
+       }
 
-       var Renderer_1 = /*#__PURE__*/function () {
-         function Renderer(options) {
-           _classCallCheck(this, Renderer);
+       //
+       // `turn` must be an `osmTurn` object
+       // see osm/intersection.js, pathToTurn()
+       //
+       // This specifies a restriction of type `restriction` when traveling from
+       // `turn.from.way` toward `turn.to.way` via `turn.via.node` OR `turn.via.ways`.
+       // (The action does not check that these entities form a valid intersection.)
+       //
+       // From, to, and via ways should be split before calling this action.
+       // (old versions of the code would split the ways here, but we no longer do it)
+       //
+       // For testing convenience, accepts a restrictionID to assign to the new
+       // relation. Normally, this will be undefined and the relation will
+       // automatically be assigned a new ID.
+       //
 
-           this.options = options || defaults$3;
-         }
+       function actionRestrictTurn(turn, restrictionType, restrictionID) {
+         return function (graph) {
+           var fromWay = graph.entity(turn.from.way);
+           var toWay = graph.entity(turn.to.way);
+           var viaNode = turn.via.node && graph.entity(turn.via.node);
+           var viaWays = turn.via.ways && turn.via.ways.map(function (id) {
+             return graph.entity(id);
+           });
+           var members = [];
+           members.push({
+             id: fromWay.id,
+             type: 'way',
+             role: 'from'
+           });
 
-         _createClass(Renderer, [{
-           key: "code",
-           value: function code(_code, infostring, escaped) {
-             var lang = (infostring || '').match(/\S*/)[0];
+           if (viaNode) {
+             members.push({
+               id: viaNode.id,
+               type: 'node',
+               role: 'via'
+             });
+           } else if (viaWays) {
+             viaWays.forEach(function (viaWay) {
+               members.push({
+                 id: viaWay.id,
+                 type: 'way',
+                 role: 'via'
+               });
+             });
+           }
 
-             if (this.options.highlight) {
-               var out = this.options.highlight(_code, lang);
+           members.push({
+             id: toWay.id,
+             type: 'way',
+             role: 'to'
+           });
+           return graph.replace(osmRelation({
+             id: restrictionID,
+             tags: {
+               type: 'restriction',
+               restriction: restrictionType
+             },
+             members: members
+           }));
+         };
+       }
 
-               if (out != null && out !== _code) {
-                 escaped = true;
-                 _code = out;
-               }
-             }
+       function actionRevert(id) {
+         var action = function action(graph) {
+           var entity = graph.hasEntity(id),
+               base = graph.base().entities[id];
 
-             _code = _code.replace(/\n$/, '') + '\n';
+           if (entity && !base) {
+             // entity will be removed..
+             if (entity.type === 'node') {
+               graph.parentWays(entity).forEach(function (parent) {
+                 parent = parent.removeNode(id);
+                 graph = graph.replace(parent);
 
-             if (!lang) {
-               return '<pre><code>' + (escaped ? _code : escape$2(_code, true)) + '</code></pre>\n';
+                 if (parent.isDegenerate()) {
+                   graph = actionDeleteWay(parent.id)(graph);
+                 }
+               });
              }
 
-             return '<pre><code class="' + this.options.langPrefix + escape$2(lang, true) + '">' + (escaped ? _code : escape$2(_code, true)) + '</code></pre>\n';
-           }
-         }, {
-           key: "blockquote",
-           value: function blockquote(quote) {
-             return '<blockquote>\n' + quote + '</blockquote>\n';
-           }
-         }, {
-           key: "html",
-           value: function html(_html) {
-             return _html;
-           }
-         }, {
-           key: "heading",
-           value: function heading(text, level, raw, slugger) {
-             if (this.options.headerIds) {
-               return '<h' + level + ' id="' + this.options.headerPrefix + slugger.slug(raw) + '">' + text + '</h' + level + '>\n';
-             } // ignore IDs
-
-
-             return '<h' + level + '>' + text + '</h' + level + '>\n';
-           }
-         }, {
-           key: "hr",
-           value: function hr() {
-             return this.options.xhtml ? '<hr/>\n' : '<hr>\n';
-           }
-         }, {
-           key: "list",
-           value: function list(body, ordered, start) {
-             var type = ordered ? 'ol' : 'ul',
-                 startatt = ordered && start !== 1 ? ' start="' + start + '"' : '';
-             return '<' + type + startatt + '>\n' + body + '</' + type + '>\n';
-           }
-         }, {
-           key: "listitem",
-           value: function listitem(text) {
-             return '<li>' + text + '</li>\n';
-           }
-         }, {
-           key: "checkbox",
-           value: function checkbox(checked) {
-             return '<input ' + (checked ? 'checked="" ' : '') + 'disabled="" type="checkbox"' + (this.options.xhtml ? ' /' : '') + '> ';
-           }
-         }, {
-           key: "paragraph",
-           value: function paragraph(text) {
-             return '<p>' + text + '</p>\n';
-           }
-         }, {
-           key: "table",
-           value: function table(header, body) {
-             if (body) body = '<tbody>' + body + '</tbody>';
-             return '<table>\n' + '<thead>\n' + header + '</thead>\n' + body + '</table>\n';
-           }
-         }, {
-           key: "tablerow",
-           value: function tablerow(content) {
-             return '<tr>\n' + content + '</tr>\n';
-           }
-         }, {
-           key: "tablecell",
-           value: function tablecell(content, flags) {
-             var type = flags.header ? 'th' : 'td';
-             var tag = flags.align ? '<' + type + ' align="' + flags.align + '">' : '<' + type + '>';
-             return tag + content + '</' + type + '>\n';
-           } // span level renderer
+             graph.parentRelations(entity).forEach(function (parent) {
+               parent = parent.removeMembersWithID(id);
+               graph = graph.replace(parent);
 
-         }, {
-           key: "strong",
-           value: function strong(text) {
-             return '<strong>' + text + '</strong>';
-           }
-         }, {
-           key: "em",
-           value: function em(text) {
-             return '<em>' + text + '</em>';
-           }
-         }, {
-           key: "codespan",
-           value: function codespan(text) {
-             return '<code>' + text + '</code>';
-           }
-         }, {
-           key: "br",
-           value: function br() {
-             return this.options.xhtml ? '<br/>' : '<br>';
-           }
-         }, {
-           key: "del",
-           value: function del(text) {
-             return '<del>' + text + '</del>';
+               if (parent.isDegenerate()) {
+                 graph = actionDeleteRelation(parent.id)(graph);
+               }
+             });
            }
-         }, {
-           key: "link",
-           value: function link(href, title, text) {
-             href = cleanUrl$1(this.options.sanitize, this.options.baseUrl, href);
 
-             if (href === null) {
-               return text;
-             }
+           return graph.revert(id);
+         };
 
-             var out = '<a href="' + escape$2(href) + '"';
+         return action;
+       }
 
-             if (title) {
-               out += ' title="' + title + '"';
-             }
+       function actionRotate(rotateIds, pivot, angle, projection) {
+         var action = function action(graph) {
+           return graph.update(function (graph) {
+             utilGetAllNodes(rotateIds, graph).forEach(function (node) {
+               var point = geoRotate([projection(node.loc)], angle, pivot)[0];
+               graph = graph.replace(node.move(projection.invert(point)));
+             });
+           });
+         };
 
-             out += '>' + text + '</a>';
-             return out;
-           }
-         }, {
-           key: "image",
-           value: function image(href, title, text) {
-             href = cleanUrl$1(this.options.sanitize, this.options.baseUrl, href);
+         return action;
+       }
 
-             if (href === null) {
-               return text;
-             }
+       function actionScale(ids, pivotLoc, scaleFactor, projection) {
+         return function (graph) {
+           return graph.update(function (graph) {
+             var point, radial;
+             utilGetAllNodes(ids, graph).forEach(function (node) {
+               point = projection(node.loc);
+               radial = [point[0] - pivotLoc[0], point[1] - pivotLoc[1]];
+               point = [pivotLoc[0] + scaleFactor * radial[0], pivotLoc[1] + scaleFactor * radial[1]];
+               graph = graph.replace(node.move(projection.invert(point)));
+             });
+           });
+         };
+       }
 
-             var out = '<img src="' + href + '" alt="' + text + '"';
+       /* Align nodes along their common axis */
 
-             if (title) {
-               out += ' title="' + title + '"';
-             }
+       function actionStraightenNodes(nodeIDs, projection) {
+         function positionAlongWay(a, o, b) {
+           return geoVecDot(a, b, o) / geoVecDot(b, b, o);
+         } // returns the endpoints of the long axis of symmetry of the `points` bounding rect
 
-             out += this.options.xhtml ? '/>' : '>';
-             return out;
-           }
-         }, {
-           key: "text",
-           value: function text(_text) {
-             return _text;
-           }
-         }]);
 
-         return Renderer;
-       }();
+         function getEndpoints(points) {
+           var ssr = geoGetSmallestSurroundingRectangle(points); // Choose line pq = axis of symmetry.
+           // The shape's surrounding rectangle has 2 axes of symmetry.
+           // Snap points to the long axis
 
-       /**
-        * TextRenderer
-        * returns only the textual part of the token
-        */
-       var TextRenderer_1 = /*#__PURE__*/function () {
-         function TextRenderer() {
-           _classCallCheck(this, TextRenderer);
-         }
+           var p1 = [(ssr.poly[0][0] + ssr.poly[1][0]) / 2, (ssr.poly[0][1] + ssr.poly[1][1]) / 2];
+           var q1 = [(ssr.poly[2][0] + ssr.poly[3][0]) / 2, (ssr.poly[2][1] + ssr.poly[3][1]) / 2];
+           var p2 = [(ssr.poly[3][0] + ssr.poly[4][0]) / 2, (ssr.poly[3][1] + ssr.poly[4][1]) / 2];
+           var q2 = [(ssr.poly[1][0] + ssr.poly[2][0]) / 2, (ssr.poly[1][1] + ssr.poly[2][1]) / 2];
+           var isLong = geoVecLength(p1, q1) > geoVecLength(p2, q2);
 
-         _createClass(TextRenderer, [{
-           key: "strong",
-           value: // no need for block level renderers
-           function strong(text) {
-             return text;
-           }
-         }, {
-           key: "em",
-           value: function em(text) {
-             return text;
-           }
-         }, {
-           key: "codespan",
-           value: function codespan(text) {
-             return text;
-           }
-         }, {
-           key: "del",
-           value: function del(text) {
-             return text;
-           }
-         }, {
-           key: "html",
-           value: function html(text) {
-             return text;
-           }
-         }, {
-           key: "text",
-           value: function text(_text) {
-             return _text;
-           }
-         }, {
-           key: "link",
-           value: function link(href, title, text) {
-             return '' + text;
-           }
-         }, {
-           key: "image",
-           value: function image(href, title, text) {
-             return '' + text;
-           }
-         }, {
-           key: "br",
-           value: function br() {
-             return '';
+           if (isLong) {
+             return [p1, q1];
            }
-         }]);
-
-         return TextRenderer;
-       }();
-
-       /**
-        * Slugger generates header id
-        */
-       var Slugger_1 = /*#__PURE__*/function () {
-         function Slugger() {
-           _classCallCheck(this, Slugger);
 
-           this.seen = {};
+           return [p2, q2];
          }
 
-         _createClass(Slugger, [{
-           key: "serialize",
-           value: function serialize(value) {
-             return value.toLowerCase().trim() // remove html tags
-             .replace(/<[!\/a-z].*?>/ig, '') // remove unwanted chars
-             .replace(/[\u2000-\u206F\u2E00-\u2E7F\\'!"#$%&()*+,./:;<=>?@[\]^`{|}~]/g, '').replace(/\s/g, '-');
+         var action = function action(graph, t) {
+           if (t === null || !isFinite(t)) t = 1;
+           t = Math.min(Math.max(+t, 0), 1);
+           var nodes = nodeIDs.map(function (id) {
+             return graph.entity(id);
+           });
+           var points = nodes.map(function (n) {
+             return projection(n.loc);
+           });
+           var endpoints = getEndpoints(points);
+           var startPoint = endpoints[0];
+           var endPoint = endpoints[1]; // Move points onto the line connecting the endpoints
+
+           for (var i = 0; i < points.length; i++) {
+             var node = nodes[i];
+             var point = points[i];
+             var u = positionAlongWay(point, startPoint, endPoint);
+             var point2 = geoVecInterp(startPoint, endPoint, u);
+             var loc2 = projection.invert(point2);
+             graph = graph.replace(node.move(geoVecInterp(node.loc, loc2, t)));
            }
-           /**
-            * Finds the next safe (unique) slug to use
-            */
 
-         }, {
-           key: "getNextSafeSlug",
-           value: function getNextSafeSlug(originalSlug, isDryRun) {
-             var slug = originalSlug;
-             var occurenceAccumulator = 0;
+           return graph;
+         };
 
-             if (this.seen.hasOwnProperty(slug)) {
-               occurenceAccumulator = this.seen[originalSlug];
+         action.disabled = function (graph) {
+           var nodes = nodeIDs.map(function (id) {
+             return graph.entity(id);
+           });
+           var points = nodes.map(function (n) {
+             return projection(n.loc);
+           });
+           var endpoints = getEndpoints(points);
+           var startPoint = endpoints[0];
+           var endPoint = endpoints[1];
+           var maxDistance = 0;
 
-               do {
-                 occurenceAccumulator++;
-                 slug = originalSlug + '-' + occurenceAccumulator;
-               } while (this.seen.hasOwnProperty(slug));
-             }
+           for (var i = 0; i < points.length; i++) {
+             var point = points[i];
+             var u = positionAlongWay(point, startPoint, endPoint);
+             var p = geoVecInterp(startPoint, endPoint, u);
+             var dist = geoVecLength(p, point);
 
-             if (!isDryRun) {
-               this.seen[originalSlug] = occurenceAccumulator;
-               this.seen[slug] = 0;
+             if (!isNaN(dist) && dist > maxDistance) {
+               maxDistance = dist;
              }
-
-             return slug;
            }
-           /**
-            * Convert string to unique id
-            * @param {object} options
-            * @param {boolean} options.dryrun Generates the next unique slug without updating the internal accumulator.
-            */
 
-         }, {
-           key: "slug",
-           value: function slug(value) {
-             var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
-             var slug = this.serialize(value);
-             return this.getNextSafeSlug(slug, options.dryrun);
+           if (maxDistance < 0.0001) {
+             return 'straight_enough';
            }
-         }]);
+         };
 
-         return Slugger;
-       }();
+         action.transitionable = true;
+         return action;
+       }
 
-       var defaults$4 = defaults.defaults;
-       var unescape$2 = helpers.unescape;
-       /**
-        * Parsing & Compiling
+       /*
+        * Based on https://github.com/openstreetmap/potlatch2/net/systemeD/potlatch2/tools/Straighten.as
         */
 
-       var Parser_1 = /*#__PURE__*/function () {
-         function Parser(options) {
-           _classCallCheck(this, Parser);
+       function actionStraightenWay(selectedIDs, projection) {
+         function positionAlongWay(a, o, b) {
+           return geoVecDot(a, b, o) / geoVecDot(b, b, o);
+         } // Return all selected ways as a continuous, ordered array of nodes
 
-           this.options = options || defaults$4;
-           this.options.renderer = this.options.renderer || new Renderer_1();
-           this.renderer = this.options.renderer;
-           this.renderer.options = this.options;
-           this.textRenderer = new TextRenderer_1();
-           this.slugger = new Slugger_1();
-         }
-         /**
-          * Static Parse Method
-          */
 
+         function allNodes(graph) {
+           var nodes = [];
+           var startNodes = [];
+           var endNodes = [];
+           var remainingWays = [];
+           var selectedWays = selectedIDs.filter(function (w) {
+             return graph.entity(w).type === 'way';
+           });
+           var selectedNodes = selectedIDs.filter(function (n) {
+             return graph.entity(n).type === 'node';
+           });
+
+           for (var i = 0; i < selectedWays.length; i++) {
+             var way = graph.entity(selectedWays[i]);
+             nodes = way.nodes.slice(0);
+             remainingWays.push(nodes);
+             startNodes.push(nodes[0]);
+             endNodes.push(nodes[nodes.length - 1]);
+           } // Remove duplicate end/startNodes (duplicate nodes cannot be at the line end,
+           //   and need to be removed so currNode difference calculation below works)
+           // i.e. ["n-1", "n-1", "n-2"] => ["n-2"]
 
-         _createClass(Parser, [{
-           key: "parse",
-           value:
-           /**
-            * Parse Loop
-            */
-           function parse(tokens) {
-             var top = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : true;
-             var out = '',
-                 i,
-                 j,
-                 k,
-                 l2,
-                 l3,
-                 row,
-                 cell,
-                 header,
-                 body,
-                 token,
-                 ordered,
-                 start,
-                 loose,
-                 itemBody,
-                 item,
-                 checked,
-                 task,
-                 checkbox;
-             var l = tokens.length;
 
-             for (i = 0; i < l; i++) {
-               token = tokens[i];
+           startNodes = startNodes.filter(function (n) {
+             return startNodes.indexOf(n) === startNodes.lastIndexOf(n);
+           });
+           endNodes = endNodes.filter(function (n) {
+             return endNodes.indexOf(n) === endNodes.lastIndexOf(n);
+           }); // Choose the initial endpoint to start from
+
+           var currNode = utilArrayDifference(startNodes, endNodes).concat(utilArrayDifference(endNodes, startNodes))[0];
+           var nextWay = [];
+           nodes = []; // Create nested function outside of loop to avoid "function in loop" lint error
+
+           var getNextWay = function getNextWay(currNode, remainingWays) {
+             return remainingWays.filter(function (way) {
+               return way[0] === currNode || way[way.length - 1] === currNode;
+             })[0];
+           }; // Add nodes to end of nodes array, until all ways are added
+
 
-               switch (token.type) {
-                 case 'space':
-                   {
-                     continue;
-                   }
+           while (remainingWays.length) {
+             nextWay = getNextWay(currNode, remainingWays);
+             remainingWays = utilArrayDifference(remainingWays, [nextWay]);
 
-                 case 'hr':
-                   {
-                     out += this.renderer.hr();
-                     continue;
-                   }
+             if (nextWay[0] !== currNode) {
+               nextWay.reverse();
+             }
 
-                 case 'heading':
-                   {
-                     out += this.renderer.heading(this.parseInline(token.tokens), token.depth, unescape$2(this.parseInline(token.tokens, this.textRenderer)), this.slugger);
-                     continue;
-                   }
+             nodes = nodes.concat(nextWay);
+             currNode = nodes[nodes.length - 1];
+           } // If user selected 2 nodes to straighten between, then slice nodes array to those nodes
 
-                 case 'code':
-                   {
-                     out += this.renderer.code(token.text, token.lang, token.escaped);
-                     continue;
-                   }
 
-                 case 'table':
-                   {
-                     header = ''; // header
+           if (selectedNodes.length === 2) {
+             var startNodeIdx = nodes.indexOf(selectedNodes[0]);
+             var endNodeIdx = nodes.indexOf(selectedNodes[1]);
+             var sortedStartEnd = [startNodeIdx, endNodeIdx];
+             sortedStartEnd.sort(function (a, b) {
+               return a - b;
+             });
+             nodes = nodes.slice(sortedStartEnd[0], sortedStartEnd[1] + 1);
+           }
 
-                     cell = '';
-                     l2 = token.header.length;
+           return nodes.map(function (n) {
+             return graph.entity(n);
+           });
+         }
 
-                     for (j = 0; j < l2; j++) {
-                       cell += this.renderer.tablecell(this.parseInline(token.tokens.header[j]), {
-                         header: true,
-                         align: token.align[j]
-                       });
-                     }
+         function shouldKeepNode(node, graph) {
+           return graph.parentWays(node).length > 1 || graph.parentRelations(node).length || node.hasInterestingTags();
+         }
 
-                     header += this.renderer.tablerow(cell);
-                     body = '';
-                     l2 = token.cells.length;
+         var action = function action(graph, t) {
+           if (t === null || !isFinite(t)) t = 1;
+           t = Math.min(Math.max(+t, 0), 1);
+           var nodes = allNodes(graph);
+           var points = nodes.map(function (n) {
+             return projection(n.loc);
+           });
+           var startPoint = points[0];
+           var endPoint = points[points.length - 1];
+           var toDelete = [];
+           var i;
 
-                     for (j = 0; j < l2; j++) {
-                       row = token.tokens.cells[j];
-                       cell = '';
-                       l3 = row.length;
+           for (i = 1; i < points.length - 1; i++) {
+             var node = nodes[i];
+             var point = points[i];
 
-                       for (k = 0; k < l3; k++) {
-                         cell += this.renderer.tablecell(this.parseInline(row[k]), {
-                           header: false,
-                           align: token.align[k]
-                         });
-                       }
+             if (t < 1 || shouldKeepNode(node, graph)) {
+               var u = positionAlongWay(point, startPoint, endPoint);
+               var p = geoVecInterp(startPoint, endPoint, u);
+               var loc2 = projection.invert(p);
+               graph = graph.replace(node.move(geoVecInterp(node.loc, loc2, t)));
+             } else {
+               // safe to delete
+               if (toDelete.indexOf(node) === -1) {
+                 toDelete.push(node);
+               }
+             }
+           }
 
-                       body += this.renderer.tablerow(cell);
-                     }
+           for (i = 0; i < toDelete.length; i++) {
+             graph = actionDeleteNode(toDelete[i].id)(graph);
+           }
 
-                     out += this.renderer.table(header, body);
-                     continue;
-                   }
+           return graph;
+         };
 
-                 case 'blockquote':
-                   {
-                     body = this.parse(token.tokens);
-                     out += this.renderer.blockquote(body);
-                     continue;
-                   }
+         action.disabled = function (graph) {
+           // check way isn't too bendy
+           var nodes = allNodes(graph);
+           var points = nodes.map(function (n) {
+             return projection(n.loc);
+           });
+           var startPoint = points[0];
+           var endPoint = points[points.length - 1];
+           var threshold = 0.2 * geoVecLength(startPoint, endPoint);
+           var i;
 
-                 case 'list':
-                   {
-                     ordered = token.ordered;
-                     start = token.start;
-                     loose = token.loose;
-                     l2 = token.items.length;
-                     body = '';
+           if (threshold === 0) {
+             return 'too_bendy';
+           }
 
-                     for (j = 0; j < l2; j++) {
-                       item = token.items[j];
-                       checked = item.checked;
-                       task = item.task;
-                       itemBody = '';
+           var maxDistance = 0;
 
-                       if (item.task) {
-                         checkbox = this.renderer.checkbox(checked);
+           for (i = 1; i < points.length - 1; i++) {
+             var point = points[i];
+             var u = positionAlongWay(point, startPoint, endPoint);
+             var p = geoVecInterp(startPoint, endPoint, u);
+             var dist = geoVecLength(p, point); // to bendy if point is off by 20% of total start/end distance in projected space
 
-                         if (loose) {
-                           if (item.tokens.length > 0 && item.tokens[0].type === 'text') {
-                             item.tokens[0].text = checkbox + ' ' + item.tokens[0].text;
+             if (isNaN(dist) || dist > threshold) {
+               return 'too_bendy';
+             } else if (dist > maxDistance) {
+               maxDistance = dist;
+             }
+           }
 
-                             if (item.tokens[0].tokens && item.tokens[0].tokens.length > 0 && item.tokens[0].tokens[0].type === 'text') {
-                               item.tokens[0].tokens[0].text = checkbox + ' ' + item.tokens[0].tokens[0].text;
-                             }
-                           } else {
-                             item.tokens.unshift({
-                               type: 'text',
-                               text: checkbox
-                             });
-                           }
-                         } else {
-                           itemBody += checkbox;
-                         }
-                       }
+           var keepingAllNodes = nodes.every(function (node, i) {
+             return i === 0 || i === nodes.length - 1 || shouldKeepNode(node, graph);
+           });
 
-                       itemBody += this.parse(item.tokens, loose);
-                       body += this.renderer.listitem(itemBody, task, checked);
-                     }
+           if (maxDistance < 0.0001 && // Allow straightening even if already straight in order to remove extraneous nodes
+           keepingAllNodes) {
+             return 'straight_enough';
+           }
+         };
 
-                     out += this.renderer.list(body, ordered, start);
-                     continue;
-                   }
+         action.transitionable = true;
+         return action;
+       }
 
-                 case 'html':
-                   {
-                     // TODO parse inline content if parameter markdown=1
-                     out += this.renderer.html(token.text);
-                     continue;
-                   }
+       //
+       // `turn` must be an `osmTurn` object with a `restrictionID` property.
+       // see osm/intersection.js, pathToTurn()
+       //
 
-                 case 'paragraph':
-                   {
-                     out += this.renderer.paragraph(this.parseInline(token.tokens));
-                     continue;
-                   }
+       function actionUnrestrictTurn(turn) {
+         return function (graph) {
+           return actionDeleteRelation(turn.restrictionID)(graph);
+         };
+       }
 
-                 case 'text':
-                   {
-                     body = token.tokens ? this.parseInline(token.tokens) : token.text;
+       /* Reflect the given area around its axis of symmetry */
 
-                     while (i + 1 < l && tokens[i + 1].type === 'text') {
-                       token = tokens[++i];
-                       body += '\n' + (token.tokens ? this.parseInline(token.tokens) : token.text);
-                     }
+       function actionReflect(reflectIds, projection) {
+         var _useLongAxis = true;
 
-                     out += top ? this.renderer.paragraph(body) : body;
-                     continue;
-                   }
+         var action = function action(graph, t) {
+           if (t === null || !isFinite(t)) t = 1;
+           t = Math.min(Math.max(+t, 0), 1);
+           var nodes = utilGetAllNodes(reflectIds, graph);
+           var points = nodes.map(function (n) {
+             return projection(n.loc);
+           });
+           var ssr = geoGetSmallestSurroundingRectangle(points); // Choose line pq = axis of symmetry.
+           // The shape's surrounding rectangle has 2 axes of symmetry.
+           // Reflect across the longer axis by default.
 
-                 default:
-                   {
-                     var errMsg = 'Token with "' + token.type + '" type was not found.';
+           var p1 = [(ssr.poly[0][0] + ssr.poly[1][0]) / 2, (ssr.poly[0][1] + ssr.poly[1][1]) / 2];
+           var q1 = [(ssr.poly[2][0] + ssr.poly[3][0]) / 2, (ssr.poly[2][1] + ssr.poly[3][1]) / 2];
+           var p2 = [(ssr.poly[3][0] + ssr.poly[4][0]) / 2, (ssr.poly[3][1] + ssr.poly[4][1]) / 2];
+           var q2 = [(ssr.poly[1][0] + ssr.poly[2][0]) / 2, (ssr.poly[1][1] + ssr.poly[2][1]) / 2];
+           var p, q;
+           var isLong = geoVecLength(p1, q1) > geoVecLength(p2, q2);
 
-                     if (this.options.silent) {
-                       console.error(errMsg);
-                       return;
-                     } else {
-                       throw new Error(errMsg);
-                     }
-                   }
-               }
-             }
+           if (_useLongAxis && isLong || !_useLongAxis && !isLong) {
+             p = p1;
+             q = q1;
+           } else {
+             p = p2;
+             q = q2;
+           } // reflect c across pq
+           // http://math.stackexchange.com/questions/65503/point-reflection-over-a-line
 
-             return out;
-           }
-           /**
-            * Parse Inline Tokens
-            */
 
-         }, {
-           key: "parseInline",
-           value: function parseInline(tokens, renderer) {
-             renderer = renderer || this.renderer;
-             var out = '',
-                 i,
-                 token;
-             var l = tokens.length;
+           var dx = q[0] - p[0];
+           var dy = q[1] - p[1];
+           var a = (dx * dx - dy * dy) / (dx * dx + dy * dy);
+           var b = 2 * dx * dy / (dx * dx + dy * dy);
 
-             for (i = 0; i < l; i++) {
-               token = tokens[i];
+           for (var i = 0; i < nodes.length; i++) {
+             var node = nodes[i];
+             var c = projection(node.loc);
+             var c2 = [a * (c[0] - p[0]) + b * (c[1] - p[1]) + p[0], b * (c[0] - p[0]) - a * (c[1] - p[1]) + p[1]];
+             var loc2 = projection.invert(c2);
+             node = node.move(geoVecInterp(node.loc, loc2, t));
+             graph = graph.replace(node);
+           }
 
-               switch (token.type) {
-                 case 'escape':
-                   {
-                     out += renderer.text(token.text);
-                     break;
-                   }
+           return graph;
+         };
 
-                 case 'html':
-                   {
-                     out += renderer.html(token.text);
-                     break;
-                   }
+         action.useLongAxis = function (val) {
+           if (!arguments.length) return _useLongAxis;
+           _useLongAxis = val;
+           return action;
+         };
 
-                 case 'link':
-                   {
-                     out += renderer.link(token.href, token.title, this.parseInline(token.tokens, renderer));
-                     break;
-                   }
+         action.transitionable = true;
+         return action;
+       }
 
-                 case 'image':
-                   {
-                     out += renderer.image(token.href, token.title, token.text);
-                     break;
-                   }
+       function actionUpgradeTags(entityId, oldTags, replaceTags) {
+         return function (graph) {
+           var entity = graph.entity(entityId);
+           var tags = Object.assign({}, entity.tags); // shallow copy
 
-                 case 'strong':
-                   {
-                     out += renderer.strong(this.parseInline(token.tokens, renderer));
-                     break;
-                   }
+           var transferValue;
+           var semiIndex;
 
-                 case 'em':
-                   {
-                     out += renderer.em(this.parseInline(token.tokens, renderer));
-                     break;
-                   }
+           for (var oldTagKey in oldTags) {
+             if (!(oldTagKey in tags)) continue; // wildcard match
 
-                 case 'codespan':
-                   {
-                     out += renderer.codespan(token.text);
-                     break;
-                   }
+             if (oldTags[oldTagKey] === '*') {
+               // note the value since we might need to transfer it
+               transferValue = tags[oldTagKey];
+               delete tags[oldTagKey]; // exact match
+             } else if (oldTags[oldTagKey] === tags[oldTagKey]) {
+               delete tags[oldTagKey]; // match is within semicolon-delimited values
+             } else {
+               var vals = tags[oldTagKey].split(';').filter(Boolean);
+               var oldIndex = vals.indexOf(oldTags[oldTagKey]);
 
-                 case 'br':
-                   {
-                     out += renderer.br();
-                     break;
-                   }
+               if (vals.length === 1 || oldIndex === -1) {
+                 delete tags[oldTagKey];
+               } else {
+                 if (replaceTags && replaceTags[oldTagKey]) {
+                   // replacing a value within a semicolon-delimited value, note the index
+                   semiIndex = oldIndex;
+                 }
 
-                 case 'del':
-                   {
-                     out += renderer.del(this.parseInline(token.tokens, renderer));
-                     break;
-                   }
+                 vals.splice(oldIndex, 1);
+                 tags[oldTagKey] = vals.join(';');
+               }
+             }
+           }
 
-                 case 'text':
-                   {
-                     out += renderer.text(token.text);
-                     break;
-                   }
+           if (replaceTags) {
+             for (var replaceKey in replaceTags) {
+               var replaceValue = replaceTags[replaceKey];
 
-                 default:
-                   {
-                     var errMsg = 'Token with "' + token.type + '" type was not found.';
+               if (replaceValue === '*') {
+                 if (tags[replaceKey] && tags[replaceKey] !== 'no') {
+                   // allow any pre-existing value except `no` (troll tag)
+                   continue;
+                 } else {
+                   // otherwise assume `yes` is okay
+                   tags[replaceKey] = 'yes';
+                 }
+               } else if (replaceValue === '$1') {
+                 tags[replaceKey] = transferValue;
+               } else {
+                 if (tags[replaceKey] && oldTags[replaceKey] && semiIndex !== undefined) {
+                   // don't override preexisting values
+                   var existingVals = tags[replaceKey].split(';').filter(Boolean);
 
-                     if (this.options.silent) {
-                       console.error(errMsg);
-                       return;
-                     } else {
-                       throw new Error(errMsg);
-                     }
+                   if (existingVals.indexOf(replaceValue) === -1) {
+                     existingVals.splice(semiIndex, 0, replaceValue);
+                     tags[replaceKey] = existingVals.join(';');
                    }
+                 } else {
+                   tags[replaceKey] = replaceValue;
+                 }
                }
              }
-
-             return out;
-           }
-         }], [{
-           key: "parse",
-           value: function parse(tokens, options) {
-             var parser = new Parser(options);
-             return parser.parse(tokens);
            }
-           /**
-            * Static Parse Inline Method
-            */
 
-         }, {
-           key: "parseInline",
-           value: function parseInline(tokens, options) {
-             var parser = new Parser(options);
-             return parser.parseInline(tokens);
-           }
-         }]);
+           return graph.replace(entity.update({
+             tags: tags
+           }));
+         };
+       }
 
-         return Parser;
-       }();
+       function behaviorEdit(context) {
+         function behavior() {
+           context.map().minzoom(context.minEditableZoom());
+         }
 
-       var merge$3 = helpers.merge,
-           checkSanitizeDeprecation$1 = helpers.checkSanitizeDeprecation,
-           escape$3 = helpers.escape;
-       var getDefaults = defaults.getDefaults,
-           changeDefaults = defaults.changeDefaults,
-           defaults$5 = defaults.defaults;
-       /**
-        * Marked
+         behavior.off = function () {
+           context.map().minzoom(0);
+         };
+
+         return behavior;
+       }
+
+       /*
+          The hover behavior adds the `.hover` class on pointerover to all elements to which
+          the identical datum is bound, and removes it on pointerout.
+
+          The :hover pseudo-class is insufficient for iD's purposes because a datum's visual
+          representation may consist of several elements scattered throughout the DOM hierarchy.
+          Only one of these elements can have the :hover pseudo-class, but all of them will
+          have the .hover class.
         */
 
-       function marked(src, opt, callback) {
-         // throw error in case of non string input
-         if (typeof src === 'undefined' || src === null) {
-           throw new Error('marked(): input parameter is undefined or null');
-         }
+       function behaviorHover(context) {
+         var dispatch = dispatch$8('hover');
 
-         if (typeof src !== 'string') {
-           throw new Error('marked(): input parameter is of type ' + Object.prototype.toString.call(src) + ', string expected');
-         }
+         var _selection = select(null);
 
-         if (typeof opt === 'function') {
-           callback = opt;
-           opt = null;
-         }
+         var _newNodeId = null;
+         var _initialNodeID = null;
 
-         opt = merge$3({}, marked.defaults, opt || {});
-         checkSanitizeDeprecation$1(opt);
+         var _altDisables;
 
-         if (callback) {
-           var highlight = opt.highlight;
-           var tokens;
+         var _ignoreVertex;
 
-           try {
-             tokens = Lexer_1.lex(src, opt);
-           } catch (e) {
-             return callback(e);
+         var _targets = []; // use pointer events on supported platforms; fallback to mouse events
+
+         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
+
+         function keydown(d3_event) {
+           if (_altDisables && d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
+             _selection.selectAll('.hover').classed('hover-suppressed', true).classed('hover', false);
+
+             _selection.classed('hover-disabled', true);
+
+             dispatch.call('hover', this, null);
            }
+         }
 
-           var done = function done(err) {
-             var out;
+         function keyup(d3_event) {
+           if (_altDisables && d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
+             _selection.selectAll('.hover-suppressed').classed('hover-suppressed', false).classed('hover', true);
 
-             if (!err) {
-               try {
-                 out = Parser_1.parse(tokens, opt);
-               } catch (e) {
-                 err = e;
-               }
-             }
+             _selection.classed('hover-disabled', false);
 
-             opt.highlight = highlight;
-             return err ? callback(err) : callback(null, out);
-           };
+             dispatch.call('hover', this, _targets);
+           }
+         }
 
-           if (!highlight || highlight.length < 3) {
-             return done();
+         function behavior(selection) {
+           _selection = selection;
+           _targets = [];
+
+           if (_initialNodeID) {
+             _newNodeId = _initialNodeID;
+             _initialNodeID = null;
+           } else {
+             _newNodeId = null;
            }
 
-           delete opt.highlight;
-           if (!tokens.length) return done();
-           var pending = 0;
-           marked.walkTokens(tokens, function (token) {
-             if (token.type === 'code') {
-               pending++;
-               setTimeout(function () {
-                 highlight(token.text, token.lang, function (err, code) {
-                   if (err) {
-                     return done(err);
-                   }
+           _selection.on(_pointerPrefix + 'over.hover', pointerover).on(_pointerPrefix + 'out.hover', pointerout) // treat pointerdown as pointerover for touch devices
+           .on(_pointerPrefix + 'down.hover', pointerover);
 
-                   if (code != null && code !== token.text) {
-                     token.text = code;
-                     token.escaped = true;
-                   }
+           select(window).on(_pointerPrefix + 'up.hover pointercancel.hover', pointerout, true).on('keydown.hover', keydown).on('keyup.hover', keyup);
 
-                   pending--;
+           function eventTarget(d3_event) {
+             var datum = d3_event.target && d3_event.target.__data__;
+             if (_typeof(datum) !== 'object') return null;
 
-                   if (pending === 0) {
-                     done();
-                   }
-                 });
-               }, 0);
+             if (!(datum instanceof osmEntity) && datum.properties && datum.properties.entity instanceof osmEntity) {
+               return datum.properties.entity;
              }
-           });
 
-           if (pending === 0) {
-             done();
+             return datum;
            }
 
-           return;
-         }
-
-         try {
-           var _tokens = Lexer_1.lex(src, opt);
-
-           if (opt.walkTokens) {
-             marked.walkTokens(_tokens, opt.walkTokens);
-           }
+           function pointerover(d3_event) {
+             // ignore mouse hovers with buttons pressed unless dragging
+             if (context.mode().id.indexOf('drag') === -1 && (!d3_event.pointerType || d3_event.pointerType === 'mouse') && d3_event.buttons) return;
+             var target = eventTarget(d3_event);
 
-           return Parser_1.parse(_tokens, opt);
-         } catch (e) {
-           e.message += '\nPlease report this to https://github.com/markedjs/marked.';
+             if (target && _targets.indexOf(target) === -1) {
+               _targets.push(target);
 
-           if (opt.silent) {
-             return '<p>An error occurred:</p><pre>' + escape$3(e.message + '', true) + '</pre>';
+               updateHover(d3_event, _targets);
+             }
            }
 
-           throw e;
-         }
-       }
-       /**
-        * Options
-        */
+           function pointerout(d3_event) {
+             var target = eventTarget(d3_event);
 
+             var index = _targets.indexOf(target);
 
-       marked.options = marked.setOptions = function (opt) {
-         merge$3(marked.defaults, opt);
-         changeDefaults(marked.defaults);
-         return marked;
-       };
+             if (index !== -1) {
+               _targets.splice(index);
 
-       marked.getDefaults = getDefaults;
-       marked.defaults = defaults$5;
-       /**
-        * Use Extension
-        */
+               updateHover(d3_event, _targets);
+             }
+           }
 
-       marked.use = function (extension) {
-         var opts = merge$3({}, extension);
+           function allowsVertex(d) {
+             return d.geometry(context.graph()) === 'vertex' || _mainPresetIndex.allowsVertex(d, context.graph());
+           }
 
-         if (extension.renderer) {
-           (function () {
-             var renderer = marked.defaults.renderer || new Renderer_1();
+           function modeAllowsHover(target) {
+             var mode = context.mode();
 
-             var _loop = function _loop(prop) {
-               var prevRenderer = renderer[prop];
+             if (mode.id === 'add-point') {
+               return mode.preset.matchGeometry('vertex') || target.type !== 'way' && target.geometry(context.graph()) !== 'vertex';
+             }
 
-               renderer[prop] = function () {
-                 for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {
-                   args[_key] = arguments[_key];
-                 }
+             return true;
+           }
 
-                 var ret = extension.renderer[prop].apply(renderer, args);
+           function updateHover(d3_event, targets) {
+             _selection.selectAll('.hover').classed('hover', false);
 
-                 if (ret === false) {
-                   ret = prevRenderer.apply(renderer, args);
-                 }
+             _selection.selectAll('.hover-suppressed').classed('hover-suppressed', false);
 
-                 return ret;
-               };
-             };
+             var mode = context.mode();
 
-             for (var prop in extension.renderer) {
-               _loop(prop);
+             if (!_newNodeId && (mode.id === 'draw-line' || mode.id === 'draw-area')) {
+               var node = targets.find(function (target) {
+                 return target instanceof osmEntity && target.type === 'node';
+               });
+               _newNodeId = node && node.id;
              }
 
-             opts.renderer = renderer;
-           })();
-         }
-
-         if (extension.tokenizer) {
-           (function () {
-             var tokenizer = marked.defaults.tokenizer || new Tokenizer_1();
+             targets = targets.filter(function (datum) {
+               if (datum instanceof osmEntity) {
+                 // If drawing a way, don't hover on a node that was just placed. #3974
+                 return datum.id !== _newNodeId && (datum.type !== 'node' || !_ignoreVertex || allowsVertex(datum)) && modeAllowsHover(datum);
+               }
 
-             var _loop2 = function _loop2(prop) {
-               var prevTokenizer = tokenizer[prop];
+               return true;
+             });
+             var selector = '';
 
-               tokenizer[prop] = function () {
-                 for (var _len2 = arguments.length, args = new Array(_len2), _key2 = 0; _key2 < _len2; _key2++) {
-                   args[_key2] = arguments[_key2];
-                 }
+             for (var i in targets) {
+               var datum = targets[i]; // What are we hovering over?
 
-                 var ret = extension.tokenizer[prop].apply(tokenizer, args);
+               if (datum.__featurehash__) {
+                 // hovering custom data
+                 selector += ', .data' + datum.__featurehash__;
+               } else if (datum instanceof QAItem) {
+                 selector += ', .' + datum.service + '.itemId-' + datum.id;
+               } else if (datum instanceof osmNote) {
+                 selector += ', .note-' + datum.id;
+               } else if (datum instanceof osmEntity) {
+                 selector += ', .' + datum.id;
 
-                 if (ret === false) {
-                   ret = prevTokenizer.apply(tokenizer, args);
+                 if (datum.type === 'relation') {
+                   for (var j in datum.members) {
+                     selector += ', .' + datum.members[j].id;
+                   }
                  }
+               }
+             }
 
-                 return ret;
-               };
-             };
+             var suppressed = _altDisables && d3_event && d3_event.altKey;
 
-             for (var prop in extension.tokenizer) {
-               _loop2(prop);
+             if (selector.trim().length) {
+               // remove the first comma
+               selector = selector.slice(1);
+
+               _selection.selectAll(selector).classed(suppressed ? 'hover-suppressed' : 'hover', true);
              }
 
-             opts.tokenizer = tokenizer;
-           })();
+             dispatch.call('hover', this, !suppressed && targets);
+           }
          }
 
-         if (extension.walkTokens) {
-           var walkTokens = marked.defaults.walkTokens;
+         behavior.off = function (selection) {
+           selection.selectAll('.hover').classed('hover', false);
+           selection.selectAll('.hover-suppressed').classed('hover-suppressed', false);
+           selection.classed('hover-disabled', false);
+           selection.on(_pointerPrefix + 'over.hover', null).on(_pointerPrefix + 'out.hover', null).on(_pointerPrefix + 'down.hover', null);
+           select(window).on(_pointerPrefix + 'up.hover pointercancel.hover', null, true).on('keydown.hover', null).on('keyup.hover', null);
+         };
 
-           opts.walkTokens = function (token) {
-             extension.walkTokens(token);
+         behavior.altDisables = function (val) {
+           if (!arguments.length) return _altDisables;
+           _altDisables = val;
+           return behavior;
+         };
 
-             if (walkTokens) {
-               walkTokens(token);
-             }
-           };
-         }
+         behavior.ignoreVertex = function (val) {
+           if (!arguments.length) return _ignoreVertex;
+           _ignoreVertex = val;
+           return behavior;
+         };
 
-         marked.setOptions(opts);
-       };
-       /**
-        * Run callback for every token
-        */
+         behavior.initialNodeID = function (nodeId) {
+           _initialNodeID = nodeId;
+           return behavior;
+         };
 
+         return utilRebind(behavior, dispatch, 'on');
+       }
 
-       marked.walkTokens = function (tokens, callback) {
-         var _iterator = _createForOfIteratorHelper(tokens),
-             _step;
+       var _disableSpace = false;
+       var _lastSpace = null;
+       function behaviorDraw(context) {
+         var dispatch = dispatch$8('move', 'down', 'downcancel', 'click', 'clickWay', 'clickNode', 'undo', 'cancel', 'finish');
+         var keybinding = utilKeybinding('draw');
 
-         try {
-           for (_iterator.s(); !(_step = _iterator.n()).done;) {
-             var token = _step.value;
-             callback(token);
+         var _hover = behaviorHover(context).altDisables(true).ignoreVertex(true).on('hover', context.ui().sidebar.hover);
 
-             switch (token.type) {
-               case 'table':
-                 {
-                   var _iterator2 = _createForOfIteratorHelper(token.tokens.header),
-                       _step2;
+         var _edit = behaviorEdit(context);
 
-                   try {
-                     for (_iterator2.s(); !(_step2 = _iterator2.n()).done;) {
-                       var cell = _step2.value;
-                       marked.walkTokens(cell, callback);
-                     }
-                   } catch (err) {
-                     _iterator2.e(err);
-                   } finally {
-                     _iterator2.f();
-                   }
+         var _closeTolerance = 4;
+         var _tolerance = 12;
+         var _mouseLeave = false;
+         var _lastMouse = null;
 
-                   var _iterator3 = _createForOfIteratorHelper(token.tokens.cells),
-                       _step3;
+         var _lastPointerUpEvent;
 
-                   try {
-                     for (_iterator3.s(); !(_step3 = _iterator3.n()).done;) {
-                       var row = _step3.value;
+         var _downPointer; // use pointer events on supported platforms; fallback to mouse events
 
-                       var _iterator4 = _createForOfIteratorHelper(row),
-                           _step4;
 
-                       try {
-                         for (_iterator4.s(); !(_step4 = _iterator4.n()).done;) {
-                           var _cell = _step4.value;
-                           marked.walkTokens(_cell, callback);
-                         }
-                       } catch (err) {
-                         _iterator4.e(err);
-                       } finally {
-                         _iterator4.f();
-                       }
-                     }
-                   } catch (err) {
-                     _iterator3.e(err);
-                   } finally {
-                     _iterator3.f();
-                   }
+         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse'; // related code
+         // - `mode/drag_node.js` `datum()`
 
-                   break;
-                 }
 
-               case 'list':
-                 {
-                   marked.walkTokens(token.items, callback);
-                   break;
-                 }
+         function datum(d3_event) {
+           var mode = context.mode();
+           var isNote = mode && mode.id.indexOf('note') !== -1;
+           if (d3_event.altKey || isNote) return {};
+           var element;
 
-               default:
-                 {
-                   if (token.tokens) {
-                     marked.walkTokens(token.tokens, callback);
-                   }
-                 }
-             }
-           }
-         } catch (err) {
-           _iterator.e(err);
-         } finally {
-           _iterator.f();
-         }
-       };
-       /**
-        * Parse Inline
-        */
+           if (d3_event.type === 'keydown') {
+             element = _lastMouse && _lastMouse.target;
+           } else {
+             element = d3_event.target;
+           } // When drawing, snap only to touch targets..
+           // (this excludes area fills and active drawing elements)
 
 
-       marked.parseInline = function (src, opt) {
-         // throw error in case of non string input
-         if (typeof src === 'undefined' || src === null) {
-           throw new Error('marked.parseInline(): input parameter is undefined or null');
+           var d = element.__data__;
+           return d && d.properties && d.properties.target ? d : {};
          }
 
-         if (typeof src !== 'string') {
-           throw new Error('marked.parseInline(): input parameter is of type ' + Object.prototype.toString.call(src) + ', string expected');
+         function pointerdown(d3_event) {
+           if (_downPointer) return;
+           var pointerLocGetter = utilFastMouse(this);
+           _downPointer = {
+             id: d3_event.pointerId || 'mouse',
+             pointerLocGetter: pointerLocGetter,
+             downTime: +new Date(),
+             downLoc: pointerLocGetter(d3_event)
+           };
+           dispatch.call('down', this, d3_event, datum(d3_event));
          }
 
-         opt = merge$3({}, marked.defaults, opt || {});
-         checkSanitizeDeprecation$1(opt);
-
-         try {
-           var tokens = Lexer_1.lexInline(src, opt);
+         function pointerup(d3_event) {
+           if (!_downPointer || _downPointer.id !== (d3_event.pointerId || 'mouse')) return;
+           var downPointer = _downPointer;
+           _downPointer = null;
+           _lastPointerUpEvent = d3_event;
+           if (downPointer.isCancelled) return;
+           var t2 = +new Date();
+           var p2 = downPointer.pointerLocGetter(d3_event);
+           var dist = geoVecLength(downPointer.downLoc, p2);
 
-           if (opt.walkTokens) {
-             marked.walkTokens(tokens, opt.walkTokens);
+           if (dist < _closeTolerance || dist < _tolerance && t2 - downPointer.downTime < 500) {
+             // Prevent a quick second click
+             select(window).on('click.draw-block', function () {
+               d3_event.stopPropagation();
+             }, true);
+             context.map().dblclickZoomEnable(false);
+             window.setTimeout(function () {
+               context.map().dblclickZoomEnable(true);
+               select(window).on('click.draw-block', null);
+             }, 500);
+             click(d3_event, p2);
            }
+         }
 
-           return Parser_1.parseInline(tokens, opt);
-         } catch (e) {
-           e.message += '\nPlease report this to https://github.com/markedjs/marked.';
+         function pointermove(d3_event) {
+           if (_downPointer && _downPointer.id === (d3_event.pointerId || 'mouse') && !_downPointer.isCancelled) {
+             var p2 = _downPointer.pointerLocGetter(d3_event);
 
-           if (opt.silent) {
-             return '<p>An error occurred:</p><pre>' + escape$3(e.message + '', true) + '</pre>';
+             var dist = geoVecLength(_downPointer.downLoc, p2);
+
+             if (dist >= _closeTolerance) {
+               _downPointer.isCancelled = true;
+               dispatch.call('downcancel', this);
+             }
            }
 
-           throw e;
-         }
-       };
-       /**
-        * Expose
-        */
+           if (d3_event.pointerType && d3_event.pointerType !== 'mouse' || d3_event.buttons || _downPointer) return; // HACK: Mobile Safari likes to send one or more `mouse` type pointermove
+           // events immediately after non-mouse pointerup events; detect and ignore them.
 
+           if (_lastPointerUpEvent && _lastPointerUpEvent.pointerType !== 'mouse' && d3_event.timeStamp - _lastPointerUpEvent.timeStamp < 100) return;
+           _lastMouse = d3_event;
+           dispatch.call('move', this, d3_event, datum(d3_event));
+         }
 
-       marked.Parser = Parser_1;
-       marked.parser = Parser_1.parse;
-       marked.Renderer = Renderer_1;
-       marked.TextRenderer = TextRenderer_1;
-       marked.Lexer = Lexer_1;
-       marked.lexer = Lexer_1.lex;
-       marked.Tokenizer = Tokenizer_1;
-       marked.Slugger = Slugger_1;
-       marked.parse = marked;
-       var marked_1 = marked;
+         function pointercancel(d3_event) {
+           if (_downPointer && _downPointer.id === (d3_event.pointerId || 'mouse')) {
+             if (!_downPointer.isCancelled) {
+               dispatch.call('downcancel', this);
+             }
 
-       var tiler$2 = utilTiler();
-       var dispatch$3 = dispatch('loaded');
-       var _tileZoom$2 = 14;
-       var _osmoseUrlRoot = 'https://osmose.openstreetmap.fr/api/0.3';
-       var _osmoseData = {
-         icons: {},
-         items: []
-       }; // This gets reassigned if reset
+             _downPointer = null;
+           }
+         }
 
-       var _cache$2;
+         function mouseenter() {
+           _mouseLeave = false;
+         }
 
-       function abortRequest$2(controller) {
-         if (controller) {
-           controller.abort();
+         function mouseleave() {
+           _mouseLeave = true;
          }
-       }
 
-       function abortUnwantedRequests$2(cache, tiles) {
-         Object.keys(cache.inflightTile).forEach(function (k) {
-           var wanted = tiles.find(function (tile) {
-             return k === tile.id;
-           });
+         function allowsVertex(d) {
+           return d.geometry(context.graph()) === 'vertex' || _mainPresetIndex.allowsVertex(d, context.graph());
+         } // related code
+         // - `mode/drag_node.js`     `doMove()`
+         // - `behavior/draw.js`      `click()`
+         // - `behavior/draw_way.js`  `move()`
 
-           if (!wanted) {
-             abortRequest$2(cache.inflightTile[k]);
-             delete cache.inflightTile[k];
+
+         function click(d3_event, loc) {
+           var d = datum(d3_event);
+           var target = d && d.properties && d.properties.entity;
+           var mode = context.mode();
+
+           if (target && target.type === 'node' && allowsVertex(target)) {
+             // Snap to a node
+             dispatch.call('clickNode', this, target, d);
+             return;
+           } else if (target && target.type === 'way' && (mode.id !== 'add-point' || mode.preset.matchGeometry('vertex'))) {
+             // Snap to a way
+             var choice = geoChooseEdge(context.graph().childNodes(target), loc, context.projection, context.activeID());
+
+             if (choice) {
+               var edge = [target.nodes[choice.index - 1], target.nodes[choice.index]];
+               dispatch.call('clickWay', this, choice.loc, edge, d);
+               return;
+             }
+           } else if (mode.id !== 'add-point' || mode.preset.matchGeometry('point')) {
+             var locLatLng = context.projection.invert(loc);
+             dispatch.call('click', this, locLatLng, d);
            }
-         });
-       }
+         } // treat a spacebar press like a click
 
-       function encodeIssueRtree$2(d) {
-         return {
-           minX: d.loc[0],
-           minY: d.loc[1],
-           maxX: d.loc[0],
-           maxY: d.loc[1],
-           data: d
-         };
-       } // Replace or remove QAItem from rtree
 
+         function space(d3_event) {
+           d3_event.preventDefault();
+           d3_event.stopPropagation();
+           var currSpace = context.map().mouse();
 
-       function updateRtree$2(item, replace) {
-         _cache$2.rtree.remove(item, function (a, b) {
-           return a.data.id === b.data.id;
-         });
+           if (_disableSpace && _lastSpace) {
+             var dist = geoVecLength(_lastSpace, currSpace);
 
-         if (replace) {
-           _cache$2.rtree.insert(item);
-         }
-       } // Issues shouldn't obscure each other
+             if (dist > _tolerance) {
+               _disableSpace = false;
+             }
+           }
 
+           if (_disableSpace || _mouseLeave || !_lastMouse) return; // user must move mouse or release space bar to allow another click
 
-       function preventCoincident$1(loc) {
-         var coincident = false;
+           _lastSpace = currSpace;
+           _disableSpace = true;
+           select(window).on('keyup.space-block', function () {
+             d3_event.preventDefault();
+             d3_event.stopPropagation();
+             _disableSpace = false;
+             select(window).on('keyup.space-block', null);
+           }); // get the current mouse position
 
-         do {
-           // first time, move marker up. after that, move marker right.
-           var delta = coincident ? [0.00001, 0] : [0, 0.00001];
-           loc = geoVecAdd(loc, delta);
-           var bbox = geoExtent(loc).bbox();
-           coincident = _cache$2.rtree.search(bbox).length;
-         } while (coincident);
+           var loc = context.map().mouse() || // or the map center if the mouse has never entered the map
+           context.projection(context.map().center());
+           click(d3_event, loc);
+         }
 
-         return loc;
-       }
+         function backspace(d3_event) {
+           d3_event.preventDefault();
+           dispatch.call('undo');
+         }
 
-       var serviceOsmose = {
-         title: 'osmose',
-         init: function init() {
-           _mainFileFetcher.get('qa_data').then(function (d) {
-             _osmoseData = d.osmose;
-             _osmoseData.items = Object.keys(d.osmose.icons).map(function (s) {
-               return s.split('-')[0];
-             }).reduce(function (unique, item) {
-               return unique.indexOf(item) !== -1 ? unique : [].concat(_toConsumableArray(unique), [item]);
-             }, []);
-           });
+         function del(d3_event) {
+           d3_event.preventDefault();
+           dispatch.call('cancel');
+         }
 
-           if (!_cache$2) {
-             this.reset();
-           }
+         function ret(d3_event) {
+           d3_event.preventDefault();
+           dispatch.call('finish');
+         }
 
-           this.event = utilRebind(this, dispatch$3, 'on');
-         },
-         reset: function reset() {
-           var _strings = {};
-           var _colors = {};
+         function behavior(selection) {
+           context.install(_hover);
+           context.install(_edit);
+           _downPointer = null;
+           keybinding.on('⌫', backspace).on('⌦', del).on('⎋', ret).on('↩', ret).on('space', space).on('⌥space', space);
+           selection.on('mouseenter.draw', mouseenter).on('mouseleave.draw', mouseleave).on(_pointerPrefix + 'down.draw', pointerdown).on(_pointerPrefix + 'move.draw', pointermove);
+           select(window).on(_pointerPrefix + 'up.draw', pointerup, true).on('pointercancel.draw', pointercancel, true);
+           select(document).call(keybinding);
+           return behavior;
+         }
 
-           if (_cache$2) {
-             Object.values(_cache$2.inflightTile).forEach(abortRequest$2); // Strings and colors are static and should not be re-populated
+         behavior.off = function (selection) {
+           context.ui().sidebar.hover.cancel();
+           context.uninstall(_hover);
+           context.uninstall(_edit);
+           selection.on('mouseenter.draw', null).on('mouseleave.draw', null).on(_pointerPrefix + 'down.draw', null).on(_pointerPrefix + 'move.draw', null);
+           select(window).on(_pointerPrefix + 'up.draw', null).on('pointercancel.draw', null); // note: keyup.space-block, click.draw-block should remain
 
-             _strings = _cache$2.strings;
-             _colors = _cache$2.colors;
-           }
+           select(document).call(keybinding.unbind);
+         };
 
-           _cache$2 = {
-             data: {},
-             loadedTile: {},
-             inflightTile: {},
-             inflightPost: {},
-             closed: {},
-             rtree: new RBush(),
-             strings: _strings,
-             colors: _colors
-           };
-         },
-         loadIssues: function loadIssues(projection) {
-           var _this = this;
+         behavior.hover = function () {
+           return _hover;
+         };
 
-           var params = {
-             // Tiles return a maximum # of issues
-             // So we want to filter our request for only types iD supports
-             item: _osmoseData.items
-           }; // determine the needed tiles to cover the view
+         return utilRebind(behavior, dispatch, 'on');
+       }
 
-           var tiles = tiler$2.zoomExtent([_tileZoom$2, _tileZoom$2]).getTiles(projection); // abort inflight requests that are no longer needed
+       function initRange(domain, range) {
+         switch (arguments.length) {
+           case 0:
+             break;
 
-           abortUnwantedRequests$2(_cache$2, tiles); // issue new requests..
+           case 1:
+             this.range(domain);
+             break;
 
-           tiles.forEach(function (tile) {
-             if (_cache$2.loadedTile[tile.id] || _cache$2.inflightTile[tile.id]) return;
+           default:
+             this.range(range).domain(domain);
+             break;
+         }
 
-             var _tile$xyz = _slicedToArray(tile.xyz, 3),
-                 x = _tile$xyz[0],
-                 y = _tile$xyz[1],
-                 z = _tile$xyz[2];
+         return this;
+       }
 
-             var url = "".concat(_osmoseUrlRoot, "/issues/").concat(z, "/").concat(x, "/").concat(y, ".json?") + utilQsString(params);
-             var controller = new AbortController();
-             _cache$2.inflightTile[tile.id] = controller;
-             d3_json(url, {
-               signal: controller.signal
-             }).then(function (data) {
-               delete _cache$2.inflightTile[tile.id];
-               _cache$2.loadedTile[tile.id] = true;
+       function constants(x) {
+         return function () {
+           return x;
+         };
+       }
 
-               if (data.features) {
-                 data.features.forEach(function (issue) {
-                   var _issue$properties = issue.properties,
-                       item = _issue$properties.item,
-                       cl = _issue$properties["class"],
-                       id = _issue$properties.uuid;
-                   /* Osmose issues are uniquely identified by a unique
-                     `item` and `class` combination (both integer values) */
+       function number(x) {
+         return +x;
+       }
 
-                   var itemType = "".concat(item, "-").concat(cl); // Filter out unsupported issue types (some are too specific or advanced)
+       var unit = [0, 1];
+       function identity$1(x) {
+         return x;
+       }
 
-                   if (itemType in _osmoseData.icons) {
-                     var loc = issue.geometry.coordinates; // lon, lat
+       function normalize(a, b) {
+         return (b -= a = +a) ? function (x) {
+           return (x - a) / b;
+         } : constants(isNaN(b) ? NaN : 0.5);
+       }
 
-                     loc = preventCoincident$1(loc);
-                     var d = new QAItem(loc, _this, itemType, id, {
-                       item: item
-                     }); // Setting elems here prevents UI detail requests
+       function clamper(a, b) {
+         var t;
+         if (a > b) t = a, a = b, b = t;
+         return function (x) {
+           return Math.max(a, Math.min(b, x));
+         };
+       } // normalize(a, b)(x) takes a domain value x in [a,b] and returns the corresponding parameter t in [0,1].
+       // interpolate(a, b)(t) takes a parameter t in [0,1] and returns the corresponding range value x in [a,b].
 
-                     if (item === 8300 || item === 8360) {
-                       d.elems = [];
-                     }
 
-                     _cache$2.data[d.id] = d;
+       function bimap(domain, range, interpolate) {
+         var d0 = domain[0],
+             d1 = domain[1],
+             r0 = range[0],
+             r1 = range[1];
+         if (d1 < d0) d0 = normalize(d1, d0), r0 = interpolate(r1, r0);else d0 = normalize(d0, d1), r0 = interpolate(r0, r1);
+         return function (x) {
+           return r0(d0(x));
+         };
+       }
 
-                     _cache$2.rtree.insert(encodeIssueRtree$2(d));
-                   }
-                 });
-               }
+       function polymap(domain, range, interpolate) {
+         var j = Math.min(domain.length, range.length) - 1,
+             d = new Array(j),
+             r = new Array(j),
+             i = -1; // Reverse descending domains.
 
-               dispatch$3.call('loaded');
-             })["catch"](function () {
-               delete _cache$2.inflightTile[tile.id];
-               _cache$2.loadedTile[tile.id] = true;
-             });
-           });
-         },
-         loadIssueDetail: function loadIssueDetail(issue) {
-           var _this2 = this;
+         if (domain[j] < domain[0]) {
+           domain = domain.slice().reverse();
+           range = range.slice().reverse();
+         }
 
-           // Issue details only need to be fetched once
-           if (issue.elems !== undefined) {
-             return Promise.resolve(issue);
-           }
+         while (++i < j) {
+           d[i] = normalize(domain[i], domain[i + 1]);
+           r[i] = interpolate(range[i], range[i + 1]);
+         }
 
-           var url = "".concat(_osmoseUrlRoot, "/issue/").concat(issue.id, "?langs=").concat(_mainLocalizer.localeCode());
+         return function (x) {
+           var i = bisectRight(domain, x, 1, j) - 1;
+           return r[i](d[i](x));
+         };
+       }
 
-           var cacheDetails = function cacheDetails(data) {
-             // Associated elements used for highlighting
-             // Assign directly for immediate use in the callback
-             issue.elems = data.elems.map(function (e) {
-               return e.type.substring(0, 1) + e.id;
-             }); // Some issues have instance specific detail in a subtitle
+       function copy$1(source, target) {
+         return target.domain(source.domain()).range(source.range()).interpolate(source.interpolate()).clamp(source.clamp()).unknown(source.unknown());
+       }
+       function transformer() {
+         var domain = unit,
+             range = unit,
+             interpolate = interpolate$1,
+             transform,
+             untransform,
+             unknown,
+             clamp = identity$1,
+             piecewise,
+             output,
+             input;
 
-             issue.detail = data.subtitle ? marked_1(data.subtitle.auto) : '';
+         function rescale() {
+           var n = Math.min(domain.length, range.length);
+           if (clamp !== identity$1) clamp = clamper(domain[0], domain[n - 1]);
+           piecewise = n > 2 ? polymap : bimap;
+           output = input = null;
+           return scale;
+         }
 
-             _this2.replaceItem(issue);
-           };
+         function scale(x) {
+           return x == null || isNaN(x = +x) ? unknown : (output || (output = piecewise(domain.map(transform), range, interpolate)))(transform(clamp(x)));
+         }
 
-           return d3_json(url).then(cacheDetails).then(function () {
-             return issue;
-           });
-         },
-         loadStrings: function loadStrings() {
-           var locale = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : _mainLocalizer.localeCode();
-           var items = Object.keys(_osmoseData.icons);
+         scale.invert = function (y) {
+           return clamp(untransform((input || (input = piecewise(range, domain.map(transform), d3_interpolateNumber)))(y)));
+         };
 
-           if (locale in _cache$2.strings && Object.keys(_cache$2.strings[locale]).length === items.length) {
-             return Promise.resolve(_cache$2.strings[locale]);
-           } // May be partially populated already if some requests were successful
+         scale.domain = function (_) {
+           return arguments.length ? (domain = Array.from(_, number), rescale()) : domain.slice();
+         };
 
+         scale.range = function (_) {
+           return arguments.length ? (range = Array.from(_), rescale()) : range.slice();
+         };
 
-           if (!(locale in _cache$2.strings)) {
-             _cache$2.strings[locale] = {};
-           } // Only need to cache strings for supported issue types
-           // Using multiple individual item + class requests to reduce fetched data size
+         scale.rangeRound = function (_) {
+           return range = Array.from(_), interpolate = interpolateRound, rescale();
+         };
 
+         scale.clamp = function (_) {
+           return arguments.length ? (clamp = _ ? true : identity$1, rescale()) : clamp !== identity$1;
+         };
 
-           var allRequests = items.map(function (itemType) {
-             // No need to request data we already have
-             if (itemType in _cache$2.strings[locale]) return null;
+         scale.interpolate = function (_) {
+           return arguments.length ? (interpolate = _, rescale()) : interpolate;
+         };
 
-             var cacheData = function cacheData(data) {
-               // Bunch of nested single value arrays of objects
-               var _data$categories = _slicedToArray(data.categories, 1),
-                   _data$categories$ = _data$categories[0],
-                   cat = _data$categories$ === void 0 ? {
-                 items: []
-               } : _data$categories$;
+         scale.unknown = function (_) {
+           return arguments.length ? (unknown = _, scale) : unknown;
+         };
 
-               var _cat$items = _slicedToArray(cat.items, 1),
-                   _cat$items$ = _cat$items[0],
-                   item = _cat$items$ === void 0 ? {
-                 "class": []
-               } : _cat$items$;
+         return function (t, u) {
+           transform = t, untransform = u;
+           return rescale();
+         };
+       }
+       function continuous() {
+         return transformer()(identity$1, identity$1);
+       }
 
-               var _item$class = _slicedToArray(item["class"], 1),
-                   _item$class$ = _item$class[0],
-                   cl = _item$class$ === void 0 ? null : _item$class$; // If null default value is reached, data wasn't as expected (or was empty)
+       function formatDecimal (x) {
+         return Math.abs(x = Math.round(x)) >= 1e21 ? x.toLocaleString("en").replace(/,/g, "") : x.toString(10);
+       } // Computes the decimal coefficient and exponent of the specified number x with
+       // significant digits p, where x is positive and p is in [1, 21] or undefined.
+       // For example, formatDecimalParts(1.23) returns ["123", 0].
 
+       function formatDecimalParts(x, p) {
+         if ((i = (x = p ? x.toExponential(p - 1) : x.toExponential()).indexOf("e")) < 0) return null; // NaN, ±Infinity
 
-               if (!cl) {
-                 /* eslint-disable no-console */
-                 console.log("Osmose strings request (".concat(itemType, ") had unexpected data"));
-                 /* eslint-enable no-console */
+         var i,
+             coefficient = x.slice(0, i); // The string returned by toExponential either has the form \d\.\d+e[-+]\d+
+         // (e.g., 1.2e+3) or the form \de[-+]\d+ (e.g., 1e+3).
 
-                 return;
-               } // Cache served item colors to automatically style issue markers later
+         return [coefficient.length > 1 ? coefficient[0] + coefficient.slice(2) : coefficient, +x.slice(i + 1)];
+       }
 
+       function exponent (x) {
+         return x = formatDecimalParts(Math.abs(x)), x ? x[1] : NaN;
+       }
 
-               var itemInt = item.item,
-                   color = item.color;
+       function formatGroup (grouping, thousands) {
+         return function (value, width) {
+           var i = value.length,
+               t = [],
+               j = 0,
+               g = grouping[0],
+               length = 0;
 
-               if (/^#[A-Fa-f0-9]{6}|[A-Fa-f0-9]{3}/.test(color)) {
-                 _cache$2.colors[itemInt] = color;
-               } // Value of root key will be null if no string exists
-               // If string exists, value is an object with key 'auto' for string
+           while (i > 0 && g > 0) {
+             if (length + g + 1 > width) g = Math.max(1, width - length);
+             t.push(value.substring(i -= g, i + g));
+             if ((length += g + 1) > width) break;
+             g = grouping[j = (j + 1) % grouping.length];
+           }
 
+           return t.reverse().join(thousands);
+         };
+       }
 
-               var title = cl.title,
-                   detail = cl.detail,
-                   fix = cl.fix,
-                   trap = cl.trap; // Osmose titles shouldn't contain markdown
+       function formatNumerals (numerals) {
+         return function (value) {
+           return value.replace(/[0-9]/g, function (i) {
+             return numerals[+i];
+           });
+         };
+       }
 
-               var issueStrings = {};
-               if (title) issueStrings.title = title.auto;
-               if (detail) issueStrings.detail = marked_1(detail.auto);
-               if (trap) issueStrings.trap = marked_1(trap.auto);
-               if (fix) issueStrings.fix = marked_1(fix.auto);
-               _cache$2.strings[locale][itemType] = issueStrings;
-             };
+       // [[fill]align][sign][symbol][0][width][,][.precision][~][type]
+       var re = /^(?:(.)?([<>=^]))?([+\-( ])?([$#])?(0)?(\d+)?(,)?(\.\d+)?(~)?([a-z%])?$/i;
+       function formatSpecifier(specifier) {
+         if (!(match = re.exec(specifier))) throw new Error("invalid format: " + specifier);
+         var match;
+         return new FormatSpecifier({
+           fill: match[1],
+           align: match[2],
+           sign: match[3],
+           symbol: match[4],
+           zero: match[5],
+           width: match[6],
+           comma: match[7],
+           precision: match[8] && match[8].slice(1),
+           trim: match[9],
+           type: match[10]
+         });
+       }
+       formatSpecifier.prototype = FormatSpecifier.prototype; // instanceof
 
-             var _itemType$split = itemType.split('-'),
-                 _itemType$split2 = _slicedToArray(_itemType$split, 2),
-                 item = _itemType$split2[0],
-                 cl = _itemType$split2[1]; // Osmose API falls back to English strings where untranslated or if locale doesn't exist
+       function FormatSpecifier(specifier) {
+         this.fill = specifier.fill === undefined ? " " : specifier.fill + "";
+         this.align = specifier.align === undefined ? ">" : specifier.align + "";
+         this.sign = specifier.sign === undefined ? "-" : specifier.sign + "";
+         this.symbol = specifier.symbol === undefined ? "" : specifier.symbol + "";
+         this.zero = !!specifier.zero;
+         this.width = specifier.width === undefined ? undefined : +specifier.width;
+         this.comma = !!specifier.comma;
+         this.precision = specifier.precision === undefined ? undefined : +specifier.precision;
+         this.trim = !!specifier.trim;
+         this.type = specifier.type === undefined ? "" : specifier.type + "";
+       }
 
+       FormatSpecifier.prototype.toString = function () {
+         return this.fill + this.align + this.sign + this.symbol + (this.zero ? "0" : "") + (this.width === undefined ? "" : Math.max(1, this.width | 0)) + (this.comma ? "," : "") + (this.precision === undefined ? "" : "." + Math.max(0, this.precision | 0)) + (this.trim ? "~" : "") + this.type;
+       };
 
-             var url = "".concat(_osmoseUrlRoot, "/items/").concat(item, "/class/").concat(cl, "?langs=").concat(locale);
-             return d3_json(url).then(cacheData);
-           }).filter(Boolean);
-           return Promise.all(allRequests).then(function () {
-             return _cache$2.strings[locale];
-           });
-         },
-         getStrings: function getStrings(itemType) {
-           var locale = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : _mainLocalizer.localeCode();
-           // No need to fallback to English, Osmose API handles this for us
-           return locale in _cache$2.strings ? _cache$2.strings[locale][itemType] : {};
-         },
-         getColor: function getColor(itemType) {
-           return itemType in _cache$2.colors ? _cache$2.colors[itemType] : '#FFFFFF';
-         },
-         postUpdate: function postUpdate(issue, callback) {
-           var _this3 = this;
+       // Trims insignificant zeros, e.g., replaces 1.2000k with 1.2k.
+       function formatTrim (s) {
+         out: for (var n = s.length, i = 1, i0 = -1, i1; i < n; ++i) {
+           switch (s[i]) {
+             case ".":
+               i0 = i1 = i;
+               break;
 
-           if (_cache$2.inflightPost[issue.id]) {
-             return callback({
-               message: 'Issue update already inflight',
-               status: -2
-             }, issue);
-           } // UI sets the status to either 'done' or 'false'
+             case "0":
+               if (i0 === 0) i0 = i;
+               i1 = i;
+               break;
 
+             default:
+               if (!+s[i]) break out;
+               if (i0 > 0) i0 = 0;
+               break;
+           }
+         }
 
-           var url = "".concat(_osmoseUrlRoot, "/issue/").concat(issue.id, "/").concat(issue.newStatus);
-           var controller = new AbortController();
+         return i0 > 0 ? s.slice(0, i0) + s.slice(i1 + 1) : s;
+       }
 
-           var after = function after() {
-             delete _cache$2.inflightPost[issue.id];
+       var nativeToPrecision = 1.0.toPrecision;
 
-             _this3.removeItem(issue);
+       var FORCED$1 = fails(function () {
+         // IE7-
+         return nativeToPrecision.call(1, undefined) !== '1';
+       }) || !fails(function () {
+         // V8 ~ Android 4.3-
+         nativeToPrecision.call({});
+       });
 
-             if (issue.newStatus === 'done') {
-               // Keep track of the number of issues closed per `item` to tag the changeset
-               if (!(issue.item in _cache$2.closed)) {
-                 _cache$2.closed[issue.item] = 0;
-               }
+       // `Number.prototype.toPrecision` method
+       // https://tc39.es/ecma262/#sec-number.prototype.toprecision
+       _export({ target: 'Number', proto: true, forced: FORCED$1 }, {
+         toPrecision: function toPrecision(precision) {
+           return precision === undefined
+             ? nativeToPrecision.call(thisNumberValue(this))
+             : nativeToPrecision.call(thisNumberValue(this), precision);
+         }
+       });
 
-               _cache$2.closed[issue.item] += 1;
-             }
+       var prefixExponent;
+       function formatPrefixAuto (x, p) {
+         var d = formatDecimalParts(x, p);
+         if (!d) return x + "";
+         var coefficient = d[0],
+             exponent = d[1],
+             i = exponent - (prefixExponent = Math.max(-8, Math.min(8, Math.floor(exponent / 3))) * 3) + 1,
+             n = coefficient.length;
+         return i === n ? coefficient : i > n ? coefficient + new Array(i - n + 1).join("0") : i > 0 ? coefficient.slice(0, i) + "." + coefficient.slice(i) : "0." + new Array(1 - i).join("0") + formatDecimalParts(x, Math.max(0, p + i - 1))[0]; // less than 1y!
+       }
 
-             if (callback) callback(null, issue);
-           };
+       function formatRounded (x, p) {
+         var d = formatDecimalParts(x, p);
+         if (!d) return x + "";
+         var coefficient = d[0],
+             exponent = d[1];
+         return exponent < 0 ? "0." + new Array(-exponent).join("0") + coefficient : coefficient.length > exponent + 1 ? coefficient.slice(0, exponent + 1) + "." + coefficient.slice(exponent + 1) : coefficient + new Array(exponent - coefficient.length + 2).join("0");
+       }
 
-           _cache$2.inflightPost[issue.id] = controller;
-           fetch(url, {
-             signal: controller.signal
-           }).then(after)["catch"](function (err) {
-             delete _cache$2.inflightPost[issue.id];
-             if (callback) callback(err.message);
-           });
+       var formatTypes = {
+         "%": function _(x, p) {
+           return (x * 100).toFixed(p);
          },
-         // Get all cached QAItems covering the viewport
-         getItems: function getItems(projection) {
-           var viewport = projection.clipExtent();
-           var min = [viewport[0][0], viewport[1][1]];
-           var max = [viewport[1][0], viewport[0][1]];
-           var bbox = geoExtent(projection.invert(min), projection.invert(max)).bbox();
-           return _cache$2.rtree.search(bbox).map(function (d) {
-             return d.data;
-           });
+         "b": function b(x) {
+           return Math.round(x).toString(2);
          },
-         // Get a QAItem from cache
-         // NOTE: Don't change method name until UI v3 is merged
-         getError: function getError(id) {
-           return _cache$2.data[id];
+         "c": function c(x) {
+           return x + "";
          },
-         // get the name of the icon to display for this item
-         getIcon: function getIcon(itemType) {
-           return _osmoseData.icons[itemType];
+         "d": formatDecimal,
+         "e": function e(x, p) {
+           return x.toExponential(p);
          },
-         // Replace a single QAItem in the cache
-         replaceItem: function replaceItem(item) {
-           if (!(item instanceof QAItem) || !item.id) return;
-           _cache$2.data[item.id] = item;
-           updateRtree$2(encodeIssueRtree$2(item), true); // true = replace
-
-           return item;
+         "f": function f(x, p) {
+           return x.toFixed(p);
          },
-         // Remove a single QAItem from the cache
-         removeItem: function removeItem(item) {
-           if (!(item instanceof QAItem) || !item.id) return;
-           delete _cache$2.data[item.id];
-           updateRtree$2(encodeIssueRtree$2(item), false); // false = remove
+         "g": function g(x, p) {
+           return x.toPrecision(p);
          },
-         // Used to populate `closed:osmose:*` changeset tags
-         getClosedCounts: function getClosedCounts() {
-           return _cache$2.closed;
+         "o": function o(x) {
+           return Math.round(x).toString(8);
          },
-         itemURL: function itemURL(item) {
-           return "https://osmose.openstreetmap.fr/en/error/".concat(item.id);
+         "p": function p(x, _p) {
+           return formatRounded(x * 100, _p);
+         },
+         "r": formatRounded,
+         "s": formatPrefixAuto,
+         "X": function X(x) {
+           return Math.round(x).toString(16).toUpperCase();
+         },
+         "x": function x(_x) {
+           return Math.round(_x).toString(16);
          }
        };
 
-       var apibase = 'https://a.mapillary.com/v3/';
-       var viewercss = 'mapillary-js/mapillary.min.css';
-       var viewerjs = 'mapillary-js/mapillary.min.js';
-       var clientId = 'NzNRM2otQkR2SHJzaXJmNmdQWVQ0dzo1ZWYyMmYwNjdmNDdlNmVi';
-       var mapFeatureConfig = {
-         values: ['construction--flat--crosswalk-plain', 'marking--discrete--crosswalk-zebra', 'object--banner', 'object--bench', 'object--bike-rack', 'object--billboard', 'object--catch-basin', 'object--cctv-camera', 'object--fire-hydrant', 'object--mailbox', 'object--manhole', 'object--phone-booth', 'object--sign--advertisement', 'object--sign--information', 'object--sign--store', 'object--street-light', 'object--support--utility-pole', 'object--traffic-light--*', 'object--traffic-light--pedestrians', 'object--trash-can'].join(',')
-       };
-       var maxResults = 1000;
-       var tileZoom = 14;
-       var tiler$3 = utilTiler().zoomExtent([tileZoom, tileZoom]).skipNullIsland(true);
-       var dispatch$4 = dispatch('change', 'loadedImages', 'loadedSigns', 'loadedMapFeatures', 'bearingChanged', 'nodeChanged');
-       var _mlyFallback = false;
+       function identity (x) {
+         return x;
+       }
 
-       var _mlyCache;
+       var map$1 = Array.prototype.map,
+           prefixes = ["y", "z", "a", "f", "p", "n", "µ", "m", "", "k", "M", "G", "T", "P", "E", "Z", "Y"];
+       function formatLocale (locale) {
+         var group = locale.grouping === undefined || locale.thousands === undefined ? identity : formatGroup(map$1.call(locale.grouping, Number), locale.thousands + ""),
+             currencyPrefix = locale.currency === undefined ? "" : locale.currency[0] + "",
+             currencySuffix = locale.currency === undefined ? "" : locale.currency[1] + "",
+             decimal = locale.decimal === undefined ? "." : locale.decimal + "",
+             numerals = locale.numerals === undefined ? identity : formatNumerals(map$1.call(locale.numerals, String)),
+             percent = locale.percent === undefined ? "%" : locale.percent + "",
+             minus = locale.minus === undefined ? "−" : locale.minus + "",
+             nan = locale.nan === undefined ? "NaN" : locale.nan + "";
+
+         function newFormat(specifier) {
+           specifier = formatSpecifier(specifier);
+           var fill = specifier.fill,
+               align = specifier.align,
+               sign = specifier.sign,
+               symbol = specifier.symbol,
+               zero = specifier.zero,
+               width = specifier.width,
+               comma = specifier.comma,
+               precision = specifier.precision,
+               trim = specifier.trim,
+               type = specifier.type; // The "n" type is an alias for ",g".
 
-       var _mlyClicks;
+           if (type === "n") comma = true, type = "g"; // The "" type, and any invalid type, is an alias for ".12~g".
+           else if (!formatTypes[type]) precision === undefined && (precision = 12), trim = true, type = "g"; // If zero fill is specified, padding goes after sign and before digits.
 
-       var _mlyActiveImage;
+           if (zero || fill === "0" && align === "=") zero = true, fill = "0", align = "="; // Compute the prefix and suffix.
+           // For SI-prefix, the suffix is lazily computed.
 
-       var _mlySelectedImageKey;
+           var prefix = symbol === "$" ? currencyPrefix : symbol === "#" && /[boxX]/.test(type) ? "0" + type.toLowerCase() : "",
+               suffix = symbol === "$" ? currencySuffix : /[%p]/.test(type) ? percent : ""; // What format function should we use?
+           // Is this an integer type?
+           // Can this type generate exponential notation?
 
-       var _mlyViewer;
+           var formatType = formatTypes[type],
+               maybeSuffix = /[defgprs%]/.test(type); // Set the default precision if not specified,
+           // or clamp the specified precision to the supported range.
+           // For significant precision, it must be in [1, 21].
+           // For fixed precision, it must be in [0, 20].
+
+           precision = precision === undefined ? 6 : /[gprs]/.test(type) ? Math.max(1, Math.min(21, precision)) : Math.max(0, Math.min(20, precision));
+
+           function format(value) {
+             var valuePrefix = prefix,
+                 valueSuffix = suffix,
+                 i,
+                 n,
+                 c;
 
-       var _mlyViewerFilter = ['all'];
+             if (type === "c") {
+               valueSuffix = formatType(value) + valueSuffix;
+               value = "";
+             } else {
+               value = +value; // Determine the sign. -0 is not less than 0, but 1 / -0 is!
 
-       var _loadViewerPromise;
+               var valueNegative = value < 0 || 1 / value < 0; // Perform the initial formatting.
 
-       var _mlyHighlightedDetection;
+               value = isNaN(value) ? nan : formatType(Math.abs(value), precision); // Trim insignificant zeros.
 
-       var _mlyShowFeatureDetections = false;
-       var _mlyShowSignDetections = false;
+               if (trim) value = formatTrim(value); // If a negative value rounds to zero after formatting, and no explicit positive sign is requested, hide the sign.
 
-       function abortRequest$3(controller) {
-         controller.abort();
-       }
+               if (valueNegative && +value === 0 && sign !== "+") valueNegative = false; // Compute the prefix and suffix.
 
-       function loadTiles(which, url, projection) {
-         var currZoom = Math.floor(geoScaleToZoom(projection.scale()));
-         var tiles = tiler$3.getTiles(projection); // abort inflight requests that are no longer needed
+               valuePrefix = (valueNegative ? sign === "(" ? sign : minus : sign === "-" || sign === "(" ? "" : sign) + valuePrefix;
+               valueSuffix = (type === "s" ? prefixes[8 + prefixExponent / 3] : "") + valueSuffix + (valueNegative && sign === "(" ? ")" : ""); // Break the formatted value into the integer “value” part that can be
+               // grouped, and fractional or exponential “suffix” part that is not.
 
-         var cache = _mlyCache[which];
-         Object.keys(cache.inflight).forEach(function (k) {
-           var wanted = tiles.find(function (tile) {
-             return k.indexOf(tile.id + ',') === 0;
-           });
+               if (maybeSuffix) {
+                 i = -1, n = value.length;
 
-           if (!wanted) {
-             abortRequest$3(cache.inflight[k]);
-             delete cache.inflight[k];
-           }
-         });
-         tiles.forEach(function (tile) {
-           loadNextTilePage(which, currZoom, url, tile);
-         });
-       }
+                 while (++i < n) {
+                   if (c = value.charCodeAt(i), 48 > c || c > 57) {
+                     valueSuffix = (c === 46 ? decimal + value.slice(i + 1) : value.slice(i)) + valueSuffix;
+                     value = value.slice(0, i);
+                     break;
+                   }
+                 }
+               }
+             } // If the fill character is not "0", grouping is applied before padding.
 
-       function loadNextTilePage(which, currZoom, url, tile) {
-         var cache = _mlyCache[which];
-         var rect = tile.extent.rectangle();
-         var maxPages = maxPageAtZoom(currZoom);
-         var nextPage = cache.nextPage[tile.id] || 0;
-         var nextURL = cache.nextURL[tile.id] || url + utilQsString({
-           per_page: maxResults,
-           page: nextPage,
-           client_id: clientId,
-           bbox: [rect[0], rect[1], rect[2], rect[3]].join(',')
-         });
-         if (nextPage > maxPages) return;
-         var id = tile.id + ',' + String(nextPage);
-         if (cache.loaded[id] || cache.inflight[id]) return;
-         var controller = new AbortController();
-         cache.inflight[id] = controller;
-         var options = {
-           method: 'GET',
-           signal: controller.signal,
-           headers: {
-             'Content-Type': 'application/json'
-           }
-         };
-         fetch(nextURL, options).then(function (response) {
-           if (!response.ok) {
-             throw new Error(response.status + ' ' + response.statusText);
-           }
 
-           var linkHeader = response.headers.get('Link');
+             if (comma && !zero) value = group(value, Infinity); // Compute the padding.
 
-           if (linkHeader) {
-             var pagination = parsePagination(linkHeader);
+             var length = valuePrefix.length + value.length + valueSuffix.length,
+                 padding = length < width ? new Array(width - length + 1).join(fill) : ""; // If the fill character is "0", grouping is applied after padding.
 
-             if (pagination.next) {
-               cache.nextURL[tile.id] = pagination.next;
-             }
-           }
+             if (comma && zero) value = group(padding + value, padding.length ? width - valueSuffix.length : Infinity), padding = ""; // Reconstruct the final output based on the desired alignment.
 
-           return response.json();
-         }).then(function (data) {
-           cache.loaded[id] = true;
-           delete cache.inflight[id];
+             switch (align) {
+               case "<":
+                 value = valuePrefix + value + valueSuffix + padding;
+                 break;
 
-           if (!data || !data.features || !data.features.length) {
-             throw new Error('No Data');
-           }
+               case "=":
+                 value = valuePrefix + padding + value + valueSuffix;
+                 break;
 
-           var features = data.features.map(function (feature) {
-             var loc = feature.geometry.coordinates;
-             var d; // An image (shown as a green dot on the map) is a single street photo with extra
-             // information such as location, camera angle (CA), camera model, and so on.
-             // Each image feature is a GeoJSON Point
+               case "^":
+                 value = padding.slice(0, length = padding.length >> 1) + valuePrefix + value + valueSuffix + padding.slice(length);
+                 break;
 
-             if (which === 'images') {
-               d = {
-                 loc: loc,
-                 key: feature.properties.key,
-                 ca: feature.properties.ca,
-                 captured_at: feature.properties.captured_at,
-                 captured_by: feature.properties.username,
-                 pano: feature.properties.pano
-               };
-               cache.forImageKey[d.key] = d; // cache imageKey -> image
-               // Mapillary organizes images as sequences. A sequence of images are continuously captured
-               // by a user at a give time. Sequences are shown on the map as green lines.
-               // Each sequence feature is a GeoJSON LineString
-             } else if (which === 'sequences') {
-               var sequenceKey = feature.properties.key;
-               cache.lineString[sequenceKey] = feature; // cache sequenceKey -> lineString
-
-               feature.properties.coordinateProperties.image_keys.forEach(function (imageKey) {
-                 cache.forImageKey[imageKey] = sequenceKey; // cache imageKey -> sequenceKey
-               });
-               return false; // because no `d` data worth loading into an rbush
-               // A map feature is a real world object that can be shown on a map. It could be any object
-               // recognized from images, manually added in images, or added on the map.
-               // Each map feature is a GeoJSON Point (located where the feature is)
-             } else if (which === 'map_features' || which === 'points') {
-               d = {
-                 loc: loc,
-                 key: feature.properties.key,
-                 value: feature.properties.value,
-                 detections: feature.properties.detections,
-                 direction: feature.properties.direction,
-                 accuracy: feature.properties.accuracy,
-                 first_seen_at: feature.properties.first_seen_at,
-                 last_seen_at: feature.properties.last_seen_at
-               };
+               default:
+                 value = padding + valuePrefix + value + valueSuffix;
+                 break;
              }
 
-             return {
-               minX: loc[0],
-               minY: loc[1],
-               maxX: loc[0],
-               maxY: loc[1],
-               data: d
-             };
-           }).filter(Boolean);
-
-           if (cache.rtree && features) {
-             cache.rtree.load(features);
-           }
-
-           if (data.features.length === maxResults) {
-             // more pages to load
-             cache.nextPage[tile.id] = nextPage + 1;
-             loadNextTilePage(which, currZoom, url, tile);
-           } else {
-             cache.nextPage[tile.id] = Infinity; // no more pages to load
+             return numerals(value);
            }
 
-           if (which === 'images' || which === 'sequences') {
-             dispatch$4.call('loadedImages');
-           } else if (which === 'map_features') {
-             dispatch$4.call('loadedSigns');
-           } else if (which === 'points') {
-             dispatch$4.call('loadedMapFeatures');
-           }
-         })["catch"](function () {
-           cache.loaded[id] = true;
-           delete cache.inflight[id];
-         });
-       }
+           format.toString = function () {
+             return specifier + "";
+           };
 
-       function loadData(which, url) {
-         var cache = _mlyCache[which];
-         var options = {
-           method: 'GET',
-           headers: {
-             'Content-Type': 'application/json'
-           }
-         };
-         var nextUrl = url + '&client_id=' + clientId;
-         return fetch(nextUrl, options).then(function (response) {
-           if (!response.ok) {
-             throw new Error(response.status + ' ' + response.statusText);
-           }
+           return format;
+         }
 
-           return response.json();
-         }).then(function (data) {
-           if (!data || !data.features || !data.features.length) {
-             throw new Error('No Data');
-           }
+         function formatPrefix(specifier, value) {
+           var f = newFormat((specifier = formatSpecifier(specifier), specifier.type = "f", specifier)),
+               e = Math.max(-8, Math.min(8, Math.floor(exponent(value) / 3))) * 3,
+               k = Math.pow(10, -e),
+               prefix = prefixes[8 + e / 3];
+           return function (value) {
+             return f(k * value) + prefix;
+           };
+         }
 
-           data.features.forEach(function (feature) {
-             var d;
+         return {
+           format: newFormat,
+           formatPrefix: formatPrefix
+         };
+       }
 
-             if (which === 'image_detections') {
-               d = {
-                 key: feature.properties.key,
-                 image_key: feature.properties.image_key,
-                 value: feature.properties.value,
-                 shape: feature.properties.shape
-               };
+       var locale;
+       var format;
+       var formatPrefix;
+       defaultLocale({
+         thousands: ",",
+         grouping: [3],
+         currency: ["$", ""]
+       });
+       function defaultLocale(definition) {
+         locale = formatLocale(definition);
+         format = locale.format;
+         formatPrefix = locale.formatPrefix;
+         return locale;
+       }
 
-               if (!cache.forImageKey[d.image_key]) {
-                 cache.forImageKey[d.image_key] = [];
-               }
+       function precisionFixed (step) {
+         return Math.max(0, -exponent(Math.abs(step)));
+       }
 
-               cache.forImageKey[d.image_key].push(d);
-             }
-           });
-         });
+       function precisionPrefix (step, value) {
+         return Math.max(0, Math.max(-8, Math.min(8, Math.floor(exponent(value) / 3))) * 3 - exponent(Math.abs(step)));
        }
 
-       function maxPageAtZoom(z) {
-         if (z < 15) return 2;
-         if (z === 15) return 5;
-         if (z === 16) return 10;
-         if (z === 17) return 20;
-         if (z === 18) return 40;
-         if (z > 18) return 80;
-       } // extract links to pages of API results
+       function precisionRound (step, max) {
+         step = Math.abs(step), max = Math.abs(max) - step;
+         return Math.max(0, exponent(max) - exponent(step)) + 1;
+       }
 
+       function tickFormat(start, stop, count, specifier) {
+         var step = tickStep(start, stop, count),
+             precision;
+         specifier = formatSpecifier(specifier == null ? ",f" : specifier);
 
-       function parsePagination(links) {
-         return links.split(',').map(function (rel) {
-           var elements = rel.split(';');
+         switch (specifier.type) {
+           case "s":
+             {
+               var value = Math.max(Math.abs(start), Math.abs(stop));
+               if (specifier.precision == null && !isNaN(precision = precisionPrefix(step, value))) specifier.precision = precision;
+               return formatPrefix(specifier, value);
+             }
 
-           if (elements.length === 2) {
-             return [/<(.+)>/.exec(elements[0])[1], /rel="(.+)"/.exec(elements[1])[1]];
-           } else {
-             return ['', ''];
-           }
-         }).reduce(function (pagination, val) {
-           pagination[val[1]] = val[0];
-           return pagination;
-         }, {});
-       } // partition viewport into higher zoom tiles
+           case "":
+           case "e":
+           case "g":
+           case "p":
+           case "r":
+             {
+               if (specifier.precision == null && !isNaN(precision = precisionRound(step, Math.max(Math.abs(start), Math.abs(stop))))) specifier.precision = precision - (specifier.type === "e");
+               break;
+             }
 
+           case "f":
+           case "%":
+             {
+               if (specifier.precision == null && !isNaN(precision = precisionFixed(step))) specifier.precision = precision - (specifier.type === "%") * 2;
+               break;
+             }
+         }
 
-       function partitionViewport(projection) {
-         var z = geoScaleToZoom(projection.scale());
-         var z2 = Math.ceil(z * 2) / 2 + 2.5; // round to next 0.5 and add 2.5
+         return format(specifier);
+       }
 
-         var tiler = utilTiler().zoomExtent([z2, z2]);
-         return tiler.getTiles(projection).map(function (tile) {
-           return tile.extent;
-         });
-       } // no more than `limit` results per partition.
+       function linearish(scale) {
+         var domain = scale.domain;
 
+         scale.ticks = function (count) {
+           var d = domain();
+           return ticks(d[0], d[d.length - 1], count == null ? 10 : count);
+         };
 
-       function searchLimited(limit, projection, rtree) {
-         limit = limit || 5;
-         return partitionViewport(projection).reduce(function (result, extent) {
-           var found = rtree.search(extent.bbox()).slice(0, limit).map(function (d) {
-             return d.data;
-           });
-           return found.length ? result.concat(found) : result;
-         }, []);
-       }
+         scale.tickFormat = function (count, specifier) {
+           var d = domain();
+           return tickFormat(d[0], d[d.length - 1], count == null ? 10 : count, specifier);
+         };
 
-       var serviceMapillary = {
-         init: function init() {
-           if (!_mlyCache) {
-             this.reset();
-           }
+         scale.nice = function (count) {
+           if (count == null) count = 10;
+           var d = domain();
+           var i0 = 0;
+           var i1 = d.length - 1;
+           var start = d[i0];
+           var stop = d[i1];
+           var prestep;
+           var step;
+           var maxIter = 10;
 
-           this.event = utilRebind(this, dispatch$4, 'on');
-         },
-         reset: function reset() {
-           if (_mlyCache) {
-             Object.values(_mlyCache.images.inflight).forEach(abortRequest$3);
-             Object.values(_mlyCache.image_detections.inflight).forEach(abortRequest$3);
-             Object.values(_mlyCache.map_features.inflight).forEach(abortRequest$3);
-             Object.values(_mlyCache.points.inflight).forEach(abortRequest$3);
-             Object.values(_mlyCache.sequences.inflight).forEach(abortRequest$3);
+           if (stop < start) {
+             step = start, start = stop, stop = step;
+             step = i0, i0 = i1, i1 = step;
            }
 
-           _mlyCache = {
-             images: {
-               inflight: {},
-               loaded: {},
-               nextPage: {},
-               nextURL: {},
-               rtree: new RBush(),
-               forImageKey: {}
-             },
-             image_detections: {
-               inflight: {},
-               loaded: {},
-               nextPage: {},
-               nextURL: {},
-               forImageKey: {}
-             },
-             map_features: {
-               inflight: {},
-               loaded: {},
-               nextPage: {},
-               nextURL: {},
-               rtree: new RBush()
-             },
-             points: {
-               inflight: {},
-               loaded: {},
-               nextPage: {},
-               nextURL: {},
-               rtree: new RBush()
-             },
-             sequences: {
-               inflight: {},
-               loaded: {},
-               nextPage: {},
-               nextURL: {},
-               rtree: new RBush(),
-               forImageKey: {},
-               lineString: {}
-             }
-           };
-           _mlySelectedImageKey = null;
-           _mlyActiveImage = null;
-           _mlyClicks = [];
-         },
-         images: function images(projection) {
-           var limit = 5;
-           return searchLimited(limit, projection, _mlyCache.images.rtree);
-         },
-         signs: function signs(projection) {
-           var limit = 5;
-           return searchLimited(limit, projection, _mlyCache.map_features.rtree);
-         },
-         mapFeatures: function mapFeatures(projection) {
-           var limit = 5;
-           return searchLimited(limit, projection, _mlyCache.points.rtree);
-         },
-         cachedImage: function cachedImage(imageKey) {
-           return _mlyCache.images.forImageKey[imageKey];
-         },
-         sequences: function sequences(projection) {
-           var viewport = projection.clipExtent();
-           var min = [viewport[0][0], viewport[1][1]];
-           var max = [viewport[1][0], viewport[0][1]];
-           var bbox = geoExtent(projection.invert(min), projection.invert(max)).bbox();
-           var sequenceKeys = {}; // all sequences for images in viewport
-
-           _mlyCache.images.rtree.search(bbox).forEach(function (d) {
-             var sequenceKey = _mlyCache.sequences.forImageKey[d.data.key];
+           while (maxIter-- > 0) {
+             step = tickIncrement(start, stop, count);
 
-             if (sequenceKey) {
-               sequenceKeys[sequenceKey] = true;
+             if (step === prestep) {
+               d[i0] = start;
+               d[i1] = stop;
+               return domain(d);
+             } else if (step > 0) {
+               start = Math.floor(start / step) * step;
+               stop = Math.ceil(stop / step) * step;
+             } else if (step < 0) {
+               start = Math.ceil(start * step) / step;
+               stop = Math.floor(stop * step) / step;
+             } else {
+               break;
              }
-           }); // Return lineStrings for the sequences
-
-
-           return Object.keys(sequenceKeys).map(function (sequenceKey) {
-             return _mlyCache.sequences.lineString[sequenceKey];
-           });
-         },
-         signsSupported: function signsSupported() {
-           return true;
-         },
-         loadImages: function loadImages(projection) {
-           loadTiles('images', apibase + 'images?sort_by=key&', projection);
-           loadTiles('sequences', apibase + 'sequences?sort_by=key&', projection);
-         },
-         loadSigns: function loadSigns(projection) {
-           loadTiles('map_features', apibase + 'map_features?layers=trafficsigns&min_nbr_image_detections=2&sort_by=key&', projection);
-         },
-         loadMapFeatures: function loadMapFeatures(projection) {
-           loadTiles('points', apibase + 'map_features?layers=points&min_nbr_image_detections=2&sort_by=key&values=' + mapFeatureConfig.values + '&', projection);
-         },
-         ensureViewerLoaded: function ensureViewerLoaded(context) {
-           if (_loadViewerPromise) return _loadViewerPromise; // add mly-wrapper
-
-           var wrap = context.container().select('.photoviewer').selectAll('.mly-wrapper').data([0]);
-           wrap.enter().append('div').attr('id', 'ideditor-mly').attr('class', 'photo-wrapper mly-wrapper').classed('hide', true);
-           var that = this;
-           _loadViewerPromise = new Promise(function (resolve, reject) {
-             var loadedCount = 0;
 
-             function loaded() {
-               loadedCount += 1; // wait until both files are loaded
+             prestep = step;
+           }
 
-               if (loadedCount === 2) resolve();
-             }
+           return scale;
+         };
 
-             var head = select('head'); // load mapillary-viewercss
+         return scale;
+       }
+       function linear() {
+         var scale = continuous();
 
-             head.selectAll('#ideditor-mapillary-viewercss').data([0]).enter().append('link').attr('id', 'ideditor-mapillary-viewercss').attr('rel', 'stylesheet').attr('crossorigin', 'anonymous').attr('href', context.asset(viewercss)).on('load.serviceMapillary', loaded).on('error.serviceMapillary', function () {
-               reject();
-             }); // load mapillary-viewerjs
+         scale.copy = function () {
+           return copy$1(scale, linear());
+         };
 
-             head.selectAll('#ideditor-mapillary-viewerjs').data([0]).enter().append('script').attr('id', 'ideditor-mapillary-viewerjs').attr('crossorigin', 'anonymous').attr('src', context.asset(viewerjs)).on('load.serviceMapillary', loaded).on('error.serviceMapillary', function () {
-               reject();
-             });
-           })["catch"](function () {
-             _loadViewerPromise = null;
-           }).then(function () {
-             that.initViewer(context);
-           });
-           return _loadViewerPromise;
-         },
-         loadSignResources: function loadSignResources(context) {
-           context.ui().svgDefs.addSprites(['mapillary-sprite'], false
-           /* don't override colors */
-           );
-           return this;
-         },
-         loadObjectResources: function loadObjectResources(context) {
-           context.ui().svgDefs.addSprites(['mapillary-object-sprite'], false
-           /* don't override colors */
-           );
-           return this;
-         },
-         resetTags: function resetTags() {
-           if (_mlyViewer && !_mlyFallback) {
-             _mlyViewer.getComponent('tag').removeAll(); // remove previous detections
+         initRange.apply(scale, arguments);
+         return linearish(scale);
+       }
 
-           }
-         },
-         showFeatureDetections: function showFeatureDetections(value) {
-           _mlyShowFeatureDetections = value;
+       // eslint-disable-next-line es/no-math-expm1 -- safe
+       var $expm1 = Math.expm1;
+       var exp$1 = Math.exp;
 
-           if (!_mlyShowFeatureDetections && !_mlyShowSignDetections) {
-             this.resetTags();
-           }
-         },
-         showSignDetections: function showSignDetections(value) {
-           _mlyShowSignDetections = value;
+       // `Math.expm1` method implementation
+       // https://tc39.es/ecma262/#sec-math.expm1
+       var mathExpm1 = (!$expm1
+         // Old FF bug
+         || $expm1(10) > 22025.465794806719 || $expm1(10) < 22025.4657948067165168
+         // Tor Browser bug
+         || $expm1(-2e-17) != -2e-17
+       ) ? function expm1(x) {
+         return (x = +x) == 0 ? x : x > -1e-6 && x < 1e-6 ? x + x * x / 2 : exp$1(x) - 1;
+       } : $expm1;
 
-           if (!_mlyShowFeatureDetections && !_mlyShowSignDetections) {
-             this.resetTags();
-           }
-         },
-         filterViewer: function filterViewer(context) {
-           var showsPano = context.photos().showsPanoramic();
-           var showsFlat = context.photos().showsFlat();
-           var fromDate = context.photos().fromDate();
-           var toDate = context.photos().toDate();
-           var usernames = context.photos().usernames();
-           var filter = ['all'];
-           if (!showsPano) filter.push(['==', 'pano', false]);
-           if (!showsFlat && showsPano) filter.push(['==', 'pano', true]);
-           if (usernames && usernames.length) filter.push(['==', 'username', usernames[0]]);
+       function quantize() {
+         var x0 = 0,
+             x1 = 1,
+             n = 1,
+             domain = [0.5],
+             range = [0, 1],
+             unknown;
 
-           if (fromDate) {
-             var fromTimestamp = new Date(fromDate).getTime();
-             filter.push(['>=', 'capturedAt', fromTimestamp]);
-           }
+         function scale(x) {
+           return x != null && x <= x ? range[bisectRight(domain, x, 0, n)] : unknown;
+         }
 
-           if (toDate) {
-             var toTimestamp = new Date(toDate).getTime();
-             filter.push(['>=', 'capturedAt', toTimestamp]);
-           }
+         function rescale() {
+           var i = -1;
+           domain = new Array(n);
 
-           if (_mlyViewer) {
-             _mlyViewer.setFilter(filter);
+           while (++i < n) {
+             domain[i] = ((i + 1) * x1 - (i - n) * x0) / (n + 1);
            }
 
-           _mlyViewerFilter = filter;
-           return filter;
-         },
-         showViewer: function showViewer(context) {
-           var wrap = context.container().select('.photoviewer').classed('hide', false);
-           var isHidden = wrap.selectAll('.photo-wrapper.mly-wrapper.hide').size();
-
-           if (isHidden && _mlyViewer) {
-             wrap.selectAll('.photo-wrapper:not(.mly-wrapper)').classed('hide', true);
-             wrap.selectAll('.photo-wrapper.mly-wrapper').classed('hide', false);
+           return scale;
+         }
 
-             _mlyViewer.resize();
-           }
+         scale.domain = function (_) {
+           var _ref, _ref2;
 
-           return this;
-         },
-         hideViewer: function hideViewer(context) {
-           _mlyActiveImage = null;
-           _mlySelectedImageKey = null;
+           return arguments.length ? ((_ref = _, _ref2 = _slicedToArray(_ref, 2), x0 = _ref2[0], x1 = _ref2[1], _ref), x0 = +x0, x1 = +x1, rescale()) : [x0, x1];
+         };
 
-           if (!_mlyFallback && _mlyViewer) {
-             _mlyViewer.getComponent('sequence').stop();
-           }
+         scale.range = function (_) {
+           return arguments.length ? (n = (range = Array.from(_)).length - 1, rescale()) : range.slice();
+         };
 
-           var viewer = context.container().select('.photoviewer');
-           if (!viewer.empty()) viewer.datum(null);
-           viewer.classed('hide', true).selectAll('.photo-wrapper').classed('hide', true);
-           this.updateUrlImage(null);
-           dispatch$4.call('nodeChanged');
-           return this.setStyles(context, null, true);
-         },
-         parsePagination: parsePagination,
-         updateUrlImage: function updateUrlImage(imageKey) {
-           if (!window.mocha) {
-             var hash = utilStringQs(window.location.hash);
+         scale.invertExtent = function (y) {
+           var i = range.indexOf(y);
+           return i < 0 ? [NaN, NaN] : i < 1 ? [x0, domain[0]] : i >= n ? [domain[n - 1], x1] : [domain[i - 1], domain[i]];
+         };
 
-             if (imageKey) {
-               hash.photo = 'mapillary/' + imageKey;
-             } else {
-               delete hash.photo;
-             }
+         scale.unknown = function (_) {
+           return arguments.length ? (unknown = _, scale) : scale;
+         };
 
-             window.location.replace('#' + utilQsString(hash, true));
-           }
-         },
-         highlightDetection: function highlightDetection(detection) {
-           if (detection) {
-             _mlyHighlightedDetection = detection.detection_key;
-           }
+         scale.thresholds = function () {
+           return domain.slice();
+         };
 
-           return this;
-         },
-         initViewer: function initViewer(context) {
-           var that = this;
-           if (!window.Mapillary) return;
-           var opts = {
-             baseImageSize: 320,
-             component: {
-               cover: false,
-               keyboard: false,
-               tag: true
-             }
-           }; // Disable components requiring WebGL support
+         scale.copy = function () {
+           return quantize().domain([x0, x1]).range(range).unknown(unknown);
+         };
 
-           if (!Mapillary.isSupported() && Mapillary.isFallbackSupported()) {
-             _mlyFallback = true;
-             opts.component = {
-               cover: false,
-               direction: false,
-               imagePlane: false,
-               keyboard: false,
-               mouse: false,
-               sequence: false,
-               tag: false,
-               image: true,
-               // fallback
-               navigation: true // fallback
+         return initRange.apply(linearish(scale), arguments);
+       }
 
-             };
-           }
+       // https://github.com/tc39/proposal-string-pad-start-end
 
-           _mlyViewer = new Mapillary.Viewer('ideditor-mly', clientId, null, opts);
 
-           _mlyViewer.on('nodechanged', nodeChanged);
 
-           _mlyViewer.on('bearingchanged', bearingChanged);
 
-           if (_mlyViewerFilter) {
-             _mlyViewer.setFilter(_mlyViewerFilter);
-           } // Register viewer resize handler
+       var ceil = Math.ceil;
 
+       // `String.prototype.{ padStart, padEnd }` methods implementation
+       var createMethod = function (IS_END) {
+         return function ($this, maxLength, fillString) {
+           var S = String(requireObjectCoercible($this));
+           var stringLength = S.length;
+           var fillStr = fillString === undefined ? ' ' : String(fillString);
+           var intMaxLength = toLength(maxLength);
+           var fillLen, stringFiller;
+           if (intMaxLength <= stringLength || fillStr == '') return S;
+           fillLen = intMaxLength - stringLength;
+           stringFiller = stringRepeat.call(fillStr, ceil(fillLen / fillStr.length));
+           if (stringFiller.length > fillLen) stringFiller = stringFiller.slice(0, fillLen);
+           return IS_END ? S + stringFiller : stringFiller + S;
+         };
+       };
 
-           context.ui().photoviewer.on('resize.mapillary', function () {
-             if (_mlyViewer) _mlyViewer.resize();
-           }); // nodeChanged: called after the viewer has changed images and is ready.
-           //
-           // 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 asynchronously.
-           //
-           // Clicks are added to the array in `selectedImage` and removed here.
-           //
+       var stringPad = {
+         // `String.prototype.padStart` method
+         // https://tc39.es/ecma262/#sec-string.prototype.padstart
+         start: createMethod(false),
+         // `String.prototype.padEnd` method
+         // https://tc39.es/ecma262/#sec-string.prototype.padend
+         end: createMethod(true)
+       };
 
-           function nodeChanged(node) {
-             that.resetTags();
-             var clicks = _mlyClicks;
-             var index = clicks.indexOf(node.key);
-             var selectedKey = _mlySelectedImageKey;
-             that.setActiveImage(node);
+       var padStart = stringPad.start;
 
-             if (index > -1) {
-               // `nodechanged` initiated from clicking on a marker..
-               clicks.splice(index, 1); // remove the click
-               // If `node.key` matches the current _mlySelectedImageKey, call `selectImage()`
-               // one more time to update the detections and attribution..
+       var abs$1 = Math.abs;
+       var DatePrototype = Date.prototype;
+       var getTime = DatePrototype.getTime;
+       var nativeDateToISOString = DatePrototype.toISOString;
 
-               if (node.key === selectedKey) {
-                 that.selectImage(context, _mlySelectedImageKey, true);
-               }
-             } else {
-               // `nodechanged` initiated from the Mapillary viewer controls..
-               var loc = node.computedLatLon ? [node.computedLatLon.lon, node.computedLatLon.lat] : [node.latLon.lon, node.latLon.lat];
-               context.map().centerEase(loc);
-               that.selectImage(context, node.key, true);
-             }
+       // `Date.prototype.toISOString` method implementation
+       // https://tc39.es/ecma262/#sec-date.prototype.toisostring
+       // PhantomJS / old WebKit fails here:
+       var dateToIsoString = (fails(function () {
+         return nativeDateToISOString.call(new Date(-5e13 - 1)) != '0385-07-25T07:06:39.999Z';
+       }) || !fails(function () {
+         nativeDateToISOString.call(new Date(NaN));
+       })) ? function toISOString() {
+         if (!isFinite(getTime.call(this))) throw RangeError('Invalid time value');
+         var date = this;
+         var year = date.getUTCFullYear();
+         var milliseconds = date.getUTCMilliseconds();
+         var sign = year < 0 ? '-' : year > 9999 ? '+' : '';
+         return sign + padStart(abs$1(year), sign ? 6 : 4, 0) +
+           '-' + padStart(date.getUTCMonth() + 1, 2, 0) +
+           '-' + padStart(date.getUTCDate(), 2, 0) +
+           'T' + padStart(date.getUTCHours(), 2, 0) +
+           ':' + padStart(date.getUTCMinutes(), 2, 0) +
+           ':' + padStart(date.getUTCSeconds(), 2, 0) +
+           '.' + padStart(milliseconds, 3, 0) +
+           'Z';
+       } : nativeDateToISOString;
 
-             dispatch$4.call('nodeChanged');
-           }
+       // `Date.prototype.toISOString` method
+       // https://tc39.es/ecma262/#sec-date.prototype.toisostring
+       // PhantomJS / old WebKit has a broken implementations
+       _export({ target: 'Date', proto: true, forced: Date.prototype.toISOString !== dateToIsoString }, {
+         toISOString: dateToIsoString
+       });
 
-           function bearingChanged(e) {
-             dispatch$4.call('bearingChanged', undefined, e);
-           }
-         },
-         // Pass in the image key string as `imageKey`.
-         // This allows images to be selected from places that dont have access
-         // to the full image datum (like the street signs layer or the js viewer)
-         selectImage: function selectImage(context, imageKey, fromViewer) {
-           _mlySelectedImageKey = imageKey;
-           this.updateUrlImage(imageKey);
-           var d = _mlyCache.images.forImageKey[imageKey];
-           var viewer = context.container().select('.photoviewer');
-           if (!viewer.empty()) viewer.datum(d);
-           imageKey = d && d.key || imageKey;
+       function behaviorBreathe() {
+         var duration = 800;
+         var steps = 4;
+         var selector = '.selected.shadow, .selected .shadow';
 
-           if (!fromViewer && imageKey) {
-             _mlyClicks.push(imageKey);
-           }
+         var _selected = select(null);
 
-           this.setStyles(context, null, true);
+         var _classed = '';
+         var _params = {};
+         var _done = false;
 
-           if (_mlyShowFeatureDetections) {
-             this.updateDetections(imageKey, apibase + 'image_detections?layers=points&values=' + mapFeatureConfig.values + '&image_keys=' + imageKey);
-           }
+         var _timer;
 
-           if (_mlyShowSignDetections) {
-             this.updateDetections(imageKey, apibase + 'image_detections?layers=trafficsigns&image_keys=' + imageKey);
-           }
+         function ratchetyInterpolator(a, b, steps, units) {
+           a = parseFloat(a);
+           b = parseFloat(b);
+           var sample = quantize().domain([0, 1]).range(d3_quantize(d3_interpolateNumber(a, b), steps));
+           return function (t) {
+             return String(sample(t)) + (units || '');
+           };
+         }
 
-           if (_mlyViewer && imageKey) {
-             _mlyViewer.moveToKey(imageKey)["catch"](function (e) {
-               console.error('mly3', e);
-             }); // eslint-disable-line no-console
+         function reset(selection) {
+           selection.style('stroke-opacity', null).style('stroke-width', null).style('fill-opacity', null).style('r', null);
+         }
 
-           }
+         function setAnimationParams(transition, fromTo) {
+           var toFrom = fromTo === 'from' ? 'to' : 'from';
+           transition.styleTween('stroke-opacity', function (d) {
+             return ratchetyInterpolator(_params[d.id][toFrom].opacity, _params[d.id][fromTo].opacity, steps);
+           }).styleTween('stroke-width', function (d) {
+             return ratchetyInterpolator(_params[d.id][toFrom].width, _params[d.id][fromTo].width, steps, 'px');
+           }).styleTween('fill-opacity', function (d) {
+             return ratchetyInterpolator(_params[d.id][toFrom].opacity, _params[d.id][fromTo].opacity, steps);
+           }).styleTween('r', function (d) {
+             return ratchetyInterpolator(_params[d.id][toFrom].width, _params[d.id][fromTo].width, steps, 'px');
+           });
+         }
 
-           return this;
-         },
-         getActiveImage: function getActiveImage() {
-           return _mlyActiveImage;
-         },
-         getSelectedImageKey: function getSelectedImageKey() {
-           return _mlySelectedImageKey;
-         },
-         getSequenceKeyForImageKey: function getSequenceKeyForImageKey(imageKey) {
-           return _mlyCache.sequences.forImageKey[imageKey];
-         },
-         setActiveImage: function setActiveImage(node) {
-           if (node) {
-             _mlyActiveImage = {
-               ca: node.originalCA,
-               key: node.key,
-               loc: [node.originalLatLon.lon, node.originalLatLon.lat],
-               pano: node.pano
+         function calcAnimationParams(selection) {
+           selection.call(reset).each(function (d) {
+             var s = select(this);
+             var tag = s.node().tagName;
+             var p = {
+               'from': {},
+               'to': {}
              };
-           } else {
-             _mlyActiveImage = null;
-           }
-         },
-         // Updates the currently highlighted sequence and selected bubble.
-         // Reset is only necessary when interacting with the viewport because
-         // this implicitly changes the currently selected bubble/sequence
-         setStyles: function setStyles(context, hovered, reset) {
-           if (reset) {
-             // reset all layers
-             context.container().selectAll('.viewfield-group').classed('highlighted', false).classed('hovered', false);
-             context.container().selectAll('.sequence').classed('highlighted', false).classed('currentView', false);
-           }
+             var opacity;
+             var width; // determine base opacity and width
 
-           var hoveredImageKey = hovered && hovered.key;
-           var hoveredSequenceKey = hoveredImageKey && this.getSequenceKeyForImageKey(hoveredImageKey);
-           var hoveredLineString = hoveredSequenceKey && _mlyCache.sequences.lineString[hoveredSequenceKey];
-           var hoveredImageKeys = hoveredLineString && hoveredLineString.properties.coordinateProperties.image_keys || [];
-           var selectedImageKey = _mlySelectedImageKey;
-           var selectedSequenceKey = selectedImageKey && this.getSequenceKeyForImageKey(selectedImageKey);
-           var selectedLineString = selectedSequenceKey && _mlyCache.sequences.lineString[selectedSequenceKey];
-           var selectedImageKeys = selectedLineString && selectedLineString.properties.coordinateProperties.image_keys || []; // highlight sibling viewfields on either the selected or the hovered sequences
+             if (tag === 'circle') {
+               opacity = parseFloat(s.style('fill-opacity') || 0.5);
+               width = parseFloat(s.style('r') || 15.5);
+             } else {
+               opacity = parseFloat(s.style('stroke-opacity') || 0.7);
+               width = parseFloat(s.style('stroke-width') || 10);
+             } // calculate from/to interpolation params..
 
-           var highlightedImageKeys = utilArrayUnion(hoveredImageKeys, selectedImageKeys);
-           context.container().selectAll('.layer-mapillary .viewfield-group').classed('highlighted', function (d) {
-             return highlightedImageKeys.indexOf(d.key) !== -1;
-           }).classed('hovered', function (d) {
-             return d.key === hoveredImageKey;
+
+             p.tag = tag;
+             p.from.opacity = opacity * 0.6;
+             p.to.opacity = opacity * 1.25;
+             p.from.width = width * 0.7;
+             p.to.width = width * (tag === 'circle' ? 1.5 : 1);
+             _params[d.id] = p;
            });
-           context.container().selectAll('.layer-mapillary .sequence').classed('highlighted', function (d) {
-             return d.properties.key === hoveredSequenceKey;
-           }).classed('currentView', function (d) {
-             return d.properties.key === selectedSequenceKey;
-           }); // update viewfields if needed
+         }
 
-           context.container().selectAll('.viewfield-group .viewfield').attr('d', viewfieldPath);
+         function run(surface, fromTo) {
+           var toFrom = fromTo === 'from' ? 'to' : 'from';
+           var currSelected = surface.selectAll(selector);
+           var currClassed = surface.attr('class');
 
-           function viewfieldPath() {
-             var d = this.parentNode.__data__;
+           if (_done || currSelected.empty()) {
+             _selected.call(reset);
 
-             if (d.pano && d.key !== selectedImageKey) {
-               return 'M 8,13 m -10,0 a 10,10 0 1,0 20,0 a 10,10 0 1,0 -20,0';
-             } else {
-               return 'M 6,9 C 8,8.4 8,8.4 10,9 L 16,-2 C 12,-5 4,-5 0,-2 z';
-             }
+             _selected = select(null);
+             return;
            }
 
-           return this;
-         },
-         updateDetections: function updateDetections(imageKey, url) {
-           if (!_mlyViewer || _mlyFallback) return;
-           if (!imageKey) return;
+           if (!fastDeepEqual(currSelected.data(), _selected.data()) || currClassed !== _classed) {
+             _selected.call(reset);
 
-           if (!_mlyCache.image_detections.forImageKey[imageKey]) {
-             loadData('image_detections', url).then(function () {
-               showDetections(_mlyCache.image_detections.forImageKey[imageKey] || []);
-             });
-           } else {
-             showDetections(_mlyCache.image_detections.forImageKey[imageKey]);
+             _classed = currClassed;
+             _selected = currSelected.call(calcAnimationParams);
            }
 
-           function showDetections(detections) {
-             detections.forEach(function (data) {
-               var tag = makeTag(data);
+           var didCallNextRun = false;
 
-               if (tag) {
-                 var tagComponent = _mlyViewer.getComponent('tag');
+           _selected.transition().duration(duration).call(setAnimationParams, fromTo).on('end', function () {
+             // `end` event is called for each selected element, but we want
+             // it to run only once
+             if (!didCallNextRun) {
+               surface.call(run, toFrom);
+               didCallNextRun = true;
+             } // if entity was deselected, remove breathe styling
 
-                 tagComponent.add([tag]);
-               }
-             });
-           }
 
-           function makeTag(data) {
-             var valueParts = data.value.split('--');
-             if (!valueParts.length) return;
-             var tag;
-             var text;
-             var color = 0xffffff;
+             if (!select(this).classed('selected')) {
+               reset(select(this));
+             }
+           });
+         }
 
-             if (_mlyHighlightedDetection === data.key) {
-               color = 0xffff00;
-               text = valueParts[1];
+         function behavior(surface) {
+           _done = false;
+           _timer = timer(function () {
+             // wait for elements to actually become selected
+             if (surface.selectAll(selector).empty()) {
+               return false;
+             }
 
-               if (text === 'flat' || text === 'discrete' || text === 'sign') {
-                 text = valueParts[2];
-               }
+             surface.call(run, 'from');
 
-               text = text.replace(/-/g, ' ');
-               text = text.charAt(0).toUpperCase() + text.slice(1);
-               _mlyHighlightedDetection = null;
-             }
+             _timer.stop();
 
-             if (data.shape.type === 'Polygon') {
-               var polygonGeometry = new Mapillary.TagComponent.PolygonGeometry(data.shape.coordinates[0]);
-               tag = new Mapillary.TagComponent.OutlineTag(data.key, polygonGeometry, {
-                 text: text,
-                 textColor: color,
-                 lineColor: color,
-                 lineWidth: 2,
-                 fillColor: color,
-                 fillOpacity: 0.3
-               });
-             } else if (data.shape.type === 'Point') {
-               var pointGeometry = new Mapillary.TagComponent.PointGeometry(data.shape.coordinates[0]);
-               tag = new Mapillary.TagComponent.SpotTag(data.key, pointGeometry, {
-                 text: text,
-                 color: color,
-                 textColor: color
-               });
-             }
+             return true;
+           }, 20);
+         }
 
-             return tag;
+         behavior.restartIfNeeded = function (surface) {
+           if (_selected.empty()) {
+             surface.call(run, 'from');
+
+             if (_timer) {
+               _timer.stop();
+             }
            }
-         },
-         cache: function cache() {
-           return _mlyCache;
-         }
-       };
+         };
 
-       function validationIssue(attrs) {
-         this.type = attrs.type; // required - name of rule that created the issue (e.g. 'missing_tag')
+         behavior.off = function () {
+           _done = true;
 
-         this.subtype = attrs.subtype; // optional - category of the issue within the type (e.g. 'relation_type' under 'missing_tag')
+           if (_timer) {
+             _timer.stop();
+           }
 
-         this.severity = attrs.severity; // required - 'warning' or 'error'
+           _selected.interrupt().call(reset);
+         };
 
-         this.message = attrs.message; // required - function returning localized string
+         return behavior;
+       }
 
-         this.reference = attrs.reference; // optional - function(selection) to render reference information
+       /* Creates a keybinding behavior for an operation */
+       function behaviorOperation(context) {
+         var _operation;
 
-         this.entityIds = attrs.entityIds; // optional - array of IDs of entities involved in the issue
+         function keypress(d3_event) {
+           // prevent operations during low zoom selection
+           if (!context.map().withinEditableZoom()) return;
+           if (_operation.availableForKeypress && !_operation.availableForKeypress()) return;
+           d3_event.preventDefault();
 
-         this.loc = attrs.loc; // optional - [lon, lat] to zoom in on to see the issue
+           var disabled = _operation.disabled();
 
-         this.data = attrs.data; // optional - object containing extra data for the fixes
+           if (disabled) {
+             context.ui().flash.duration(4000).iconName('#iD-operation-' + _operation.id).iconClass('operation disabled').label(_operation.tooltip)();
+           } else {
+             context.ui().flash.duration(2000).iconName('#iD-operation-' + _operation.id).iconClass('operation').label(_operation.annotation() || _operation.title)();
+             if (_operation.point) _operation.point(null);
 
-         this.dynamicFixes = attrs.dynamicFixes; // optional - function(context) returning fixes
+             _operation();
+           }
+         }
 
-         this.hash = attrs.hash; // optional - string to further differentiate the issue
+         function behavior() {
+           if (_operation && _operation.available()) {
+             context.keybinding().on(_operation.keys, keypress);
+           }
 
-         this.id = generateID.apply(this); // generated - see below
+           return behavior;
+         }
 
-         this.autoFix = null; // generated - if autofix exists, will be set below
-         // A unique, deterministic string hash.
-         // Issues with identical id values are considered identical.
+         behavior.off = function () {
+           context.keybinding().off(_operation.keys);
+         };
 
-         function generateID() {
-           var parts = [this.type];
+         behavior.which = function (_) {
+           if (!arguments.length) return _operation;
+           _operation = _;
+           return behavior;
+         };
 
-           if (this.hash) {
-             // subclasses can pass in their own differentiator
-             parts.push(this.hash);
-           }
+         return behavior;
+       }
 
-           if (this.subtype) {
-             parts.push(this.subtype);
-           } // include the entities this issue is for
-           // (sort them so the id is deterministic)
+       function operationCircularize(context, selectedIDs) {
+         var _extent;
 
+         var _actions = selectedIDs.map(getAction).filter(Boolean);
 
-           if (this.entityIds) {
-             var entityKeys = this.entityIds.slice().sort();
-             parts.push.apply(parts, entityKeys);
-           }
+         var _amount = _actions.length === 1 ? 'single' : 'multiple';
 
-           return parts.join(':');
-         }
+         var _coords = utilGetAllNodes(selectedIDs, context.graph()).map(function (n) {
+           return n.loc;
+         });
 
-         this.extent = function (resolver) {
-           if (this.loc) {
-             return geoExtent(this.loc);
-           }
+         function getAction(entityID) {
+           var entity = context.entity(entityID);
+           if (entity.type !== 'way' || new Set(entity.nodes).size <= 1) return null;
 
-           if (this.entityIds && this.entityIds.length) {
-             return this.entityIds.reduce(function (extent, entityId) {
-               return extent.extend(resolver.entity(entityId).extent(resolver));
-             }, geoExtent());
+           if (!_extent) {
+             _extent = entity.extent(context.graph());
+           } else {
+             _extent = _extent.extend(entity.extent(context.graph()));
            }
 
-           return null;
-         };
+           return actionCircularize(entityID, context.projection);
+         }
 
-         this.fixes = function (context) {
-           var fixes = this.dynamicFixes ? this.dynamicFixes(context) : [];
-           var issue = this;
+         var operation = function operation() {
+           if (!_actions.length) return;
 
-           if (issue.severity === 'warning') {
-             // allow ignoring any issue that's not an error
-             fixes.push(new validationIssueFix({
-               title: _t.html('issues.fix.ignore_issue.title'),
-               icon: 'iD-icon-close',
-               onClick: function onClick() {
-                 context.validator().ignoreIssue(this.issue.id);
+           var combinedAction = function combinedAction(graph, t) {
+             _actions.forEach(function (action) {
+               if (!action.disabled(graph)) {
+                 graph = action(graph, t);
                }
-             }));
-           }
-
-           fixes.forEach(function (fix) {
-             // the id doesn't matter as long as it's unique to this issue/fix
-             fix.id = fix.title; // add a reference to the issue for use in actions
+             });
 
-             fix.issue = issue;
+             return graph;
+           };
 
-             if (fix.autoArgs) {
-               issue.autoFix = fix;
-             }
-           });
-           return fixes;
+           combinedAction.transitionable = true;
+           context.perform(combinedAction, operation.annotation());
+           window.setTimeout(function () {
+             context.validator().validate();
+           }, 300); // after any transition
          };
-       }
-       function validationIssueFix(attrs) {
-         this.title = attrs.title; // Required
 
-         this.onClick = attrs.onClick; // Optional - the function to run to apply the fix
+         operation.available = function () {
+           return _actions.length && selectedIDs.length === _actions.length;
+         }; // don't cache this because the visible extent could change
 
-         this.disabledReason = attrs.disabledReason; // Optional - a string explaining why the fix is unavailable, if any
 
-         this.icon = attrs.icon; // Optional - shows 'iD-icon-wrench' if not set
+         operation.disabled = function () {
+           if (!_actions.length) return '';
 
-         this.entityIds = attrs.entityIds || []; // Optional - used for hover-higlighting.
+           var actionDisableds = _actions.map(function (action) {
+             return action.disabled(context.graph());
+           }).filter(Boolean);
 
-         this.autoArgs = attrs.autoArgs; // Optional - pass [actions, annotation] arglist if this fix can automatically run
+           if (actionDisableds.length === _actions.length) {
+             // none of the features can be circularized
+             if (new Set(actionDisableds).size > 1) {
+               return 'multiple_blockers';
+             }
 
-         this.issue = null; // Generated link - added by validationIssue
-       }
+             return actionDisableds[0];
+           } else if (_extent.percentContainedIn(context.map().extent()) < 0.8) {
+             return 'too_large';
+           } else if (someMissing()) {
+             return 'not_downloaded';
+           } else if (selectedIDs.some(context.hasHiddenConnections)) {
+             return 'connected_to_hidden';
+           }
 
-       var buildRuleChecks = function buildRuleChecks() {
-         return {
-           equals: function equals(_equals) {
-             return function (tags) {
-               return Object.keys(_equals).every(function (k) {
-                 return _equals[k] === tags[k];
-               });
-             };
-           },
-           notEquals: function notEquals(_notEquals) {
-             return function (tags) {
-               return Object.keys(_notEquals).some(function (k) {
-                 return _notEquals[k] !== tags[k];
-               });
-             };
-           },
-           absence: function absence(_absence) {
-             return function (tags) {
-               return Object.keys(tags).indexOf(_absence) === -1;
-             };
-           },
-           presence: function presence(_presence) {
-             return function (tags) {
-               return Object.keys(tags).indexOf(_presence) > -1;
-             };
-           },
-           greaterThan: function greaterThan(_greaterThan) {
-             var key = Object.keys(_greaterThan)[0];
-             var value = _greaterThan[key];
-             return function (tags) {
-               return tags[key] > value;
-             };
-           },
-           greaterThanEqual: function greaterThanEqual(_greaterThanEqual) {
-             var key = Object.keys(_greaterThanEqual)[0];
-             var value = _greaterThanEqual[key];
-             return function (tags) {
-               return tags[key] >= value;
-             };
-           },
-           lessThan: function lessThan(_lessThan) {
-             var key = Object.keys(_lessThan)[0];
-             var value = _lessThan[key];
-             return function (tags) {
-               return tags[key] < value;
-             };
-           },
-           lessThanEqual: function lessThanEqual(_lessThanEqual) {
-             var key = Object.keys(_lessThanEqual)[0];
-             var value = _lessThanEqual[key];
-             return function (tags) {
-               return tags[key] <= value;
-             };
-           },
-           positiveRegex: function positiveRegex(_positiveRegex) {
-             var tagKey = Object.keys(_positiveRegex)[0];
+           return false;
 
-             var expression = _positiveRegex[tagKey].join('|');
+           function someMissing() {
+             if (context.inIntro()) return false;
+             var osm = context.connection();
 
-             var regex = new RegExp(expression);
-             return function (tags) {
-               return regex.test(tags[tagKey]);
-             };
-           },
-           negativeRegex: function negativeRegex(_negativeRegex) {
-             var tagKey = Object.keys(_negativeRegex)[0];
+             if (osm) {
+               var missing = _coords.filter(function (loc) {
+                 return !osm.isDataLoaded(loc);
+               });
 
-             var expression = _negativeRegex[tagKey].join('|');
+               if (missing.length) {
+                 missing.forEach(function (loc) {
+                   context.loadTileAtLoc(loc);
+                 });
+                 return true;
+               }
+             }
 
-             var regex = new RegExp(expression);
-             return function (tags) {
-               return !regex.test(tags[tagKey]);
-             };
+             return false;
            }
          };
-       };
 
-       var buildLineKeys = function buildLineKeys() {
-         return {
-           highway: {
-             rest_area: true,
-             services: true
-           },
-           railway: {
-             roundhouse: true,
-             station: true,
-             traverser: true,
-             turntable: true,
-             wash: true
-           }
+         operation.tooltip = function () {
+           var disable = operation.disabled();
+           return disable ? _t('operations.circularize.' + disable + '.' + _amount) : _t('operations.circularize.description.' + _amount);
          };
-       };
 
-       var serviceMapRules = {
-         init: function init() {
-           this._ruleChecks = buildRuleChecks();
-           this._validationRules = [];
-           this._areaKeys = osmAreaKeys;
-           this._lineKeys = buildLineKeys();
-         },
-         // list of rules only relevant to tag checks...
-         filterRuleChecks: function filterRuleChecks(selector) {
-           var _ruleChecks = this._ruleChecks;
-           return Object.keys(selector).reduce(function (rules, key) {
-             if (['geometry', 'error', 'warning'].indexOf(key) === -1) {
-               rules.push(_ruleChecks[key](selector[key]));
-             }
+         operation.annotation = function () {
+           return _t('operations.circularize.annotation.feature', {
+             n: _actions.length
+           });
+         };
 
-             return rules;
-           }, []);
-         },
-         // builds tagMap from mapcss-parse selector object...
-         buildTagMap: function buildTagMap(selector) {
-           var getRegexValues = function getRegexValues(regexes) {
-             return regexes.map(function (regex) {
-               return regex.replace(/\$|\^/g, '');
-             });
-           };
+         operation.id = 'circularize';
+         operation.keys = [_t('operations.circularize.key')];
+         operation.title = _t('operations.circularize.title');
+         operation.behavior = behaviorOperation(context).which(operation);
+         return operation;
+       }
 
-           var tagMap = Object.keys(selector).reduce(function (expectedTags, key) {
-             var values;
-             var isRegex = /regex/gi.test(key);
-             var isEqual = /equals/gi.test(key);
+       // For example, ⌘Z -> Ctrl+Z
 
-             if (isRegex || isEqual) {
-               Object.keys(selector[key]).forEach(function (selectorKey) {
-                 values = isEqual ? [selector[key][selectorKey]] : getRegexValues(selector[key][selectorKey]);
+       var uiCmd = function uiCmd(code) {
+         var detected = utilDetect();
 
-                 if (expectedTags.hasOwnProperty(selectorKey)) {
-                   values = values.concat(expectedTags[selectorKey]);
-                 }
+         if (detected.os === 'mac') {
+           return code;
+         }
 
-                 expectedTags[selectorKey] = values;
-               });
-             } else if (/(greater|less)Than(Equal)?|presence/g.test(key)) {
-               var tagKey = /presence/.test(key) ? selector[key] : Object.keys(selector[key])[0];
-               values = [selector[key][tagKey]];
+         if (detected.os === 'win') {
+           if (code === '⌘⇧Z') return 'Ctrl+Y';
+         }
 
-               if (expectedTags.hasOwnProperty(tagKey)) {
-                 values = values.concat(expectedTags[tagKey]);
-               }
+         var result = '',
+             replacements = {
+           '⌘': 'Ctrl',
+           '⇧': 'Shift',
+           '⌥': 'Alt',
+           '⌫': 'Backspace',
+           '⌦': 'Delete'
+         };
 
-               expectedTags[tagKey] = values;
-             }
+         for (var i = 0; i < code.length; i++) {
+           if (code[i] in replacements) {
+             result += replacements[code[i]] + (i < code.length - 1 ? '+' : '');
+           } else {
+             result += code[i];
+           }
+         }
 
-             return expectedTags;
-           }, {});
-           return tagMap;
-         },
-         // inspired by osmWay#isArea()
-         inferGeometry: function inferGeometry(tagMap) {
-           var _lineKeys = this._lineKeys;
-           var _areaKeys = this._areaKeys;
+         return result;
+       }; // return a display-focused string for a given keyboard code
 
-           var keyValueDoesNotImplyArea = function keyValueDoesNotImplyArea(key) {
-             return utilArrayIntersection(tagMap[key], Object.keys(_areaKeys[key])).length > 0;
-           };
+       uiCmd.display = function (code) {
+         if (code.length !== 1) return code;
+         var detected = utilDetect();
+         var mac = detected.os === 'mac';
+         var replacements = {
+           '⌘': mac ? '⌘ ' + _t('shortcuts.key.cmd') : _t('shortcuts.key.ctrl'),
+           '⇧': mac ? '⇧ ' + _t('shortcuts.key.shift') : _t('shortcuts.key.shift'),
+           '⌥': mac ? '⌥ ' + _t('shortcuts.key.option') : _t('shortcuts.key.alt'),
+           '⌃': mac ? '⌃ ' + _t('shortcuts.key.ctrl') : _t('shortcuts.key.ctrl'),
+           '⌫': mac ? '⌫ ' + _t('shortcuts.key.delete') : _t('shortcuts.key.backspace'),
+           '⌦': mac ? '⌦ ' + _t('shortcuts.key.del') : _t('shortcuts.key.del'),
+           '↖': mac ? '↖ ' + _t('shortcuts.key.pgup') : _t('shortcuts.key.pgup'),
+           '↘': mac ? '↘ ' + _t('shortcuts.key.pgdn') : _t('shortcuts.key.pgdn'),
+           '⇞': mac ? '⇞ ' + _t('shortcuts.key.home') : _t('shortcuts.key.home'),
+           '⇟': mac ? '⇟ ' + _t('shortcuts.key.end') : _t('shortcuts.key.end'),
+           '↵': mac ? '⏎ ' + _t('shortcuts.key.return') : _t('shortcuts.key.enter'),
+           '⎋': mac ? '⎋ ' + _t('shortcuts.key.esc') : _t('shortcuts.key.esc'),
+           '☰': mac ? '☰ ' + _t('shortcuts.key.menu') : _t('shortcuts.key.menu')
+         };
+         return replacements[code] || code;
+       };
 
-           var keyValueImpliesLine = function keyValueImpliesLine(key) {
-             return utilArrayIntersection(tagMap[key], Object.keys(_lineKeys[key])).length > 0;
-           };
+       function operationDelete(context, selectedIDs) {
+         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 = utilTotalExtent(selectedIDs, context.graph());
 
-           if (tagMap.hasOwnProperty('area')) {
-             if (tagMap.area.indexOf('yes') > -1) {
-               return 'area';
-             }
+         var operation = function operation() {
+           var nextSelectedID;
+           var nextSelectedLoc;
 
-             if (tagMap.area.indexOf('no') > -1) {
-               return 'line';
+           if (selectedIDs.length === 1) {
+             var id = selectedIDs[0];
+             var entity = context.entity(id);
+             var geometry = entity.geometry(context.graph());
+             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;
              }
            }
 
-           for (var key in tagMap) {
-             if (key in _areaKeys && !keyValueDoesNotImplyArea(key)) {
-               return 'area';
-             }
+           context.perform(action, operation.annotation());
+           context.validator().validate();
 
-             if (key in _lineKeys && keyValueImpliesLine(key)) {
-               return 'area';
+           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));
            }
+         };
 
-           return 'line';
-         },
-         // adds from mapcss-parse selector check...
-         addRule: function addRule(selector) {
-           var rule = {
-             // checks relevant to mapcss-selector
-             checks: this.filterRuleChecks(selector),
-             // true if all conditions for a tag error are true..
-             matches: function matches(entity) {
-               return this.checks.every(function (check) {
-                 return check(entity.tags);
-               });
-             },
-             // borrowed from Way#isArea()
-             inferredGeometry: this.inferGeometry(this.buildTagMap(selector), this._areaKeys),
-             geometryMatches: function geometryMatches(entity, graph) {
-               if (entity.type === 'node' || entity.type === 'relation') {
-                 return selector.geometry === entity.type;
-               } else if (entity.type === 'way') {
-                 return this.inferredGeometry === entity.geometry(graph);
-               }
-             },
-             // when geometries match and tag matches are present, return a warning...
-             findIssues: function findIssues(entity, graph, issues) {
-               if (this.geometryMatches(entity, graph) && this.matches(entity)) {
-                 var severity = Object.keys(selector).indexOf('error') > -1 ? 'error' : 'warning';
-                 var _message = selector[severity];
-                 issues.push(new validationIssue({
-                   type: 'maprules',
-                   severity: severity,
-                   message: function message() {
-                     return _message;
-                   },
-                   entityIds: [entity.id]
-                 }));
-               }
-             }
-           };
+         operation.available = function () {
+           return true;
+         };
 
-           this._validationRules.push(rule);
-         },
-         clearRules: function clearRules() {
-           this._validationRules = [];
-         },
-         // returns validationRules...
-         validationRules: function validationRules() {
-           return this._validationRules;
-         },
-         // returns ruleChecks
-         ruleChecks: function ruleChecks() {
-           return this._ruleChecks;
-         }
-       };
+         operation.disabled = function () {
+           if (extent.percentContainedIn(context.map().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';
+           }
 
-       var apibase$1 = 'https://nominatim.openstreetmap.org/';
-       var _inflight = {};
+           return false;
 
-       var _nominatimCache;
+           function someMissing() {
+             if (context.inIntro()) return false;
+             var osm = context.connection();
 
-       var serviceNominatim = {
-         init: function init() {
-           _inflight = {};
-           _nominatimCache = new RBush();
-         },
-         reset: function reset() {
-           Object.values(_inflight).forEach(function (controller) {
-             controller.abort();
-           });
-           _inflight = {};
-           _nominatimCache = new RBush();
-         },
-         countryCode: function countryCode(location, callback) {
-           this.reverse(location, function (err, result) {
-             if (err) {
-               return callback(err);
-             } else if (result.address) {
-               return callback(null, result.address.country_code);
-             } else {
-               return callback('Unable to geocode', null);
+             if (osm) {
+               var missing = coords.filter(function (loc) {
+                 return !osm.isDataLoaded(loc);
+               });
+
+               if (missing.length) {
+                 missing.forEach(function (loc) {
+                   context.loadTileAtLoc(loc);
+                 });
+                 return true;
+               }
              }
-           });
-         },
-         reverse: function reverse(loc, callback) {
-           var cached = _nominatimCache.search({
-             minX: loc[0],
-             minY: loc[1],
-             maxX: loc[0],
-             maxY: loc[1]
-           });
 
-           if (cached.length > 0) {
-             if (callback) callback(null, cached[0].data);
-             return;
+             return false;
            }
 
-           var params = {
-             zoom: 13,
-             format: 'json',
-             addressdetails: 1,
-             lat: loc[1],
-             lon: loc[0]
-           };
-           var url = apibase$1 + 'reverse?' + utilQsString(params);
-           if (_inflight[url]) return;
-           var controller = new AbortController();
-           _inflight[url] = controller;
-           d3_json(url, {
-             signal: controller.signal
-           }).then(function (result) {
-             delete _inflight[url];
-
-             if (result && result.error) {
-               throw new Error(result.error);
-             }
+           function hasWikidataTag(id) {
+             var entity = context.entity(id);
+             return entity.tags.wikidata && entity.tags.wikidata.trim().length > 0;
+           }
 
-             var extent = geoExtent(loc).padByMeters(200);
+           function incompleteRelation(id) {
+             var entity = context.entity(id);
+             return entity.type === 'relation' && !entity.isComplete(context.graph());
+           }
 
-             _nominatimCache.insert(Object.assign(extent.bbox(), {
-               data: result
-             }));
+           function protectedMember(id) {
+             var entity = context.entity(id);
+             if (entity.type !== 'way') return false;
+             var parents = context.graph().parentRelations(entity);
 
-             if (callback) callback(null, result);
-           })["catch"](function (err) {
-             delete _inflight[url];
-             if (err.name === 'AbortError') return;
-             if (callback) callback(err.message);
-           });
-         },
-         search: function search(val, callback) {
-           var searchVal = encodeURIComponent(val);
-           var url = apibase$1 + 'search/' + searchVal + '?limit=10&format=json';
-           if (_inflight[url]) return;
-           var controller = new AbortController();
-           _inflight[url] = controller;
-           d3_json(url, {
-             signal: controller.signal
-           }).then(function (result) {
-             delete _inflight[url];
+             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 (result && result.error) {
-               throw new Error(result.error);
+               if (type === 'route' || type === 'boundary' || type === 'multipolygon' && role === 'outer') {
+                 return true;
+               }
              }
 
-             if (callback) callback(null, result);
-           })["catch"](function (err) {
-             delete _inflight[url];
-             if (err.name === 'AbortError') return;
-             if (callback) callback(err.message);
-           });
-         }
-       };
-
-       var apibase$2 = 'https://openstreetcam.org';
-       var maxResults$1 = 1000;
-       var tileZoom$1 = 14;
-       var tiler$4 = utilTiler().zoomExtent([tileZoom$1, tileZoom$1]).skipNullIsland(true);
-       var dispatch$5 = dispatch('loadedImages');
-       var imgZoom = d3_zoom().extent([[0, 0], [320, 240]]).translateExtent([[0, 0], [320, 240]]).scaleExtent([1, 15]);
-
-       var _oscCache;
+             return false;
+           }
+         };
 
-       var _oscSelectedImage;
+         operation.tooltip = function () {
+           var disable = operation.disabled();
+           return disable ? _t('operations.delete.' + disable + '.' + multi) : _t('operations.delete.description.' + multi);
+         };
 
-       var _loadViewerPromise$1;
+         operation.annotation = function () {
+           return selectedIDs.length === 1 ? _t('operations.delete.annotation.' + context.graph().geometry(selectedIDs[0])) : _t('operations.delete.annotation.feature', {
+             n: selectedIDs.length
+           });
+         };
 
-       function abortRequest$4(controller) {
-         controller.abort();
+         operation.id = 'delete';
+         operation.keys = [uiCmd('⌘⌫'), uiCmd('⌘⌦'), uiCmd('⌦')];
+         operation.title = _t('operations.delete.title');
+         operation.behavior = behaviorOperation(context).which(operation);
+         return operation;
        }
 
-       function maxPageAtZoom$1(z) {
-         if (z < 15) return 2;
-         if (z === 15) return 5;
-         if (z === 16) return 10;
-         if (z === 17) return 20;
-         if (z === 18) return 40;
-         if (z > 18) return 80;
-       }
+       function operationOrthogonalize(context, selectedIDs) {
+         var _extent;
 
-       function loadTiles$1(which, url, projection) {
-         var currZoom = Math.floor(geoScaleToZoom(projection.scale()));
-         var tiles = tiler$4.getTiles(projection); // abort inflight requests that are no longer needed
+         var _type;
 
-         var cache = _oscCache[which];
-         Object.keys(cache.inflight).forEach(function (k) {
-           var wanted = tiles.find(function (tile) {
-             return k.indexOf(tile.id + ',') === 0;
-           });
+         var _actions = selectedIDs.map(chooseAction).filter(Boolean);
 
-           if (!wanted) {
-             abortRequest$4(cache.inflight[k]);
-             delete cache.inflight[k];
-           }
-         });
-         tiles.forEach(function (tile) {
-           loadNextTilePage$1(which, currZoom, url, tile);
+         var _amount = _actions.length === 1 ? 'single' : 'multiple';
+
+         var _coords = utilGetAllNodes(selectedIDs, context.graph()).map(function (n) {
+           return n.loc;
          });
-       }
 
-       function loadNextTilePage$1(which, currZoom, url, tile) {
-         var cache = _oscCache[which];
-         var bbox = tile.extent.bbox();
-         var maxPages = maxPageAtZoom$1(currZoom);
-         var nextPage = cache.nextPage[tile.id] || 1;
-         var params = utilQsString({
-           ipp: maxResults$1,
-           page: nextPage,
-           // client_id: clientId,
-           bbTopLeft: [bbox.maxY, bbox.minX].join(','),
-           bbBottomRight: [bbox.minY, bbox.maxX].join(',')
-         }, true);
-         if (nextPage > maxPages) return;
-         var id = tile.id + ',' + String(nextPage);
-         if (cache.loaded[id] || cache.inflight[id]) return;
-         var controller = new AbortController();
-         cache.inflight[id] = controller;
-         var options = {
-           method: 'POST',
-           signal: controller.signal,
-           body: params,
-           headers: {
-             'Content-Type': 'application/x-www-form-urlencoded'
-           }
-         };
-         d3_json(url, options).then(function (data) {
-           cache.loaded[id] = true;
-           delete cache.inflight[id];
+         function chooseAction(entityID) {
+           var entity = context.entity(entityID);
+           var geometry = entity.geometry(context.graph());
 
-           if (!data || !data.currentPageItems || !data.currentPageItems.length) {
-             throw new Error('No Data');
-           }
+           if (!_extent) {
+             _extent = entity.extent(context.graph());
+           } else {
+             _extent = _extent.extend(entity.extent(context.graph()));
+           } // square a line/area
 
-           var features = data.currentPageItems.map(function (item) {
-             var loc = [+item.lng, +item.lat];
-             var d;
 
-             if (which === 'images') {
-               d = {
-                 loc: loc,
-                 key: item.id,
-                 ca: +item.heading,
-                 captured_at: item.shot_date || item.date_added,
-                 captured_by: item.username,
-                 imagePath: item.lth_name,
-                 sequence_id: item.sequence_id,
-                 sequence_index: +item.sequence_index
-               }; // cache sequence info
+           if (entity.type === 'way' && new Set(entity.nodes).size > 2) {
+             if (_type && _type !== 'feature') return null;
+             _type = 'feature';
+             return actionOrthogonalize(entityID, context.projection); // square a single vertex
+           } else if (geometry === 'vertex') {
+             if (_type && _type !== 'corner') return null;
+             _type = 'corner';
+             var graph = context.graph();
+             var parents = graph.parentWays(entity);
 
-               var seq = _oscCache.sequences[d.sequence_id];
+             if (parents.length === 1) {
+               var way = parents[0];
 
-               if (!seq) {
-                 seq = {
-                   rotation: 0,
-                   images: []
-                 };
-                 _oscCache.sequences[d.sequence_id] = seq;
+               if (way.nodes.indexOf(entityID) !== -1) {
+                 return actionOrthogonalize(way.id, context.projection, entityID);
                }
-
-               seq.images[d.sequence_index] = d;
-               _oscCache.images.forImageKey[d.key] = d; // cache imageKey -> image
              }
-
-             return {
-               minX: loc[0],
-               minY: loc[1],
-               maxX: loc[0],
-               maxY: loc[1],
-               data: d
-             };
-           });
-           cache.rtree.load(features);
-
-           if (data.currentPageItems.length === maxResults$1) {
-             // more pages to load
-             cache.nextPage[tile.id] = nextPage + 1;
-             loadNextTilePage$1(which, currZoom, url, tile);
-           } else {
-             cache.nextPage[tile.id] = Infinity; // no more pages to load
            }
 
-           if (which === 'images') {
-             dispatch$5.call('loadedImages');
-           }
-         })["catch"](function () {
-           cache.loaded[id] = true;
-           delete cache.inflight[id];
-         });
-       } // partition viewport into higher zoom tiles
+           return null;
+         }
 
+         var operation = function operation() {
+           if (!_actions.length) return;
 
-       function partitionViewport$1(projection) {
-         var z = geoScaleToZoom(projection.scale());
-         var z2 = Math.ceil(z * 2) / 2 + 2.5; // round to next 0.5 and add 2.5
+           var combinedAction = function combinedAction(graph, t) {
+             _actions.forEach(function (action) {
+               if (!action.disabled(graph)) {
+                 graph = action(graph, t);
+               }
+             });
 
-         var tiler = utilTiler().zoomExtent([z2, z2]);
-         return tiler.getTiles(projection).map(function (tile) {
-           return tile.extent;
-         });
-       } // no more than `limit` results per partition.
+             return graph;
+           };
+
+           combinedAction.transitionable = true;
+           context.perform(combinedAction, operation.annotation());
+           window.setTimeout(function () {
+             context.validator().validate();
+           }, 300); // after any transition
+         };
 
+         operation.available = function () {
+           return _actions.length && selectedIDs.length === _actions.length;
+         }; // don't cache this because the visible extent could change
 
-       function searchLimited$1(limit, projection, rtree) {
-         limit = limit || 5;
-         return partitionViewport$1(projection).reduce(function (result, extent) {
-           var found = rtree.search(extent.bbox()).slice(0, limit).map(function (d) {
-             return d.data;
-           });
-           return found.length ? result.concat(found) : result;
-         }, []);
-       }
 
-       var serviceOpenstreetcam = {
-         init: function init() {
-           if (!_oscCache) {
-             this.reset();
-           }
+         operation.disabled = function () {
+           if (!_actions.length) return '';
 
-           this.event = utilRebind(this, dispatch$5, 'on');
-         },
-         reset: function reset() {
-           if (_oscCache) {
-             Object.values(_oscCache.images.inflight).forEach(abortRequest$4);
-           }
+           var actionDisableds = _actions.map(function (action) {
+             return action.disabled(context.graph());
+           }).filter(Boolean);
 
-           _oscCache = {
-             images: {
-               inflight: {},
-               loaded: {},
-               nextPage: {},
-               rtree: new RBush(),
-               forImageKey: {}
-             },
-             sequences: {}
-           };
-           _oscSelectedImage = null;
-         },
-         images: function images(projection) {
-           var limit = 5;
-           return searchLimited$1(limit, projection, _oscCache.images.rtree);
-         },
-         sequences: function sequences(projection) {
-           var viewport = projection.clipExtent();
-           var min = [viewport[0][0], viewport[1][1]];
-           var max = [viewport[1][0], viewport[0][1]];
-           var bbox = geoExtent(projection.invert(min), projection.invert(max)).bbox();
-           var sequenceKeys = {}; // all sequences for images in viewport
+           if (actionDisableds.length === _actions.length) {
+             // none of the features can be squared
+             if (new Set(actionDisableds).size > 1) {
+               return 'multiple_blockers';
+             }
 
-           _oscCache.images.rtree.search(bbox).forEach(function (d) {
-             sequenceKeys[d.data.sequence_id] = true;
-           }); // make linestrings from those sequences
+             return actionDisableds[0];
+           } else if (_extent && _extent.percentContainedIn(context.map().extent()) < 0.8) {
+             return 'too_large';
+           } else if (someMissing()) {
+             return 'not_downloaded';
+           } else if (selectedIDs.some(context.hasHiddenConnections)) {
+             return 'connected_to_hidden';
+           }
 
+           return false;
 
-           var lineStrings = [];
-           Object.keys(sequenceKeys).forEach(function (sequenceKey) {
-             var seq = _oscCache.sequences[sequenceKey];
-             var images = seq && seq.images;
+           function someMissing() {
+             if (context.inIntro()) return false;
+             var osm = context.connection();
 
-             if (images) {
-               lineStrings.push({
-                 type: 'LineString',
-                 coordinates: images.map(function (d) {
-                   return d.loc;
-                 }).filter(Boolean),
-                 properties: {
-                   captured_at: images[0] ? images[0].captured_at : null,
-                   captured_by: images[0] ? images[0].captured_by : null,
-                   key: sequenceKey
-                 }
+             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 lineStrings;
-         },
-         cachedImage: function cachedImage(imageKey) {
-           return _oscCache.images.forImageKey[imageKey];
-         },
-         loadImages: function loadImages(projection) {
-           var url = apibase$2 + '/1.0/list/nearby-photos/';
-           loadTiles$1('images', url, projection);
-         },
-         ensureViewerLoaded: function ensureViewerLoaded(context) {
-           if (_loadViewerPromise$1) return _loadViewerPromise$1; // add osc-wrapper
 
-           var wrap = context.container().select('.photoviewer').selectAll('.osc-wrapper').data([0]);
-           var that = this;
-           var wrapEnter = wrap.enter().append('div').attr('class', 'photo-wrapper osc-wrapper').classed('hide', true).call(imgZoom.on('zoom', zoomPan)).on('dblclick.zoom', null);
-           wrapEnter.append('div').attr('class', 'photo-attribution fillD');
-           var controlsEnter = wrapEnter.append('div').attr('class', 'photo-controls-wrap').append('div').attr('class', 'photo-controls');
-           controlsEnter.append('button').on('click.back', step(-1)).html('◄');
-           controlsEnter.append('button').on('click.rotate-ccw', rotate(-90)).html('⤿');
-           controlsEnter.append('button').on('click.rotate-cw', rotate(90)).html('⤾');
-           controlsEnter.append('button').on('click.forward', step(1)).html('►');
-           wrapEnter.append('div').attr('class', 'osc-image-wrap'); // Register viewer resize handler
+             return false;
+           }
+         };
 
-           context.ui().photoviewer.on('resize.openstreetcam', function (dimensions) {
-             imgZoom = d3_zoom().extent([[0, 0], dimensions]).translateExtent([[0, 0], dimensions]).scaleExtent([1, 15]).on('zoom', zoomPan);
+         operation.tooltip = function () {
+           var disable = operation.disabled();
+           return disable ? _t('operations.orthogonalize.' + disable + '.' + _amount) : _t('operations.orthogonalize.description.' + _type + '.' + _amount);
+         };
+
+         operation.annotation = function () {
+           return _t('operations.orthogonalize.annotation.' + _type, {
+             n: _actions.length
            });
+         };
 
-           function zoomPan(d3_event) {
-             var t = d3_event.transform;
-             context.container().select('.photoviewer .osc-image-wrap').call(utilSetTransform, t.x, t.y, t.k);
-           }
+         operation.id = 'orthogonalize';
+         operation.keys = [_t('operations.orthogonalize.key')];
+         operation.title = _t('operations.orthogonalize.title');
+         operation.behavior = behaviorOperation(context).which(operation);
+         return operation;
+       }
 
-           function rotate(deg) {
-             return function () {
-               if (!_oscSelectedImage) return;
-               var sequenceKey = _oscSelectedImage.sequence_id;
-               var sequence = _oscCache.sequences[sequenceKey];
-               if (!sequence) return;
-               var r = sequence.rotation || 0;
-               r += deg;
-               if (r > 180) r -= 360;
-               if (r < -180) r += 360;
-               sequence.rotation = r;
-               var wrap = context.container().select('.photoviewer .osc-wrapper');
-               wrap.transition().duration(100).call(imgZoom.transform, identity$2);
-               wrap.selectAll('.osc-image').transition().duration(100).style('transform', 'rotate(' + r + 'deg)');
-             };
-           }
+       function operationReflectShort(context, selectedIDs) {
+         return operationReflect(context, selectedIDs, 'short');
+       }
+       function operationReflectLong(context, selectedIDs) {
+         return operationReflect(context, selectedIDs, 'long');
+       }
+       function operationReflect(context, selectedIDs, axis) {
+         axis = axis || 'long';
+         var multi = selectedIDs.length === 1 ? 'single' : 'multiple';
+         var nodes = utilGetAllNodes(selectedIDs, context.graph());
+         var coords = nodes.map(function (n) {
+           return n.loc;
+         });
+         var extent = utilTotalExtent(selectedIDs, context.graph());
 
-           function step(stepBy) {
-             return function () {
-               if (!_oscSelectedImage) return;
-               var sequenceKey = _oscSelectedImage.sequence_id;
-               var sequence = _oscCache.sequences[sequenceKey];
-               if (!sequence) return;
-               var nextIndex = _oscSelectedImage.sequence_index + stepBy;
-               var nextImage = sequence.images[nextIndex];
-               if (!nextImage) return;
-               context.map().centerEase(nextImage.loc);
-               that.selectImage(context, nextImage.key);
-             };
-           } // don't need any async loading so resolve immediately
+         var operation = function operation() {
+           var action = actionReflect(selectedIDs, context.projection).useLongAxis(Boolean(axis === 'long'));
+           context.perform(action, operation.annotation());
+           window.setTimeout(function () {
+             context.validator().validate();
+           }, 300); // after any transition
+         };
 
+         operation.available = function () {
+           return nodes.length >= 3;
+         }; // don't cache this because the visible extent could change
 
-           _loadViewerPromise$1 = Promise.resolve();
-           return _loadViewerPromise$1;
-         },
-         showViewer: function showViewer(context) {
-           var viewer = context.container().select('.photoviewer').classed('hide', false);
-           var isHidden = viewer.selectAll('.photo-wrapper.osc-wrapper.hide').size();
 
-           if (isHidden) {
-             viewer.selectAll('.photo-wrapper:not(.osc-wrapper)').classed('hide', true);
-             viewer.selectAll('.photo-wrapper.osc-wrapper').classed('hide', false);
+         operation.disabled = function () {
+           if (extent.percentContainedIn(context.map().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(incompleteRelation)) {
+             return 'incomplete_relation';
            }
 
-           return this;
-         },
-         hideViewer: function hideViewer(context) {
-           _oscSelectedImage = null;
-           this.updateUrlImage(null);
-           var viewer = context.container().select('.photoviewer');
-           if (!viewer.empty()) viewer.datum(null);
-           viewer.classed('hide', true).selectAll('.photo-wrapper').classed('hide', true);
-           context.container().selectAll('.viewfield-group, .sequence, .icon-sign').classed('currentView', false);
-           return this.setStyles(context, null, true);
-         },
-         selectImage: function selectImage(context, imageKey) {
-           var d = this.cachedImage(imageKey);
-           _oscSelectedImage = d;
-           this.updateUrlImage(imageKey);
-           var viewer = context.container().select('.photoviewer');
-           if (!viewer.empty()) viewer.datum(d);
-           this.setStyles(context, null, true);
-           context.container().selectAll('.icon-sign').classed('currentView', false);
-           if (!d) return this;
-           var wrap = context.container().select('.photoviewer .osc-wrapper');
-           var imageWrap = wrap.selectAll('.osc-image-wrap');
-           var attribution = wrap.selectAll('.photo-attribution').html('');
-           wrap.transition().duration(100).call(imgZoom.transform, identity$2);
-           imageWrap.selectAll('.osc-image').remove();
+           return false;
 
-           if (d) {
-             var sequence = _oscCache.sequences[d.sequence_id];
-             var r = sequence && sequence.rotation || 0;
-             imageWrap.append('img').attr('class', 'osc-image').attr('src', apibase$2 + '/' + d.imagePath).style('transform', 'rotate(' + r + 'deg)');
+           function someMissing() {
+             if (context.inIntro()) return false;
+             var osm = context.connection();
 
-             if (d.captured_by) {
-               attribution.append('a').attr('class', 'captured_by').attr('target', '_blank').attr('href', 'https://openstreetcam.org/user/' + encodeURIComponent(d.captured_by)).html('@' + d.captured_by);
-               attribution.append('span').html('|');
-             }
+             if (osm) {
+               var missing = coords.filter(function (loc) {
+                 return !osm.isDataLoaded(loc);
+               });
 
-             if (d.captured_at) {
-               attribution.append('span').attr('class', 'captured_at').html(localeDateString(d.captured_at));
-               attribution.append('span').html('|');
+               if (missing.length) {
+                 missing.forEach(function (loc) {
+                   context.loadTileAtLoc(loc);
+                 });
+                 return true;
+               }
              }
 
-             attribution.append('a').attr('class', 'image-link').attr('target', '_blank').attr('href', 'https://openstreetcam.org/details/' + d.sequence_id + '/' + d.sequence_index).html('openstreetcam.org');
+             return false;
            }
 
-           return this;
-
-           function localeDateString(s) {
-             if (!s) return null;
-             var options = {
-               day: 'numeric',
-               month: 'short',
-               year: 'numeric'
-             };
-             var d = new Date(s);
-             if (isNaN(d.getTime())) return null;
-             return d.toLocaleDateString(_mainLocalizer.localeCode(), options);
-           }
-         },
-         getSelectedImage: function getSelectedImage() {
-           return _oscSelectedImage;
-         },
-         getSequenceKeyForImage: function getSequenceKeyForImage(d) {
-           return d && d.sequence_id;
-         },
-         // Updates the currently highlighted sequence and selected bubble.
-         // Reset is only necessary when interacting with the viewport because
-         // this implicitly changes the currently selected bubble/sequence
-         setStyles: function setStyles(context, hovered, reset) {
-           if (reset) {
-             // reset all layers
-             context.container().selectAll('.viewfield-group').classed('highlighted', false).classed('hovered', false).classed('currentView', false);
-             context.container().selectAll('.sequence').classed('highlighted', false).classed('currentView', false);
+           function incompleteRelation(id) {
+             var entity = context.entity(id);
+             return entity.type === 'relation' && !entity.isComplete(context.graph());
            }
+         };
 
-           var hoveredImageKey = hovered && hovered.key;
-           var hoveredSequenceKey = this.getSequenceKeyForImage(hovered);
-           var hoveredSequence = hoveredSequenceKey && _oscCache.sequences[hoveredSequenceKey];
-           var hoveredImageKeys = hoveredSequence && hoveredSequence.images.map(function (d) {
-             return d.key;
-           }) || [];
-           var viewer = context.container().select('.photoviewer');
-           var selected = viewer.empty() ? undefined : viewer.datum();
-           var selectedImageKey = selected && selected.key;
-           var selectedSequenceKey = this.getSequenceKeyForImage(selected);
-           var selectedSequence = selectedSequenceKey && _oscCache.sequences[selectedSequenceKey];
-           var selectedImageKeys = selectedSequence && selectedSequence.images.map(function (d) {
-             return d.key;
-           }) || []; // highlight sibling viewfields on either the selected or the hovered sequences
+         operation.tooltip = function () {
+           var disable = operation.disabled();
+           return disable ? _t('operations.reflect.' + disable + '.' + multi) : _t('operations.reflect.description.' + axis + '.' + multi);
+         };
 
-           var highlightedImageKeys = utilArrayUnion(hoveredImageKeys, selectedImageKeys);
-           context.container().selectAll('.layer-openstreetcam .viewfield-group').classed('highlighted', function (d) {
-             return highlightedImageKeys.indexOf(d.key) !== -1;
-           }).classed('hovered', function (d) {
-             return d.key === hoveredImageKey;
-           }).classed('currentView', function (d) {
-             return d.key === selectedImageKey;
+         operation.annotation = function () {
+           return _t('operations.reflect.annotation.' + axis + '.feature', {
+             n: selectedIDs.length
            });
-           context.container().selectAll('.layer-openstreetcam .sequence').classed('highlighted', function (d) {
-             return d.properties.key === hoveredSequenceKey;
-           }).classed('currentView', function (d) {
-             return d.properties.key === selectedSequenceKey;
-           }); // update viewfields if needed
-
-           context.container().selectAll('.viewfield-group .viewfield').attr('d', viewfieldPath);
+         };
 
-           function viewfieldPath() {
-             var d = this.parentNode.__data__;
+         operation.id = 'reflect-' + axis;
+         operation.keys = [_t('operations.reflect.key.' + axis)];
+         operation.title = _t('operations.reflect.title.' + axis);
+         operation.behavior = behaviorOperation(context).which(operation);
+         return operation;
+       }
 
-             if (d.pano && d.key !== selectedImageKey) {
-               return 'M 8,13 m -10,0 a 10,10 0 1,0 20,0 a 10,10 0 1,0 -20,0';
-             } else {
-               return 'M 6,9 C 8,8.4 8,8.4 10,9 L 16,-2 C 12,-5 4,-5 0,-2 z';
-             }
-           }
+       function operationMove(context, selectedIDs) {
+         var multi = selectedIDs.length === 1 ? 'single' : 'multiple';
+         var nodes = utilGetAllNodes(selectedIDs, context.graph());
+         var coords = nodes.map(function (n) {
+           return n.loc;
+         });
+         var extent = utilTotalExtent(selectedIDs, context.graph());
 
-           return this;
-         },
-         updateUrlImage: function updateUrlImage(imageKey) {
-           if (!window.mocha) {
-             var hash = utilStringQs(window.location.hash);
+         var operation = function operation() {
+           context.enter(modeMove(context, selectedIDs));
+         };
 
-             if (imageKey) {
-               hash.photo = 'openstreetcam/' + imageKey;
-             } else {
-               delete hash.photo;
-             }
+         operation.available = function () {
+           return selectedIDs.length > 0;
+         };
 
-             window.location.replace('#' + utilQsString(hash, true));
+         operation.disabled = function () {
+           if (extent.percentContainedIn(context.map().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(incompleteRelation)) {
+             return 'incomplete_relation';
            }
-         },
-         cache: function cache() {
-           return _oscCache;
-         }
-       };
 
-       var FORCED$f = fails(function () {
-         return new Date(NaN).toJSON() !== null
-           || Date.prototype.toJSON.call({ toISOString: function () { return 1; } }) !== 1;
-       });
+           return false;
 
-       // `Date.prototype.toJSON` method
-       // https://tc39.es/ecma262/#sec-date.prototype.tojson
-       _export({ target: 'Date', proto: true, forced: FORCED$f }, {
-         // eslint-disable-next-line no-unused-vars -- required for `.length`
-         toJSON: function toJSON(key) {
-           var O = toObject(this);
-           var pv = toPrimitive(O);
-           return typeof pv == 'number' && !isFinite(pv) ? null : O.toISOString();
-         }
-       });
+           function someMissing() {
+             if (context.inIntro()) return false;
+             var osm = context.connection();
 
-       // `URL.prototype.toJSON` method
-       // https://url.spec.whatwg.org/#dom-url-tojson
-       _export({ target: 'URL', proto: true, enumerable: true }, {
-         toJSON: function toJSON() {
-           return URL.prototype.toString.call(this);
-         }
-       });
+             if (osm) {
+               var missing = coords.filter(function (loc) {
+                 return !osm.isDataLoaded(loc);
+               });
 
-       /**
-        * Checks if `value` is the
-        * [language type](http://www.ecma-international.org/ecma-262/7.0/#sec-ecmascript-language-types)
-        * of `Object`. (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`)
-        *
-        * @static
-        * @memberOf _
-        * @since 0.1.0
-        * @category Lang
-        * @param {*} value The value to check.
-        * @returns {boolean} Returns `true` if `value` is an object, else `false`.
-        * @example
-        *
-        * _.isObject({});
-        * // => true
-        *
-        * _.isObject([1, 2, 3]);
-        * // => true
-        *
-        * _.isObject(_.noop);
-        * // => true
-        *
-        * _.isObject(null);
-        * // => false
-        */
-       function isObject$1(value) {
-         var type = _typeof(value);
+               if (missing.length) {
+                 missing.forEach(function (loc) {
+                   context.loadTileAtLoc(loc);
+                 });
+                 return true;
+               }
+             }
+
+             return false;
+           }
+
+           function incompleteRelation(id) {
+             var entity = context.entity(id);
+             return entity.type === 'relation' && !entity.isComplete(context.graph());
+           }
+         };
 
-         return value != null && (type == 'object' || type == 'function');
-       }
+         operation.tooltip = function () {
+           var disable = operation.disabled();
+           return disable ? _t('operations.move.' + disable + '.' + multi) : _t('operations.move.description.' + multi);
+         };
 
-       /** Detect free variable `global` from Node.js. */
-       var freeGlobal = (typeof global === "undefined" ? "undefined" : _typeof(global)) == 'object' && global && global.Object === Object && global;
+         operation.annotation = function () {
+           return selectedIDs.length === 1 ? _t('operations.move.annotation.' + context.graph().geometry(selectedIDs[0])) : _t('operations.move.annotation.feature', {
+             n: selectedIDs.length
+           });
+         };
 
-       /** Detect free variable `self`. */
+         operation.id = 'move';
+         operation.keys = [_t('operations.move.key')];
+         operation.title = _t('operations.move.title');
+         operation.behavior = behaviorOperation(context).which(operation);
+         operation.mouseOnly = true;
+         return operation;
+       }
 
-       var freeSelf = (typeof self === "undefined" ? "undefined" : _typeof(self)) == 'object' && self && self.Object === Object && self;
-       /** Used as a reference to the global object. */
+       function modeRotate(context, entityIDs) {
+         var _tolerancePx = 4; // see also behaviorDrag, behaviorSelect, modeMove
 
-       var root$1 = freeGlobal || freeSelf || Function('return this')();
+         var mode = {
+           id: 'rotate',
+           button: 'browse'
+         };
+         var keybinding = utilKeybinding('rotate');
+         var behaviors = [behaviorEdit(context), operationCircularize(context, entityIDs).behavior, operationDelete(context, entityIDs).behavior, operationMove(context, entityIDs).behavior, operationOrthogonalize(context, entityIDs).behavior, operationReflectLong(context, entityIDs).behavior, operationReflectShort(context, entityIDs).behavior];
+         var annotation = entityIDs.length === 1 ? _t('operations.rotate.annotation.' + context.graph().geometry(entityIDs[0])) : _t('operations.rotate.annotation.feature', {
+           n: entityIDs.length
+         });
 
-       /**
-        * Gets the timestamp of the number of milliseconds that have elapsed since
-        * the Unix epoch (1 January 1970 00:00:00 UTC).
-        *
-        * @static
-        * @memberOf _
-        * @since 2.4.0
-        * @category Date
-        * @returns {number} Returns the timestamp.
-        * @example
-        *
-        * _.defer(function(stamp) {
-        *   console.log(_.now() - stamp);
-        * }, _.now());
-        * // => Logs the number of milliseconds it took for the deferred invocation.
-        */
+         var _prevGraph;
 
-       var now$1 = function now() {
-         return root$1.Date.now();
-       };
+         var _prevAngle;
 
-       /** Used to match a single whitespace character. */
-       var reWhitespace = /\s/;
-       /**
-        * Used by `_.trim` and `_.trimEnd` to get the index of the last non-whitespace
-        * character of `string`.
-        *
-        * @private
-        * @param {string} string The string to inspect.
-        * @returns {number} Returns the index of the last non-whitespace character.
-        */
+         var _prevTransform;
 
-       function trimmedEndIndex(string) {
-         var index = string.length;
+         var _pivot; // use pointer events on supported platforms; fallback to mouse events
 
-         while (index-- && reWhitespace.test(string.charAt(index))) {}
 
-         return index;
-       }
+         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
 
-       /** Used to match leading whitespace. */
+         function doRotate(d3_event) {
+           var fn;
 
-       var reTrimStart = /^\s+/;
-       /**
-        * The base implementation of `_.trim`.
-        *
-        * @private
-        * @param {string} string The string to trim.
-        * @returns {string} Returns the trimmed string.
-        */
+           if (context.graph() !== _prevGraph) {
+             fn = context.perform;
+           } else {
+             fn = context.replace;
+           } // projection changed, recalculate _pivot
 
-       function baseTrim(string) {
-         return string ? string.slice(0, trimmedEndIndex(string) + 1).replace(reTrimStart, '') : string;
-       }
 
-       /** Built-in value references. */
+           var projection = context.projection;
+           var currTransform = projection.transform();
 
-       var _Symbol = root$1.Symbol;
+           if (!_prevTransform || currTransform.k !== _prevTransform.k || currTransform.x !== _prevTransform.x || currTransform.y !== _prevTransform.y) {
+             var nodes = utilGetAllNodes(entityIDs, context.graph());
+             var points = nodes.map(function (n) {
+               return projection(n.loc);
+             });
+             _pivot = getPivot(points);
+             _prevAngle = undefined;
+           }
 
-       /** Used for built-in method references. */
+           var currMouse = context.map().mouse(d3_event);
+           var currAngle = Math.atan2(currMouse[1] - _pivot[1], currMouse[0] - _pivot[0]);
+           if (typeof _prevAngle === 'undefined') _prevAngle = currAngle;
+           var delta = currAngle - _prevAngle;
+           fn(actionRotate(entityIDs, _pivot, delta, projection));
+           _prevTransform = currTransform;
+           _prevAngle = currAngle;
+           _prevGraph = context.graph();
+         }
 
-       var objectProto = Object.prototype;
-       /** Used to check objects for own properties. */
+         function getPivot(points) {
+           var _pivot;
 
-       var hasOwnProperty$1 = objectProto.hasOwnProperty;
-       /**
-        * Used to resolve the
-        * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)
-        * of values.
-        */
+           if (points.length === 1) {
+             _pivot = points[0];
+           } else if (points.length === 2) {
+             _pivot = geoVecInterp(points[0], points[1], 0.5);
+           } else {
+             var polygonHull = d3_polygonHull(points);
 
-       var nativeObjectToString = objectProto.toString;
-       /** Built-in value references. */
+             if (polygonHull.length === 2) {
+               _pivot = geoVecInterp(points[0], points[1], 0.5);
+             } else {
+               _pivot = d3_polygonCentroid(d3_polygonHull(points));
+             }
+           }
 
-       var symToStringTag = _Symbol ? _Symbol.toStringTag : undefined;
-       /**
-        * A specialized version of `baseGetTag` which ignores `Symbol.toStringTag` values.
-        *
-        * @private
-        * @param {*} value The value to query.
-        * @returns {string} Returns the raw `toStringTag`.
-        */
+           return _pivot;
+         }
 
-       function getRawTag(value) {
-         var isOwn = hasOwnProperty$1.call(value, symToStringTag),
-             tag = value[symToStringTag];
+         function finish(d3_event) {
+           d3_event.stopPropagation();
+           context.replace(actionNoop(), annotation);
+           context.enter(modeSelect(context, entityIDs));
+         }
 
-         try {
-           value[symToStringTag] = undefined;
-           var unmasked = true;
-         } catch (e) {}
+         function cancel() {
+           if (_prevGraph) context.pop(); // remove the rotate
 
-         var result = nativeObjectToString.call(value);
+           context.enter(modeSelect(context, entityIDs));
+         }
 
-         if (unmasked) {
-           if (isOwn) {
-             value[symToStringTag] = tag;
-           } else {
-             delete value[symToStringTag];
-           }
+         function undone() {
+           context.enter(modeBrowse(context));
          }
 
-         return result;
-       }
+         mode.enter = function () {
+           _prevGraph = null;
+           context.features().forceVisible(entityIDs);
+           behaviors.forEach(context.install);
+           var downEvent;
+           context.surface().on(_pointerPrefix + 'down.modeRotate', function (d3_event) {
+             downEvent = d3_event;
+           });
+           select(window).on(_pointerPrefix + 'move.modeRotate', doRotate, true).on(_pointerPrefix + 'up.modeRotate', function (d3_event) {
+             if (!downEvent) return;
+             var mapNode = context.container().select('.main-map').node();
+             var pointGetter = utilFastMouse(mapNode);
+             var p1 = pointGetter(downEvent);
+             var p2 = pointGetter(d3_event);
+             var dist = geoVecLength(p1, p2);
+             if (dist <= _tolerancePx) finish(d3_event);
+             downEvent = null;
+           }, true);
+           context.history().on('undone.modeRotate', undone);
+           keybinding.on('⎋', cancel).on('↩', finish);
+           select(document).call(keybinding);
+         };
 
-       /** Used for built-in method references. */
-       var objectProto$1 = Object.prototype;
-       /**
-        * Used to resolve the
-        * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)
-        * of values.
-        */
+         mode.exit = function () {
+           behaviors.forEach(context.uninstall);
+           context.surface().on(_pointerPrefix + 'down.modeRotate', null);
+           select(window).on(_pointerPrefix + 'move.modeRotate', null, true).on(_pointerPrefix + 'up.modeRotate', null, true);
+           context.history().on('undone.modeRotate', null);
+           select(document).call(keybinding.unbind);
+           context.features().forceVisible([]);
+         };
 
-       var nativeObjectToString$1 = objectProto$1.toString;
-       /**
-        * Converts `value` to a string using `Object.prototype.toString`.
-        *
-        * @private
-        * @param {*} value The value to convert.
-        * @returns {string} Returns the converted string.
-        */
+         mode.selectedIDs = function () {
+           if (!arguments.length) return entityIDs; // no assign
 
-       function objectToString$1(value) {
-         return nativeObjectToString$1.call(value);
-       }
+           return mode;
+         };
 
-       /** `Object#toString` result references. */
+         return mode;
+       }
 
-       var nullTag = '[object Null]',
-           undefinedTag = '[object Undefined]';
-       /** Built-in value references. */
+       function operationRotate(context, selectedIDs) {
+         var multi = selectedIDs.length === 1 ? 'single' : 'multiple';
+         var nodes = utilGetAllNodes(selectedIDs, context.graph());
+         var coords = nodes.map(function (n) {
+           return n.loc;
+         });
+         var extent = utilTotalExtent(selectedIDs, context.graph());
 
-       var symToStringTag$1 = _Symbol ? _Symbol.toStringTag : undefined;
-       /**
-        * The base implementation of `getTag` without fallbacks for buggy environments.
-        *
-        * @private
-        * @param {*} value The value to query.
-        * @returns {string} Returns the `toStringTag`.
-        */
+         var operation = function operation() {
+           context.enter(modeRotate(context, selectedIDs));
+         };
 
-       function baseGetTag(value) {
-         if (value == null) {
-           return value === undefined ? undefinedTag : nullTag;
-         }
+         operation.available = function () {
+           return nodes.length >= 2;
+         };
 
-         return symToStringTag$1 && symToStringTag$1 in Object(value) ? getRawTag(value) : objectToString$1(value);
-       }
+         operation.disabled = function () {
+           if (extent.percentContainedIn(context.map().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(incompleteRelation)) {
+             return 'incomplete_relation';
+           }
 
-       /**
-        * Checks if `value` is object-like. A value is object-like if it's not `null`
-        * and has a `typeof` result of "object".
-        *
-        * @static
-        * @memberOf _
-        * @since 4.0.0
-        * @category Lang
-        * @param {*} value The value to check.
-        * @returns {boolean} Returns `true` if `value` is object-like, else `false`.
-        * @example
-        *
-        * _.isObjectLike({});
-        * // => true
-        *
-        * _.isObjectLike([1, 2, 3]);
-        * // => true
-        *
-        * _.isObjectLike(_.noop);
-        * // => false
-        *
-        * _.isObjectLike(null);
-        * // => false
-        */
-       function isObjectLike(value) {
-         return value != null && _typeof(value) == 'object';
-       }
+           return false;
 
-       /** `Object#toString` result references. */
+           function someMissing() {
+             if (context.inIntro()) return false;
+             var osm = context.connection();
 
-       var symbolTag = '[object Symbol]';
-       /**
-        * Checks if `value` is classified as a `Symbol` primitive or object.
-        *
-        * @static
-        * @memberOf _
-        * @since 4.0.0
-        * @category Lang
-        * @param {*} value The value to check.
-        * @returns {boolean} Returns `true` if `value` is a symbol, else `false`.
-        * @example
-        *
-        * _.isSymbol(Symbol.iterator);
-        * // => true
-        *
-        * _.isSymbol('abc');
-        * // => false
-        */
+             if (osm) {
+               var missing = coords.filter(function (loc) {
+                 return !osm.isDataLoaded(loc);
+               });
 
-       function isSymbol$1(value) {
-         return _typeof(value) == 'symbol' || isObjectLike(value) && baseGetTag(value) == symbolTag;
-       }
+               if (missing.length) {
+                 missing.forEach(function (loc) {
+                   context.loadTileAtLoc(loc);
+                 });
+                 return true;
+               }
+             }
 
-       /** Used as references for various `Number` constants. */
+             return false;
+           }
 
-       var NAN = 0 / 0;
-       /** Used to detect bad signed hexadecimal string values. */
+           function incompleteRelation(id) {
+             var entity = context.entity(id);
+             return entity.type === 'relation' && !entity.isComplete(context.graph());
+           }
+         };
 
-       var reIsBadHex = /^[-+]0x[0-9a-f]+$/i;
-       /** Used to detect binary string values. */
+         operation.tooltip = function () {
+           var disable = operation.disabled();
+           return disable ? _t('operations.rotate.' + disable + '.' + multi) : _t('operations.rotate.description.' + multi);
+         };
 
-       var reIsBinary = /^0b[01]+$/i;
-       /** Used to detect octal string values. */
+         operation.annotation = function () {
+           return selectedIDs.length === 1 ? _t('operations.rotate.annotation.' + context.graph().geometry(selectedIDs[0])) : _t('operations.rotate.annotation.feature', {
+             n: selectedIDs.length
+           });
+         };
 
-       var reIsOctal = /^0o[0-7]+$/i;
-       /** Built-in method references without a dependency on `root`. */
+         operation.id = 'rotate';
+         operation.keys = [_t('operations.rotate.key')];
+         operation.title = _t('operations.rotate.title');
+         operation.behavior = behaviorOperation(context).which(operation);
+         operation.mouseOnly = true;
+         return operation;
+       }
 
-       var freeParseInt = parseInt;
-       /**
-        * Converts `value` to a number.
-        *
-        * @static
-        * @memberOf _
-        * @since 4.0.0
-        * @category Lang
-        * @param {*} value The value to process.
-        * @returns {number} Returns the number.
-        * @example
-        *
-        * _.toNumber(3.2);
-        * // => 3.2
-        *
-        * _.toNumber(Number.MIN_VALUE);
-        * // => 5e-324
-        *
-        * _.toNumber(Infinity);
-        * // => Infinity
-        *
-        * _.toNumber('3.2');
-        * // => 3.2
-        */
+       function modeMove(context, entityIDs, baseGraph) {
+         var _tolerancePx = 4; // see also behaviorDrag, behaviorSelect, modeRotate
 
-       function toNumber$1(value) {
-         if (typeof value == 'number') {
-           return value;
-         }
+         var mode = {
+           id: 'move',
+           button: 'browse'
+         };
+         var keybinding = utilKeybinding('move');
+         var behaviors = [behaviorEdit(context), operationCircularize(context, entityIDs).behavior, operationDelete(context, entityIDs).behavior, operationOrthogonalize(context, entityIDs).behavior, operationReflectLong(context, entityIDs).behavior, operationReflectShort(context, entityIDs).behavior, operationRotate(context, entityIDs).behavior];
+         var annotation = entityIDs.length === 1 ? _t('operations.move.annotation.' + context.graph().geometry(entityIDs[0])) : _t('operations.move.annotation.feature', {
+           n: entityIDs.length
+         });
 
-         if (isSymbol$1(value)) {
-           return NAN;
-         }
+         var _prevGraph;
 
-         if (isObject$1(value)) {
-           var other = typeof value.valueOf == 'function' ? value.valueOf() : value;
-           value = isObject$1(other) ? other + '' : other;
-         }
+         var _cache;
 
-         if (typeof value != 'string') {
-           return value === 0 ? value : +value;
-         }
+         var _origin;
 
-         value = baseTrim(value);
-         var isBinary = reIsBinary.test(value);
-         return isBinary || reIsOctal.test(value) ? freeParseInt(value.slice(2), isBinary ? 2 : 8) : reIsBadHex.test(value) ? NAN : +value;
-       }
+         var _nudgeInterval; // use pointer events on supported platforms; fallback to mouse events
 
-       /** Error message constants. */
 
-       var FUNC_ERROR_TEXT = 'Expected a function';
-       /* Built-in method references for those with the same name as other `lodash` methods. */
+         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
 
-       var nativeMax = Math.max,
-           nativeMin = Math.min;
-       /**
-        * Creates a debounced function that delays invoking `func` until after `wait`
-        * milliseconds have elapsed since the last time the debounced function was
-        * invoked. The debounced function comes with a `cancel` method to cancel
-        * delayed `func` invocations and a `flush` method to immediately invoke them.
-        * Provide `options` to indicate whether `func` should be invoked on the
-        * leading and/or trailing edge of the `wait` timeout. The `func` is invoked
-        * with the last arguments provided to the debounced function. Subsequent
-        * calls to the debounced function return the result of the last `func`
-        * invocation.
-        *
-        * **Note:** If `leading` and `trailing` options are `true`, `func` is
-        * invoked on the trailing edge of the timeout only if the debounced function
-        * is invoked more than once during the `wait` timeout.
-        *
-        * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
-        * until to the next tick, similar to `setTimeout` with a timeout of `0`.
-        *
-        * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
-        * for details over the differences between `_.debounce` and `_.throttle`.
-        *
-        * @static
-        * @memberOf _
-        * @since 0.1.0
-        * @category Function
-        * @param {Function} func The function to debounce.
-        * @param {number} [wait=0] The number of milliseconds to delay.
-        * @param {Object} [options={}] The options object.
-        * @param {boolean} [options.leading=false]
-        *  Specify invoking on the leading edge of the timeout.
-        * @param {number} [options.maxWait]
-        *  The maximum time `func` is allowed to be delayed before it's invoked.
-        * @param {boolean} [options.trailing=true]
-        *  Specify invoking on the trailing edge of the timeout.
-        * @returns {Function} Returns the new debounced function.
-        * @example
-        *
-        * // Avoid costly calculations while the window size is in flux.
-        * jQuery(window).on('resize', _.debounce(calculateLayout, 150));
-        *
-        * // Invoke `sendMail` when clicked, debouncing subsequent calls.
-        * jQuery(element).on('click', _.debounce(sendMail, 300, {
-        *   'leading': true,
-        *   'trailing': false
-        * }));
-        *
-        * // Ensure `batchLog` is invoked once after 1 second of debounced calls.
-        * var debounced = _.debounce(batchLog, 250, { 'maxWait': 1000 });
-        * var source = new EventSource('/stream');
-        * jQuery(source).on('message', debounced);
-        *
-        * // Cancel the trailing debounced invocation.
-        * jQuery(window).on('popstate', debounced.cancel);
-        */
+         function doMove(nudge) {
+           nudge = nudge || [0, 0];
+           var fn;
 
-       function debounce(func, wait, options) {
-         var lastArgs,
-             lastThis,
-             maxWait,
-             result,
-             timerId,
-             lastCallTime,
-             lastInvokeTime = 0,
-             leading = false,
-             maxing = false,
-             trailing = true;
+           if (_prevGraph !== context.graph()) {
+             _cache = {};
+             _origin = context.map().mouseCoordinates();
+             fn = context.perform;
+           } else {
+             fn = context.overwrite;
+           }
 
-         if (typeof func != 'function') {
-           throw new TypeError(FUNC_ERROR_TEXT);
+           var currMouse = context.map().mouse();
+           var origMouse = context.projection(_origin);
+           var delta = geoVecSubtract(geoVecSubtract(currMouse, origMouse), nudge);
+           fn(actionMove(entityIDs, delta, context.projection, _cache));
+           _prevGraph = context.graph();
          }
 
-         wait = toNumber$1(wait) || 0;
+         function startNudge(nudge) {
+           if (_nudgeInterval) window.clearInterval(_nudgeInterval);
+           _nudgeInterval = window.setInterval(function () {
+             context.map().pan(nudge);
+             doMove(nudge);
+           }, 50);
+         }
 
-         if (isObject$1(options)) {
-           leading = !!options.leading;
-           maxing = 'maxWait' in options;
-           maxWait = maxing ? nativeMax(toNumber$1(options.maxWait) || 0, wait) : maxWait;
-           trailing = 'trailing' in options ? !!options.trailing : trailing;
+         function stopNudge() {
+           if (_nudgeInterval) {
+             window.clearInterval(_nudgeInterval);
+             _nudgeInterval = null;
+           }
          }
 
-         function invokeFunc(time) {
-           var args = lastArgs,
-               thisArg = lastThis;
-           lastArgs = lastThis = undefined;
-           lastInvokeTime = time;
-           result = func.apply(thisArg, args);
-           return result;
+         function move() {
+           doMove();
+           var nudge = geoViewportEdge(context.map().mouse(), context.map().dimensions());
+
+           if (nudge) {
+             startNudge(nudge);
+           } else {
+             stopNudge();
+           }
          }
 
-         function leadingEdge(time) {
-           // Reset any `maxWait` timer.
-           lastInvokeTime = time; // Start the timer for the trailing edge.
+         function finish(d3_event) {
+           d3_event.stopPropagation();
+           context.replace(actionNoop(), annotation);
+           context.enter(modeSelect(context, entityIDs));
+           stopNudge();
+         }
 
-           timerId = setTimeout(timerExpired, wait); // Invoke the leading edge.
+         function cancel() {
+           if (baseGraph) {
+             while (context.graph() !== baseGraph) {
+               context.pop();
+             } // reset to baseGraph
 
-           return leading ? invokeFunc(time) : result;
-         }
 
-         function remainingWait(time) {
-           var timeSinceLastCall = time - lastCallTime,
-               timeSinceLastInvoke = time - lastInvokeTime,
-               timeWaiting = wait - timeSinceLastCall;
-           return maxing ? nativeMin(timeWaiting, maxWait - timeSinceLastInvoke) : timeWaiting;
-         }
+             context.enter(modeBrowse(context));
+           } else {
+             if (_prevGraph) context.pop(); // remove the move
 
-         function shouldInvoke(time) {
-           var timeSinceLastCall = time - lastCallTime,
-               timeSinceLastInvoke = time - lastInvokeTime; // Either this is the first call, activity has stopped and we're at the
-           // trailing edge, the system time has gone backwards and we're treating
-           // it as the trailing edge, or we've hit the `maxWait` limit.
+             context.enter(modeSelect(context, entityIDs));
+           }
 
-           return lastCallTime === undefined || timeSinceLastCall >= wait || timeSinceLastCall < 0 || maxing && timeSinceLastInvoke >= maxWait;
+           stopNudge();
          }
 
-         function timerExpired() {
-           var time = now$1();
+         function undone() {
+           context.enter(modeBrowse(context));
+         }
 
-           if (shouldInvoke(time)) {
-             return trailingEdge(time);
-           } // Restart the timer.
+         mode.enter = function () {
+           _origin = context.map().mouseCoordinates();
+           _prevGraph = null;
+           _cache = {};
+           context.features().forceVisible(entityIDs);
+           behaviors.forEach(context.install);
+           var downEvent;
+           context.surface().on(_pointerPrefix + 'down.modeMove', function (d3_event) {
+             downEvent = d3_event;
+           });
+           select(window).on(_pointerPrefix + 'move.modeMove', move, true).on(_pointerPrefix + 'up.modeMove', function (d3_event) {
+             if (!downEvent) return;
+             var mapNode = context.container().select('.main-map').node();
+             var pointGetter = utilFastMouse(mapNode);
+             var p1 = pointGetter(downEvent);
+             var p2 = pointGetter(d3_event);
+             var dist = geoVecLength(p1, p2);
+             if (dist <= _tolerancePx) finish(d3_event);
+             downEvent = null;
+           }, true);
+           context.history().on('undone.modeMove', undone);
+           keybinding.on('⎋', cancel).on('↩', finish);
+           select(document).call(keybinding);
+         };
 
+         mode.exit = function () {
+           stopNudge();
+           behaviors.forEach(function (behavior) {
+             context.uninstall(behavior);
+           });
+           context.surface().on(_pointerPrefix + 'down.modeMove', null);
+           select(window).on(_pointerPrefix + 'move.modeMove', null, true).on(_pointerPrefix + 'up.modeMove', null, true);
+           context.history().on('undone.modeMove', null);
+           select(document).call(keybinding.unbind);
+           context.features().forceVisible([]);
+         };
 
-           timerId = setTimeout(timerExpired, remainingWait(time));
-         }
+         mode.selectedIDs = function () {
+           if (!arguments.length) return entityIDs; // no assign
 
-         function trailingEdge(time) {
-           timerId = undefined; // Only invoke if we have `lastArgs` which means `func` has been
-           // debounced at least once.
+           return mode;
+         };
 
-           if (trailing && lastArgs) {
-             return invokeFunc(time);
-           }
+         return mode;
+       }
 
-           lastArgs = lastThis = undefined;
-           return result;
-         }
+       function behaviorPaste(context) {
+         function doPaste(d3_event) {
+           // prevent paste during low zoom selection
+           if (!context.map().withinEditableZoom()) return;
+           d3_event.preventDefault();
+           var baseGraph = context.graph();
+           var mouse = context.map().mouse();
+           var projection = context.projection;
+           var viewport = geoExtent(projection.clipExtent()).polygon();
+           if (!geoPointInPolygon(mouse, viewport)) return;
+           var oldIDs = context.copyIDs();
+           if (!oldIDs.length) return;
+           var extent = geoExtent();
+           var oldGraph = context.copyGraph();
+           var newIDs = [];
+           var action = actionCopyEntities(oldIDs, oldGraph);
+           context.perform(action);
+           var copies = action.copies();
+           var originals = new Set();
+           Object.values(copies).forEach(function (entity) {
+             originals.add(entity.id);
+           });
 
-         function cancel() {
-           if (timerId !== undefined) {
-             clearTimeout(timerId);
-           }
+           for (var id in copies) {
+             var oldEntity = oldGraph.entity(id);
+             var newEntity = copies[id];
 
-           lastInvokeTime = 0;
-           lastArgs = lastCallTime = lastThis = timerId = undefined;
-         }
+             extent._extend(oldEntity.extent(oldGraph)); // Exclude child nodes from newIDs if their parent way was also copied.
 
-         function flush() {
-           return timerId === undefined ? result : trailingEdge(now$1());
-         }
 
-         function debounced() {
-           var time = now$1(),
-               isInvoking = shouldInvoke(time);
-           lastArgs = arguments;
-           lastThis = this;
-           lastCallTime = time;
+             var parents = context.graph().parentWays(newEntity);
+             var parentCopied = parents.some(function (parent) {
+               return originals.has(parent.id);
+             });
 
-           if (isInvoking) {
-             if (timerId === undefined) {
-               return leadingEdge(lastCallTime);
+             if (!parentCopied) {
+               newIDs.push(newEntity.id);
              }
+           } // Put pasted objects where mouse pointer is..
 
-             if (maxing) {
-               // Handle invocations in a tight loop.
-               clearTimeout(timerId);
-               timerId = setTimeout(timerExpired, wait);
-               return invokeFunc(lastCallTime);
-             }
-           }
 
-           if (timerId === undefined) {
-             timerId = setTimeout(timerExpired, wait);
-           }
+           var copyPoint = context.copyLonLat() && projection(context.copyLonLat()) || projection(extent.center());
+           var delta = geoVecSubtract(mouse, copyPoint);
+           context.perform(actionMove(newIDs, delta, projection));
+           context.enter(modeMove(context, newIDs, baseGraph));
+         }
 
-           return result;
+         function behavior() {
+           context.keybinding().on(uiCmd('⌘V'), doPaste);
+           return behavior;
          }
 
-         debounced.cancel = cancel;
-         debounced.flush = flush;
-         return debounced;
+         behavior.off = function () {
+           context.keybinding().off(uiCmd('⌘V'));
+         };
+
+         return behavior;
        }
 
-       /** Error message constants. */
+       // `String.prototype.repeat` method
+       // https://tc39.es/ecma262/#sec-string.prototype.repeat
+       _export({ target: 'String', proto: true }, {
+         repeat: stringRepeat
+       });
 
-       var FUNC_ERROR_TEXT$1 = 'Expected a function';
-       /**
-        * Creates a throttled function that only invokes `func` at most once per
-        * every `wait` milliseconds. The throttled function comes with a `cancel`
-        * method to cancel delayed `func` invocations and a `flush` method to
-        * immediately invoke them. Provide `options` to indicate whether `func`
-        * should be invoked on the leading and/or trailing edge of the `wait`
-        * timeout. The `func` is invoked with the last arguments provided to the
-        * throttled function. Subsequent calls to the throttled function return the
-        * result of the last `func` invocation.
-        *
-        * **Note:** If `leading` and `trailing` options are `true`, `func` is
-        * invoked on the trailing edge of the timeout only if the throttled function
-        * is invoked more than once during the `wait` timeout.
-        *
-        * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
-        * until to the next tick, similar to `setTimeout` with a timeout of `0`.
-        *
-        * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
-        * for details over the differences between `_.throttle` and `_.debounce`.
-        *
-        * @static
-        * @memberOf _
-        * @since 0.1.0
-        * @category Function
-        * @param {Function} func The function to throttle.
-        * @param {number} [wait=0] The number of milliseconds to throttle invocations to.
-        * @param {Object} [options={}] The options object.
-        * @param {boolean} [options.leading=true]
-        *  Specify invoking on the leading edge of the timeout.
-        * @param {boolean} [options.trailing=true]
-        *  Specify invoking on the trailing edge of the timeout.
-        * @returns {Function} Returns the new throttled function.
-        * @example
-        *
-        * // Avoid excessively updating the position while scrolling.
-        * jQuery(window).on('scroll', _.throttle(updatePosition, 100));
-        *
-        * // Invoke `renewToken` when the click event is fired, but not more than once every 5 minutes.
-        * var throttled = _.throttle(renewToken, 300000, { 'trailing': false });
-        * jQuery(element).on('click', throttled);
-        *
-        * // Cancel the trailing throttled invocation.
-        * jQuery(window).on('popstate', throttled.cancel);
+       /*
+           `behaviorDrag` is like `d3_behavior.drag`, with the following differences:
+
+           * The `origin` function is expected to return an [x, y] tuple rather than an
+             {x, y} object.
+           * The events are `start`, `move`, and `end`.
+             (https://github.com/mbostock/d3/issues/563)
+           * The `start` event is not dispatched until the first cursor movement occurs.
+             (https://github.com/mbostock/d3/pull/368)
+           * The `move` event has a `point` and `delta` [x, y] tuple properties rather
+             than `x`, `y`, `dx`, and `dy` properties.
+           * The `end` event is not dispatched if no movement occurs.
+           * An `off` function is available that unbinds the drag's internal event handlers.
         */
 
-       function throttle(func, wait, options) {
-         var leading = true,
-             trailing = true;
+       function behaviorDrag() {
+         var dispatch = dispatch$8('start', 'move', 'end'); // see also behaviorSelect
 
-         if (typeof func != 'function') {
-           throw new TypeError(FUNC_ERROR_TEXT$1);
-         }
+         var _tolerancePx = 1; // keep this low to facilitate pixel-perfect micromapping
 
-         if (isObject$1(options)) {
-           leading = 'leading' in options ? !!options.leading : leading;
-           trailing = 'trailing' in options ? !!options.trailing : trailing;
-         }
+         var _penTolerancePx = 4; // styluses can be touchy so require greater movement - #1981
 
-         return debounce(func, wait, {
-           'leading': leading,
-           'maxWait': wait,
-           'trailing': trailing
-         });
-       }
+         var _origin = null;
+         var _selector = '';
 
-       var hashes = createCommonjsModule(function (module, exports) {
-         /**
-          * jshashes - https://github.com/h2non/jshashes
-          * Released under the "New BSD" license
-          *
-          * Algorithms specification:
-          *
-          * MD5 - http://www.ietf.org/rfc/rfc1321.txt
-          * RIPEMD-160 - http://homes.esat.kuleuven.be/~bosselae/ripemd160.html
-          * SHA1   - http://csrc.nist.gov/publications/fips/fips180-4/fips-180-4.pdf
-          * SHA256 - http://csrc.nist.gov/publications/fips/fips180-4/fips-180-4.pdf
-          * SHA512 - http://csrc.nist.gov/publications/fips/fips180-4/fips-180-4.pdf
-          * HMAC - http://www.ietf.org/rfc/rfc2104.txt
-          */
-         (function () {
-           var Hashes;
+         var _targetNode;
 
-           function utf8Encode(str) {
-             var x,
-                 y,
-                 output = '',
-                 i = -1,
-                 l;
+         var _targetEntity;
 
-             if (str && str.length) {
-               l = str.length;
+         var _surface;
 
-               while ((i += 1) < l) {
-                 /* Decode utf-16 surrogate pairs */
-                 x = str.charCodeAt(i);
-                 y = i + 1 < l ? str.charCodeAt(i + 1) : 0;
+         var _pointerId; // use pointer events on supported platforms; fallback to mouse events
 
-                 if (0xD800 <= x && x <= 0xDBFF && 0xDC00 <= y && y <= 0xDFFF) {
-                   x = 0x10000 + ((x & 0x03FF) << 10) + (y & 0x03FF);
-                   i += 1;
-                 }
-                 /* Encode output as utf-8 */
 
+         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
 
-                 if (x <= 0x7F) {
-                   output += String.fromCharCode(x);
-                 } else if (x <= 0x7FF) {
-                   output += String.fromCharCode(0xC0 | x >>> 6 & 0x1F, 0x80 | x & 0x3F);
-                 } else if (x <= 0xFFFF) {
-                   output += String.fromCharCode(0xE0 | x >>> 12 & 0x0F, 0x80 | x >>> 6 & 0x3F, 0x80 | x & 0x3F);
-                 } else if (x <= 0x1FFFFF) {
-                   output += String.fromCharCode(0xF0 | x >>> 18 & 0x07, 0x80 | x >>> 12 & 0x3F, 0x80 | x >>> 6 & 0x3F, 0x80 | x & 0x3F);
-                 }
-               }
-             }
+         var d3_event_userSelectProperty = utilPrefixCSSProperty('UserSelect');
 
-             return output;
+         var d3_event_userSelectSuppress = function d3_event_userSelectSuppress() {
+           var selection$1 = selection();
+           var select = selection$1.style(d3_event_userSelectProperty);
+           selection$1.style(d3_event_userSelectProperty, 'none');
+           return function () {
+             selection$1.style(d3_event_userSelectProperty, select);
+           };
+         };
+
+         function pointerdown(d3_event) {
+           if (_pointerId) return;
+           _pointerId = d3_event.pointerId || 'mouse';
+           _targetNode = this; // only force reflow once per drag
+
+           var pointerLocGetter = utilFastMouse(_surface || _targetNode.parentNode);
+           var offset;
+           var startOrigin = pointerLocGetter(d3_event);
+           var started = false;
+           var selectEnable = d3_event_userSelectSuppress();
+           select(window).on(_pointerPrefix + 'move.drag', pointermove).on(_pointerPrefix + 'up.drag pointercancel.drag', pointerup, true);
+
+           if (_origin) {
+             offset = _origin.call(_targetNode, _targetEntity);
+             offset = [offset[0] - startOrigin[0], offset[1] - startOrigin[1]];
+           } else {
+             offset = [0, 0];
            }
 
-           function utf8Decode(str) {
-             var i,
-                 ac,
-                 c1,
-                 c2,
-                 c3,
-                 arr = [],
-                 l;
-             i = ac = c1 = c2 = c3 = 0;
+           d3_event.stopPropagation();
 
-             if (str && str.length) {
-               l = str.length;
-               str += '';
+           function pointermove(d3_event) {
+             if (_pointerId !== (d3_event.pointerId || 'mouse')) return;
+             var p = pointerLocGetter(d3_event);
 
-               while (i < l) {
-                 c1 = str.charCodeAt(i);
-                 ac += 1;
+             if (!started) {
+               var dist = geoVecLength(startOrigin, p);
+               var tolerance = d3_event.pointerType === 'pen' ? _penTolerancePx : _tolerancePx; // don't start until the drag has actually moved somewhat
 
-                 if (c1 < 128) {
-                   arr[ac] = String.fromCharCode(c1);
-                   i += 1;
-                 } else if (c1 > 191 && c1 < 224) {
-                   c2 = str.charCodeAt(i + 1);
-                   arr[ac] = String.fromCharCode((c1 & 31) << 6 | c2 & 63);
-                   i += 2;
-                 } else {
-                   c2 = str.charCodeAt(i + 1);
-                   c3 = str.charCodeAt(i + 2);
-                   arr[ac] = String.fromCharCode((c1 & 15) << 12 | (c2 & 63) << 6 | c3 & 63);
-                   i += 3;
-                 }
-               }
+               if (dist < tolerance) return;
+               started = true;
+               dispatch.call('start', this, d3_event, _targetEntity); // Don't send a `move` event in the same cycle as `start` since dragging
+               // a midpoint will convert the target to a node.
+             } else {
+               startOrigin = p;
+               d3_event.stopPropagation();
+               d3_event.preventDefault();
+               var dx = p[0] - startOrigin[0];
+               var dy = p[1] - startOrigin[1];
+               dispatch.call('move', this, d3_event, _targetEntity, [p[0] + offset[0], p[1] + offset[1]], [dx, dy]);
              }
-
-             return arr.join('');
            }
-           /**
-            * Add integers, wrapping at 2^32. This uses 16-bit operations internally
-            * to work around bugs in some JS interpreters.
-            */
 
+           function pointerup(d3_event) {
+             if (_pointerId !== (d3_event.pointerId || 'mouse')) return;
+             _pointerId = null;
+
+             if (started) {
+               dispatch.call('end', this, d3_event, _targetEntity);
+               d3_event.preventDefault();
+             }
 
-           function safe_add(x, y) {
-             var lsw = (x & 0xFFFF) + (y & 0xFFFF),
-                 msw = (x >> 16) + (y >> 16) + (lsw >> 16);
-             return msw << 16 | lsw & 0xFFFF;
+             select(window).on(_pointerPrefix + 'move.drag', null).on(_pointerPrefix + 'up.drag pointercancel.drag', null);
+             selectEnable();
            }
-           /**
-            * Bitwise rotate a 32-bit number to the left.
-            */
+         }
+
+         function behavior(selection) {
+           var matchesSelector = utilPrefixDOMProperty('matchesSelector');
+           var delegate = pointerdown;
 
+           if (_selector) {
+             delegate = function delegate(d3_event) {
+               var root = this;
+               var target = d3_event.target;
 
-           function bit_rol(num, cnt) {
-             return num << cnt | num >>> 32 - cnt;
+               for (; target && target !== root; target = target.parentNode) {
+                 var datum = target.__data__;
+                 _targetEntity = datum instanceof osmNote ? datum : datum && datum.properties && datum.properties.entity;
+
+                 if (_targetEntity && target[matchesSelector](_selector)) {
+                   return pointerdown.call(target, d3_event);
+                 }
+               }
+             };
            }
-           /**
-            * Convert a raw string to a hex string
-            */
 
+           selection.on(_pointerPrefix + 'down.drag' + _selector, delegate);
+         }
 
-           function rstr2hex(input, hexcase) {
-             var hex_tab = hexcase ? '0123456789ABCDEF' : '0123456789abcdef',
-                 output = '',
-                 x,
-                 i = 0,
-                 l = input.length;
+         behavior.off = function (selection) {
+           selection.on(_pointerPrefix + 'down.drag' + _selector, null);
+         };
 
-             for (; i < l; i += 1) {
-               x = input.charCodeAt(i);
-               output += hex_tab.charAt(x >>> 4 & 0x0F) + hex_tab.charAt(x & 0x0F);
-             }
+         behavior.selector = function (_) {
+           if (!arguments.length) return _selector;
+           _selector = _;
+           return behavior;
+         };
 
-             return output;
-           }
-           /**
-            * Convert an array of big-endian words to a string
-            */
+         behavior.origin = function (_) {
+           if (!arguments.length) return _origin;
+           _origin = _;
+           return behavior;
+         };
 
+         behavior.cancel = function () {
+           select(window).on(_pointerPrefix + 'move.drag', null).on(_pointerPrefix + 'up.drag pointercancel.drag', null);
+           return behavior;
+         };
 
-           function binb2rstr(input) {
-             var i,
-                 l = input.length * 32,
-                 output = '';
+         behavior.targetNode = function (_) {
+           if (!arguments.length) return _targetNode;
+           _targetNode = _;
+           return behavior;
+         };
 
-             for (i = 0; i < l; i += 8) {
-               output += String.fromCharCode(input[i >> 5] >>> 24 - i % 32 & 0xFF);
-             }
+         behavior.targetEntity = function (_) {
+           if (!arguments.length) return _targetEntity;
+           _targetEntity = _;
+           return behavior;
+         };
 
-             return output;
-           }
-           /**
-            * Convert an array of little-endian words to a string
-            */
+         behavior.surface = function (_) {
+           if (!arguments.length) return _surface;
+           _surface = _;
+           return behavior;
+         };
 
+         return utilRebind(behavior, dispatch, 'on');
+       }
 
-           function binl2rstr(input) {
-             var i,
-                 l = input.length * 32,
-                 output = '';
+       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);
 
-             for (i = 0; i < l; i += 8) {
-               output += String.fromCharCode(input[i >> 5] >>> i % 32 & 0xFF);
-             }
+         var _nudgeInterval;
 
-             return output;
-           }
-           /**
-            * Convert a raw string to an array of little-endian words
-            * Characters >255 have their high-byte silently ignored.
-            */
+         var _restoreSelectedIDs = [];
+         var _wasMidpoint = false;
+         var _isCancelled = false;
 
+         var _activeEntity;
 
-           function rstr2binl(input) {
-             var i,
-                 l = input.length * 8,
-                 output = Array(input.length >> 2),
-                 lo = output.length;
+         var _startLoc;
 
-             for (i = 0; i < lo; i += 1) {
-               output[i] = 0;
-             }
+         var _lastLoc;
 
-             for (i = 0; i < l; i += 8) {
-               output[i >> 5] |= (input.charCodeAt(i / 8) & 0xFF) << i % 32;
-             }
+         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);
+         }
 
-             return output;
+         function stopNudge() {
+           if (_nudgeInterval) {
+             window.clearInterval(_nudgeInterval);
+             _nudgeInterval = null;
            }
-           /**
-            * Convert a raw string to an array of big-endian words
-            * Characters >255 have their high-byte silently ignored.
-            */
+         }
 
+         function moveAnnotation(entity) {
+           return _t('operations.move.annotation.' + entity.geometry(context.graph()));
+         }
 
-           function rstr2binb(input) {
-             var i,
-                 l = input.length * 8,
-                 output = Array(input.length >> 2),
-                 lo = output.length;
+         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
 
-             for (i = 0; i < lo; i += 1) {
-               output[i] = 0;
-             }
+             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');
+               }
 
-             for (i = 0; i < l; i += 8) {
-               output[i >> 5] |= (input.charCodeAt(i / 8) & 0xFF) << 24 - i % 32;
+               return _t('operations.connect.annotation.from_vertex.to_sibling_vertex');
              }
-
-             return output;
            }
-           /**
-            * Convert a raw string to an arbitrary string encoding
-            */
 
+           return _t('operations.connect.annotation.from_' + nodeGeometry + '.to_' + targetGeometry);
+         }
 
-           function rstr2any(input, encoding) {
-             var divisor = encoding.length,
-                 remainders = Array(),
-                 i,
-                 q,
-                 x,
-                 ld,
-                 quotient,
-                 dividend,
-                 output,
-                 full_length;
-             /* Convert to an array of 16-bit big-endian values, forming the dividend */
+         function shouldSnapToNode(target) {
+           if (!_activeEntity) return false;
+           return _activeEntity.geometry(context.graph()) !== 'vertex' || target.geometry(context.graph()) === 'vertex' || _mainPresetIndex.allowsVertex(target, context.graph());
+         }
 
-             dividend = Array(Math.ceil(input.length / 2));
-             ld = dividend.length;
+         function origin(entity) {
+           return context.projection(entity.loc);
+         }
 
-             for (i = 0; i < ld; i += 1) {
-               dividend[i] = input.charCodeAt(i * 2) << 8 | input.charCodeAt(i * 2 + 1);
+         function keydown(d3_event) {
+           if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
+             if (context.surface().classed('nope')) {
+               context.surface().classed('nope-suppressed', true);
              }
-             /**
-              * Repeatedly perform a long division. The binary array forms the dividend,
-              * the length of the encoding is the divisor. Once computed, the quotient
-              * forms the dividend for the next step. We stop when the dividend is zerHashes.
-              * All remainders are stored for later use.
-              */
 
+             context.surface().classed('nope', false).classed('nope-disabled', true);
+           }
+         }
 
-             while (dividend.length > 0) {
-               quotient = Array();
-               x = 0;
+         function keyup(d3_event) {
+           if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
+             if (context.surface().classed('nope-suppressed')) {
+               context.surface().classed('nope', true);
+             }
 
-               for (i = 0; i < dividend.length; i += 1) {
-                 x = (x << 16) + dividend[i];
-                 q = Math.floor(x / divisor);
-                 x -= q * divisor;
+             context.surface().classed('nope-suppressed', false).classed('nope-disabled', false);
+           }
+         }
 
-                 if (quotient.length > 0 || q > 0) {
-                   quotient[quotient.length] = q;
-                 }
-               }
+         function start(d3_event, entity) {
+           _wasMidpoint = entity.type === 'midpoint';
+           var hasHidden = context.features().hasHiddenConnections(entity, context.graph());
+           _isCancelled = !context.editable() || d3_event.shiftKey || hasHidden;
 
-               remainders[remainders.length] = x;
-               dividend = quotient;
+           if (_isCancelled) {
+             if (hasHidden) {
+               context.ui().flash.duration(4000).iconName('#iD-icon-no').label(_t('modes.drag_node.connected_to_hidden'))();
              }
-             /* Convert the remainders to the output string */
 
+             return drag.cancel();
+           }
 
-             output = '';
-
-             for (i = remainders.length - 1; i >= 0; i--) {
-               output += encoding.charAt(remainders[i]);
-             }
-             /* Append leading zero equivalents */
+           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());
+           }
 
-             full_length = Math.ceil(input.length * 8 / (Math.log(encoding.length) / Math.log(2)));
+           _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()`
 
-             for (i = output.length; i < full_length; i += 1) {
-               output = encoding[0] + output;
-             }
 
-             return output;
+         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 : {};
            }
-           /**
-            * Convert a raw string to a base-64 string
-            */
+         }
 
+         function doMove(d3_event, entity, nudge) {
+           nudge = nudge || [0, 0];
+           var currPoint = d3_event && d3_event.point || context.projection(_lastLoc);
+           var currMouse = geoVecSubtract(currPoint, nudge);
+           var loc = context.projection.invert(currMouse);
+           var target, edge;
 
-           function rstr2b64(input, b64pad) {
-             var tab = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/',
-                 output = '',
-                 len = input.length,
-                 i,
-                 j,
-                 triplet;
-             b64pad = b64pad || '=';
+           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;
 
-             for (i = 0; i < len; i += 3) {
-               triplet = input.charCodeAt(i) << 16 | (i + 1 < len ? input.charCodeAt(i + 1) << 8 : 0) | (i + 2 < len ? input.charCodeAt(i + 2) : 0);
+             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);
 
-               for (j = 0; j < 4; j += 1) {
-                 if (i * 8 + j * 6 > input.length * 8) {
-                   output += b64pad;
-                 } else {
-                   output += tab.charAt(triplet >>> 6 * (3 - j) & 0x3F);
-                 }
+               if (edge) {
+                 loc = edge.loc;
                }
              }
-
-             return output;
            }
 
-           Hashes = {
-             /**
-              * @property {String} version
-              * @readonly
-              */
-             VERSION: '1.0.6',
+           context.replace(actionMoveNode(entity.id, loc)); // Below here: validations
 
-             /**
-              * @member Hashes
-              * @class Base64
-              * @constructor
-              */
-             Base64: function Base64() {
-               // private properties
-               var tab = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/',
-                   pad = '=',
-                   // URL encoding support @todo
-               utf8 = true; // by default enable UTF-8 support encoding
-               // public method for encoding
+           var isInvalid = false; // Check if this connection to `target` could cause relations to break..
 
-               this.encode = function (input) {
-                 var i,
-                     j,
-                     triplet,
-                     output = '',
-                     len = input.length;
-                 pad = pad || '=';
-                 input = utf8 ? utf8Encode(input) : input;
+           if (target) {
+             isInvalid = hasRelationConflict(entity, target, edge, context.graph());
+           } // Check if this drag causes the geometry to break..
 
-                 for (i = 0; i < len; i += 3) {
-                   triplet = input.charCodeAt(i) << 16 | (i + 1 < len ? input.charCodeAt(i + 1) << 8 : 0) | (i + 2 < len ? input.charCodeAt(i + 2) : 0);
 
-                   for (j = 0; j < 4; j += 1) {
-                     if (i * 8 + j * 6 > len * 8) {
-                       output += pad;
-                     } else {
-                       output += tab.charAt(triplet >>> 6 * (3 - j) & 0x3F);
-                     }
-                   }
-                 }
+           if (!isInvalid) {
+             isInvalid = hasInvalidGeometry(entity, context.graph());
+           }
 
-                 return output;
-               }; // public method for decoding
+           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: _mainPresetIndex.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('')();
+             }
+           }
 
-               this.decode = function (input) {
-                 // var b64 = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';
-                 var i,
-                     o1,
-                     o2,
-                     o3,
-                     h1,
-                     h2,
-                     h3,
-                     h4,
-                     bits,
-                     ac,
-                     dec = '',
-                     arr = [];
+           var nopeDisabled = context.surface().classed('nope-disabled');
 
-                 if (!input) {
-                   return input;
-                 }
+           if (nopeDisabled) {
+             context.surface().classed('nope', false).classed('nope-suppressed', isInvalid);
+           } else {
+             context.surface().classed('nope', isInvalid).classed('nope-suppressed', false);
+           }
 
-                 i = ac = 0;
-                 input = input.replace(new RegExp('\\' + pad, 'gi'), ''); // use '='
-                 //input += '';
+           _lastLoc = loc;
+         } // Uses `actionConnect.disabled()` to know whether this connection is ok..
 
-                 do {
-                   // unpack four hexets into three octets using index points in b64
-                   h1 = tab.indexOf(input.charAt(i += 1));
-                   h2 = tab.indexOf(input.charAt(i += 1));
-                   h3 = tab.indexOf(input.charAt(i += 1));
-                   h4 = tab.indexOf(input.charAt(i += 1));
-                   bits = h1 << 18 | h2 << 12 | h3 << 6 | h4;
-                   o1 = bits >> 16 & 0xff;
-                   o2 = bits >> 8 & 0xff;
-                   o3 = bits & 0xff;
-                   ac += 1;
 
-                   if (h3 === 64) {
-                     arr[ac] = String.fromCharCode(o1);
-                   } else if (h4 === 64) {
-                     arr[ac] = String.fromCharCode(o1, o2);
-                   } else {
-                     arr[ac] = String.fromCharCode(o1, o2, o3);
-                   }
-                 } while (i < input.length);
+         function hasRelationConflict(entity, target, edge, graph) {
+           var testGraph = graph.update(); // copy
+           // if snapping to way - add midpoint there and consider that the target..
 
-                 dec = arr.join('');
-                 dec = utf8 ? utf8Decode(dec) : dec;
-                 return dec;
-               }; // set custom pad string
+           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?
 
 
-               this.setPad = function (str) {
-                 pad = str || pad;
-                 return this;
-               }; // set custom tab string characters
+           var ids = [entity.id, target.id];
+           return actionConnect(ids).disabled(testGraph);
+         }
 
+         function hasInvalidGeometry(entity, graph) {
+           var parents = graph.parentWays(entity);
+           var i, j, k;
 
-               this.setTab = function (str) {
-                 tab = str || tab;
-                 return this;
-               };
+           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
 
-               this.setUTF8 = function (bool) {
-                 if (typeof bool === 'boolean') {
-                   utf8 = bool;
-                 }
+             var relations = graph.parentRelations(parent);
 
-                 return this;
-               };
-             },
+             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
 
-             /**
-              * CRC-32 calculation
-              * @member Hashes
-              * @method CRC32
-              * @static
-              * @param {String} str Input String
-              * @return {String}
-              */
-             CRC32: function CRC32(str) {
-               var crc = 0,
-                   x = 0,
-                   y = 0,
-                   table,
-                   i,
-                   iTop;
-               str = utf8Encode(str);
-               table = ['00000000 77073096 EE0E612C 990951BA 076DC419 706AF48F E963A535 9E6495A3 0EDB8832 ', '79DCB8A4 E0D5E91E 97D2D988 09B64C2B 7EB17CBD E7B82D07 90BF1D91 1DB71064 6AB020F2 F3B97148 ', '84BE41DE 1ADAD47D 6DDDE4EB F4D4B551 83D385C7 136C9856 646BA8C0 FD62F97A 8A65C9EC 14015C4F ', '63066CD9 FA0F3D63 8D080DF5 3B6E20C8 4C69105E D56041E4 A2677172 3C03E4D1 4B04D447 D20D85FD ', 'A50AB56B 35B5A8FA 42B2986C DBBBC9D6 ACBCF940 32D86CE3 45DF5C75 DCD60DCF ABD13D59 26D930AC ', '51DE003A C8D75180 BFD06116 21B4F4B5 56B3C423 CFBA9599 B8BDA50F 2802B89E 5F058808 C60CD9B2 ', 'B10BE924 2F6F7C87 58684C11 C1611DAB B6662D3D 76DC4190 01DB7106 98D220BC EFD5102A 71B18589 ', '06B6B51F 9FBFE4A5 E8B8D433 7807C9A2 0F00F934 9609A88E E10E9818 7F6A0DBB 086D3D2D 91646C97 ', 'E6635C01 6B6B51F4 1C6C6162 856530D8 F262004E 6C0695ED 1B01A57B 8208F4C1 F50FC457 65B0D9C6 ', '12B7E950 8BBEB8EA FCB9887C 62DD1DDF 15DA2D49 8CD37CF3 FBD44C65 4DB26158 3AB551CE A3BC0074 ', 'D4BB30E2 4ADFA541 3DD895D7 A4D1C46D D3D6F4FB 4369E96A 346ED9FC AD678846 DA60B8D0 44042D73 ', '33031DE5 AA0A4C5F DD0D7CC9 5005713C 270241AA BE0B1010 C90C2086 5768B525 206F85B3 B966D409 ', 'CE61E49F 5EDEF90E 29D9C998 B0D09822 C7D7A8B4 59B33D17 2EB40D81 B7BD5C3B C0BA6CAD EDB88320 ', '9ABFB3B6 03B6E20C 74B1D29A EAD54739 9DD277AF 04DB2615 73DC1683 E3630B12 94643B84 0D6D6A3E ', '7A6A5AA8 E40ECF0B 9309FF9D 0A00AE27 7D079EB1 F00F9344 8708A3D2 1E01F268 6906C2FE F762575D ', '806567CB 196C3671 6E6B06E7 FED41B76 89D32BE0 10DA7A5A 67DD4ACC F9B9DF6F 8EBEEFF9 17B7BE43 ', '60B08ED5 D6D6A3E8 A1D1937E 38D8C2C4 4FDFF252 D1BB67F1 A6BC5767 3FB506DD 48B2364B D80D2BDA ', 'AF0A1B4C 36034AF6 41047A60 DF60EFC3 A867DF55 316E8EEF 4669BE79 CB61B38C BC66831A 256FD2A0 ', '5268E236 CC0C7795 BB0B4703 220216B9 5505262F C5BA3BBE B2BD0B28 2BB45A92 5CB36A04 C2D7FFA7 ', 'B5D0CF31 2CD99E8B 5BDEAE1D 9B64C2B0 EC63F226 756AA39C 026D930A 9C0906A9 EB0E363F 72076785 ', '05005713 95BF4A82 E2B87A14 7BB12BAE 0CB61B38 92D28E9B E5D5BE0D 7CDCEFB7 0BDBDF21 86D3D2D4 ', 'F1D4E242 68DDB3F8 1FDA836E 81BE16CD F6B9265B 6FB077E1 18B74777 88085AE6 FF0F6A70 66063BCA ', '11010B5C 8F659EFF F862AE69 616BFFD3 166CCF45 A00AE278 D70DD2EE 4E048354 3903B3C2 A7672661 ', 'D06016F7 4969474D 3E6E77DB AED16A4A D9D65ADC 40DF0B66 37D83BF0 A9BCAE53 DEBB9EC5 47B2CF7F ', '30B5FFE9 BDBDF21C CABAC28A 53B39330 24B4A3A6 BAD03605 CDD70693 54DE5729 23D967BF B3667A2E ', 'C4614AB8 5D681B02 2A6F2B94 B40BBE37 C30C8EA1 5A05DF1B 2D02EF8D'].join('');
-               crc = crc ^ -1;
+               for (k = 0; k < rings.length; k++) {
+                 nodes = rings[k].nodes;
 
-               for (i = 0, iTop = str.length; i < iTop; i += 1) {
-                 y = (crc ^ str.charCodeAt(i)) & 0xFF;
-                 x = '0x' + table.substr(y * 9, 8);
-                 crc = crc >>> 8 ^ x;
-               } // always return a positive number (that's what >>> 0 does)
+                 if (nodes.find(function (n) {
+                   return n.id === entity.id;
+                 })) {
+                   activeIndex = k;
 
+                   if (geoHasSelfIntersections(nodes, entity.id)) {
+                     return 'multipolygonMember';
+                   }
+                 }
 
-               return (crc ^ -1) >>> 0;
-             },
+                 rings[k].coords = nodes.map(function (n) {
+                   return n.loc;
+                 });
+               } // test active ring for intersections with other rings in the multipolygon
 
-             /**
-              * @member Hashes
-              * @class MD5
-              * @constructor
-              * @param {Object} [config]
-              *
-              * A JavaScript implementation of the RSA Data Security, Inc. MD5 Message
-              * Digest Algorithm, as defined in RFC 1321.
-              * Version 2.2 Copyright (C) Paul Johnston 1999 - 2009
-              * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
-              * See <http://pajhome.org.uk/crypt/md5> for more infHashes.
-              */
-             MD5: function MD5(options) {
-               /**
-                * Private config properties. You may need to tweak these to be compatible with
-                * the server-side, but the defaults work in most cases.
-                * See {@link Hashes.MD5#method-setUpperCase} and {@link Hashes.SHA1#method-setUpperCase}
-                */
-               var hexcase = options && typeof options.uppercase === 'boolean' ? options.uppercase : false,
-                   // hexadecimal output case format. false - lowercase; true - uppercase
-               b64pad = options && typeof options.pad === 'string' ? options.pad : '=',
-                   // base-64 pad character. Defaults to '=' for strict RFC compliance
-               utf8 = options && typeof options.utf8 === 'boolean' ? options.utf8 : true; // enable/disable utf8 encoding
-               // privileged (public) methods
 
-               this.hex = function (s) {
-                 return rstr2hex(rstr(s), hexcase);
-               };
+               for (k = 0; k < rings.length; k++) {
+                 if (k === activeIndex) continue; // make sure active ring doesn't cross passive rings
 
-               this.b64 = function (s) {
-                 return rstr2b64(rstr(s), b64pad);
-               };
+                 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.
 
-               this.any = function (s, e) {
-                 return rstr2any(rstr(s), e);
-               };
 
-               this.raw = function (s) {
-                 return rstr(s);
-               };
+             if (activeIndex === null) {
+               nodes = parent.nodes.map(function (nodeID) {
+                 return graph.entity(nodeID);
+               });
 
-               this.hex_hmac = function (k, d) {
-                 return rstr2hex(rstr_hmac(k, d), hexcase);
-               };
+               if (nodes.length && geoHasSelfIntersections(nodes, entity.id)) {
+                 return parent.geometry(graph);
+               }
+             }
+           }
 
-               this.b64_hmac = function (k, d) {
-                 return rstr2b64(rstr_hmac(k, d), b64pad);
-               };
+           return false;
+         }
 
-               this.any_hmac = function (k, d, e) {
-                 return rstr2any(rstr_hmac(k, d), e);
-               };
-               /**
-                * Perform a simple self-test to see if the VM is working
-                * @return {String} Hexadecimal hash sample
-                */
+         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 = geoViewportEdge(point, context.map().dimensions());
 
+           if (nudge) {
+             startNudge(d3_event, entity, nudge);
+           } else {
+             stopNudge();
+           }
+         }
 
-               this.vm_test = function () {
-                 return hex('abc').toLowerCase() === '900150983cd24fb0d6963f7d28e17f72';
-               };
-               /**
-                * Enable/disable uppercase hexadecimal returned string
-                * @param {Boolean}
-                * @return {Object} this
-                */
+         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));
+           }
 
-               this.setUpperCase = function (a) {
-                 if (typeof a === 'boolean') {
-                   hexcase = a;
-                 }
+           if (wasPoint) {
+             context.enter(modeSelect(context, [entity.id]));
+           } else {
+             var reselection = _restoreSelectedIDs.filter(function (id) {
+               return context.graph().hasEntity(id);
+             });
 
-                 return this;
-               };
-               /**
-                * Defines a base64 pad string
-                * @param {String} Pad
-                * @return {Object} this
-                */
+             if (reselection.length) {
+               context.enter(modeSelect(context, reselection));
+             } else {
+               context.enter(modeBrowse(context));
+             }
+           }
+         }
 
+         function _actionBounceBack(nodeID, toLoc) {
+           var moveNode = actionMoveNode(nodeID, toLoc);
 
-               this.setPad = function (a) {
-                 b64pad = a || b64pad;
-                 return this;
-               };
-               /**
-                * Defines a base64 pad string
-                * @param {Boolean}
-                * @return {Object} [this]
-                */
+           var action = function action(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;
+         }
 
-               this.setUTF8 = function (a) {
-                 if (typeof a === 'boolean') {
-                   utf8 = a;
-                 }
+         function cancel() {
+           drag.cancel();
+           context.enter(modeBrowse(context));
+         }
 
-                 return this;
-               }; // private methods
+         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);
 
-               /**
-                * Calculate the MD5 of a raw string
-                */
+         mode.enter = function () {
+           context.install(hover);
+           context.install(edit);
+           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);
+           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();
+         };
 
-               function rstr(s) {
-                 s = utf8 ? utf8Encode(s) : s;
-                 return binl2rstr(binl(rstr2binl(s), s.length * 8));
-               }
-               /**
-                * Calculate the HMAC-MD5, of a key and some data (raw strings)
-                */
+         mode.selectedIDs = function () {
+           if (!arguments.length) return _activeEntity ? [_activeEntity.id] : []; // no assign
 
+           return mode;
+         };
 
-               function rstr_hmac(key, data) {
-                 var bkey, ipad, opad, hash, i;
-                 key = utf8 ? utf8Encode(key) : key;
-                 data = utf8 ? utf8Encode(data) : data;
-                 bkey = rstr2binl(key);
+         mode.activeID = function () {
+           if (!arguments.length) return _activeEntity && _activeEntity.id; // no assign
 
-                 if (bkey.length > 16) {
-                   bkey = binl(bkey, key.length * 8);
-                 }
+           return mode;
+         };
 
-                 ipad = Array(16), opad = Array(16);
+         mode.restoreSelectedIDs = function (_) {
+           if (!arguments.length) return _restoreSelectedIDs;
+           _restoreSelectedIDs = _;
+           return mode;
+         };
 
-                 for (i = 0; i < 16; i += 1) {
-                   ipad[i] = bkey[i] ^ 0x36363636;
-                   opad[i] = bkey[i] ^ 0x5C5C5C5C;
-                 }
+         mode.behavior = drag;
+         return mode;
+       }
 
-                 hash = binl(ipad.concat(rstr2binl(data)), 512 + data.length * 8);
-                 return binl2rstr(binl(opad.concat(hash), 512 + 128));
-               }
-               /**
-                * Calculate the MD5 of an array of little-endian words, and a bit length.
-                */
+       // Safari bug https://bugs.webkit.org/show_bug.cgi?id=200829
+       var NON_GENERIC = !!nativePromiseConstructor && fails(function () {
+         nativePromiseConstructor.prototype['finally'].call({ then: function () { /* empty */ } }, function () { /* empty */ });
+       });
 
+       // `Promise.prototype.finally` method
+       // https://tc39.es/ecma262/#sec-promise.prototype.finally
+       _export({ target: 'Promise', proto: true, real: true, forced: NON_GENERIC }, {
+         'finally': function (onFinally) {
+           var C = speciesConstructor(this, getBuiltIn('Promise'));
+           var isFunction = typeof onFinally == 'function';
+           return this.then(
+             isFunction ? function (x) {
+               return promiseResolve(C, onFinally()).then(function () { return x; });
+             } : onFinally,
+             isFunction ? function (e) {
+               return promiseResolve(C, onFinally()).then(function () { throw e; });
+             } : onFinally
+           );
+         }
+       });
 
-               function binl(x, len) {
-                 var i,
-                     olda,
-                     oldb,
-                     oldc,
-                     oldd,
-                     a = 1732584193,
-                     b = -271733879,
-                     c = -1732584194,
-                     d = 271733878;
-                 /* append padding */
+       // makes sure that native promise-based APIs `Promise#finally` properly works with patched `Promise#then`
+       if (typeof nativePromiseConstructor == 'function') {
+         var method = getBuiltIn('Promise').prototype['finally'];
+         if (nativePromiseConstructor.prototype['finally'] !== method) {
+           redefine(nativePromiseConstructor.prototype, 'finally', method, { unsafe: true });
+         }
+       }
 
-                 x[len >> 5] |= 0x80 << len % 32;
-                 x[(len + 64 >>> 9 << 4) + 14] = len;
+       function quickselect(arr, k, left, right, compare) {
+         quickselectStep(arr, k, left || 0, right || arr.length - 1, compare || defaultCompare);
+       }
 
-                 for (i = 0; i < x.length; i += 16) {
-                   olda = a;
-                   oldb = b;
-                   oldc = c;
-                   oldd = d;
-                   a = md5_ff(a, b, c, d, x[i + 0], 7, -680876936);
-                   d = md5_ff(d, a, b, c, x[i + 1], 12, -389564586);
-                   c = md5_ff(c, d, a, b, x[i + 2], 17, 606105819);
-                   b = md5_ff(b, c, d, a, x[i + 3], 22, -1044525330);
-                   a = md5_ff(a, b, c, d, x[i + 4], 7, -176418897);
-                   d = md5_ff(d, a, b, c, x[i + 5], 12, 1200080426);
-                   c = md5_ff(c, d, a, b, x[i + 6], 17, -1473231341);
-                   b = md5_ff(b, c, d, a, x[i + 7], 22, -45705983);
-                   a = md5_ff(a, b, c, d, x[i + 8], 7, 1770035416);
-                   d = md5_ff(d, a, b, c, x[i + 9], 12, -1958414417);
-                   c = md5_ff(c, d, a, b, x[i + 10], 17, -42063);
-                   b = md5_ff(b, c, d, a, x[i + 11], 22, -1990404162);
-                   a = md5_ff(a, b, c, d, x[i + 12], 7, 1804603682);
-                   d = md5_ff(d, a, b, c, x[i + 13], 12, -40341101);
-                   c = md5_ff(c, d, a, b, x[i + 14], 17, -1502002290);
-                   b = md5_ff(b, c, d, a, x[i + 15], 22, 1236535329);
-                   a = md5_gg(a, b, c, d, x[i + 1], 5, -165796510);
-                   d = md5_gg(d, a, b, c, x[i + 6], 9, -1069501632);
-                   c = md5_gg(c, d, a, b, x[i + 11], 14, 643717713);
-                   b = md5_gg(b, c, d, a, x[i + 0], 20, -373897302);
-                   a = md5_gg(a, b, c, d, x[i + 5], 5, -701558691);
-                   d = md5_gg(d, a, b, c, x[i + 10], 9, 38016083);
-                   c = md5_gg(c, d, a, b, x[i + 15], 14, -660478335);
-                   b = md5_gg(b, c, d, a, x[i + 4], 20, -405537848);
-                   a = md5_gg(a, b, c, d, x[i + 9], 5, 568446438);
-                   d = md5_gg(d, a, b, c, x[i + 14], 9, -1019803690);
-                   c = md5_gg(c, d, a, b, x[i + 3], 14, -187363961);
-                   b = md5_gg(b, c, d, a, x[i + 8], 20, 1163531501);
-                   a = md5_gg(a, b, c, d, x[i + 13], 5, -1444681467);
-                   d = md5_gg(d, a, b, c, x[i + 2], 9, -51403784);
-                   c = md5_gg(c, d, a, b, x[i + 7], 14, 1735328473);
-                   b = md5_gg(b, c, d, a, x[i + 12], 20, -1926607734);
-                   a = md5_hh(a, b, c, d, x[i + 5], 4, -378558);
-                   d = md5_hh(d, a, b, c, x[i + 8], 11, -2022574463);
-                   c = md5_hh(c, d, a, b, x[i + 11], 16, 1839030562);
-                   b = md5_hh(b, c, d, a, x[i + 14], 23, -35309556);
-                   a = md5_hh(a, b, c, d, x[i + 1], 4, -1530992060);
-                   d = md5_hh(d, a, b, c, x[i + 4], 11, 1272893353);
-                   c = md5_hh(c, d, a, b, x[i + 7], 16, -155497632);
-                   b = md5_hh(b, c, d, a, x[i + 10], 23, -1094730640);
-                   a = md5_hh(a, b, c, d, x[i + 13], 4, 681279174);
-                   d = md5_hh(d, a, b, c, x[i + 0], 11, -358537222);
-                   c = md5_hh(c, d, a, b, x[i + 3], 16, -722521979);
-                   b = md5_hh(b, c, d, a, x[i + 6], 23, 76029189);
-                   a = md5_hh(a, b, c, d, x[i + 9], 4, -640364487);
-                   d = md5_hh(d, a, b, c, x[i + 12], 11, -421815835);
-                   c = md5_hh(c, d, a, b, x[i + 15], 16, 530742520);
-                   b = md5_hh(b, c, d, a, x[i + 2], 23, -995338651);
-                   a = md5_ii(a, b, c, d, x[i + 0], 6, -198630844);
-                   d = md5_ii(d, a, b, c, x[i + 7], 10, 1126891415);
-                   c = md5_ii(c, d, a, b, x[i + 14], 15, -1416354905);
-                   b = md5_ii(b, c, d, a, x[i + 5], 21, -57434055);
-                   a = md5_ii(a, b, c, d, x[i + 12], 6, 1700485571);
-                   d = md5_ii(d, a, b, c, x[i + 3], 10, -1894986606);
-                   c = md5_ii(c, d, a, b, x[i + 10], 15, -1051523);
-                   b = md5_ii(b, c, d, a, x[i + 1], 21, -2054922799);
-                   a = md5_ii(a, b, c, d, x[i + 8], 6, 1873313359);
-                   d = md5_ii(d, a, b, c, x[i + 15], 10, -30611744);
-                   c = md5_ii(c, d, a, b, x[i + 6], 15, -1560198380);
-                   b = md5_ii(b, c, d, a, x[i + 13], 21, 1309151649);
-                   a = md5_ii(a, b, c, d, x[i + 4], 6, -145523070);
-                   d = md5_ii(d, a, b, c, x[i + 11], 10, -1120210379);
-                   c = md5_ii(c, d, a, b, x[i + 2], 15, 718787259);
-                   b = md5_ii(b, c, d, a, x[i + 9], 21, -343485551);
-                   a = safe_add(a, olda);
-                   b = safe_add(b, oldb);
-                   c = safe_add(c, oldc);
-                   d = safe_add(d, oldd);
-                 }
+       function quickselectStep(arr, k, left, right, compare) {
+         while (right > left) {
+           if (right - left > 600) {
+             var n = right - left + 1;
+             var m = k - left + 1;
+             var z = Math.log(n);
+             var s = 0.5 * Math.exp(2 * z / 3);
+             var sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1);
+             var newLeft = Math.max(left, Math.floor(k - m * s / n + sd));
+             var newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd));
+             quickselectStep(arr, k, newLeft, newRight, compare);
+           }
 
-                 return Array(a, b, c, d);
-               }
-               /**
-                * These functions implement the four basic operations the algorithm uses.
-                */
+           var t = arr[k];
+           var i = left;
+           var j = right;
+           swap(arr, left, k);
+           if (compare(arr[right], t) > 0) swap(arr, left, right);
 
+           while (i < j) {
+             swap(arr, i, j);
+             i++;
+             j--;
 
-               function md5_cmn(q, a, b, x, s, t) {
-                 return safe_add(bit_rol(safe_add(safe_add(a, q), safe_add(x, t)), s), b);
-               }
+             while (compare(arr[i], t) < 0) {
+               i++;
+             }
 
-               function md5_ff(a, b, c, d, x, s, t) {
-                 return md5_cmn(b & c | ~b & d, a, b, x, s, t);
-               }
+             while (compare(arr[j], t) > 0) {
+               j--;
+             }
+           }
 
-               function md5_gg(a, b, c, d, x, s, t) {
-                 return md5_cmn(b & d | c & ~d, a, b, x, s, t);
-               }
+           if (compare(arr[left], t) === 0) swap(arr, left, j);else {
+             j++;
+             swap(arr, j, right);
+           }
+           if (j <= k) left = j + 1;
+           if (k <= j) right = j - 1;
+         }
+       }
 
-               function md5_hh(a, b, c, d, x, s, t) {
-                 return md5_cmn(b ^ c ^ d, a, b, x, s, t);
-               }
+       function swap(arr, i, j) {
+         var tmp = arr[i];
+         arr[i] = arr[j];
+         arr[j] = tmp;
+       }
 
-               function md5_ii(a, b, c, d, x, s, t) {
-                 return md5_cmn(c ^ (b | ~d), a, b, x, s, t);
-               }
-             },
+       function defaultCompare(a, b) {
+         return a < b ? -1 : a > b ? 1 : 0;
+       }
 
-             /**
-              * @member Hashes
-              * @class Hashes.SHA1
-              * @param {Object} [config]
-              * @constructor
-              *
-              * A JavaScript implementation of the Secure Hash Algorithm, SHA-1, as defined in FIPS 180-1
-              * Version 2.2 Copyright Paul Johnston 2000 - 2009.
-              * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
-              * See http://pajhome.org.uk/crypt/md5 for details.
-              */
-             SHA1: function SHA1(options) {
-               /**
-                * Private config properties. You may need to tweak these to be compatible with
-                * the server-side, but the defaults work in most cases.
-                * See {@link Hashes.MD5#method-setUpperCase} and {@link Hashes.SHA1#method-setUpperCase}
-                */
-               var hexcase = options && typeof options.uppercase === 'boolean' ? options.uppercase : false,
-                   // hexadecimal output case format. false - lowercase; true - uppercase
-               b64pad = options && typeof options.pad === 'string' ? options.pad : '=',
-                   // base-64 pad character. Defaults to '=' for strict RFC compliance
-               utf8 = options && typeof options.utf8 === 'boolean' ? options.utf8 : true; // enable/disable utf8 encoding
-               // public methods
+       var RBush = /*#__PURE__*/function () {
+         function RBush() {
+           var maxEntries = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 9;
 
-               this.hex = function (s) {
-                 return rstr2hex(rstr(s), hexcase);
-               };
+           _classCallCheck$1(this, RBush);
 
-               this.b64 = function (s) {
-                 return rstr2b64(rstr(s), b64pad);
-               };
+           // max entries in a node is 9 by default; min node fill is 40% for best performance
+           this._maxEntries = Math.max(4, maxEntries);
+           this._minEntries = Math.max(2, Math.ceil(this._maxEntries * 0.4));
+           this.clear();
+         }
 
-               this.any = function (s, e) {
-                 return rstr2any(rstr(s), e);
-               };
+         _createClass$1(RBush, [{
+           key: "all",
+           value: function all() {
+             return this._all(this.data, []);
+           }
+         }, {
+           key: "search",
+           value: function search(bbox) {
+             var node = this.data;
+             var result = [];
+             if (!intersects(bbox, node)) return result;
+             var toBBox = this.toBBox;
+             var nodesToSearch = [];
 
-               this.raw = function (s) {
-                 return rstr(s);
-               };
+             while (node) {
+               for (var i = 0; i < node.children.length; i++) {
+                 var child = node.children[i];
+                 var childBBox = node.leaf ? toBBox(child) : child;
 
-               this.hex_hmac = function (k, d) {
-                 return rstr2hex(rstr_hmac(k, d));
-               };
+                 if (intersects(bbox, childBBox)) {
+                   if (node.leaf) result.push(child);else if (contains(bbox, childBBox)) this._all(child, result);else nodesToSearch.push(child);
+                 }
+               }
 
-               this.b64_hmac = function (k, d) {
-                 return rstr2b64(rstr_hmac(k, d), b64pad);
-               };
+               node = nodesToSearch.pop();
+             }
 
-               this.any_hmac = function (k, d, e) {
-                 return rstr2any(rstr_hmac(k, d), e);
-               };
-               /**
-                * Perform a simple self-test to see if the VM is working
-                * @return {String} Hexadecimal hash sample
-                * @public
-                */
+             return result;
+           }
+         }, {
+           key: "collides",
+           value: function collides(bbox) {
+             var node = this.data;
+             if (!intersects(bbox, node)) return false;
+             var nodesToSearch = [];
 
+             while (node) {
+               for (var i = 0; i < node.children.length; i++) {
+                 var child = node.children[i];
+                 var childBBox = node.leaf ? this.toBBox(child) : child;
 
-               this.vm_test = function () {
-                 return hex('abc').toLowerCase() === '900150983cd24fb0d6963f7d28e17f72';
-               };
-               /**
-                * @description Enable/disable uppercase hexadecimal returned string
-                * @param {boolean}
-                * @return {Object} this
-                * @public
-                */
+                 if (intersects(bbox, childBBox)) {
+                   if (node.leaf || contains(bbox, childBBox)) return true;
+                   nodesToSearch.push(child);
+                 }
+               }
 
+               node = nodesToSearch.pop();
+             }
 
-               this.setUpperCase = function (a) {
-                 if (typeof a === 'boolean') {
-                   hexcase = a;
-                 }
+             return false;
+           }
+         }, {
+           key: "load",
+           value: function load(data) {
+             if (!(data && data.length)) return this;
 
-                 return this;
-               };
-               /**
-                * @description Defines a base64 pad string
-                * @param {string} Pad
-                * @return {Object} this
-                * @public
-                */
+             if (data.length < this._minEntries) {
+               for (var i = 0; i < data.length; i++) {
+                 this.insert(data[i]);
+               }
 
+               return this;
+             } // recursively build the tree with the given data from scratch using OMT algorithm
 
-               this.setPad = function (a) {
-                 b64pad = a || b64pad;
-                 return this;
-               };
-               /**
-                * @description Defines a base64 pad string
-                * @param {boolean}
-                * @return {Object} this
-                * @public
-                */
 
+             var node = this._build(data.slice(), 0, data.length - 1, 0);
 
-               this.setUTF8 = function (a) {
-                 if (typeof a === 'boolean') {
-                   utf8 = a;
-                 }
+             if (!this.data.children.length) {
+               // save as is if tree is empty
+               this.data = node;
+             } else if (this.data.height === node.height) {
+               // split root if trees have the same height
+               this._splitRoot(this.data, node);
+             } else {
+               if (this.data.height < node.height) {
+                 // swap trees if inserted one is bigger
+                 var tmpNode = this.data;
+                 this.data = node;
+                 node = tmpNode;
+               } // insert the small tree into the large tree at appropriate level
 
-                 return this;
-               }; // private methods
 
-               /**
-                * Calculate the SHA-512 of a raw string
-                */
+               this._insert(node, this.data.height - node.height - 1, true);
+             }
 
+             return this;
+           }
+         }, {
+           key: "insert",
+           value: function insert(item) {
+             if (item) this._insert(item, this.data.height - 1);
+             return this;
+           }
+         }, {
+           key: "clear",
+           value: function clear() {
+             this.data = createNode([]);
+             return this;
+           }
+         }, {
+           key: "remove",
+           value: function remove(item, equalsFn) {
+             if (!item) return this;
+             var node = this.data;
+             var bbox = this.toBBox(item);
+             var path = [];
+             var indexes = [];
+             var i, parent, goingUp; // depth-first iterative tree traversal
 
-               function rstr(s) {
-                 s = utf8 ? utf8Encode(s) : s;
-                 return binb2rstr(binb(rstr2binb(s), s.length * 8));
+             while (node || path.length) {
+               if (!node) {
+                 // go up
+                 node = path.pop();
+                 parent = path[path.length - 1];
+                 i = indexes.pop();
+                 goingUp = true;
                }
-               /**
-                * Calculate the HMAC-SHA1 of a key and some data (raw strings)
-                */
 
+               if (node.leaf) {
+                 // check current node
+                 var index = findItem(item, node.children, equalsFn);
 
-               function rstr_hmac(key, data) {
-                 var bkey, ipad, opad, i, hash;
-                 key = utf8 ? utf8Encode(key) : key;
-                 data = utf8 ? utf8Encode(data) : data;
-                 bkey = rstr2binb(key);
+                 if (index !== -1) {
+                   // item found, remove the item and condense tree upwards
+                   node.children.splice(index, 1);
+                   path.push(node);
 
-                 if (bkey.length > 16) {
-                   bkey = binb(bkey, key.length * 8);
+                   this._condense(path);
+
+                   return this;
                  }
+               }
 
-                 ipad = Array(16), opad = Array(16);
+               if (!goingUp && !node.leaf && contains(node, bbox)) {
+                 // go down
+                 path.push(node);
+                 indexes.push(i);
+                 i = 0;
+                 parent = node;
+                 node = node.children[0];
+               } else if (parent) {
+                 // go right
+                 i++;
+                 node = parent.children[i];
+                 goingUp = false;
+               } else node = null; // nothing found
+
+             }
+
+             return this;
+           }
+         }, {
+           key: "toBBox",
+           value: function toBBox(item) {
+             return item;
+           }
+         }, {
+           key: "compareMinX",
+           value: function compareMinX(a, b) {
+             return a.minX - b.minX;
+           }
+         }, {
+           key: "compareMinY",
+           value: function compareMinY(a, b) {
+             return a.minY - b.minY;
+           }
+         }, {
+           key: "toJSON",
+           value: function toJSON() {
+             return this.data;
+           }
+         }, {
+           key: "fromJSON",
+           value: function fromJSON(data) {
+             this.data = data;
+             return this;
+           }
+         }, {
+           key: "_all",
+           value: function _all(node, result) {
+             var nodesToSearch = [];
 
-                 for (i = 0; i < 16; i += 1) {
-                   ipad[i] = bkey[i] ^ 0x36363636;
-                   opad[i] = bkey[i] ^ 0x5C5C5C5C;
-                 }
+             while (node) {
+               if (node.leaf) result.push.apply(result, _toConsumableArray(node.children));else nodesToSearch.push.apply(nodesToSearch, _toConsumableArray(node.children));
+               node = nodesToSearch.pop();
+             }
 
-                 hash = binb(ipad.concat(rstr2binb(data)), 512 + data.length * 8);
-                 return binb2rstr(binb(opad.concat(hash), 512 + 160));
-               }
-               /**
-                * Calculate the SHA-1 of an array of big-endian words, and a bit length
-                */
+             return result;
+           }
+         }, {
+           key: "_build",
+           value: function _build(items, left, right, height) {
+             var N = right - left + 1;
+             var M = this._maxEntries;
+             var node;
 
+             if (N <= M) {
+               // reached leaf level; return leaf
+               node = createNode(items.slice(left, right + 1));
+               calcBBox(node, this.toBBox);
+               return node;
+             }
 
-               function binb(x, len) {
-                 var i,
-                     j,
-                     t,
-                     olda,
-                     oldb,
-                     oldc,
-                     oldd,
-                     olde,
-                     w = Array(80),
-                     a = 1732584193,
-                     b = -271733879,
-                     c = -1732584194,
-                     d = 271733878,
-                     e = -1009589776;
-                 /* append padding */
+             if (!height) {
+               // target height of the bulk-loaded tree
+               height = Math.ceil(Math.log(N) / Math.log(M)); // target number of root entries to maximize storage utilization
 
-                 x[len >> 5] |= 0x80 << 24 - len % 32;
-                 x[(len + 64 >> 9 << 4) + 15] = len;
+               M = Math.ceil(N / Math.pow(M, height - 1));
+             }
 
-                 for (i = 0; i < x.length; i += 16) {
-                   olda = a;
-                   oldb = b;
-                   oldc = c;
-                   oldd = d;
-                   olde = e;
+             node = createNode([]);
+             node.leaf = false;
+             node.height = height; // split the items into M mostly square tiles
 
-                   for (j = 0; j < 80; j += 1) {
-                     if (j < 16) {
-                       w[j] = x[i + j];
-                     } else {
-                       w[j] = bit_rol(w[j - 3] ^ w[j - 8] ^ w[j - 14] ^ w[j - 16], 1);
-                     }
+             var N2 = Math.ceil(N / M);
+             var N1 = N2 * Math.ceil(Math.sqrt(M));
+             multiSelect(items, left, right, N1, this.compareMinX);
 
-                     t = safe_add(safe_add(bit_rol(a, 5), sha1_ft(j, b, c, d)), safe_add(safe_add(e, w[j]), sha1_kt(j)));
-                     e = d;
-                     d = c;
-                     c = bit_rol(b, 30);
-                     b = a;
-                     a = t;
-                   }
+             for (var i = left; i <= right; i += N1) {
+               var right2 = Math.min(i + N1 - 1, right);
+               multiSelect(items, i, right2, N2, this.compareMinY);
 
-                   a = safe_add(a, olda);
-                   b = safe_add(b, oldb);
-                   c = safe_add(c, oldc);
-                   d = safe_add(d, oldd);
-                   e = safe_add(e, olde);
-                 }
+               for (var j = i; j <= right2; j += N2) {
+                 var right3 = Math.min(j + N2 - 1, right2); // pack each entry recursively
 
-                 return Array(a, b, c, d, e);
+                 node.children.push(this._build(items, j, right3, height - 1));
                }
-               /**
-                * Perform the appropriate triplet combination function for the current
-                * iteration
-                */
-
+             }
 
-               function sha1_ft(t, b, c, d) {
-                 if (t < 20) {
-                   return b & c | ~b & d;
-                 }
+             calcBBox(node, this.toBBox);
+             return node;
+           }
+         }, {
+           key: "_chooseSubtree",
+           value: function _chooseSubtree(bbox, node, level, path) {
+             while (true) {
+               path.push(node);
+               if (node.leaf || path.length - 1 === level) break;
+               var minArea = Infinity;
+               var minEnlargement = Infinity;
+               var targetNode = void 0;
 
-                 if (t < 40) {
-                   return b ^ c ^ d;
-                 }
+               for (var i = 0; i < node.children.length; i++) {
+                 var child = node.children[i];
+                 var area = bboxArea(child);
+                 var enlargement = enlargedArea(bbox, child) - area; // choose entry with the least area enlargement
 
-                 if (t < 60) {
-                   return b & c | b & d | c & d;
+                 if (enlargement < minEnlargement) {
+                   minEnlargement = enlargement;
+                   minArea = area < minArea ? area : minArea;
+                   targetNode = child;
+                 } else if (enlargement === minEnlargement) {
+                   // otherwise choose one with the smallest area
+                   if (area < minArea) {
+                     minArea = area;
+                     targetNode = child;
+                   }
                  }
-
-                 return b ^ c ^ d;
                }
-               /**
-                * Determine the appropriate additive constant for the current iteration
-                */
 
+               node = targetNode || node.children[0];
+             }
 
-               function sha1_kt(t) {
-                 return t < 20 ? 1518500249 : t < 40 ? 1859775393 : t < 60 ? -1894007588 : -899497514;
-               }
-             },
-
-             /**
-              * @class Hashes.SHA256
-              * @param {config}
-              *
-              * A JavaScript implementation of the Secure Hash Algorithm, SHA-256, as defined in FIPS 180-2
-              * Version 2.2 Copyright Angel Marin, Paul Johnston 2000 - 2009.
-              * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
-              * See http://pajhome.org.uk/crypt/md5 for details.
-              * Also http://anmar.eu.org/projects/jssha2/
-              */
-             SHA256: function SHA256(options) {
-               /**
-                * Private properties configuration variables. You may need to tweak these to be compatible with
-                * the server-side, but the defaults work in most cases.
-                * @see this.setUpperCase() method
-                * @see this.setPad() method
-                */
-               var hexcase = options && typeof options.uppercase === 'boolean' ? options.uppercase : false,
-                   // hexadecimal output case format. false - lowercase; true - uppercase  */
-               b64pad = options && typeof options.pad === 'string' ? options.pad : '=',
+             return node;
+           }
+         }, {
+           key: "_insert",
+           value: function _insert(item, level, isNode) {
+             var bbox = isNode ? item : this.toBBox(item);
+             var insertPath = []; // find the best node for accommodating the item, saving all nodes along the path too
 
-               /* base-64 pad character. Default '=' for strict RFC compliance   */
-               utf8 = options && typeof options.utf8 === 'boolean' ? options.utf8 : true,
+             var node = this._chooseSubtree(bbox, this.data, level, insertPath); // put the item into the node
 
-               /* enable/disable utf8 encoding */
-               sha256_K;
-               /* privileged (public) methods */
 
-               this.hex = function (s) {
-                 return rstr2hex(rstr(s, utf8));
-               };
+             node.children.push(item);
+             extend$1(node, bbox); // split on node overflow; propagate upwards if necessary
 
-               this.b64 = function (s) {
-                 return rstr2b64(rstr(s, utf8), b64pad);
-               };
+             while (level >= 0) {
+               if (insertPath[level].children.length > this._maxEntries) {
+                 this._split(insertPath, level);
 
-               this.any = function (s, e) {
-                 return rstr2any(rstr(s, utf8), e);
-               };
+                 level--;
+               } else break;
+             } // adjust bboxes along the insertion path
 
-               this.raw = function (s) {
-                 return rstr(s, utf8);
-               };
 
-               this.hex_hmac = function (k, d) {
-                 return rstr2hex(rstr_hmac(k, d));
-               };
+             this._adjustParentBBoxes(bbox, insertPath, level);
+           } // split overflowed node into two
 
-               this.b64_hmac = function (k, d) {
-                 return rstr2b64(rstr_hmac(k, d), b64pad);
-               };
+         }, {
+           key: "_split",
+           value: function _split(insertPath, level) {
+             var node = insertPath[level];
+             var M = node.children.length;
+             var m = this._minEntries;
 
-               this.any_hmac = function (k, d, e) {
-                 return rstr2any(rstr_hmac(k, d), e);
-               };
-               /**
-                * Perform a simple self-test to see if the VM is working
-                * @return {String} Hexadecimal hash sample
-                * @public
-                */
+             this._chooseSplitAxis(node, m, M);
 
+             var splitIndex = this._chooseSplitIndex(node, m, M);
 
-               this.vm_test = function () {
-                 return hex('abc').toLowerCase() === '900150983cd24fb0d6963f7d28e17f72';
-               };
-               /**
-                * Enable/disable uppercase hexadecimal returned string
-                * @param {boolean}
-                * @return {Object} this
-                * @public
-                */
+             var newNode = createNode(node.children.splice(splitIndex, node.children.length - splitIndex));
+             newNode.height = node.height;
+             newNode.leaf = node.leaf;
+             calcBBox(node, this.toBBox);
+             calcBBox(newNode, this.toBBox);
+             if (level) insertPath[level - 1].children.push(newNode);else this._splitRoot(node, newNode);
+           }
+         }, {
+           key: "_splitRoot",
+           value: function _splitRoot(node, newNode) {
+             // split root node
+             this.data = createNode([node, newNode]);
+             this.data.height = node.height + 1;
+             this.data.leaf = false;
+             calcBBox(this.data, this.toBBox);
+           }
+         }, {
+           key: "_chooseSplitIndex",
+           value: function _chooseSplitIndex(node, m, M) {
+             var index;
+             var minOverlap = Infinity;
+             var minArea = Infinity;
 
+             for (var i = m; i <= M - m; i++) {
+               var bbox1 = distBBox(node, 0, i, this.toBBox);
+               var bbox2 = distBBox(node, i, M, this.toBBox);
+               var overlap = intersectionArea(bbox1, bbox2);
+               var area = bboxArea(bbox1) + bboxArea(bbox2); // choose distribution with minimum overlap
 
-               this.setUpperCase = function (a) {
-                 if (typeof a === 'boolean') {
-                   hexcase = a;
+               if (overlap < minOverlap) {
+                 minOverlap = overlap;
+                 index = i;
+                 minArea = area < minArea ? area : minArea;
+               } else if (overlap === minOverlap) {
+                 // otherwise choose distribution with minimum area
+                 if (area < minArea) {
+                   minArea = area;
+                   index = i;
                  }
+               }
+             }
 
-                 return this;
-               };
-               /**
-                * @description Defines a base64 pad string
-                * @param {string} Pad
-                * @return {Object} this
-                * @public
-                */
-
+             return index || M - m;
+           } // sorts node children by the best axis for split
 
-               this.setPad = function (a) {
-                 b64pad = a || b64pad;
-                 return this;
-               };
-               /**
-                * Defines a base64 pad string
-                * @param {boolean}
-                * @return {Object} this
-                * @public
-                */
+         }, {
+           key: "_chooseSplitAxis",
+           value: function _chooseSplitAxis(node, m, M) {
+             var compareMinX = node.leaf ? this.compareMinX : compareNodeMinX;
+             var compareMinY = node.leaf ? this.compareMinY : compareNodeMinY;
 
+             var xMargin = this._allDistMargin(node, m, M, compareMinX);
 
-               this.setUTF8 = function (a) {
-                 if (typeof a === 'boolean') {
-                   utf8 = a;
-                 }
+             var yMargin = this._allDistMargin(node, m, M, compareMinY); // if total distributions margin value is minimal for x, sort by minX,
+             // otherwise it's already sorted by minY
 
-                 return this;
-               }; // private methods
 
-               /**
-                * Calculate the SHA-512 of a raw string
-                */
+             if (xMargin < yMargin) node.children.sort(compareMinX);
+           } // total margin of all possible split distributions where each node is at least m full
 
+         }, {
+           key: "_allDistMargin",
+           value: function _allDistMargin(node, m, M, compare) {
+             node.children.sort(compare);
+             var toBBox = this.toBBox;
+             var leftBBox = distBBox(node, 0, m, toBBox);
+             var rightBBox = distBBox(node, M - m, M, toBBox);
+             var margin = bboxMargin(leftBBox) + bboxMargin(rightBBox);
 
-               function rstr(s, utf8) {
-                 s = utf8 ? utf8Encode(s) : s;
-                 return binb2rstr(binb(rstr2binb(s), s.length * 8));
-               }
-               /**
-                * Calculate the HMAC-sha256 of a key and some data (raw strings)
-                */
+             for (var i = m; i < M - m; i++) {
+               var child = node.children[i];
+               extend$1(leftBBox, node.leaf ? toBBox(child) : child);
+               margin += bboxMargin(leftBBox);
+             }
 
+             for (var _i = M - m - 1; _i >= m; _i--) {
+               var _child = node.children[_i];
+               extend$1(rightBBox, node.leaf ? toBBox(_child) : _child);
+               margin += bboxMargin(rightBBox);
+             }
 
-               function rstr_hmac(key, data) {
-                 key = utf8 ? utf8Encode(key) : key;
-                 data = utf8 ? utf8Encode(data) : data;
-                 var hash,
-                     i = 0,
-                     bkey = rstr2binb(key),
-                     ipad = Array(16),
-                     opad = Array(16);
+             return margin;
+           }
+         }, {
+           key: "_adjustParentBBoxes",
+           value: function _adjustParentBBoxes(bbox, path, level) {
+             // adjust bboxes along the given tree path
+             for (var i = level; i >= 0; i--) {
+               extend$1(path[i], bbox);
+             }
+           }
+         }, {
+           key: "_condense",
+           value: function _condense(path) {
+             // go through the path, removing empty nodes and updating bboxes
+             for (var i = path.length - 1, siblings; i >= 0; i--) {
+               if (path[i].children.length === 0) {
+                 if (i > 0) {
+                   siblings = path[i - 1].children;
+                   siblings.splice(siblings.indexOf(path[i]), 1);
+                 } else this.clear();
+               } else calcBBox(path[i], this.toBBox);
+             }
+           }
+         }]);
 
-                 if (bkey.length > 16) {
-                   bkey = binb(bkey, key.length * 8);
-                 }
+         return RBush;
+       }();
 
-                 for (; i < 16; i += 1) {
-                   ipad[i] = bkey[i] ^ 0x36363636;
-                   opad[i] = bkey[i] ^ 0x5C5C5C5C;
-                 }
+       function findItem(item, items, equalsFn) {
+         if (!equalsFn) return items.indexOf(item);
 
-                 hash = binb(ipad.concat(rstr2binb(data)), 512 + data.length * 8);
-                 return binb2rstr(binb(opad.concat(hash), 512 + 256));
-               }
-               /*
-                * Main sha256 function, with its support functions
-                */
+         for (var i = 0; i < items.length; i++) {
+           if (equalsFn(item, items[i])) return i;
+         }
 
+         return -1;
+       } // calculate node's bbox from bboxes of its children
 
-               function sha256_S(X, n) {
-                 return X >>> n | X << 32 - n;
-               }
 
-               function sha256_R(X, n) {
-                 return X >>> n;
-               }
+       function calcBBox(node, toBBox) {
+         distBBox(node, 0, node.children.length, toBBox, node);
+       } // min bounding rectangle of node children from k to p-1
 
-               function sha256_Ch(x, y, z) {
-                 return x & y ^ ~x & z;
-               }
 
-               function sha256_Maj(x, y, z) {
-                 return x & y ^ x & z ^ y & z;
-               }
+       function distBBox(node, k, p, toBBox, destNode) {
+         if (!destNode) destNode = createNode(null);
+         destNode.minX = Infinity;
+         destNode.minY = Infinity;
+         destNode.maxX = -Infinity;
+         destNode.maxY = -Infinity;
 
-               function sha256_Sigma0256(x) {
-                 return sha256_S(x, 2) ^ sha256_S(x, 13) ^ sha256_S(x, 22);
-               }
+         for (var i = k; i < p; i++) {
+           var child = node.children[i];
+           extend$1(destNode, node.leaf ? toBBox(child) : child);
+         }
 
-               function sha256_Sigma1256(x) {
-                 return sha256_S(x, 6) ^ sha256_S(x, 11) ^ sha256_S(x, 25);
-               }
+         return destNode;
+       }
 
-               function sha256_Gamma0256(x) {
-                 return sha256_S(x, 7) ^ sha256_S(x, 18) ^ sha256_R(x, 3);
-               }
+       function extend$1(a, b) {
+         a.minX = Math.min(a.minX, b.minX);
+         a.minY = Math.min(a.minY, b.minY);
+         a.maxX = Math.max(a.maxX, b.maxX);
+         a.maxY = Math.max(a.maxY, b.maxY);
+         return a;
+       }
 
-               function sha256_Gamma1256(x) {
-                 return sha256_S(x, 17) ^ sha256_S(x, 19) ^ sha256_R(x, 10);
-               }
+       function compareNodeMinX(a, b) {
+         return a.minX - b.minX;
+       }
 
-               sha256_K = [1116352408, 1899447441, -1245643825, -373957723, 961987163, 1508970993, -1841331548, -1424204075, -670586216, 310598401, 607225278, 1426881987, 1925078388, -2132889090, -1680079193, -1046744716, -459576895, -272742522, 264347078, 604807628, 770255983, 1249150122, 1555081692, 1996064986, -1740746414, -1473132947, -1341970488, -1084653625, -958395405, -710438585, 113926993, 338241895, 666307205, 773529912, 1294757372, 1396182291, 1695183700, 1986661051, -2117940946, -1838011259, -1564481375, -1474664885, -1035236496, -949202525, -778901479, -694614492, -200395387, 275423344, 430227734, 506948616, 659060556, 883997877, 958139571, 1322822218, 1537002063, 1747873779, 1955562222, 2024104815, -2067236844, -1933114872, -1866530822, -1538233109, -1090935817, -965641998];
+       function compareNodeMinY(a, b) {
+         return a.minY - b.minY;
+       }
 
-               function binb(m, l) {
-                 var HASH = [1779033703, -1150833019, 1013904242, -1521486534, 1359893119, -1694144372, 528734635, 1541459225];
-                 var W = new Array(64);
-                 var a, b, c, d, e, f, g, h;
-                 var i, j, T1, T2;
-                 /* append padding */
+       function bboxArea(a) {
+         return (a.maxX - a.minX) * (a.maxY - a.minY);
+       }
 
-                 m[l >> 5] |= 0x80 << 24 - l % 32;
-                 m[(l + 64 >> 9 << 4) + 15] = l;
+       function bboxMargin(a) {
+         return a.maxX - a.minX + (a.maxY - a.minY);
+       }
 
-                 for (i = 0; i < m.length; i += 16) {
-                   a = HASH[0];
-                   b = HASH[1];
-                   c = HASH[2];
-                   d = HASH[3];
-                   e = HASH[4];
-                   f = HASH[5];
-                   g = HASH[6];
-                   h = HASH[7];
+       function enlargedArea(a, b) {
+         return (Math.max(b.maxX, a.maxX) - Math.min(b.minX, a.minX)) * (Math.max(b.maxY, a.maxY) - Math.min(b.minY, a.minY));
+       }
 
-                   for (j = 0; j < 64; j += 1) {
-                     if (j < 16) {
-                       W[j] = m[j + i];
-                     } else {
-                       W[j] = safe_add(safe_add(safe_add(sha256_Gamma1256(W[j - 2]), W[j - 7]), sha256_Gamma0256(W[j - 15])), W[j - 16]);
-                     }
+       function intersectionArea(a, b) {
+         var minX = Math.max(a.minX, b.minX);
+         var minY = Math.max(a.minY, b.minY);
+         var maxX = Math.min(a.maxX, b.maxX);
+         var maxY = Math.min(a.maxY, b.maxY);
+         return Math.max(0, maxX - minX) * Math.max(0, maxY - minY);
+       }
 
-                     T1 = safe_add(safe_add(safe_add(safe_add(h, sha256_Sigma1256(e)), sha256_Ch(e, f, g)), sha256_K[j]), W[j]);
-                     T2 = safe_add(sha256_Sigma0256(a), sha256_Maj(a, b, c));
-                     h = g;
-                     g = f;
-                     f = e;
-                     e = safe_add(d, T1);
-                     d = c;
-                     c = b;
-                     b = a;
-                     a = safe_add(T1, T2);
-                   }
+       function contains(a, b) {
+         return a.minX <= b.minX && a.minY <= b.minY && b.maxX <= a.maxX && b.maxY <= a.maxY;
+       }
 
-                   HASH[0] = safe_add(a, HASH[0]);
-                   HASH[1] = safe_add(b, HASH[1]);
-                   HASH[2] = safe_add(c, HASH[2]);
-                   HASH[3] = safe_add(d, HASH[3]);
-                   HASH[4] = safe_add(e, HASH[4]);
-                   HASH[5] = safe_add(f, HASH[5]);
-                   HASH[6] = safe_add(g, HASH[6]);
-                   HASH[7] = safe_add(h, HASH[7]);
-                 }
+       function intersects(a, b) {
+         return b.minX <= a.maxX && b.minY <= a.maxY && b.maxX >= a.minX && b.maxY >= a.minY;
+       }
 
-                 return HASH;
-               }
-             },
+       function createNode(children) {
+         return {
+           children: children,
+           height: 1,
+           leaf: true,
+           minX: Infinity,
+           minY: Infinity,
+           maxX: -Infinity,
+           maxY: -Infinity
+         };
+       } // sort an array so that items come in groups of n unsorted items, with groups sorted between each other;
+       // combines selection algorithm with binary divide & conquer approach
 
-             /**
-              * @class Hashes.SHA512
-              * @param {config}
-              *
-              * A JavaScript implementation of the Secure Hash Algorithm, SHA-512, as defined in FIPS 180-2
-              * Version 2.2 Copyright Anonymous Contributor, Paul Johnston 2000 - 2009.
-              * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
-              * See http://pajhome.org.uk/crypt/md5 for details.
-              */
-             SHA512: function SHA512(options) {
-               /**
-                * Private properties configuration variables. You may need to tweak these to be compatible with
-                * the server-side, but the defaults work in most cases.
-                * @see this.setUpperCase() method
-                * @see this.setPad() method
-                */
-               var hexcase = options && typeof options.uppercase === 'boolean' ? options.uppercase : false,
 
-               /* hexadecimal output case format. false - lowercase; true - uppercase  */
-               b64pad = options && typeof options.pad === 'string' ? options.pad : '=',
+       function multiSelect(arr, left, right, n, compare) {
+         var stack = [left, right];
 
-               /* base-64 pad character. Default '=' for strict RFC compliance   */
-               utf8 = options && typeof options.utf8 === 'boolean' ? options.utf8 : true,
+         while (stack.length) {
+           right = stack.pop();
+           left = stack.pop();
+           if (right - left <= n) continue;
+           var mid = left + Math.ceil((right - left) / n / 2) * n;
+           quickselect(arr, mid, left, right, compare);
+           stack.push(left, mid, mid, right);
+         }
+       }
 
-               /* enable/disable utf8 encoding */
-               sha512_k;
-               /* privileged (public) methods */
+       function responseText(response) {
+         if (!response.ok) throw new Error(response.status + " " + response.statusText);
+         return response.text();
+       }
 
-               this.hex = function (s) {
-                 return rstr2hex(rstr(s));
-               };
+       function d3_text (input, init) {
+         return fetch(input, init).then(responseText);
+       }
 
-               this.b64 = function (s) {
-                 return rstr2b64(rstr(s), b64pad);
-               };
+       function responseJson(response) {
+         if (!response.ok) throw new Error(response.status + " " + response.statusText);
+         if (response.status === 204 || response.status === 205) return;
+         return response.json();
+       }
 
-               this.any = function (s, e) {
-                 return rstr2any(rstr(s), e);
-               };
+       function d3_json (input, init) {
+         return fetch(input, init).then(responseJson);
+       }
 
-               this.raw = function (s) {
-                 return rstr(s);
-               };
+       function parser(type) {
+         return function (input, init) {
+           return d3_text(input, init).then(function (text) {
+             return new DOMParser().parseFromString(text, type);
+           });
+         };
+       }
 
-               this.hex_hmac = function (k, d) {
-                 return rstr2hex(rstr_hmac(k, d));
-               };
+       var d3_xml = parser("application/xml");
+       var svg = parser("image/svg+xml");
 
-               this.b64_hmac = function (k, d) {
-                 return rstr2b64(rstr_hmac(k, d), b64pad);
-               };
+       var tiler$6 = utilTiler();
+       var dispatch$7 = dispatch$8('loaded');
+       var _tileZoom$3 = 14;
+       var _krUrlRoot = 'https://www.keepright.at';
+       var _krData = {
+         errorTypes: {},
+         localizeStrings: {}
+       }; // This gets reassigned if reset
 
-               this.any_hmac = function (k, d, e) {
-                 return rstr2any(rstr_hmac(k, d), e);
-               };
-               /**
-                * Perform a simple self-test to see if the VM is working
-                * @return {String} Hexadecimal hash sample
-                * @public
-                */
+       var _cache$2;
 
+       var _krRuleset = [// no 20 - multiple node on same spot - these are mostly boundaries overlapping roads
+       30, 40, 50, 60, 70, 90, 100, 110, 120, 130, 150, 160, 170, 180, 190, 191, 192, 193, 194, 195, 196, 197, 198, 200, 201, 202, 203, 204, 205, 206, 207, 208, 210, 220, 230, 231, 232, 270, 280, 281, 282, 283, 284, 285, 290, 291, 292, 293, 294, 295, 296, 297, 298, 300, 310, 311, 312, 313, 320, 350, 360, 370, 380, 390, 400, 401, 402, 410, 411, 412, 413];
 
-               this.vm_test = function () {
-                 return hex('abc').toLowerCase() === '900150983cd24fb0d6963f7d28e17f72';
-               };
-               /**
-                * @description Enable/disable uppercase hexadecimal returned string
-                * @param {boolean}
-                * @return {Object} this
-                * @public
-                */
+       function abortRequest$6(controller) {
+         if (controller) {
+           controller.abort();
+         }
+       }
 
+       function abortUnwantedRequests$3(cache, tiles) {
+         Object.keys(cache.inflightTile).forEach(function (k) {
+           var wanted = tiles.find(function (tile) {
+             return k === tile.id;
+           });
 
-               this.setUpperCase = function (a) {
-                 if (typeof a === 'boolean') {
-                   hexcase = a;
-                 }
+           if (!wanted) {
+             abortRequest$6(cache.inflightTile[k]);
+             delete cache.inflightTile[k];
+           }
+         });
+       }
 
-                 return this;
-               };
-               /**
-                * @description Defines a base64 pad string
-                * @param {string} Pad
-                * @return {Object} this
-                * @public
-                */
+       function encodeIssueRtree$2(d) {
+         return {
+           minX: d.loc[0],
+           minY: d.loc[1],
+           maxX: d.loc[0],
+           maxY: d.loc[1],
+           data: d
+         };
+       } // Replace or remove QAItem from rtree
 
 
-               this.setPad = function (a) {
-                 b64pad = a || b64pad;
-                 return this;
-               };
-               /**
-                * @description Defines a base64 pad string
-                * @param {boolean}
-                * @return {Object} this
-                * @public
-                */
+       function updateRtree$3(item, replace) {
+         _cache$2.rtree.remove(item, function (a, b) {
+           return a.data.id === b.data.id;
+         });
 
+         if (replace) {
+           _cache$2.rtree.insert(item);
+         }
+       }
 
-               this.setUTF8 = function (a) {
-                 if (typeof a === 'boolean') {
-                   utf8 = a;
-                 }
+       function tokenReplacements(d) {
+         if (!(d instanceof QAItem)) return;
+         var htmlRegex = new RegExp(/<\/[a-z][\s\S]*>/);
+         var replacements = {};
+         var issueTemplate = _krData.errorTypes[d.whichType];
 
-                 return this;
-               };
-               /* private methods */
+         if (!issueTemplate) {
+           /* eslint-disable no-console */
+           console.log('No Template: ', d.whichType);
+           console.log('  ', d.description);
+           /* eslint-enable no-console */
 
-               /**
-                * Calculate the SHA-512 of a raw string
-                */
+           return;
+         } // some descriptions are just fixed text
 
 
-               function rstr(s) {
-                 s = utf8 ? utf8Encode(s) : s;
-                 return binb2rstr(binb(rstr2binb(s), s.length * 8));
-               }
-               /*
-                * Calculate the HMAC-SHA-512 of a key and some data (raw strings)
-                */
+         if (!issueTemplate.regex) return; // regex pattern should match description with variable details captured
 
+         var errorRegex = new RegExp(issueTemplate.regex, 'i');
+         var errorMatch = errorRegex.exec(d.description);
 
-               function rstr_hmac(key, data) {
-                 key = utf8 ? utf8Encode(key) : key;
-                 data = utf8 ? utf8Encode(data) : data;
-                 var hash,
-                     i = 0,
-                     bkey = rstr2binb(key),
-                     ipad = Array(32),
-                     opad = Array(32);
+         if (!errorMatch) {
+           /* eslint-disable no-console */
+           console.log('Unmatched: ', d.whichType);
+           console.log('  ', d.description);
+           console.log('  ', errorRegex);
+           /* eslint-enable no-console */
 
-                 if (bkey.length > 32) {
-                   bkey = binb(bkey, key.length * 8);
-                 }
+           return;
+         }
 
-                 for (; i < 32; i += 1) {
-                   ipad[i] = bkey[i] ^ 0x36363636;
-                   opad[i] = bkey[i] ^ 0x5C5C5C5C;
-                 }
+         for (var i = 1; i < errorMatch.length; i++) {
+           // skip first
+           var capture = errorMatch[i];
+           var idType = void 0;
+           idType = 'IDs' in issueTemplate ? issueTemplate.IDs[i - 1] : '';
 
-                 hash = binb(ipad.concat(rstr2binb(data)), 1024 + data.length * 8);
-                 return binb2rstr(binb(opad.concat(hash), 1024 + 512));
-               }
-               /**
-                * Calculate the SHA-512 of an array of big-endian dwords, and a bit length
-                */
+           if (idType && capture) {
+             // link IDs if present in the capture
+             capture = parseError(capture, idType);
+           } else if (htmlRegex.test(capture)) {
+             // escape any html in non-IDs
+             capture = '\\' + capture + '\\';
+           } else {
+             var compare = capture.toLowerCase();
 
+             if (_krData.localizeStrings[compare]) {
+               // some replacement strings can be localized
+               capture = _t('QA.keepRight.error_parts.' + _krData.localizeStrings[compare]);
+             }
+           }
 
-               function binb(x, len) {
-                 var j,
-                     i,
-                     l,
-                     W = new Array(80),
-                     hash = new Array(16),
-                     //Initial hash values
-                 H = [new int64(0x6a09e667, -205731576), new int64(-1150833019, -2067093701), new int64(0x3c6ef372, -23791573), new int64(-1521486534, 0x5f1d36f1), new int64(0x510e527f, -1377402159), new int64(-1694144372, 0x2b3e6c1f), new int64(0x1f83d9ab, -79577749), new int64(0x5be0cd19, 0x137e2179)],
-                     T1 = new int64(0, 0),
-                     T2 = new int64(0, 0),
-                     a = new int64(0, 0),
-                     b = new int64(0, 0),
-                     c = new int64(0, 0),
-                     d = new int64(0, 0),
-                     e = new int64(0, 0),
-                     f = new int64(0, 0),
-                     g = new int64(0, 0),
-                     h = new int64(0, 0),
-                     //Temporary variables not specified by the document
-                 s0 = new int64(0, 0),
-                     s1 = new int64(0, 0),
-                     Ch = new int64(0, 0),
-                     Maj = new int64(0, 0),
-                     r1 = new int64(0, 0),
-                     r2 = new int64(0, 0),
-                     r3 = new int64(0, 0);
+           replacements['var' + i] = capture;
+         }
 
-                 if (sha512_k === undefined) {
-                   //SHA512 constants
-                   sha512_k = [new int64(0x428a2f98, -685199838), new int64(0x71374491, 0x23ef65cd), new int64(-1245643825, -330482897), new int64(-373957723, -2121671748), new int64(0x3956c25b, -213338824), new int64(0x59f111f1, -1241133031), new int64(-1841331548, -1357295717), new int64(-1424204075, -630357736), new int64(-670586216, -1560083902), new int64(0x12835b01, 0x45706fbe), new int64(0x243185be, 0x4ee4b28c), new int64(0x550c7dc3, -704662302), new int64(0x72be5d74, -226784913), new int64(-2132889090, 0x3b1696b1), new int64(-1680079193, 0x25c71235), new int64(-1046744716, -815192428), new int64(-459576895, -1628353838), new int64(-272742522, 0x384f25e3), new int64(0xfc19dc6, -1953704523), new int64(0x240ca1cc, 0x77ac9c65), new int64(0x2de92c6f, 0x592b0275), new int64(0x4a7484aa, 0x6ea6e483), new int64(0x5cb0a9dc, -1119749164), new int64(0x76f988da, -2096016459), new int64(-1740746414, -295247957), new int64(-1473132947, 0x2db43210), new int64(-1341970488, -1728372417), new int64(-1084653625, -1091629340), new int64(-958395405, 0x3da88fc2), new int64(-710438585, -1828018395), new int64(0x6ca6351, -536640913), new int64(0x14292967, 0xa0e6e70), new int64(0x27b70a85, 0x46d22ffc), new int64(0x2e1b2138, 0x5c26c926), new int64(0x4d2c6dfc, 0x5ac42aed), new int64(0x53380d13, -1651133473), new int64(0x650a7354, -1951439906), new int64(0x766a0abb, 0x3c77b2a8), new int64(-2117940946, 0x47edaee6), new int64(-1838011259, 0x1482353b), new int64(-1564481375, 0x4cf10364), new int64(-1474664885, -1136513023), new int64(-1035236496, -789014639), new int64(-949202525, 0x654be30), new int64(-778901479, -688958952), new int64(-694614492, 0x5565a910), new int64(-200395387, 0x5771202a), new int64(0x106aa070, 0x32bbd1b8), new int64(0x19a4c116, -1194143544), new int64(0x1e376c08, 0x5141ab53), new int64(0x2748774c, -544281703), new int64(0x34b0bcb5, -509917016), new int64(0x391c0cb3, -976659869), new int64(0x4ed8aa4a, -482243893), new int64(0x5b9cca4f, 0x7763e373), new int64(0x682e6ff3, -692930397), new int64(0x748f82ee, 0x5defb2fc), new int64(0x78a5636f, 0x43172f60), new int64(-2067236844, -1578062990), new int64(-1933114872, 0x1a6439ec), new int64(-1866530822, 0x23631e28), new int64(-1538233109, -561857047), new int64(-1090935817, -1295615723), new int64(-965641998, -479046869), new int64(-903397682, -366583396), new int64(-779700025, 0x21c0c207), new int64(-354779690, -840897762), new int64(-176337025, -294727304), new int64(0x6f067aa, 0x72176fba), new int64(0xa637dc5, -1563912026), new int64(0x113f9804, -1090974290), new int64(0x1b710b35, 0x131c471b), new int64(0x28db77f5, 0x23047d84), new int64(0x32caab7b, 0x40c72493), new int64(0x3c9ebe0a, 0x15c9bebc), new int64(0x431d67c4, -1676669620), new int64(0x4cc5d4be, -885112138), new int64(0x597f299c, -60457430), new int64(0x5fcb6fab, 0x3ad6faec), new int64(0x6c44198c, 0x4a475817)];
-                 }
+         return replacements;
+       }
 
-                 for (i = 0; i < 80; i += 1) {
-                   W[i] = new int64(0, 0);
-                 } // append padding to the source string. The format is described in the FIPS.
+       function parseError(capture, idType) {
+         var compare = capture.toLowerCase();
 
+         if (_krData.localizeStrings[compare]) {
+           // some replacement strings can be localized
+           capture = _t('QA.keepRight.error_parts.' + _krData.localizeStrings[compare]);
+         }
 
-                 x[len >> 5] |= 0x80 << 24 - (len & 0x1f);
-                 x[(len + 128 >> 10 << 5) + 31] = len;
-                 l = x.length;
+         switch (idType) {
+           // link a string like "this node"
+           case 'this':
+             capture = linkErrorObject(capture);
+             break;
 
-                 for (i = 0; i < l; i += 32) {
-                   //32 dwords is the block size
-                   int64copy(a, H[0]);
-                   int64copy(b, H[1]);
-                   int64copy(c, H[2]);
-                   int64copy(d, H[3]);
-                   int64copy(e, H[4]);
-                   int64copy(f, H[5]);
-                   int64copy(g, H[6]);
-                   int64copy(h, H[7]);
+           case 'url':
+             capture = linkURL(capture);
+             break;
+           // link an entity ID
 
-                   for (j = 0; j < 16; j += 1) {
-                     W[j].h = x[i + 2 * j];
-                     W[j].l = x[i + 2 * j + 1];
-                   }
+           case 'n':
+           case 'w':
+           case 'r':
+             capture = linkEntity(idType + capture);
+             break;
+           // some errors have more complex ID lists/variance
 
-                   for (j = 16; j < 80; j += 1) {
-                     //sigma1
-                     int64rrot(r1, W[j - 2], 19);
-                     int64revrrot(r2, W[j - 2], 29);
-                     int64shr(r3, W[j - 2], 6);
-                     s1.l = r1.l ^ r2.l ^ r3.l;
-                     s1.h = r1.h ^ r2.h ^ r3.h; //sigma0
+           case '20':
+             capture = parse20(capture);
+             break;
 
-                     int64rrot(r1, W[j - 15], 1);
-                     int64rrot(r2, W[j - 15], 8);
-                     int64shr(r3, W[j - 15], 7);
-                     s0.l = r1.l ^ r2.l ^ r3.l;
-                     s0.h = r1.h ^ r2.h ^ r3.h;
-                     int64add4(W[j], s1, W[j - 7], s0, W[j - 16]);
-                   }
+           case '211':
+             capture = parse211(capture);
+             break;
 
-                   for (j = 0; j < 80; j += 1) {
-                     //Ch
-                     Ch.l = e.l & f.l ^ ~e.l & g.l;
-                     Ch.h = e.h & f.h ^ ~e.h & g.h; //Sigma1
+           case '231':
+             capture = parse231(capture);
+             break;
 
-                     int64rrot(r1, e, 14);
-                     int64rrot(r2, e, 18);
-                     int64revrrot(r3, e, 9);
-                     s1.l = r1.l ^ r2.l ^ r3.l;
-                     s1.h = r1.h ^ r2.h ^ r3.h; //Sigma0
+           case '294':
+             capture = parse294(capture);
+             break;
 
-                     int64rrot(r1, a, 28);
-                     int64revrrot(r2, a, 2);
-                     int64revrrot(r3, a, 7);
-                     s0.l = r1.l ^ r2.l ^ r3.l;
-                     s0.h = r1.h ^ r2.h ^ r3.h; //Maj
+           case '370':
+             capture = parse370(capture);
+             break;
+         }
 
-                     Maj.l = a.l & b.l ^ a.l & c.l ^ b.l & c.l;
-                     Maj.h = a.h & b.h ^ a.h & c.h ^ b.h & c.h;
-                     int64add5(T1, h, s1, Ch, sha512_k[j], W[j]);
-                     int64add(T2, s0, Maj);
-                     int64copy(h, g);
-                     int64copy(g, f);
-                     int64copy(f, e);
-                     int64add(e, d, T1);
-                     int64copy(d, c);
-                     int64copy(c, b);
-                     int64copy(b, a);
-                     int64add(a, T1, T2);
-                   }
+         return capture;
 
-                   int64add(H[0], H[0], a);
-                   int64add(H[1], H[1], b);
-                   int64add(H[2], H[2], c);
-                   int64add(H[3], H[3], d);
-                   int64add(H[4], H[4], e);
-                   int64add(H[5], H[5], f);
-                   int64add(H[6], H[6], g);
-                   int64add(H[7], H[7], h);
-                 } //represent the hash as an array of 32-bit dwords
+         function linkErrorObject(d) {
+           return "<a class=\"error_object_link\">".concat(d, "</a>");
+         }
 
+         function linkEntity(d) {
+           return "<a class=\"error_entity_link\">".concat(d, "</a>");
+         }
 
-                 for (i = 0; i < 8; i += 1) {
-                   hash[2 * i] = H[i].h;
-                   hash[2 * i + 1] = H[i].l;
-                 }
+         function linkURL(d) {
+           return "<a class=\"kr_external_link\" target=\"_blank\" href=\"".concat(d, "\">").concat(d, "</a>");
+         } // arbitrary node list of form: #ID, #ID, #ID...
 
-                 return hash;
-               } //A constructor for 64-bit numbers
 
+         function parse211(capture) {
+           var newList = [];
+           var items = capture.split(', ');
+           items.forEach(function (item) {
+             // ID has # at the front
+             var id = linkEntity('n' + item.slice(1));
+             newList.push(id);
+           });
+           return newList.join(', ');
+         } // arbitrary way list of form: #ID(layer),#ID(layer),#ID(layer)...
 
-               function int64(h, l) {
-                 this.h = h;
-                 this.l = l; //this.toString = int64toString;
-               } //Copies src into dst, assuming both are 64-bit numbers
 
+         function parse231(capture) {
+           var newList = []; // unfortunately 'layer' can itself contain commas, so we split on '),'
 
-               function int64copy(dst, src) {
-                 dst.h = src.h;
-                 dst.l = src.l;
-               } //Right-rotates a 64-bit number by shift
-               //Won't handle cases of shift>=32
-               //The function revrrot() is for that
+           var items = capture.split('),');
+           items.forEach(function (item) {
+             var match = item.match(/\#(\d+)\((.+)\)?/);
 
+             if (match !== null && match.length > 2) {
+               newList.push(linkEntity('w' + match[1]) + ' ' + _t('QA.keepRight.errorTypes.231.layer', {
+                 layer: match[2]
+               }));
+             }
+           });
+           return newList.join(', ');
+         } // arbitrary node/relation list of form: from node #ID,to relation #ID,to node #ID...
 
-               function int64rrot(dst, x, shift) {
-                 dst.l = x.l >>> shift | x.h << 32 - shift;
-                 dst.h = x.h >>> shift | x.l << 32 - shift;
-               } //Reverses the dwords of the source and then rotates right by shift.
-               //This is equivalent to rotation by 32+shift
 
+         function parse294(capture) {
+           var newList = [];
+           var items = capture.split(',');
+           items.forEach(function (item) {
+             // item of form "from/to node/relation #ID"
+             item = item.split(' '); // to/from role is more clear in quotes
 
-               function int64revrrot(dst, x, shift) {
-                 dst.l = x.h >>> shift | x.l << 32 - shift;
-                 dst.h = x.l >>> shift | x.h << 32 - shift;
-               } //Bitwise-shifts right a 64-bit number by shift
-               //Won't handle shift>=32, but it's never needed in SHA512
+             var role = "\"".concat(item[0], "\""); // first letter of node/relation provides the type
 
+             var idType = item[1].slice(0, 1); // ID has # at the front
 
-               function int64shr(dst, x, shift) {
-                 dst.l = x.l >>> shift | x.h << 32 - shift;
-                 dst.h = x.h >>> shift;
-               } //Adds two 64-bit numbers
-               //Like the original implementation, does not rely on 32-bit operations
+             var id = item[2].slice(1);
+             id = linkEntity(idType + id);
+             newList.push("".concat(role, " ").concat(item[1], " ").concat(id));
+           });
+           return newList.join(', ');
+         } // may or may not include the string "(including the name 'name')"
 
 
-               function int64add(dst, x, y) {
-                 var w0 = (x.l & 0xffff) + (y.l & 0xffff);
-                 var w1 = (x.l >>> 16) + (y.l >>> 16) + (w0 >>> 16);
-                 var w2 = (x.h & 0xffff) + (y.h & 0xffff) + (w1 >>> 16);
-                 var w3 = (x.h >>> 16) + (y.h >>> 16) + (w2 >>> 16);
-                 dst.l = w0 & 0xffff | w1 << 16;
-                 dst.h = w2 & 0xffff | w3 << 16;
-               } //Same, except with 4 addends. Works faster than adding them one by one.
+         function parse370(capture) {
+           if (!capture) return '';
+           var match = capture.match(/\(including the name (\'.+\')\)/);
+
+           if (match && match.length) {
+             return _t('QA.keepRight.errorTypes.370.including_the_name', {
+               name: match[1]
+             });
+           }
 
+           return '';
+         } // arbitrary node list of form: #ID,#ID,#ID...
 
-               function int64add4(dst, a, b, c, d) {
-                 var w0 = (a.l & 0xffff) + (b.l & 0xffff) + (c.l & 0xffff) + (d.l & 0xffff);
-                 var w1 = (a.l >>> 16) + (b.l >>> 16) + (c.l >>> 16) + (d.l >>> 16) + (w0 >>> 16);
-                 var w2 = (a.h & 0xffff) + (b.h & 0xffff) + (c.h & 0xffff) + (d.h & 0xffff) + (w1 >>> 16);
-                 var w3 = (a.h >>> 16) + (b.h >>> 16) + (c.h >>> 16) + (d.h >>> 16) + (w2 >>> 16);
-                 dst.l = w0 & 0xffff | w1 << 16;
-                 dst.h = w2 & 0xffff | w3 << 16;
-               } //Same, except with 5 addends
 
+         function parse20(capture) {
+           var newList = [];
+           var items = capture.split(',');
+           items.forEach(function (item) {
+             // ID has # at the front
+             var id = linkEntity('n' + item.slice(1));
+             newList.push(id);
+           });
+           return newList.join(', ');
+         }
+       }
 
-               function int64add5(dst, a, b, c, d, e) {
-                 var w0 = (a.l & 0xffff) + (b.l & 0xffff) + (c.l & 0xffff) + (d.l & 0xffff) + (e.l & 0xffff),
-                     w1 = (a.l >>> 16) + (b.l >>> 16) + (c.l >>> 16) + (d.l >>> 16) + (e.l >>> 16) + (w0 >>> 16),
-                     w2 = (a.h & 0xffff) + (b.h & 0xffff) + (c.h & 0xffff) + (d.h & 0xffff) + (e.h & 0xffff) + (w1 >>> 16),
-                     w3 = (a.h >>> 16) + (b.h >>> 16) + (c.h >>> 16) + (d.h >>> 16) + (e.h >>> 16) + (w2 >>> 16);
-                 dst.l = w0 & 0xffff | w1 << 16;
-                 dst.h = w2 & 0xffff | w3 << 16;
-               }
-             },
+       var serviceKeepRight = {
+         title: 'keepRight',
+         init: function init() {
+           _mainFileFetcher.get('keepRight').then(function (d) {
+             return _krData = d;
+           });
 
-             /**
-              * @class Hashes.RMD160
-              * @constructor
-              * @param {Object} [config]
-              *
-              * A JavaScript implementation of the RIPEMD-160 Algorithm
-              * Version 2.2 Copyright Jeremy Lin, Paul Johnston 2000 - 2009.
-              * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
-              * See http://pajhome.org.uk/crypt/md5 for details.
-              * Also http://www.ocf.berkeley.edu/~jjlin/jsotp/
-              */
-             RMD160: function RMD160(options) {
-               /**
-                * Private properties configuration variables. You may need to tweak these to be compatible with
-                * the server-side, but the defaults work in most cases.
-                * @see this.setUpperCase() method
-                * @see this.setPad() method
-                */
-               var hexcase = options && typeof options.uppercase === 'boolean' ? options.uppercase : false,
+           if (!_cache$2) {
+             this.reset();
+           }
 
-               /* hexadecimal output case format. false - lowercase; true - uppercase  */
-               b64pad = options && typeof options.pad === 'string' ? options.pa : '=',
+           this.event = utilRebind(this, dispatch$7, 'on');
+         },
+         reset: function reset() {
+           if (_cache$2) {
+             Object.values(_cache$2.inflightTile).forEach(abortRequest$6);
+           }
 
-               /* base-64 pad character. Default '=' for strict RFC compliance   */
-               utf8 = options && typeof options.utf8 === 'boolean' ? options.utf8 : true,
+           _cache$2 = {
+             data: {},
+             loadedTile: {},
+             inflightTile: {},
+             inflightPost: {},
+             closed: {},
+             rtree: new RBush()
+           };
+         },
+         // KeepRight API:  http://osm.mueschelsoft.de/keepright/interfacing.php
+         loadIssues: function loadIssues(projection) {
+           var _this = this;
 
-               /* enable/disable utf8 encoding */
-               rmd160_r1 = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 7, 4, 13, 1, 10, 6, 15, 3, 12, 0, 9, 5, 2, 14, 11, 8, 3, 10, 14, 4, 9, 15, 8, 1, 2, 7, 0, 6, 13, 11, 5, 12, 1, 9, 11, 10, 0, 8, 12, 4, 13, 3, 7, 15, 14, 5, 6, 2, 4, 0, 5, 9, 7, 12, 2, 10, 14, 1, 3, 8, 11, 6, 15, 13],
-                   rmd160_r2 = [5, 14, 7, 0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12, 6, 11, 3, 7, 0, 13, 5, 10, 14, 15, 8, 12, 4, 9, 1, 2, 15, 5, 1, 3, 7, 14, 6, 9, 11, 8, 12, 2, 10, 0, 4, 13, 8, 6, 4, 1, 3, 11, 15, 0, 5, 12, 2, 13, 9, 7, 10, 14, 12, 15, 10, 4, 1, 5, 8, 7, 6, 2, 13, 14, 0, 3, 9, 11],
-                   rmd160_s1 = [11, 14, 15, 12, 5, 8, 7, 9, 11, 13, 14, 15, 6, 7, 9, 8, 7, 6, 8, 13, 11, 9, 7, 15, 7, 12, 15, 9, 11, 7, 13, 12, 11, 13, 6, 7, 14, 9, 13, 15, 14, 8, 13, 6, 5, 12, 7, 5, 11, 12, 14, 15, 14, 15, 9, 8, 9, 14, 5, 6, 8, 6, 5, 12, 9, 15, 5, 11, 6, 8, 13, 12, 5, 12, 13, 14, 11, 8, 5, 6],
-                   rmd160_s2 = [8, 9, 9, 11, 13, 15, 15, 5, 7, 7, 8, 11, 14, 14, 12, 6, 9, 13, 15, 7, 12, 8, 9, 11, 7, 7, 12, 7, 6, 15, 13, 11, 9, 7, 15, 11, 8, 6, 6, 14, 12, 13, 5, 14, 13, 13, 7, 5, 15, 5, 8, 11, 14, 14, 6, 14, 6, 9, 12, 9, 12, 5, 15, 8, 8, 5, 12, 9, 12, 5, 14, 6, 8, 13, 6, 5, 15, 13, 11, 11];
-               /* privileged (public) methods */
+           var options = {
+             format: 'geojson',
+             ch: _krRuleset
+           }; // determine the needed tiles to cover the view
 
-               this.hex = function (s) {
-                 return rstr2hex(rstr(s));
-               };
+           var tiles = tiler$6.zoomExtent([_tileZoom$3, _tileZoom$3]).getTiles(projection); // abort inflight requests that are no longer needed
 
-               this.b64 = function (s) {
-                 return rstr2b64(rstr(s), b64pad);
-               };
+           abortUnwantedRequests$3(_cache$2, tiles); // issue new requests..
 
-               this.any = function (s, e) {
-                 return rstr2any(rstr(s), e);
-               };
+           tiles.forEach(function (tile) {
+             if (_cache$2.loadedTile[tile.id] || _cache$2.inflightTile[tile.id]) return;
 
-               this.raw = function (s) {
-                 return rstr(s);
-               };
+             var _tile$extent$rectangl = tile.extent.rectangle(),
+                 _tile$extent$rectangl2 = _slicedToArray(_tile$extent$rectangl, 4),
+                 left = _tile$extent$rectangl2[0],
+                 top = _tile$extent$rectangl2[1],
+                 right = _tile$extent$rectangl2[2],
+                 bottom = _tile$extent$rectangl2[3];
 
-               this.hex_hmac = function (k, d) {
-                 return rstr2hex(rstr_hmac(k, d));
-               };
+             var params = Object.assign({}, options, {
+               left: left,
+               bottom: bottom,
+               right: right,
+               top: top
+             });
+             var url = "".concat(_krUrlRoot, "/export.php?") + utilQsString(params);
+             var controller = new AbortController();
+             _cache$2.inflightTile[tile.id] = controller;
+             d3_json(url, {
+               signal: controller.signal
+             }).then(function (data) {
+               delete _cache$2.inflightTile[tile.id];
+               _cache$2.loadedTile[tile.id] = true;
 
-               this.b64_hmac = function (k, d) {
-                 return rstr2b64(rstr_hmac(k, d), b64pad);
-               };
+               if (!data || !data.features || !data.features.length) {
+                 throw new Error('No Data');
+               }
 
-               this.any_hmac = function (k, d, e) {
-                 return rstr2any(rstr_hmac(k, d), e);
-               };
-               /**
-                * Perform a simple self-test to see if the VM is working
-                * @return {String} Hexadecimal hash sample
-                * @public
-                */
+               data.features.forEach(function (feature) {
+                 var _feature$properties = feature.properties,
+                     itemType = _feature$properties.error_type,
+                     id = _feature$properties.error_id,
+                     _feature$properties$c = _feature$properties.comment,
+                     comment = _feature$properties$c === void 0 ? null : _feature$properties$c,
+                     objectId = _feature$properties.object_id,
+                     objectType = _feature$properties.object_type,
+                     schema = _feature$properties.schema,
+                     title = _feature$properties.title;
+                 var loc = feature.geometry.coordinates,
+                     _feature$properties$d = feature.properties.description,
+                     description = _feature$properties$d === void 0 ? '' : _feature$properties$d; // if there is a parent, save its error type e.g.:
+                 //  Error 191 = "highway-highway"
+                 //  Error 190 = "intersections without junctions"  (parent)
 
+                 var issueTemplate = _krData.errorTypes[itemType];
+                 var parentIssueType = (Math.floor(itemType / 10) * 10).toString(); // try to handle error type directly, fallback to parent error type.
 
-               this.vm_test = function () {
-                 return hex('abc').toLowerCase() === '900150983cd24fb0d6963f7d28e17f72';
-               };
-               /**
-                * @description Enable/disable uppercase hexadecimal returned string
-                * @param {boolean}
-                * @return {Object} this
-                * @public
-                */
+                 var whichType = issueTemplate ? itemType : parentIssueType;
+                 var whichTemplate = _krData.errorTypes[whichType]; // Rewrite a few of the errors at this point..
+                 // This is done to make them easier to linkify and translate.
 
+                 switch (whichType) {
+                   case '170':
+                     description = "This feature has a FIXME tag: ".concat(description);
+                     break;
 
-               this.setUpperCase = function (a) {
-                 if (typeof a === 'boolean') {
-                   hexcase = a;
-                 }
+                   case '292':
+                   case '293':
+                     description = description.replace('A turn-', 'This turn-');
+                     break;
 
-                 return this;
-               };
-               /**
-                * @description Defines a base64 pad string
-                * @param {string} Pad
-                * @return {Object} this
-                * @public
-                */
+                   case '294':
+                   case '295':
+                   case '296':
+                   case '297':
+                   case '298':
+                     description = "This turn-restriction~".concat(description);
+                     break;
 
+                   case '300':
+                     description = 'This highway is missing a maxspeed tag';
+                     break;
 
-               this.setPad = function (a) {
-                 if (typeof a !== 'undefined') {
-                   b64pad = a;
-                 }
+                   case '411':
+                   case '412':
+                   case '413':
+                     description = "This feature~".concat(description);
+                     break;
+                 } // move markers slightly so it doesn't obscure the geometry,
+                 // then move markers away from other coincident markers
 
-                 return this;
-               };
-               /**
-                * @description Defines a base64 pad string
-                * @param {boolean}
-                * @return {Object} this
-                * @public
-                */
 
+                 var coincident = false;
 
-               this.setUTF8 = function (a) {
-                 if (typeof a === 'boolean') {
-                   utf8 = a;
-                 }
+                 do {
+                   // first time, move marker up. after that, move marker right.
+                   var delta = coincident ? [0.00001, 0] : [0, 0.00001];
+                   loc = geoVecAdd(loc, delta);
+                   var bbox = geoExtent(loc).bbox();
+                   coincident = _cache$2.rtree.search(bbox).length;
+                 } while (coincident);
 
-                 return this;
-               };
-               /* private methods */
+                 var d = new QAItem(loc, _this, itemType, id, {
+                   comment: comment,
+                   description: description,
+                   whichType: whichType,
+                   parentIssueType: parentIssueType,
+                   severity: whichTemplate.severity || 'error',
+                   objectId: objectId,
+                   objectType: objectType,
+                   schema: schema,
+                   title: title
+                 });
+                 d.replacements = tokenReplacements(d);
+                 _cache$2.data[id] = d;
 
-               /**
-                * Calculate the rmd160 of a raw string
-                */
+                 _cache$2.rtree.insert(encodeIssueRtree$2(d));
+               });
+               dispatch$7.call('loaded');
+             })["catch"](function () {
+               delete _cache$2.inflightTile[tile.id];
+               _cache$2.loadedTile[tile.id] = true;
+             });
+           });
+         },
+         postUpdate: function postUpdate(d, callback) {
+           var _this2 = this;
 
+           if (_cache$2.inflightPost[d.id]) {
+             return callback({
+               message: 'Error update already inflight',
+               status: -2
+             }, d);
+           }
 
-               function rstr(s) {
-                 s = utf8 ? utf8Encode(s) : s;
-                 return binl2rstr(binl(rstr2binl(s), s.length * 8));
-               }
-               /**
-                * Calculate the HMAC-rmd160 of a key and some data (raw strings)
-                */
+           var params = {
+             schema: d.schema,
+             id: d.id
+           };
 
+           if (d.newStatus) {
+             params.st = d.newStatus;
+           }
 
-               function rstr_hmac(key, data) {
-                 key = utf8 ? utf8Encode(key) : key;
-                 data = utf8 ? utf8Encode(data) : data;
-                 var i,
-                     hash,
-                     bkey = rstr2binl(key),
-                     ipad = Array(16),
-                     opad = Array(16);
+           if (d.newComment !== undefined) {
+             params.co = d.newComment;
+           } // NOTE: This throws a CORS err, but it seems successful.
+           // We don't care too much about the response, so this is fine.
 
-                 if (bkey.length > 16) {
-                   bkey = binl(bkey, key.length * 8);
-                 }
 
-                 for (i = 0; i < 16; i += 1) {
-                   ipad[i] = bkey[i] ^ 0x36363636;
-                   opad[i] = bkey[i] ^ 0x5C5C5C5C;
-                 }
+           var url = "".concat(_krUrlRoot, "/comment.php?") + utilQsString(params);
+           var controller = new AbortController();
+           _cache$2.inflightPost[d.id] = controller; // Since this is expected to throw an error just continue as if it worked
+           // (worst case scenario the request truly fails and issue will show up if iD restarts)
 
-                 hash = binl(ipad.concat(rstr2binl(data)), 512 + data.length * 8);
-                 return binl2rstr(binl(opad.concat(hash), 512 + 160));
-               }
-               /**
-                * Convert an array of little-endian words to a string
-                */
+           d3_json(url, {
+             signal: controller.signal
+           })["finally"](function () {
+             delete _cache$2.inflightPost[d.id];
 
+             if (d.newStatus === 'ignore') {
+               // ignore permanently (false positive)
+               _this2.removeItem(d);
+             } else if (d.newStatus === 'ignore_t') {
+               // ignore temporarily (error fixed)
+               _this2.removeItem(d);
 
-               function binl2rstr(input) {
-                 var i,
-                     output = '',
-                     l = input.length * 32;
+               _cache$2.closed["".concat(d.schema, ":").concat(d.id)] = true;
+             } else {
+               d = _this2.replaceItem(d.update({
+                 comment: d.newComment,
+                 newComment: undefined,
+                 newState: undefined
+               }));
+             }
 
-                 for (i = 0; i < l; i += 8) {
-                   output += String.fromCharCode(input[i >> 5] >>> i % 32 & 0xFF);
-                 }
+             if (callback) callback(null, d);
+           });
+         },
+         // Get all cached QAItems covering the viewport
+         getItems: function getItems(projection) {
+           var viewport = projection.clipExtent();
+           var min = [viewport[0][0], viewport[1][1]];
+           var max = [viewport[1][0], viewport[0][1]];
+           var bbox = geoExtent(projection.invert(min), projection.invert(max)).bbox();
+           return _cache$2.rtree.search(bbox).map(function (d) {
+             return d.data;
+           });
+         },
+         // Get a QAItem from cache
+         // NOTE: Don't change method name until UI v3 is merged
+         getError: function getError(id) {
+           return _cache$2.data[id];
+         },
+         // Replace a single QAItem in the cache
+         replaceItem: function replaceItem(item) {
+           if (!(item instanceof QAItem) || !item.id) return;
+           _cache$2.data[item.id] = item;
+           updateRtree$3(encodeIssueRtree$2(item), true); // true = replace
 
-                 return output;
-               }
-               /**
-                * Calculate the RIPE-MD160 of an array of little-endian words, and a bit length.
-                */
+           return item;
+         },
+         // Remove a single QAItem from the cache
+         removeItem: function removeItem(item) {
+           if (!(item instanceof QAItem) || !item.id) return;
+           delete _cache$2.data[item.id];
+           updateRtree$3(encodeIssueRtree$2(item), false); // false = remove
+         },
+         issueURL: function issueURL(item) {
+           return "".concat(_krUrlRoot, "/report_map.php?schema=").concat(item.schema, "&error=").concat(item.id);
+         },
+         // Get an array of issues closed during this session.
+         // Used to populate `closed:keepright` changeset tag
+         getClosedIDs: function getClosedIDs() {
+           return Object.keys(_cache$2.closed).sort();
+         }
+       };
 
+       var tiler$5 = utilTiler();
+       var dispatch$6 = dispatch$8('loaded');
+       var _tileZoom$2 = 14;
+       var _impOsmUrls = {
+         ow: 'https://grab.community.improve-osm.org/directionOfFlowService',
+         mr: 'https://grab.community.improve-osm.org/missingGeoService',
+         tr: 'https://grab.community.improve-osm.org/turnRestrictionService'
+       };
+       var _impOsmData = {
+         icons: {}
+       }; // This gets reassigned if reset
 
-               function binl(x, len) {
-                 var T,
-                     j,
-                     i,
-                     l,
-                     h0 = 0x67452301,
-                     h1 = 0xefcdab89,
-                     h2 = 0x98badcfe,
-                     h3 = 0x10325476,
-                     h4 = 0xc3d2e1f0,
-                     A1,
-                     B1,
-                     C1,
-                     D1,
-                     E1,
-                     A2,
-                     B2,
-                     C2,
-                     D2,
-                     E2;
-                 /* append padding */
+       var _cache$1;
 
-                 x[len >> 5] |= 0x80 << len % 32;
-                 x[(len + 64 >>> 9 << 4) + 14] = len;
-                 l = x.length;
+       function abortRequest$5(i) {
+         Object.values(i).forEach(function (controller) {
+           if (controller) {
+             controller.abort();
+           }
+         });
+       }
 
-                 for (i = 0; i < l; i += 16) {
-                   A1 = A2 = h0;
-                   B1 = B2 = h1;
-                   C1 = C2 = h2;
-                   D1 = D2 = h3;
-                   E1 = E2 = h4;
+       function abortUnwantedRequests$2(cache, tiles) {
+         Object.keys(cache.inflightTile).forEach(function (k) {
+           var wanted = tiles.find(function (tile) {
+             return k === tile.id;
+           });
 
-                   for (j = 0; j <= 79; j += 1) {
-                     T = safe_add(A1, rmd160_f(j, B1, C1, D1));
-                     T = safe_add(T, x[i + rmd160_r1[j]]);
-                     T = safe_add(T, rmd160_K1(j));
-                     T = safe_add(bit_rol(T, rmd160_s1[j]), E1);
-                     A1 = E1;
-                     E1 = D1;
-                     D1 = bit_rol(C1, 10);
-                     C1 = B1;
-                     B1 = T;
-                     T = safe_add(A2, rmd160_f(79 - j, B2, C2, D2));
-                     T = safe_add(T, x[i + rmd160_r2[j]]);
-                     T = safe_add(T, rmd160_K2(j));
-                     T = safe_add(bit_rol(T, rmd160_s2[j]), E2);
-                     A2 = E2;
-                     E2 = D2;
-                     D2 = bit_rol(C2, 10);
-                     C2 = B2;
-                     B2 = T;
-                   }
+           if (!wanted) {
+             abortRequest$5(cache.inflightTile[k]);
+             delete cache.inflightTile[k];
+           }
+         });
+       }
 
-                   T = safe_add(h1, safe_add(C1, D2));
-                   h1 = safe_add(h2, safe_add(D1, E2));
-                   h2 = safe_add(h3, safe_add(E1, A2));
-                   h3 = safe_add(h4, safe_add(A1, B2));
-                   h4 = safe_add(h0, safe_add(B1, C2));
-                   h0 = T;
-                 }
+       function encodeIssueRtree$1(d) {
+         return {
+           minX: d.loc[0],
+           minY: d.loc[1],
+           maxX: d.loc[0],
+           maxY: d.loc[1],
+           data: d
+         };
+       } // Replace or remove QAItem from rtree
 
-                 return [h0, h1, h2, h3, h4];
-               } // specific algorithm methods
 
+       function updateRtree$2(item, replace) {
+         _cache$1.rtree.remove(item, function (a, b) {
+           return a.data.id === b.data.id;
+         });
 
-               function rmd160_f(j, x, y, z) {
-                 return 0 <= j && j <= 15 ? x ^ y ^ z : 16 <= j && j <= 31 ? x & y | ~x & z : 32 <= j && j <= 47 ? (x | ~y) ^ z : 48 <= j && j <= 63 ? x & z | y & ~z : 64 <= j && j <= 79 ? x ^ (y | ~z) : 'rmd160_f: j out of range';
-               }
+         if (replace) {
+           _cache$1.rtree.insert(item);
+         }
+       }
 
-               function rmd160_K1(j) {
-                 return 0 <= j && j <= 15 ? 0x00000000 : 16 <= j && j <= 31 ? 0x5a827999 : 32 <= j && j <= 47 ? 0x6ed9eba1 : 48 <= j && j <= 63 ? 0x8f1bbcdc : 64 <= j && j <= 79 ? 0xa953fd4e : 'rmd160_K1: j out of range';
-               }
+       function linkErrorObject(d) {
+         return "<a class=\"error_object_link\">".concat(d, "</a>");
+       }
 
-               function rmd160_K2(j) {
-                 return 0 <= j && j <= 15 ? 0x50a28be6 : 16 <= j && j <= 31 ? 0x5c4dd124 : 32 <= j && j <= 47 ? 0x6d703ef3 : 48 <= j && j <= 63 ? 0x7a6d76e9 : 64 <= j && j <= 79 ? 0x00000000 : 'rmd160_K2: j out of range';
-               }
-             }
-           }; // exposes Hashes
+       function linkEntity(d) {
+         return "<a class=\"error_entity_link\">".concat(d, "</a>");
+       }
 
-           (function (window, undefined$1) {
-             var freeExports = false;
+       function pointAverage(points) {
+         if (points.length) {
+           var sum = points.reduce(function (acc, point) {
+             return geoVecAdd(acc, [point.lon, point.lat]);
+           }, [0, 0]);
+           return geoVecScale(sum, 1 / points.length);
+         } else {
+           return [0, 0];
+         }
+       }
 
-             {
-               freeExports = exports;
+       function relativeBearing(p1, p2) {
+         var angle = Math.atan2(p2.lon - p1.lon, p2.lat - p1.lat);
 
-               if (exports && _typeof(commonjsGlobal) === 'object' && commonjsGlobal && commonjsGlobal === commonjsGlobal.global) {
-                 window = commonjsGlobal;
-               }
-             }
+         if (angle < 0) {
+           angle += 2 * Math.PI;
+         } // Return degrees
 
-             if (typeof undefined$1 === 'function' && _typeof(undefined$1.amd) === 'object' && undefined$1.amd) {
-               // define as an anonymous module, so, through path mapping, it can be aliased
-               undefined$1(function () {
-                 return Hashes;
-               });
-             } else if (freeExports) {
-               // in Node.js or RingoJS v0.8.0+
-               if ( module && module.exports === freeExports) {
-                 module.exports = Hashes;
-               } // in Narwhal or RingoJS v0.7.0-
-               else {
-                   freeExports.Hashes = Hashes;
-                 }
-             } else {
-               // in a browser or Rhino
-               window.Hashes = Hashes;
-             }
-           })(this);
-         })(); // IIFE
 
-       });
+         return angle * 180 / Math.PI;
+       } // Assuming range [0,360)
+
 
-       var immutable = extend$2;
-       var hasOwnProperty$2 = Object.prototype.hasOwnProperty;
+       function cardinalDirection(bearing) {
+         var dir = 45 * Math.round(bearing / 45);
+         var compass = {
+           0: 'north',
+           45: 'northeast',
+           90: 'east',
+           135: 'southeast',
+           180: 'south',
+           225: 'southwest',
+           270: 'west',
+           315: 'northwest',
+           360: 'north'
+         };
+         return _t("QA.improveOSM.directions.".concat(compass[dir]));
+       } // Errors shouldn't obscure each other
 
-       function extend$2() {
-         var target = {};
 
-         for (var i = 0; i < arguments.length; i++) {
-           var source = arguments[i];
+       function preventCoincident$1(loc, bumpUp) {
+         var coincident = false;
 
-           for (var key in source) {
-             if (hasOwnProperty$2.call(source, key)) {
-               target[key] = source[key];
-             }
-           }
-         }
+         do {
+           // first time, move marker up. after that, move marker right.
+           var delta = coincident ? [0.00001, 0] : bumpUp ? [0, 0.00001] : [0, 0];
+           loc = geoVecAdd(loc, delta);
+           var bbox = geoExtent(loc).bbox();
+           coincident = _cache$1.rtree.search(bbox).length;
+         } while (coincident);
 
-         return target;
+         return loc;
        }
 
-       var sha1 = new hashes.SHA1();
-       var ohauth = {};
-
-       ohauth.qsString = function (obj) {
-         return Object.keys(obj).sort().map(function (key) {
-           return ohauth.percentEncode(key) + '=' + ohauth.percentEncode(obj[key]);
-         }).join('&');
-       };
-
-       ohauth.stringQs = function (str) {
-         return str.split('&').filter(function (pair) {
-           return pair !== '';
-         }).reduce(function (obj, pair) {
-           var parts = pair.split('=');
-           obj[decodeURIComponent(parts[0])] = null === parts[1] ? '' : decodeURIComponent(parts[1]);
-           return obj;
-         }, {});
-       };
-
-       ohauth.rawxhr = function (method, url, data, headers, callback) {
-         var xhr = new XMLHttpRequest(),
-             twoHundred = /^20\d$/;
+       var serviceImproveOSM = {
+         title: 'improveOSM',
+         init: function init() {
+           _mainFileFetcher.get('qa_data').then(function (d) {
+             return _impOsmData = d.improveOSM;
+           });
 
-         xhr.onreadystatechange = function () {
-           if (4 === xhr.readyState && 0 !== xhr.status) {
-             if (twoHundred.test(xhr.status)) callback(null, xhr);else return callback(xhr, null);
+           if (!_cache$1) {
+             this.reset();
            }
-         };
 
-         xhr.onerror = function (e) {
-           return callback(e, null);
-         };
+           this.event = utilRebind(this, dispatch$6, 'on');
+         },
+         reset: function reset() {
+           if (_cache$1) {
+             Object.values(_cache$1.inflightTile).forEach(abortRequest$5);
+           }
 
-         xhr.open(method, url, true);
+           _cache$1 = {
+             data: {},
+             loadedTile: {},
+             inflightTile: {},
+             inflightPost: {},
+             closed: {},
+             rtree: new RBush()
+           };
+         },
+         loadIssues: function loadIssues(projection) {
+           var _this = this;
 
-         for (var h in headers) {
-           xhr.setRequestHeader(h, headers[h]);
-         }
+           var options = {
+             client: 'iD',
+             status: 'OPEN',
+             zoom: '19' // Use a high zoom so that clusters aren't returned
 
-         xhr.send(data);
-         return xhr;
-       };
+           }; // determine the needed tiles to cover the view
 
-       ohauth.xhr = function (method, url, auth, data, options, callback) {
-         var headers = options && options.header || {
-           'Content-Type': 'application/x-www-form-urlencoded'
-         };
-         headers.Authorization = 'OAuth ' + ohauth.authHeader(auth);
-         return ohauth.rawxhr(method, url, data, headers, callback);
-       };
+           var tiles = tiler$5.zoomExtent([_tileZoom$2, _tileZoom$2]).getTiles(projection); // abort inflight requests that are no longer needed
 
-       ohauth.nonce = function () {
-         for (var o = ''; o.length < 6;) {
-           o += '0123456789ABCDEFGHIJKLMNOPQRSTUVWXTZabcdefghiklmnopqrstuvwxyz'[Math.floor(Math.random() * 61)];
-         }
+           abortUnwantedRequests$2(_cache$1, tiles); // issue new requests..
 
-         return o;
-       };
+           tiles.forEach(function (tile) {
+             if (_cache$1.loadedTile[tile.id] || _cache$1.inflightTile[tile.id]) return;
 
-       ohauth.authHeader = function (obj) {
-         return Object.keys(obj).sort().map(function (key) {
-           return encodeURIComponent(key) + '="' + encodeURIComponent(obj[key]) + '"';
-         }).join(', ');
-       };
+             var _tile$extent$rectangl = tile.extent.rectangle(),
+                 _tile$extent$rectangl2 = _slicedToArray(_tile$extent$rectangl, 4),
+                 east = _tile$extent$rectangl2[0],
+                 north = _tile$extent$rectangl2[1],
+                 west = _tile$extent$rectangl2[2],
+                 south = _tile$extent$rectangl2[3];
 
-       ohauth.timestamp = function () {
-         return ~~(+new Date() / 1000);
-       };
+             var params = Object.assign({}, options, {
+               east: east,
+               south: south,
+               west: west,
+               north: north
+             }); // 3 separate requests to store for each tile
 
-       ohauth.percentEncode = function (s) {
-         return encodeURIComponent(s).replace(/\!/g, '%21').replace(/\'/g, '%27').replace(/\*/g, '%2A').replace(/\(/g, '%28').replace(/\)/g, '%29');
-       };
+             var requests = {};
+             Object.keys(_impOsmUrls).forEach(function (k) {
+               // We exclude WATER from missing geometry as it doesn't seem useful
+               // We use most confident one-way and turn restrictions only, still have false positives
+               var kParams = Object.assign({}, params, k === 'mr' ? {
+                 type: 'PARKING,ROAD,BOTH,PATH'
+               } : {
+                 confidenceLevel: 'C1'
+               });
+               var url = "".concat(_impOsmUrls[k], "/search?") + utilQsString(kParams);
+               var controller = new AbortController();
+               requests[k] = controller;
+               d3_json(url, {
+                 signal: controller.signal
+               }).then(function (data) {
+                 delete _cache$1.inflightTile[tile.id][k];
 
-       ohauth.baseString = function (method, url, params) {
-         if (params.oauth_signature) delete params.oauth_signature;
-         return [method, ohauth.percentEncode(url), ohauth.percentEncode(ohauth.qsString(params))].join('&');
-       };
+                 if (!Object.keys(_cache$1.inflightTile[tile.id]).length) {
+                   delete _cache$1.inflightTile[tile.id];
+                   _cache$1.loadedTile[tile.id] = true;
+                 } // Road segments at high zoom == oneways
 
-       ohauth.signature = function (oauth_secret, token_secret, baseString) {
-         return sha1.b64_hmac(ohauth.percentEncode(oauth_secret) + '&' + ohauth.percentEncode(token_secret), baseString);
-       };
-       /**
-        * Takes an options object for configuration (consumer_key,
-        * consumer_secret, version, signature_method, token, token_secret)
-        * and returns a function that generates the Authorization header
-        * for given data.
-        *
-        * The returned function takes these parameters:
-        * - method: GET/POST/...
-        * - uri: full URI with protocol, port, path and query string
-        * - extra_params: any extra parameters (that are passed in the POST data),
-        *   can be an object or a from-urlencoded string.
-        *
-        * Returned function returns full OAuth header with "OAuth" string in it.
-        */
 
+                 if (data.roadSegments) {
+                   data.roadSegments.forEach(function (feature) {
+                     // Position error at the approximate middle of the segment
+                     var points = feature.points,
+                         wayId = feature.wayId,
+                         fromNodeId = feature.fromNodeId,
+                         toNodeId = feature.toNodeId;
+                     var itemId = "".concat(wayId).concat(fromNodeId).concat(toNodeId);
+                     var mid = points.length / 2;
+                     var loc; // Even number of points, find midpoint of the middle two
+                     // Odd number of points, use position of very middle point
 
-       ohauth.headerGenerator = function (options) {
-         options = options || {};
-         var consumer_key = options.consumer_key || '',
-             consumer_secret = options.consumer_secret || '',
-             signature_method = options.signature_method || 'HMAC-SHA1',
-             version = options.version || '1.0',
-             token = options.token || '',
-             token_secret = options.token_secret || '';
-         return function (method, uri, extra_params) {
-           method = method.toUpperCase();
+                     if (mid % 1 === 0) {
+                       loc = pointAverage([points[mid - 1], points[mid]]);
+                     } else {
+                       mid = points[Math.floor(mid)];
+                       loc = [mid.lon, mid.lat];
+                     } // One-ways can land on same segment in opposite direction
 
-           if (typeof extra_params === 'string' && extra_params.length > 0) {
-             extra_params = ohauth.stringQs(extra_params);
-           }
 
-           var uri_parts = uri.split('?', 2),
-               base_uri = uri_parts[0];
-           var query_params = uri_parts.length === 2 ? ohauth.stringQs(uri_parts[1]) : {};
-           var oauth_params = {
-             oauth_consumer_key: consumer_key,
-             oauth_signature_method: signature_method,
-             oauth_version: version,
-             oauth_timestamp: ohauth.timestamp(),
-             oauth_nonce: ohauth.nonce()
-           };
-           if (token) oauth_params.oauth_token = token;
-           var all_params = immutable({}, oauth_params, query_params, extra_params),
-               base_str = ohauth.baseString(method, base_uri, all_params);
-           oauth_params.oauth_signature = ohauth.signature(consumer_secret, token_secret, base_str);
-           return 'OAuth ' + ohauth.authHeader(oauth_params);
-         };
-       };
+                     loc = preventCoincident$1(loc, false);
+                     var d = new QAItem(loc, _this, k, itemId, {
+                       issueKey: k,
+                       // used as a category
+                       identifier: {
+                         // used to post changes
+                         wayId: wayId,
+                         fromNodeId: fromNodeId,
+                         toNodeId: toNodeId
+                       },
+                       objectId: wayId,
+                       objectType: 'way'
+                     }); // Variables used in the description
 
-       var ohauth_1 = ohauth;
+                     d.replacements = {
+                       percentage: feature.percentOfTrips,
+                       num_trips: feature.numberOfTrips,
+                       highway: linkErrorObject(_t('QA.keepRight.error_parts.highway')),
+                       from_node: linkEntity('n' + feature.fromNodeId),
+                       to_node: linkEntity('n' + feature.toNodeId)
+                     };
+                     _cache$1.data[d.id] = d;
 
-       var resolveUrl$1 = createCommonjsModule(function (module, exports) {
-         // Copyright 2014 Simon Lydell
-         // X11 (“MIT”) Licensed. (See LICENSE.)
-         void function (root, factory) {
-           {
-             module.exports = factory();
-           }
-         }(commonjsGlobal, function () {
-           function resolveUrl()
-           /* ...urls */
-           {
-             var numUrls = arguments.length;
+                     _cache$1.rtree.insert(encodeIssueRtree$1(d));
+                   });
+                 } // Tiles at high zoom == missing roads
 
-             if (numUrls === 0) {
-               throw new Error("resolveUrl requires at least one argument; got none.");
-             }
 
-             var base = document.createElement("base");
-             base.href = arguments[0];
+                 if (data.tiles) {
+                   data.tiles.forEach(function (feature) {
+                     var type = feature.type,
+                         x = feature.x,
+                         y = feature.y,
+                         numberOfTrips = feature.numberOfTrips;
+                     var geoType = type.toLowerCase();
+                     var itemId = "".concat(geoType).concat(x).concat(y).concat(numberOfTrips); // Average of recorded points should land on the missing geometry
+                     // Missing geometry could happen to land on another error
 
-             if (numUrls === 1) {
-               return base.href;
-             }
+                     var loc = pointAverage(feature.points);
+                     loc = preventCoincident$1(loc, false);
+                     var d = new QAItem(loc, _this, "".concat(k, "-").concat(geoType), itemId, {
+                       issueKey: k,
+                       identifier: {
+                         x: x,
+                         y: y
+                       }
+                     });
+                     d.replacements = {
+                       num_trips: numberOfTrips,
+                       geometry_type: _t("QA.improveOSM.geometry_types.".concat(geoType))
+                     }; // -1 trips indicates data came from a 3rd party
 
-             var head = document.getElementsByTagName("head")[0];
-             head.insertBefore(base, head.firstChild);
-             var a = document.createElement("a");
-             var resolved;
+                     if (numberOfTrips === -1) {
+                       d.desc = _t('QA.improveOSM.error_types.mr.description_alt', d.replacements);
+                     }
 
-             for (var index = 1; index < numUrls; index++) {
-               a.href = arguments[index];
-               resolved = a.href;
-               base.href = resolved;
-             }
+                     _cache$1.data[d.id] = d;
 
-             head.removeChild(base);
-             return resolved;
-           }
+                     _cache$1.rtree.insert(encodeIssueRtree$1(d));
+                   });
+                 } // Entities at high zoom == turn restrictions
 
-           return resolveUrl;
-         });
-       });
 
-       var assign = make_assign();
-       var create$1 = make_create();
-       var trim$3 = make_trim();
-       var Global = typeof window !== 'undefined' ? window : commonjsGlobal;
-       var util = {
-         assign: assign,
-         create: create$1,
-         trim: trim$3,
-         bind: bind$1,
-         slice: slice$2,
-         each: each,
-         map: map$1,
-         pluck: pluck,
-         isList: isList,
-         isFunction: isFunction,
-         isObject: isObject$2,
-         Global: Global
-       };
+                 if (data.entities) {
+                   data.entities.forEach(function (feature) {
+                     var point = feature.point,
+                         id = feature.id,
+                         segments = feature.segments,
+                         numberOfPasses = feature.numberOfPasses,
+                         turnType = feature.turnType;
+                     var itemId = "".concat(id.replace(/[,:+#]/g, '_')); // Turn restrictions could be missing at same junction
+                     // We also want to bump the error up so node is accessible
 
-       function make_assign() {
-         if (Object.assign) {
-           return Object.assign;
-         } else {
-           return function shimAssign(obj, props1, props2, etc) {
-             for (var i = 1; i < arguments.length; i++) {
-               each(Object(arguments[i]), function (val, key) {
-                 obj[key] = val;
-               });
-             }
+                     var loc = preventCoincident$1([point.lon, point.lat], true); // Elements are presented in a strange way
 
-             return obj;
-           };
-         }
-       }
+                     var ids = id.split(',');
+                     var from_way = ids[0];
+                     var via_node = ids[3];
+                     var to_way = ids[2].split(':')[1];
+                     var d = new QAItem(loc, _this, k, itemId, {
+                       issueKey: k,
+                       identifier: id,
+                       objectId: via_node,
+                       objectType: 'node'
+                     }); // Travel direction along from_way clarifies the turn restriction
 
-       function make_create() {
-         if (Object.create) {
-           return function create(obj, assignProps1, assignProps2, etc) {
-             var assignArgsList = slice$2(arguments, 1);
-             return assign.apply(this, [Object.create(obj)].concat(assignArgsList));
-           };
-         } else {
-           var F = function F() {}; // eslint-disable-line no-inner-declarations
+                     var _segments$0$points = _slicedToArray(segments[0].points, 2),
+                         p1 = _segments$0$points[0],
+                         p2 = _segments$0$points[1];
 
+                     var dir_of_travel = cardinalDirection(relativeBearing(p1, p2)); // Variables used in the description
 
-           return function create(obj, assignProps1, assignProps2, etc) {
-             var assignArgsList = slice$2(arguments, 1);
-             F.prototype = obj;
-             return assign.apply(this, [new F()].concat(assignArgsList));
-           };
-         }
-       }
+                     d.replacements = {
+                       num_passed: numberOfPasses,
+                       num_trips: segments[0].numberOfTrips,
+                       turn_restriction: turnType.toLowerCase(),
+                       from_way: linkEntity('w' + from_way),
+                       to_way: linkEntity('w' + to_way),
+                       travel_direction: dir_of_travel,
+                       junction: linkErrorObject(_t('QA.keepRight.error_parts.this_node'))
+                     };
+                     _cache$1.data[d.id] = d;
 
-       function make_trim() {
-         if (String.prototype.trim) {
-           return function trim(str) {
-             return String.prototype.trim.call(str);
-           };
-         } else {
-           return function trim(str) {
-             return str.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, '');
-           };
-         }
-       }
+                     _cache$1.rtree.insert(encodeIssueRtree$1(d));
 
-       function bind$1(obj, fn) {
-         return function () {
-           return fn.apply(obj, Array.prototype.slice.call(arguments, 0));
-         };
-       }
+                     dispatch$6.call('loaded');
+                   });
+                 }
+               })["catch"](function () {
+                 delete _cache$1.inflightTile[tile.id][k];
 
-       function slice$2(arr, index) {
-         return Array.prototype.slice.call(arr, index || 0);
-       }
+                 if (!Object.keys(_cache$1.inflightTile[tile.id]).length) {
+                   delete _cache$1.inflightTile[tile.id];
+                   _cache$1.loadedTile[tile.id] = true;
+                 }
+               });
+             });
+             _cache$1.inflightTile[tile.id] = requests;
+           });
+         },
+         getComments: function getComments(item) {
+           var _this2 = this;
 
-       function each(obj, fn) {
-         pluck(obj, function (val, key) {
-           fn(val, key);
-           return false;
-         });
-       }
+           // If comments already retrieved no need to do so again
+           if (item.comments) {
+             return Promise.resolve(item);
+           }
 
-       function map$1(obj, fn) {
-         var res = isList(obj) ? [] : {};
-         pluck(obj, function (v, k) {
-           res[k] = fn(v, k);
-           return false;
-         });
-         return res;
-       }
+           var key = item.issueKey;
+           var qParams = {};
 
-       function pluck(obj, fn) {
-         if (isList(obj)) {
-           for (var i = 0; i < obj.length; i++) {
-             if (fn(obj[i], i)) {
-               return obj[i];
-             }
-           }
-         } else {
-           for (var key in obj) {
-             if (obj.hasOwnProperty(key)) {
-               if (fn(obj[key], key)) {
-                 return obj[key];
-               }
-             }
+           if (key === 'ow') {
+             qParams = item.identifier;
+           } else if (key === 'mr') {
+             qParams.tileX = item.identifier.x;
+             qParams.tileY = item.identifier.y;
+           } else if (key === 'tr') {
+             qParams.targetId = item.identifier;
            }
-         }
-       }
-
-       function isList(val) {
-         return val != null && typeof val != 'function' && typeof val.length == 'number';
-       }
 
-       function isFunction(val) {
-         return val && {}.toString.call(val) === '[object Function]';
-       }
+           var url = "".concat(_impOsmUrls[key], "/retrieveComments?") + utilQsString(qParams);
 
-       function isObject$2(val) {
-         return val && {}.toString.call(val) === '[object Object]';
-       }
+           var cacheComments = function cacheComments(data) {
+             // Assign directly for immediate use afterwards
+             // comments are served newest to oldest
+             item.comments = data.comments ? data.comments.reverse() : [];
 
-       var slice$3 = util.slice;
-       var pluck$1 = util.pluck;
-       var each$1 = util.each;
-       var bind$2 = util.bind;
-       var create$2 = util.create;
-       var isList$1 = util.isList;
-       var isFunction$1 = util.isFunction;
-       var isObject$3 = util.isObject;
-       var storeEngine = {
-         createStore: _createStore
-       };
-       var storeAPI = {
-         version: '2.0.12',
-         enabled: false,
-         // get returns the value of the given key. If that value
-         // is undefined, it returns optionalDefaultValue instead.
-         get: function get(key, optionalDefaultValue) {
-           var data = this.storage.read(this._namespacePrefix + key);
-           return this._deserialize(data, optionalDefaultValue);
-         },
-         // set will store the given value at key and returns value.
-         // Calling set with value === undefined is equivalent to calling remove.
-         set: function set(key, value) {
-           if (value === undefined) {
-             return this.remove(key);
-           }
+             _this2.replaceItem(item);
+           };
 
-           this.storage.write(this._namespacePrefix + key, this._serialize(value));
-           return value;
-         },
-         // remove deletes the key and value stored at the given key.
-         remove: function remove(key) {
-           this.storage.remove(this._namespacePrefix + key);
-         },
-         // each will call the given callback once for each key-value pair
-         // in this store.
-         each: function each(callback) {
-           var self = this;
-           this.storage.each(function (val, namespacedKey) {
-             callback.call(self, self._deserialize(val), (namespacedKey || '').replace(self._namespaceRegexp, ''));
+           return d3_json(url).then(cacheComments).then(function () {
+             return item;
            });
          },
-         // clearAll will remove all the stored key-value pairs in this store.
-         clearAll: function clearAll() {
-           this.storage.clearAll();
-         },
-         // additional functionality that can't live in plugins
-         // ---------------------------------------------------
-         // hasNamespace returns true if this store instance has the given namespace.
-         hasNamespace: function hasNamespace(namespace) {
-           return this._namespacePrefix == '__storejs_' + namespace + '_';
-         },
-         // createStore creates a store.js instance with the first
-         // functioning storage in the list of storage candidates,
-         // and applies the the given mixins to the instance.
-         createStore: function createStore() {
-           return _createStore.apply(this, arguments);
-         },
-         addPlugin: function addPlugin(plugin) {
-           this._addPlugin(plugin);
-         },
-         namespace: function namespace(_namespace) {
-           return _createStore(this.storage, this.plugins, _namespace);
-         }
-       };
+         postUpdate: function postUpdate(d, callback) {
+           if (!serviceOsm.authenticated()) {
+             // Username required in payload
+             return callback({
+               message: 'Not Authenticated',
+               status: -3
+             }, d);
+           }
 
-       function _warn() {
-         var _console = typeof console == 'undefined' ? null : console;
+           if (_cache$1.inflightPost[d.id]) {
+             return callback({
+               message: 'Error update already inflight',
+               status: -2
+             }, d);
+           } // Payload can only be sent once username is established
 
-         if (!_console) {
-           return;
-         }
 
-         var fn = _console.warn ? _console.warn : _console.log;
-         fn.apply(_console, arguments);
-       }
+           serviceOsm.userDetails(sendPayload.bind(this));
 
-       function _createStore(storages, plugins, namespace) {
-         if (!namespace) {
-           namespace = '';
-         }
+           function sendPayload(err, user) {
+             var _this3 = this;
 
-         if (storages && !isList$1(storages)) {
-           storages = [storages];
-         }
+             if (err) {
+               return callback(err, d);
+             }
 
-         if (plugins && !isList$1(plugins)) {
-           plugins = [plugins];
-         }
+             var key = d.issueKey;
+             var url = "".concat(_impOsmUrls[key], "/comment");
+             var payload = {
+               username: user.display_name,
+               targetIds: [d.identifier]
+             };
 
-         var namespacePrefix = namespace ? '__storejs_' + namespace + '_' : '';
-         var namespaceRegexp = namespace ? new RegExp('^' + namespacePrefix) : null;
-         var legalNamespaces = /^[a-zA-Z0-9_\-]*$/; // alpha-numeric + underscore and dash
+             if (d.newStatus) {
+               payload.status = d.newStatus;
+               payload.text = 'status changed';
+             } // Comment take place of default text
 
-         if (!legalNamespaces.test(namespace)) {
-           throw new Error('store.js namespaces can only have alphanumerics + underscores and dashes');
-         }
 
-         var _privateStoreProps = {
-           _namespacePrefix: namespacePrefix,
-           _namespaceRegexp: namespaceRegexp,
-           _testStorage: function _testStorage(storage) {
-             try {
-               var testStr = '__storejs__test__';
-               storage.write(testStr, testStr);
-               var ok = storage.read(testStr) === testStr;
-               storage.remove(testStr);
-               return ok;
-             } catch (e) {
-               return false;
+             if (d.newComment) {
+               payload.text = d.newComment;
              }
-           },
-           _assignPluginFnProp: function _assignPluginFnProp(pluginFnProp, propName) {
-             var oldFn = this[propName];
-
-             this[propName] = function pluginFn() {
-               var args = slice$3(arguments, 0);
-               var self = this; // super_fn calls the old function which was overwritten by
-               // this mixin.
 
-               function super_fn() {
-                 if (!oldFn) {
-                   return;
-                 }
+             var controller = new AbortController();
+             _cache$1.inflightPost[d.id] = controller;
+             var options = {
+               method: 'POST',
+               signal: controller.signal,
+               body: JSON.stringify(payload)
+             };
+             d3_json(url, options).then(function () {
+               delete _cache$1.inflightPost[d.id]; // Just a comment, update error in cache
 
-                 each$1(arguments, function (arg, i) {
-                   args[i] = arg;
+               if (!d.newStatus) {
+                 var now = new Date();
+                 var comments = d.comments ? d.comments : [];
+                 comments.push({
+                   username: payload.username,
+                   text: payload.text,
+                   timestamp: now.getTime() / 1000
                  });
-                 return oldFn.apply(self, args);
-               } // Give mixing function access to super_fn by prefixing all mixin function
-               // arguments with super_fn.
 
+                 _this3.replaceItem(d.update({
+                   comments: comments,
+                   newComment: undefined
+                 }));
+               } else {
+                 _this3.removeItem(d);
 
-               var newFnArgs = [super_fn].concat(args);
-               return pluginFnProp.apply(self, newFnArgs);
-             };
-           },
-           _serialize: function _serialize(obj) {
-             return JSON.stringify(obj);
-           },
-           _deserialize: function _deserialize(strVal, defaultVal) {
-             if (!strVal) {
-               return defaultVal;
-             } // It is possible that a raw string value has been previously stored
-             // in a storage without using store.js, meaning it will be a raw
-             // string value instead of a JSON serialized string. By defaulting
-             // to the raw string value in case of a JSON parse error, we allow
-             // for past stored values to be forwards-compatible with store.js
+                 if (d.newStatus === 'SOLVED') {
+                   // Keep track of the number of issues closed per type to tag the changeset
+                   if (!(d.issueKey in _cache$1.closed)) {
+                     _cache$1.closed[d.issueKey] = 0;
+                   }
 
+                   _cache$1.closed[d.issueKey] += 1;
+                 }
+               }
 
-             var val = '';
+               if (callback) callback(null, d);
+             })["catch"](function (err) {
+               delete _cache$1.inflightPost[d.id];
+               if (callback) callback(err.message);
+             });
+           }
+         },
+         // Get all cached QAItems covering the viewport
+         getItems: function getItems(projection) {
+           var viewport = projection.clipExtent();
+           var min = [viewport[0][0], viewport[1][1]];
+           var max = [viewport[1][0], viewport[0][1]];
+           var bbox = geoExtent(projection.invert(min), projection.invert(max)).bbox();
+           return _cache$1.rtree.search(bbox).map(function (d) {
+             return d.data;
+           });
+         },
+         // Get a QAItem from cache
+         // NOTE: Don't change method name until UI v3 is merged
+         getError: function getError(id) {
+           return _cache$1.data[id];
+         },
+         // get the name of the icon to display for this item
+         getIcon: function getIcon(itemType) {
+           return _impOsmData.icons[itemType];
+         },
+         // Replace a single QAItem in the cache
+         replaceItem: function replaceItem(issue) {
+           if (!(issue instanceof QAItem) || !issue.id) return;
+           _cache$1.data[issue.id] = issue;
+           updateRtree$2(encodeIssueRtree$1(issue), true); // true = replace
 
-             try {
-               val = JSON.parse(strVal);
-             } catch (e) {
-               val = strVal;
-             }
+           return issue;
+         },
+         // Remove a single QAItem from the cache
+         removeItem: function removeItem(issue) {
+           if (!(issue instanceof QAItem) || !issue.id) return;
+           delete _cache$1.data[issue.id];
+           updateRtree$2(encodeIssueRtree$1(issue), false); // false = remove
+         },
+         // Used to populate `closed:improveosm:*` changeset tags
+         getClosedCounts: function getClosedCounts() {
+           return _cache$1.closed;
+         }
+       };
 
-             return val !== undefined ? val : defaultVal;
-           },
-           _addStorage: function _addStorage(storage) {
-             if (this.enabled) {
-               return;
-             }
+       var defaults$5 = createCommonjsModule(function (module) {
+         function getDefaults() {
+           return {
+             baseUrl: null,
+             breaks: false,
+             gfm: true,
+             headerIds: true,
+             headerPrefix: '',
+             highlight: null,
+             langPrefix: 'language-',
+             mangle: true,
+             pedantic: false,
+             renderer: null,
+             sanitize: false,
+             sanitizer: null,
+             silent: false,
+             smartLists: false,
+             smartypants: false,
+             tokenizer: null,
+             walkTokens: null,
+             xhtml: false
+           };
+         }
 
-             if (this._testStorage(storage)) {
-               this.storage = storage;
-               this.enabled = true;
-             }
-           },
-           _addPlugin: function _addPlugin(plugin) {
-             var self = this; // If the plugin is an array, then add all plugins in the array.
-             // This allows for a plugin to depend on other plugins.
+         function changeDefaults(newDefaults) {
+           module.exports.defaults = newDefaults;
+         }
 
-             if (isList$1(plugin)) {
-               each$1(plugin, function (plugin) {
-                 self._addPlugin(plugin);
-               });
-               return;
-             } // Keep track of all plugins we've seen so far, so that we
-             // don't add any of them twice.
+         module.exports = {
+           defaults: getDefaults(),
+           getDefaults: getDefaults,
+           changeDefaults: changeDefaults
+         };
+       });
 
+       /**
+        * Helpers
+        */
+       var escapeTest = /[&<>"']/;
+       var escapeReplace = /[&<>"']/g;
+       var escapeTestNoEncode = /[<>"']|&(?!#?\w+;)/;
+       var escapeReplaceNoEncode = /[<>"']|&(?!#?\w+;)/g;
+       var escapeReplacements = {
+         '&': '&amp;',
+         '<': '&lt;',
+         '>': '&gt;',
+         '"': '&quot;',
+         "'": '&#39;'
+       };
 
-             var seenPlugin = pluck$1(this.plugins, function (seenPlugin) {
-               return plugin === seenPlugin;
-             });
+       var getEscapeReplacement = function getEscapeReplacement(ch) {
+         return escapeReplacements[ch];
+       };
 
-             if (seenPlugin) {
-               return;
-             }
+       function escape$3(html, encode) {
+         if (encode) {
+           if (escapeTest.test(html)) {
+             return html.replace(escapeReplace, getEscapeReplacement);
+           }
+         } else {
+           if (escapeTestNoEncode.test(html)) {
+             return html.replace(escapeReplaceNoEncode, getEscapeReplacement);
+           }
+         }
 
-             this.plugins.push(plugin); // Check that the plugin is properly formed
+         return html;
+       }
 
-             if (!isFunction$1(plugin)) {
-               throw new Error('Plugins must be function values that return objects');
-             }
+       var unescapeTest = /&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/ig;
 
-             var pluginProperties = plugin.call(this);
+       function unescape$2(html) {
+         // explicitly match decimal, hex, and named HTML entities
+         return html.replace(unescapeTest, function (_, n) {
+           n = n.toLowerCase();
+           if (n === 'colon') return ':';
 
-             if (!isObject$3(pluginProperties)) {
-               throw new Error('Plugins must return an object of function properties');
-             } // Add the plugin function properties to this store instance.
+           if (n.charAt(0) === '#') {
+             return n.charAt(1) === 'x' ? String.fromCharCode(parseInt(n.substring(2), 16)) : String.fromCharCode(+n.substring(1));
+           }
 
+           return '';
+         });
+       }
 
-             each$1(pluginProperties, function (pluginFnProp, propName) {
-               if (!isFunction$1(pluginFnProp)) {
-                 throw new Error('Bad plugin property: ' + propName + ' from plugin ' + plugin.name + '. Plugins should only return functions.');
-               }
+       var caret = /(^|[^\[])\^/g;
 
-               self._assignPluginFnProp(pluginFnProp, propName);
-             });
+       function edit$1(regex, opt) {
+         regex = regex.source || regex;
+         opt = opt || '';
+         var obj = {
+           replace: function replace(name, val) {
+             val = val.source || val;
+             val = val.replace(caret, '$1');
+             regex = regex.replace(name, val);
+             return obj;
            },
-           // Put deprecated properties in the private API, so as to not expose it to accidential
-           // discovery through inspection of the store object.
-           // Deprecated: addStorage
-           addStorage: function addStorage(storage) {
-             _warn('store.addStorage(storage) is deprecated. Use createStore([storages])');
-
-             this._addStorage(storage);
+           getRegex: function getRegex() {
+             return new RegExp(regex, opt);
            }
          };
-         var store = create$2(_privateStoreProps, storeAPI, {
-           plugins: []
-         });
-         store.raw = {};
-         each$1(store, function (prop, propName) {
-           if (isFunction$1(prop)) {
-             store.raw[propName] = bind$2(store, prop);
-           }
-         });
-         each$1(storages, function (storage) {
-           store._addStorage(storage);
-         });
-         each$1(plugins, function (plugin) {
-           store._addPlugin(plugin);
-         });
-         return store;
+         return obj;
        }
 
-       var Global$1 = util.Global;
-       var localStorage_1 = {
-         name: 'localStorage',
-         read: read,
-         write: write,
-         each: each$2,
-         remove: remove$2,
-         clearAll: clearAll
-       };
+       var nonWordAndColonTest = /[^\w:]/g;
+       var originIndependentUrl = /^$|^[a-z][a-z0-9+.-]*:|^[?#]/i;
 
-       function localStorage$1() {
-         return Global$1.localStorage;
-       }
+       function cleanUrl$1(sanitize, base, href) {
+         if (sanitize) {
+           var prot;
 
-       function read(key) {
-         return localStorage$1().getItem(key);
-       }
+           try {
+             prot = decodeURIComponent(unescape$2(href)).replace(nonWordAndColonTest, '').toLowerCase();
+           } catch (e) {
+             return null;
+           }
 
-       function write(key, data) {
-         return localStorage$1().setItem(key, data);
-       }
+           if (prot.indexOf('javascript:') === 0 || prot.indexOf('vbscript:') === 0 || prot.indexOf('data:') === 0) {
+             return null;
+           }
+         }
 
-       function each$2(fn) {
-         for (var i = localStorage$1().length - 1; i >= 0; i--) {
-           var key = localStorage$1().key(i);
-           fn(read(key), key);
+         if (base && !originIndependentUrl.test(href)) {
+           href = resolveUrl$1(base, href);
          }
-       }
 
-       function remove$2(key) {
-         return localStorage$1().removeItem(key);
-       }
+         try {
+           href = encodeURI(href).replace(/%25/g, '%');
+         } catch (e) {
+           return null;
+         }
 
-       function clearAll() {
-         return localStorage$1().clear();
+         return href;
        }
 
-       // versions 6 and 7, where no localStorage, etc
-       // is available.
-
-       var Global$2 = util.Global;
-       var oldFFGlobalStorage = {
-         name: 'oldFF-globalStorage',
-         read: read$1,
-         write: write$1,
-         each: each$3,
-         remove: remove$3,
-         clearAll: clearAll$1
-       };
-       var globalStorage = Global$2.globalStorage;
+       var baseUrls = {};
+       var justDomain = /^[^:]+:\/*[^/]*$/;
+       var protocol = /^([^:]+:)[\s\S]*$/;
+       var domain = /^([^:]+:\/*[^/]*)[\s\S]*$/;
 
-       function read$1(key) {
-         return globalStorage[key];
-       }
+       function resolveUrl$1(base, href) {
+         if (!baseUrls[' ' + base]) {
+           // we can ignore everything in base after the last slash of its path component,
+           // but we might need to add _that_
+           // https://tools.ietf.org/html/rfc3986#section-3
+           if (justDomain.test(base)) {
+             baseUrls[' ' + base] = base + '/';
+           } else {
+             baseUrls[' ' + base] = rtrim$1(base, '/', true);
+           }
+         }
 
-       function write$1(key, data) {
-         globalStorage[key] = data;
-       }
+         base = baseUrls[' ' + base];
+         var relativeBase = base.indexOf(':') === -1;
 
-       function each$3(fn) {
-         for (var i = globalStorage.length - 1; i >= 0; i--) {
-           var key = globalStorage.key(i);
-           fn(globalStorage[key], key);
-         }
-       }
+         if (href.substring(0, 2) === '//') {
+           if (relativeBase) {
+             return href;
+           }
 
-       function remove$3(key) {
-         return globalStorage.removeItem(key);
-       }
+           return base.replace(protocol, '$1') + href;
+         } else if (href.charAt(0) === '/') {
+           if (relativeBase) {
+             return href;
+           }
 
-       function clearAll$1() {
-         each$3(function (key, _) {
-           delete globalStorage[key];
-         });
+           return base.replace(domain, '$1') + href;
+         } else {
+           return base + href;
+         }
        }
 
-       // versions 6 and 7, where no localStorage, sessionStorage, etc
-       // is available.
-
-       var Global$3 = util.Global;
-       var oldIEUserDataStorage = {
-         name: 'oldIE-userDataStorage',
-         write: write$2,
-         read: read$2,
-         each: each$4,
-         remove: remove$4,
-         clearAll: clearAll$2
+       var noopTest$1 = {
+         exec: function noopTest() {}
        };
-       var storageName = 'storejs';
-       var doc = Global$3.document;
 
-       var _withStorageEl = _makeIEStorageElFunction();
+       function merge$2(obj) {
+         var i = 1,
+             target,
+             key;
 
-       var disable = (Global$3.navigator ? Global$3.navigator.userAgent : '').match(/ (MSIE 8|MSIE 9|MSIE 10)\./); // MSIE 9.x, MSIE 10.x
+         for (; i < arguments.length; i++) {
+           target = arguments[i];
 
-       function write$2(unfixedKey, data) {
-         if (disable) {
-           return;
+           for (key in target) {
+             if (Object.prototype.hasOwnProperty.call(target, key)) {
+               obj[key] = target[key];
+             }
+           }
          }
 
-         var fixedKey = fixKey(unfixedKey);
-
-         _withStorageEl(function (storageEl) {
-           storageEl.setAttribute(fixedKey, data);
-           storageEl.save(storageName);
-         });
+         return obj;
        }
 
-       function read$2(unfixedKey) {
-         if (disable) {
-           return;
-         }
+       function splitCells$1(tableRow, count) {
+         // ensure that every cell-delimiting pipe has a space
+         // before it to distinguish it from an escaped pipe
+         var row = tableRow.replace(/\|/g, function (match, offset, str) {
+           var escaped = false,
+               curr = offset;
 
-         var fixedKey = fixKey(unfixedKey);
-         var res = null;
+           while (--curr >= 0 && str[curr] === '\\') {
+             escaped = !escaped;
+           }
 
-         _withStorageEl(function (storageEl) {
-           res = storageEl.getAttribute(fixedKey);
-         });
+           if (escaped) {
+             // odd number of slashes means | is escaped
+             // so we leave it alone
+             return '|';
+           } else {
+             // add space before unescaped |
+             return ' |';
+           }
+         }),
+             cells = row.split(/ \|/);
+         var i = 0;
 
-         return res;
-       }
+         if (cells.length > count) {
+           cells.splice(count);
+         } else {
+           while (cells.length < count) {
+             cells.push('');
+           }
+         }
 
-       function each$4(callback) {
-         _withStorageEl(function (storageEl) {
-           var attributes = storageEl.XMLDocument.documentElement.attributes;
+         for (; i < cells.length; i++) {
+           // leading or trailing whitespace is ignored per the gfm spec
+           cells[i] = cells[i].trim().replace(/\\\|/g, '|');
+         }
 
-           for (var i = attributes.length - 1; i >= 0; i--) {
-             var attr = attributes[i];
-             callback(storageEl.getAttribute(attr.name), attr.name);
-           }
-         });
-       }
+         return cells;
+       } // Remove trailing 'c's. Equivalent to str.replace(/c*$/, '').
+       // /c*$/ is vulnerable to REDOS.
+       // invert: Remove suffix of non-c chars instead. Default falsey.
 
-       function remove$4(unfixedKey) {
-         var fixedKey = fixKey(unfixedKey);
 
-         _withStorageEl(function (storageEl) {
-           storageEl.removeAttribute(fixedKey);
-           storageEl.save(storageName);
-         });
-       }
+       function rtrim$1(str, c, invert) {
+         var l = str.length;
 
-       function clearAll$2() {
-         _withStorageEl(function (storageEl) {
-           var attributes = storageEl.XMLDocument.documentElement.attributes;
-           storageEl.load(storageName);
+         if (l === 0) {
+           return '';
+         } // Length of suffix matching the invert condition.
 
-           for (var i = attributes.length - 1; i >= 0; i--) {
-             storageEl.removeAttribute(attributes[i].name);
-           }
 
-           storageEl.save(storageName);
-         });
-       } // Helpers
-       //////////
-       // In IE7, keys cannot start with a digit or contain certain chars.
-       // See https://github.com/marcuswestin/store.js/issues/40
-       // See https://github.com/marcuswestin/store.js/issues/83
+         var suffLen = 0; // Step left until we fail to match the invert condition.
 
+         while (suffLen < l) {
+           var currChar = str.charAt(l - suffLen - 1);
 
-       var forbiddenCharsRegex = new RegExp("[!\"#$%&'()*+,/\\\\:;<=>?@[\\]^`{|}~]", "g");
+           if (currChar === c && !invert) {
+             suffLen++;
+           } else if (currChar !== c && invert) {
+             suffLen++;
+           } else {
+             break;
+           }
+         }
 
-       function fixKey(key) {
-         return key.replace(/^\d/, '___$&').replace(forbiddenCharsRegex, '___');
+         return str.substr(0, l - suffLen);
        }
 
-       function _makeIEStorageElFunction() {
-         if (!doc || !doc.documentElement || !doc.documentElement.addBehavior) {
-           return null;
+       function findClosingBracket$1(str, b) {
+         if (str.indexOf(b[1]) === -1) {
+           return -1;
          }
 
-         var scriptTag = 'script',
-             storageOwner,
-             storageContainer,
-             storageEl; // Since #userData storage applies only to specific paths, we need to
-         // somehow link our data to a specific path.  We choose /favicon.ico
-         // as a pretty safe option, since all browsers already make a request to
-         // this URL anyway and being a 404 will not hurt us here.  We wrap an
-         // iframe pointing to the favicon in an ActiveXObject(htmlfile) object
-         // (see: http://msdn.microsoft.com/en-us/library/aa752574(v=VS.85).aspx)
-         // since the iframe access rules appear to allow direct access and
-         // manipulation of the document element, even for a 404 page.  This
-         // document can be used instead of the current document (which would
-         // have been limited to the current path) to perform #userData storage.
+         var l = str.length;
+         var level = 0,
+             i = 0;
 
-         try {
-           /* global ActiveXObject */
-           storageContainer = new ActiveXObject('htmlfile');
-           storageContainer.open();
-           storageContainer.write('<' + scriptTag + '>document.w=window</' + scriptTag + '><iframe src="/favicon.ico"></iframe>');
-           storageContainer.close();
-           storageOwner = storageContainer.w.frames[0].document;
-           storageEl = storageOwner.createElement('div');
-         } catch (e) {
-           // somehow ActiveXObject instantiation failed (perhaps some special
-           // security settings or otherwse), fall back to per-path storage
-           storageEl = doc.createElement('div');
-           storageOwner = doc.body;
-         }
+         for (; i < l; i++) {
+           if (str[i] === '\\') {
+             i++;
+           } else if (str[i] === b[0]) {
+             level++;
+           } else if (str[i] === b[1]) {
+             level--;
 
-         return function (storeFunction) {
-           var args = [].slice.call(arguments, 0);
-           args.unshift(storageEl); // See http://msdn.microsoft.com/en-us/library/ms531081(v=VS.85).aspx
-           // and http://msdn.microsoft.com/en-us/library/ms531424(v=VS.85).aspx
+             if (level < 0) {
+               return i;
+             }
+           }
+         }
 
-           storageOwner.appendChild(storageEl);
-           storageEl.addBehavior('#default#userData');
-           storageEl.load(storageName);
-           storeFunction.apply(this, args);
-           storageOwner.removeChild(storageEl);
-           return;
-         };
+         return -1;
        }
 
-       // doesn't work but cookies do. This implementation is adopted from
-       // https://developer.mozilla.org/en-US/docs/Web/API/Storage/LocalStorage
+       function checkSanitizeDeprecation$1(opt) {
+         if (opt && opt.sanitize && !opt.silent) {
+           console.warn('marked(): sanitize and sanitizer parameters are deprecated since version 0.7.0, should not be used and will be removed in the future. Read more here: https://marked.js.org/#/USING_ADVANCED.md#options');
+         }
+       } // copied from https://stackoverflow.com/a/5450113/806777
 
-       var Global$4 = util.Global;
-       var trim$4 = util.trim;
-       var cookieStorage = {
-         name: 'cookieStorage',
-         read: read$3,
-         write: write$3,
-         each: each$5,
-         remove: remove$5,
-         clearAll: clearAll$3
-       };
-       var doc$1 = Global$4.document;
 
-       function read$3(key) {
-         if (!key || !_has(key)) {
-           return null;
+       function repeatString$1(pattern, count) {
+         if (count < 1) {
+           return '';
          }
 
-         var regexpStr = "(?:^|.*;\\s*)" + escape(key).replace(/[\-\.\+\*]/g, "\\$&") + "\\s*\\=\\s*((?:[^;](?!;))*[^;]?).*";
-         return unescape(doc$1.cookie.replace(new RegExp(regexpStr), "$1"));
-       }
-
-       function each$5(callback) {
-         var cookies = doc$1.cookie.split(/; ?/g);
+         var result = '';
 
-         for (var i = cookies.length - 1; i >= 0; i--) {
-           if (!trim$4(cookies[i])) {
-             continue;
+         while (count > 1) {
+           if (count & 1) {
+             result += pattern;
            }
 
-           var kvp = cookies[i].split('=');
-           var key = unescape(kvp[0]);
-           var val = unescape(kvp[1]);
-           callback(val, key);
+           count >>= 1;
+           pattern += pattern;
          }
+
+         return result + pattern;
        }
 
-       function write$3(key, data) {
-         if (!key) {
-           return;
-         }
+       var helpers = {
+         escape: escape$3,
+         unescape: unescape$2,
+         edit: edit$1,
+         cleanUrl: cleanUrl$1,
+         resolveUrl: resolveUrl$1,
+         noopTest: noopTest$1,
+         merge: merge$2,
+         splitCells: splitCells$1,
+         rtrim: rtrim$1,
+         findClosingBracket: findClosingBracket$1,
+         checkSanitizeDeprecation: checkSanitizeDeprecation$1,
+         repeatString: repeatString$1
+       };
+
+       var defaults$4 = defaults$5.defaults;
+       var rtrim = helpers.rtrim,
+           splitCells = helpers.splitCells,
+           _escape = helpers.escape,
+           findClosingBracket = helpers.findClosingBracket;
 
-         doc$1.cookie = escape(key) + "=" + escape(data) + "; expires=Tue, 19 Jan 2038 03:14:07 GMT; path=/";
+       function outputLink(cap, link, raw) {
+         var href = link.href;
+         var title = link.title ? _escape(link.title) : null;
+         var text = cap[1].replace(/\\([\[\]])/g, '$1');
+
+         if (cap[0].charAt(0) !== '!') {
+           return {
+             type: 'link',
+             raw: raw,
+             href: href,
+             title: title,
+             text: text
+           };
+         } else {
+           return {
+             type: 'image',
+             raw: raw,
+             href: href,
+             title: title,
+             text: _escape(text)
+           };
+         }
        }
 
-       function remove$5(key) {
-         if (!key || !_has(key)) {
-           return;
+       function indentCodeCompensation(raw, text) {
+         var matchIndentToCode = raw.match(/^(\s+)(?:```)/);
+
+         if (matchIndentToCode === null) {
+           return text;
          }
 
-         doc$1.cookie = escape(key) + "=; expires=Thu, 01 Jan 1970 00:00:00 GMT; path=/";
-       }
+         var indentToCode = matchIndentToCode[1];
+         return text.split('\n').map(function (node) {
+           var matchIndentInNode = node.match(/^\s+/);
 
-       function clearAll$3() {
-         each$5(function (_, key) {
-           remove$5(key);
-         });
-       }
+           if (matchIndentInNode === null) {
+             return node;
+           }
 
-       function _has(key) {
-         return new RegExp("(?:^|;\\s*)" + escape(key).replace(/[\-\.\+\*]/g, "\\$&") + "\\s*\\=").test(doc$1.cookie);
-       }
+           var _matchIndentInNode = _slicedToArray(matchIndentInNode, 1),
+               indentInNode = _matchIndentInNode[0];
 
-       var Global$5 = util.Global;
-       var sessionStorage_1 = {
-         name: 'sessionStorage',
-         read: read$4,
-         write: write$4,
-         each: each$6,
-         remove: remove$6,
-         clearAll: clearAll$4
-       };
+           if (indentInNode.length >= indentToCode.length) {
+             return node.slice(indentToCode.length);
+           }
 
-       function sessionStorage() {
-         return Global$5.sessionStorage;
+           return node;
+         }).join('\n');
        }
+       /**
+        * Tokenizer
+        */
 
-       function read$4(key) {
-         return sessionStorage().getItem(key);
-       }
 
-       function write$4(key, data) {
-         return sessionStorage().setItem(key, data);
-       }
+       var Tokenizer_1 = /*#__PURE__*/function () {
+         function Tokenizer(options) {
+           _classCallCheck$1(this, Tokenizer);
 
-       function each$6(fn) {
-         for (var i = sessionStorage().length - 1; i >= 0; i--) {
-           var key = sessionStorage().key(i);
-           fn(read$4(key), key);
+           this.options = options || defaults$4;
          }
-       }
 
-       function remove$6(key) {
-         return sessionStorage().removeItem(key);
-       }
-
-       function clearAll$4() {
-         return sessionStorage().clear();
-       }
+         _createClass$1(Tokenizer, [{
+           key: "space",
+           value: function space(src) {
+             var cap = this.rules.block.newline.exec(src);
 
-       // memoryStorage is a useful last fallback to ensure that the store
-       // is functions (meaning store.get(), store.set(), etc will all function).
-       // However, stored values will not persist when the browser navigates to
-       // a new page or reloads the current page.
-       var memoryStorage_1 = {
-         name: 'memoryStorage',
-         read: read$5,
-         write: write$5,
-         each: each$7,
-         remove: remove$7,
-         clearAll: clearAll$5
-       };
-       var memoryStorage = {};
+             if (cap) {
+               if (cap[0].length > 1) {
+                 return {
+                   type: 'space',
+                   raw: cap[0]
+                 };
+               }
 
-       function read$5(key) {
-         return memoryStorage[key];
-       }
+               return {
+                 raw: '\n'
+               };
+             }
+           }
+         }, {
+           key: "code",
+           value: function code(src) {
+             var cap = this.rules.block.code.exec(src);
 
-       function write$5(key, data) {
-         memoryStorage[key] = data;
-       }
+             if (cap) {
+               var text = cap[0].replace(/^ {1,4}/gm, '');
+               return {
+                 type: 'code',
+                 raw: cap[0],
+                 codeBlockStyle: 'indented',
+                 text: !this.options.pedantic ? rtrim(text, '\n') : text
+               };
+             }
+           }
+         }, {
+           key: "fences",
+           value: function fences(src) {
+             var cap = this.rules.block.fences.exec(src);
 
-       function each$7(callback) {
-         for (var key in memoryStorage) {
-           if (memoryStorage.hasOwnProperty(key)) {
-             callback(memoryStorage[key], key);
+             if (cap) {
+               var raw = cap[0];
+               var text = indentCodeCompensation(raw, cap[3] || '');
+               return {
+                 type: 'code',
+                 raw: raw,
+                 lang: cap[2] ? cap[2].trim() : cap[2],
+                 text: text
+               };
+             }
            }
-         }
-       }
+         }, {
+           key: "heading",
+           value: function heading(src) {
+             var cap = this.rules.block.heading.exec(src);
 
-       function remove$7(key) {
-         delete memoryStorage[key];
-       }
+             if (cap) {
+               var text = cap[2].trim(); // remove trailing #s
 
-       function clearAll$5(key) {
-         memoryStorage = {};
-       }
+               if (/#$/.test(text)) {
+                 var trimmed = rtrim(text, '#');
 
-       var all = [// Listed in order of usage preference
-       localStorage_1, oldFFGlobalStorage, oldIEUserDataStorage, cookieStorage, sessionStorage_1, memoryStorage_1];
+                 if (this.options.pedantic) {
+                   text = trimmed.trim();
+                 } else if (!trimmed || / $/.test(trimmed)) {
+                   // CommonMark requires space before trailing #s
+                   text = trimmed.trim();
+                 }
+               }
 
-       /* eslint-disable */
-       //  json2.js
-       //  2016-10-28
-       //  Public Domain.
-       //  NO WARRANTY EXPRESSED OR IMPLIED. USE AT YOUR OWN RISK.
-       //  See http://www.JSON.org/js.html
-       //  This code should be minified before deployment.
-       //  See http://javascript.crockford.com/jsmin.html
-       //  USE YOUR OWN COPY. IT IS EXTREMELY UNWISE TO LOAD CODE FROM SERVERS YOU DO
-       //  NOT CONTROL.
-       //  This file creates a global JSON object containing two methods: stringify
-       //  and parse. This file provides the ES5 JSON capability to ES3 systems.
-       //  If a project might run on IE8 or earlier, then this file should be included.
-       //  This file does nothing on ES5 systems.
-       //      JSON.stringify(value, replacer, space)
-       //          value       any JavaScript value, usually an object or array.
-       //          replacer    an optional parameter that determines how object
-       //                      values are stringified for objects. It can be a
-       //                      function or an array of strings.
-       //          space       an optional parameter that specifies the indentation
-       //                      of nested structures. If it is omitted, the text will
-       //                      be packed without extra whitespace. If it is a number,
-       //                      it will specify the number of spaces to indent at each
-       //                      level. If it is a string (such as "\t" or "&nbsp;"),
-       //                      it contains the characters used to indent at each level.
-       //          This method produces a JSON text from a JavaScript value.
-       //          When an object value is found, if the object contains a toJSON
-       //          method, its toJSON method will be called and the result will be
-       //          stringified. A toJSON method does not serialize: it returns the
-       //          value represented by the name/value pair that should be serialized,
-       //          or undefined if nothing should be serialized. The toJSON method
-       //          will be passed the key associated with the value, and this will be
-       //          bound to the value.
-       //          For example, this would serialize Dates as ISO strings.
-       //              Date.prototype.toJSON = function (key) {
-       //                  function f(n) {
-       //                      // Format integers to have at least two digits.
-       //                      return (n < 10)
-       //                          ? "0" + n
-       //                          : n;
-       //                  }
-       //                  return this.getUTCFullYear()   + "-" +
-       //                       f(this.getUTCMonth() + 1) + "-" +
-       //                       f(this.getUTCDate())      + "T" +
-       //                       f(this.getUTCHours())     + ":" +
-       //                       f(this.getUTCMinutes())   + ":" +
-       //                       f(this.getUTCSeconds())   + "Z";
-       //              };
-       //          You can provide an optional replacer method. It will be passed the
-       //          key and value of each member, with this bound to the containing
-       //          object. The value that is returned from your method will be
-       //          serialized. If your method returns undefined, then the member will
-       //          be excluded from the serialization.
-       //          If the replacer parameter is an array of strings, then it will be
-       //          used to select the members to be serialized. It filters the results
-       //          such that only members with keys listed in the replacer array are
-       //          stringified.
-       //          Values that do not have JSON representations, such as undefined or
-       //          functions, will not be serialized. Such values in objects will be
-       //          dropped; in arrays they will be replaced with null. You can use
-       //          a replacer function to replace those with JSON values.
-       //          JSON.stringify(undefined) returns undefined.
-       //          The optional space parameter produces a stringification of the
-       //          value that is filled with line breaks and indentation to make it
-       //          easier to read.
-       //          If the space parameter is a non-empty string, then that string will
-       //          be used for indentation. If the space parameter is a number, then
-       //          the indentation will be that many spaces.
-       //          Example:
-       //          text = JSON.stringify(["e", {pluribus: "unum"}]);
-       //          // text is '["e",{"pluribus":"unum"}]'
-       //          text = JSON.stringify(["e", {pluribus: "unum"}], null, "\t");
-       //          // text is '[\n\t"e",\n\t{\n\t\t"pluribus": "unum"\n\t}\n]'
-       //          text = JSON.stringify([new Date()], function (key, value) {
-       //              return this[key] instanceof Date
-       //                  ? "Date(" + this[key] + ")"
-       //                  : value;
-       //          });
-       //          // text is '["Date(---current time---)"]'
-       //      JSON.parse(text, reviver)
-       //          This method parses a JSON text to produce an object or array.
-       //          It can throw a SyntaxError exception.
-       //          The optional reviver parameter is a function that can filter and
-       //          transform the results. It receives each of the keys and values,
-       //          and its return value is used instead of the original value.
-       //          If it returns what it received, then the structure is not modified.
-       //          If it returns undefined then the member is deleted.
-       //          Example:
-       //          // Parse the text. Values that look like ISO date strings will
-       //          // be converted to Date objects.
-       //          myData = JSON.parse(text, function (key, value) {
-       //              var a;
-       //              if (typeof value === "string") {
-       //                  a =
-       //   /^(\d{4})-(\d{2})-(\d{2})T(\d{2}):(\d{2}):(\d{2}(?:\.\d*)?)Z$/.exec(value);
-       //                  if (a) {
-       //                      return new Date(Date.UTC(+a[1], +a[2] - 1, +a[3], +a[4],
-       //                          +a[5], +a[6]));
-       //                  }
-       //              }
-       //              return value;
-       //          });
-       //          myData = JSON.parse('["Date(09/09/2001)"]', function (key, value) {
-       //              var d;
-       //              if (typeof value === "string" &&
-       //                      value.slice(0, 5) === "Date(" &&
-       //                      value.slice(-1) === ")") {
-       //                  d = new Date(value.slice(5, -1));
-       //                  if (d) {
-       //                      return d;
-       //                  }
-       //              }
-       //              return value;
-       //          });
-       //  This is a reference implementation. You are free to copy, modify, or
-       //  redistribute.
+               return {
+                 type: 'heading',
+                 raw: cap[0],
+                 depth: cap[1].length,
+                 text: text
+               };
+             }
+           }
+         }, {
+           key: "nptable",
+           value: function nptable(src) {
+             var cap = this.rules.block.nptable.exec(src);
 
-       /*jslint
-           eval, for, this
-       */
+             if (cap) {
+               var item = {
+                 type: 'table',
+                 header: splitCells(cap[1].replace(/^ *| *\| *$/g, '')),
+                 align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
+                 cells: cap[3] ? cap[3].replace(/\n$/, '').split('\n') : [],
+                 raw: cap[0]
+               };
 
-       /*property
-           JSON, apply, call, charCodeAt, getUTCDate, getUTCFullYear, getUTCHours,
-           getUTCMinutes, getUTCMonth, getUTCSeconds, hasOwnProperty, join,
-           lastIndex, length, parse, prototype, push, replace, slice, stringify,
-           test, toJSON, toString, valueOf
-       */
-       // Create a JSON object only if one does not already exist. We create the
-       // methods in a closure to avoid creating global variables.
-       if ((typeof JSON === "undefined" ? "undefined" : _typeof(JSON)) !== "object") {
-         JSON = {};
-       }
+               if (item.header.length === item.align.length) {
+                 var l = item.align.length;
+                 var i;
 
-       (function () {
+                 for (i = 0; i < l; i++) {
+                   if (/^ *-+: *$/.test(item.align[i])) {
+                     item.align[i] = 'right';
+                   } else if (/^ *:-+: *$/.test(item.align[i])) {
+                     item.align[i] = 'center';
+                   } else if (/^ *:-+ *$/.test(item.align[i])) {
+                     item.align[i] = 'left';
+                   } else {
+                     item.align[i] = null;
+                   }
+                 }
 
-         var rx_one = /^[\],:{}\s]*$/;
-         var rx_two = /\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g;
-         var rx_three = /"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g;
-         var rx_four = /(?:^|:|,)(?:\s*\[)+/g;
-         var rx_escapable = /[\\"\u0000-\u001f\u007f-\u009f\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g;
-         var rx_dangerous = /[\u0000\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g;
+                 l = item.cells.length;
 
-         function f(n) {
-           // Format integers to have at least two digits.
-           return n < 10 ? "0" + n : n;
-         }
+                 for (i = 0; i < l; i++) {
+                   item.cells[i] = splitCells(item.cells[i], item.header.length);
+                 }
 
-         function this_value() {
-           return this.valueOf();
-         }
+                 return item;
+               }
+             }
+           }
+         }, {
+           key: "hr",
+           value: function hr(src) {
+             var cap = this.rules.block.hr.exec(src);
 
-         if (typeof Date.prototype.toJSON !== "function") {
-           Date.prototype.toJSON = function () {
-             return isFinite(this.valueOf()) ? this.getUTCFullYear() + "-" + f(this.getUTCMonth() + 1) + "-" + f(this.getUTCDate()) + "T" + f(this.getUTCHours()) + ":" + f(this.getUTCMinutes()) + ":" + f(this.getUTCSeconds()) + "Z" : null;
-           };
+             if (cap) {
+               return {
+                 type: 'hr',
+                 raw: cap[0]
+               };
+             }
+           }
+         }, {
+           key: "blockquote",
+           value: function blockquote(src) {
+             var cap = this.rules.block.blockquote.exec(src);
+
+             if (cap) {
+               var text = cap[0].replace(/^ *> ?/gm, '');
+               return {
+                 type: 'blockquote',
+                 raw: cap[0],
+                 text: text
+               };
+             }
+           }
+         }, {
+           key: "list",
+           value: function list(src) {
+             var cap = this.rules.block.list.exec(src);
+
+             if (cap) {
+               var raw = cap[0];
+               var bull = cap[2];
+               var isordered = bull.length > 1;
+               var list = {
+                 type: 'list',
+                 raw: raw,
+                 ordered: isordered,
+                 start: isordered ? +bull.slice(0, -1) : '',
+                 loose: false,
+                 items: []
+               }; // Get each top-level item.
+
+               var itemMatch = cap[0].match(this.rules.block.item);
+               var next = false,
+                   item,
+                   space,
+                   bcurr,
+                   bnext,
+                   addBack,
+                   loose,
+                   istask,
+                   ischecked,
+                   endMatch;
+               var l = itemMatch.length;
+               bcurr = this.rules.block.listItemStart.exec(itemMatch[0]);
 
-           Boolean.prototype.toJSON = this_value;
-           Number.prototype.toJSON = this_value;
-           String.prototype.toJSON = this_value;
-         }
+               for (var i = 0; i < l; i++) {
+                 item = itemMatch[i];
+                 raw = item;
+
+                 if (!this.options.pedantic) {
+                   // Determine if current item contains the end of the list
+                   endMatch = item.match(new RegExp('\\n\\s*\\n {0,' + (bcurr[0].length - 1) + '}\\S'));
+
+                   if (endMatch) {
+                     addBack = item.length - endMatch.index + itemMatch.slice(i + 1).join('\n').length;
+                     list.raw = list.raw.substring(0, list.raw.length - addBack);
+                     item = item.substring(0, endMatch.index);
+                     raw = item;
+                     l = i + 1;
+                   }
+                 } // Determine whether the next list item belongs here.
+                 // Backpedal if it does not belong in this list.
 
-         var gap;
-         var indent;
-         var meta;
-         var rep;
 
-         function quote(string) {
-           // If the string contains no control characters, no quote characters, and no
-           // backslash characters, then we can safely slap some quotes around it.
-           // Otherwise we must also replace the offending characters with safe escape
-           // sequences.
-           rx_escapable.lastIndex = 0;
-           return rx_escapable.test(string) ? "\"" + string.replace(rx_escapable, function (a) {
-             var c = meta[a];
-             return typeof c === "string" ? c : "\\u" + ("0000" + a.charCodeAt(0).toString(16)).slice(-4);
-           }) + "\"" : "\"" + string + "\"";
-         }
+                 if (i !== l - 1) {
+                   bnext = this.rules.block.listItemStart.exec(itemMatch[i + 1]);
 
-         function str(key, holder) {
-           // Produce a string from holder[key].
-           var i; // The loop counter.
+                   if (!this.options.pedantic ? bnext[1].length >= bcurr[0].length || bnext[1].length > 3 : bnext[1].length > bcurr[1].length) {
+                     // nested list or continuation
+                     itemMatch.splice(i, 2, itemMatch[i] + (!this.options.pedantic && bnext[1].length < bcurr[0].length && !itemMatch[i].match(/\n$/) ? '' : '\n') + itemMatch[i + 1]);
+                     i--;
+                     l--;
+                     continue;
+                   } else if ( // different bullet style
+                   !this.options.pedantic || this.options.smartLists ? bnext[2][bnext[2].length - 1] !== bull[bull.length - 1] : isordered === (bnext[2].length === 1)) {
+                     addBack = itemMatch.slice(i + 1).join('\n').length;
+                     list.raw = list.raw.substring(0, list.raw.length - addBack);
+                     i = l - 1;
+                   }
 
-           var k; // The member key.
+                   bcurr = bnext;
+                 } // Remove the list item's bullet
+                 // so it is seen as the next token.
 
-           var v; // The member value.
 
-           var length;
-           var mind = gap;
-           var partial;
-           var value = holder[key]; // If the value has a toJSON method, call it to obtain a replacement value.
+                 space = item.length;
+                 item = item.replace(/^ *([*+-]|\d+[.)]) ?/, ''); // Outdent whatever the
+                 // list item contains. Hacky.
 
-           if (value && _typeof(value) === "object" && typeof value.toJSON === "function") {
-             value = value.toJSON(key);
-           } // If we were called with a replacer function, then call the replacer to
-           // obtain a replacement value.
+                 if (~item.indexOf('\n ')) {
+                   space -= item.length;
+                   item = !this.options.pedantic ? item.replace(new RegExp('^ {1,' + space + '}', 'gm'), '') : item.replace(/^ {1,4}/gm, '');
+                 } // trim item newlines at end
 
 
-           if (typeof rep === "function") {
-             value = rep.call(holder, key, value);
-           } // What happens next depends on the value's type.
+                 item = rtrim(item, '\n');
 
+                 if (i !== l - 1) {
+                   raw = raw + '\n';
+                 } // Determine whether item is loose or not.
+                 // Use: /(^|\n)(?! )[^\n]+\n\n(?!\s*$)/
+                 // for discount behavior.
 
-           switch (_typeof(value)) {
-             case "string":
-               return quote(value);
 
-             case "number":
-               // JSON numbers must be finite. Encode non-finite numbers as null.
-               return isFinite(value) ? String(value) : "null";
+                 loose = next || /\n\n(?!\s*$)/.test(raw);
 
-             case "boolean":
-             case "null":
-               // If the value is a boolean or null, convert it to a string. Note:
-               // typeof null does not produce "null". The case is included here in
-               // the remote chance that this gets fixed someday.
-               return String(value);
-             // If the type is "object", we might be dealing with an object or an array or
-             // null.
+                 if (i !== l - 1) {
+                   next = raw.slice(-2) === '\n\n';
+                   if (!loose) loose = next;
+                 }
 
-             case "object":
-               // Due to a specification blunder in ECMAScript, typeof null is "object",
-               // so watch out for that case.
-               if (!value) {
-                 return "null";
-               } // Make an array to hold the partial results of stringifying this object value.
+                 if (loose) {
+                   list.loose = true;
+                 } // Check for task list items
 
 
-               gap += indent;
-               partial = []; // Is the value an array?
+                 if (this.options.gfm) {
+                   istask = /^\[[ xX]\] /.test(item);
+                   ischecked = undefined;
 
-               if (Object.prototype.toString.apply(value) === "[object Array]") {
-                 // The value is an array. Stringify every element. Use null as a placeholder
-                 // for non-JSON values.
-                 length = value.length;
+                   if (istask) {
+                     ischecked = item[1] !== ' ';
+                     item = item.replace(/^\[[ xX]\] +/, '');
+                   }
+                 }
 
-                 for (i = 0; i < length; i += 1) {
-                   partial[i] = str(i, value) || "null";
-                 } // Join all of the elements together, separated with commas, and wrap them in
-                 // brackets.
+                 list.items.push({
+                   type: 'list_item',
+                   raw: raw,
+                   task: istask,
+                   checked: ischecked,
+                   loose: loose,
+                   text: item
+                 });
+               }
 
+               return list;
+             }
+           }
+         }, {
+           key: "html",
+           value: function html(src) {
+             var cap = this.rules.block.html.exec(src);
 
-                 v = partial.length === 0 ? "[]" : gap ? "[\n" + gap + partial.join(",\n" + gap) + "\n" + mind + "]" : "[" + partial.join(",") + "]";
-                 gap = mind;
-                 return v;
-               } // If the replacer is an array, use it to select the members to be stringified.
+             if (cap) {
+               return {
+                 type: this.options.sanitize ? 'paragraph' : 'html',
+                 raw: cap[0],
+                 pre: !this.options.sanitizer && (cap[1] === 'pre' || cap[1] === 'script' || cap[1] === 'style'),
+                 text: this.options.sanitize ? this.options.sanitizer ? this.options.sanitizer(cap[0]) : _escape(cap[0]) : cap[0]
+               };
+             }
+           }
+         }, {
+           key: "def",
+           value: function def(src) {
+             var cap = this.rules.block.def.exec(src);
 
+             if (cap) {
+               if (cap[3]) cap[3] = cap[3].substring(1, cap[3].length - 1);
+               var tag = cap[1].toLowerCase().replace(/\s+/g, ' ');
+               return {
+                 type: 'def',
+                 tag: tag,
+                 raw: cap[0],
+                 href: cap[2],
+                 title: cap[3]
+               };
+             }
+           }
+         }, {
+           key: "table",
+           value: function table(src) {
+             var cap = this.rules.block.table.exec(src);
 
-               if (rep && _typeof(rep) === "object") {
-                 length = rep.length;
+             if (cap) {
+               var item = {
+                 type: 'table',
+                 header: splitCells(cap[1].replace(/^ *| *\| *$/g, '')),
+                 align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
+                 cells: cap[3] ? cap[3].replace(/\n$/, '').split('\n') : []
+               };
 
-                 for (i = 0; i < length; i += 1) {
-                   if (typeof rep[i] === "string") {
-                     k = rep[i];
-                     v = str(k, value);
+               if (item.header.length === item.align.length) {
+                 item.raw = cap[0];
+                 var l = item.align.length;
+                 var i;
 
-                     if (v) {
-                       partial.push(quote(k) + (gap ? ": " : ":") + v);
-                     }
+                 for (i = 0; i < l; i++) {
+                   if (/^ *-+: *$/.test(item.align[i])) {
+                     item.align[i] = 'right';
+                   } else if (/^ *:-+: *$/.test(item.align[i])) {
+                     item.align[i] = 'center';
+                   } else if (/^ *:-+ *$/.test(item.align[i])) {
+                     item.align[i] = 'left';
+                   } else {
+                     item.align[i] = null;
                    }
                  }
-               } else {
-                 // Otherwise, iterate through all of the keys in the object.
-                 for (k in value) {
-                   if (Object.prototype.hasOwnProperty.call(value, k)) {
-                     v = str(k, value);
 
-                     if (v) {
-                       partial.push(quote(k) + (gap ? ": " : ":") + v);
-                     }
-                   }
+                 l = item.cells.length;
+
+                 for (i = 0; i < l; i++) {
+                   item.cells[i] = splitCells(item.cells[i].replace(/^ *\| *| *\| *$/g, ''), item.header.length);
                  }
-               } // Join all of the member texts together, separated with commas,
-               // and wrap them in braces.
 
+                 return item;
+               }
+             }
+           }
+         }, {
+           key: "lheading",
+           value: function lheading(src) {
+             var cap = this.rules.block.lheading.exec(src);
 
-               v = partial.length === 0 ? "{}" : gap ? "{\n" + gap + partial.join(",\n" + gap) + "\n" + mind + "}" : "{" + partial.join(",") + "}";
-               gap = mind;
-               return v;
+             if (cap) {
+               return {
+                 type: 'heading',
+                 raw: cap[0],
+                 depth: cap[2].charAt(0) === '=' ? 1 : 2,
+                 text: cap[1]
+               };
+             }
            }
-         } // If the JSON object does not yet have a stringify method, give it one.
+         }, {
+           key: "paragraph",
+           value: function paragraph(src) {
+             var cap = this.rules.block.paragraph.exec(src);
 
+             if (cap) {
+               return {
+                 type: 'paragraph',
+                 raw: cap[0],
+                 text: cap[1].charAt(cap[1].length - 1) === '\n' ? cap[1].slice(0, -1) : cap[1]
+               };
+             }
+           }
+         }, {
+           key: "text",
+           value: function text(src) {
+             var cap = this.rules.block.text.exec(src);
 
-         if (typeof JSON.stringify !== "function") {
-           meta = {
-             // table of character substitutions
-             "\b": "\\b",
-             "\t": "\\t",
-             "\n": "\\n",
-             "\f": "\\f",
-             "\r": "\\r",
-             "\"": "\\\"",
-             "\\": "\\\\"
-           };
+             if (cap) {
+               return {
+                 type: 'text',
+                 raw: cap[0],
+                 text: cap[0]
+               };
+             }
+           }
+         }, {
+           key: "escape",
+           value: function escape(src) {
+             var cap = this.rules.inline.escape.exec(src);
 
-           JSON.stringify = function (value, replacer, space) {
-             // The stringify method takes a value and an optional replacer, and an optional
-             // space parameter, and returns a JSON text. The replacer can be a function
-             // that can replace values, or an array of strings that will select the keys.
-             // A default replacer method can be provided. Use of the space parameter can
-             // produce text that is more easily readable.
-             var i;
-             gap = "";
-             indent = ""; // If the space parameter is a number, make an indent string containing that
-             // many spaces.
+             if (cap) {
+               return {
+                 type: 'escape',
+                 raw: cap[0],
+                 text: _escape(cap[1])
+               };
+             }
+           }
+         }, {
+           key: "tag",
+           value: function tag(src, inLink, inRawBlock) {
+             var cap = this.rules.inline.tag.exec(src);
 
-             if (typeof space === "number") {
-               for (i = 0; i < space; i += 1) {
-                 indent += " ";
-               } // If the space parameter is a string, it will be used as the indent string.
+             if (cap) {
+               if (!inLink && /^<a /i.test(cap[0])) {
+                 inLink = true;
+               } else if (inLink && /^<\/a>/i.test(cap[0])) {
+                 inLink = false;
+               }
 
-             } else if (typeof space === "string") {
-               indent = space;
-             } // If there is a replacer, it must be a function or an array.
-             // Otherwise, throw an error.
+               if (!inRawBlock && /^<(pre|code|kbd|script)(\s|>)/i.test(cap[0])) {
+                 inRawBlock = true;
+               } else if (inRawBlock && /^<\/(pre|code|kbd|script)(\s|>)/i.test(cap[0])) {
+                 inRawBlock = false;
+               }
 
+               return {
+                 type: this.options.sanitize ? 'text' : 'html',
+                 raw: cap[0],
+                 inLink: inLink,
+                 inRawBlock: inRawBlock,
+                 text: this.options.sanitize ? this.options.sanitizer ? this.options.sanitizer(cap[0]) : _escape(cap[0]) : cap[0]
+               };
+             }
+           }
+         }, {
+           key: "link",
+           value: function link(src) {
+             var cap = this.rules.inline.link.exec(src);
 
-             rep = replacer;
+             if (cap) {
+               var trimmedUrl = cap[2].trim();
 
-             if (replacer && typeof replacer !== "function" && (_typeof(replacer) !== "object" || typeof replacer.length !== "number")) {
-               throw new Error("JSON.stringify");
-             } // Make a fake root object containing our value under the key of "".
-             // Return the result of stringifying the value.
+               if (!this.options.pedantic && /^</.test(trimmedUrl)) {
+                 // commonmark requires matching angle brackets
+                 if (!/>$/.test(trimmedUrl)) {
+                   return;
+                 } // ending angle bracket cannot be escaped
 
 
-             return str("", {
-               "": value
-             });
-           };
-         } // If the JSON object does not yet have a parse method, give it one.
+                 var rtrimSlash = rtrim(trimmedUrl.slice(0, -1), '\\');
 
+                 if ((trimmedUrl.length - rtrimSlash.length) % 2 === 0) {
+                   return;
+                 }
+               } else {
+                 // find closing parenthesis
+                 var lastParenIndex = findClosingBracket(cap[2], '()');
 
-         if (typeof JSON.parse !== "function") {
-           JSON.parse = function (text, reviver) {
-             // The parse method takes a text and an optional reviver function, and returns
-             // a JavaScript value if the text is a valid JSON text.
-             var j;
+                 if (lastParenIndex > -1) {
+                   var start = cap[0].indexOf('!') === 0 ? 5 : 4;
+                   var linkLen = start + cap[1].length + lastParenIndex;
+                   cap[2] = cap[2].substring(0, lastParenIndex);
+                   cap[0] = cap[0].substring(0, linkLen).trim();
+                   cap[3] = '';
+                 }
+               }
 
-             function walk(holder, key) {
-               // The walk method is used to recursively walk the resulting structure so
-               // that modifications can be made.
-               var k;
-               var v;
-               var value = holder[key];
+               var href = cap[2];
+               var title = '';
 
-               if (value && _typeof(value) === "object") {
-                 for (k in value) {
-                   if (Object.prototype.hasOwnProperty.call(value, k)) {
-                     v = walk(value, k);
+               if (this.options.pedantic) {
+                 // split pedantic href and title
+                 var link = /^([^'"]*[^\s])\s+(['"])(.*)\2/.exec(href);
 
-                     if (v !== undefined) {
-                       value[k] = v;
-                     } else {
-                       delete value[k];
-                     }
-                   }
+                 if (link) {
+                   href = link[1];
+                   title = link[3];
                  }
+               } else {
+                 title = cap[3] ? cap[3].slice(1, -1) : '';
                }
 
-               return reviver.call(holder, key, value);
-             } // Parsing happens in four stages. In the first stage, we replace certain
-             // Unicode characters with escape sequences. JavaScript handles many characters
-             // incorrectly, either silently deleting them, or treating them as line endings.
-
+               href = href.trim();
 
-             text = String(text);
-             rx_dangerous.lastIndex = 0;
+               if (/^</.test(href)) {
+                 if (this.options.pedantic && !/>$/.test(trimmedUrl)) {
+                   // pedantic allows starting angle bracket without ending angle bracket
+                   href = href.slice(1);
+                 } else {
+                   href = href.slice(1, -1);
+                 }
+               }
 
-             if (rx_dangerous.test(text)) {
-               text = text.replace(rx_dangerous, function (a) {
-                 return "\\u" + ("0000" + a.charCodeAt(0).toString(16)).slice(-4);
-               });
-             } // In the second stage, we run the text against regular expressions that look
-             // for non-JSON patterns. We are especially concerned with "()" and "new"
-             // because they can cause invocation, and "=" because it can cause mutation.
-             // But just to be safe, we want to reject all unexpected forms.
-             // We split the second stage into 4 regexp operations in order to work around
-             // crippling inefficiencies in IE's and Safari's regexp engines. First we
-             // replace the JSON backslash pairs with "@" (a non-JSON character). Second, we
-             // replace all simple value tokens with "]" characters. Third, we delete all
-             // open brackets that follow a colon or comma or that begin the text. Finally,
-             // we look to see that the remaining characters are only whitespace or "]" or
-             // "," or ":" or "{" or "}". If that is so, then the text is safe for eval.
+               return outputLink(cap, {
+                 href: href ? href.replace(this.rules.inline._escapes, '$1') : href,
+                 title: title ? title.replace(this.rules.inline._escapes, '$1') : title
+               }, cap[0]);
+             }
+           }
+         }, {
+           key: "reflink",
+           value: function reflink(src, links) {
+             var cap;
 
+             if ((cap = this.rules.inline.reflink.exec(src)) || (cap = this.rules.inline.nolink.exec(src))) {
+               var link = (cap[2] || cap[1]).replace(/\s+/g, ' ');
+               link = links[link.toLowerCase()];
 
-             if (rx_one.test(text.replace(rx_two, "@").replace(rx_three, "]").replace(rx_four, ""))) {
-               // In the third stage we use the eval function to compile the text into a
-               // JavaScript structure. The "{" operator is subject to a syntactic ambiguity
-               // in JavaScript: it can begin a block or an object literal. We wrap the text
-               // in parens to eliminate the ambiguity.
-               j = eval("(" + text + ")"); // In the optional fourth stage, we recursively walk the new structure, passing
-               // each name/value pair to a reviver function for possible transformation.
+               if (!link || !link.href) {
+                 var text = cap[0].charAt(0);
+                 return {
+                   type: 'text',
+                   raw: text,
+                   text: text
+                 };
+               }
 
-               return typeof reviver === "function" ? walk({
-                 "": j
-               }, "") : j;
-             } // If the text is not JSON parseable, then a SyntaxError is thrown.
+               return outputLink(cap, link, cap[0]);
+             }
+           }
+         }, {
+           key: "emStrong",
+           value: function emStrong(src, maskedSrc) {
+             var prevChar = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : '';
+             var match = this.rules.inline.emStrong.lDelim.exec(src);
+             if (!match) return; // _ can't be between two alphanumerics. \p{L}\p{N} includes non-english alphabet/numbers as well
 
+             if (match[3] && prevChar.match(/(?:[0-9A-Za-z\xAA\xB2\xB3\xB5\xB9\xBA\xBC-\xBE\xC0-\xD6\xD8-\xF6\xF8-\u02C1\u02C6-\u02D1\u02E0-\u02E4\u02EC\u02EE\u0370-\u0374\u0376\u0377\u037A-\u037D\u037F\u0386\u0388-\u038A\u038C\u038E-\u03A1\u03A3-\u03F5\u03F7-\u0481\u048A-\u052F\u0531-\u0556\u0559\u0560-\u0588\u05D0-\u05EA\u05EF-\u05F2\u0620-\u064A\u0660-\u0669\u066E\u066F\u0671-\u06D3\u06D5\u06E5\u06E6\u06EE-\u06FC\u06FF\u0710\u0712-\u072F\u074D-\u07A5\u07B1\u07C0-\u07EA\u07F4\u07F5\u07FA\u0800-\u0815\u081A\u0824\u0828\u0840-\u0858\u0860-\u086A\u08A0-\u08B4\u08B6-\u08C7\u0904-\u0939\u093D\u0950\u0958-\u0961\u0966-\u096F\u0971-\u0980\u0985-\u098C\u098F\u0990\u0993-\u09A8\u09AA-\u09B0\u09B2\u09B6-\u09B9\u09BD\u09CE\u09DC\u09DD\u09DF-\u09E1\u09E6-\u09F1\u09F4-\u09F9\u09FC\u0A05-\u0A0A\u0A0F\u0A10\u0A13-\u0A28\u0A2A-\u0A30\u0A32\u0A33\u0A35\u0A36\u0A38\u0A39\u0A59-\u0A5C\u0A5E\u0A66-\u0A6F\u0A72-\u0A74\u0A85-\u0A8D\u0A8F-\u0A91\u0A93-\u0AA8\u0AAA-\u0AB0\u0AB2\u0AB3\u0AB5-\u0AB9\u0ABD\u0AD0\u0AE0\u0AE1\u0AE6-\u0AEF\u0AF9\u0B05-\u0B0C\u0B0F\u0B10\u0B13-\u0B28\u0B2A-\u0B30\u0B32\u0B33\u0B35-\u0B39\u0B3D\u0B5C\u0B5D\u0B5F-\u0B61\u0B66-\u0B6F\u0B71-\u0B77\u0B83\u0B85-\u0B8A\u0B8E-\u0B90\u0B92-\u0B95\u0B99\u0B9A\u0B9C\u0B9E\u0B9F\u0BA3\u0BA4\u0BA8-\u0BAA\u0BAE-\u0BB9\u0BD0\u0BE6-\u0BF2\u0C05-\u0C0C\u0C0E-\u0C10\u0C12-\u0C28\u0C2A-\u0C39\u0C3D\u0C58-\u0C5A\u0C60\u0C61\u0C66-\u0C6F\u0C78-\u0C7E\u0C80\u0C85-\u0C8C\u0C8E-\u0C90\u0C92-\u0CA8\u0CAA-\u0CB3\u0CB5-\u0CB9\u0CBD\u0CDE\u0CE0\u0CE1\u0CE6-\u0CEF\u0CF1\u0CF2\u0D04-\u0D0C\u0D0E-\u0D10\u0D12-\u0D3A\u0D3D\u0D4E\u0D54-\u0D56\u0D58-\u0D61\u0D66-\u0D78\u0D7A-\u0D7F\u0D85-\u0D96\u0D9A-\u0DB1\u0DB3-\u0DBB\u0DBD\u0DC0-\u0DC6\u0DE6-\u0DEF\u0E01-\u0E30\u0E32\u0E33\u0E40-\u0E46\u0E50-\u0E59\u0E81\u0E82\u0E84\u0E86-\u0E8A\u0E8C-\u0EA3\u0EA5\u0EA7-\u0EB0\u0EB2\u0EB3\u0EBD\u0EC0-\u0EC4\u0EC6\u0ED0-\u0ED9\u0EDC-\u0EDF\u0F00\u0F20-\u0F33\u0F40-\u0F47\u0F49-\u0F6C\u0F88-\u0F8C\u1000-\u102A\u103F-\u1049\u1050-\u1055\u105A-\u105D\u1061\u1065\u1066\u106E-\u1070\u1075-\u1081\u108E\u1090-\u1099\u10A0-\u10C5\u10C7\u10CD\u10D0-\u10FA\u10FC-\u1248\u124A-\u124D\u1250-\u1256\u1258\u125A-\u125D\u1260-\u1288\u128A-\u128D\u1290-\u12B0\u12B2-\u12B5\u12B8-\u12BE\u12C0\u12C2-\u12C5\u12C8-\u12D6\u12D8-\u1310\u1312-\u1315\u1318-\u135A\u1369-\u137C\u1380-\u138F\u13A0-\u13F5\u13F8-\u13FD\u1401-\u166C\u166F-\u167F\u1681-\u169A\u16A0-\u16EA\u16EE-\u16F8\u1700-\u170C\u170E-\u1711\u1720-\u1731\u1740-\u1751\u1760-\u176C\u176E-\u1770\u1780-\u17B3\u17D7\u17DC\u17E0-\u17E9\u17F0-\u17F9\u1810-\u1819\u1820-\u1878\u1880-\u1884\u1887-\u18A8\u18AA\u18B0-\u18F5\u1900-\u191E\u1946-\u196D\u1970-\u1974\u1980-\u19AB\u19B0-\u19C9\u19D0-\u19DA\u1A00-\u1A16\u1A20-\u1A54\u1A80-\u1A89\u1A90-\u1A99\u1AA7\u1B05-\u1B33\u1B45-\u1B4B\u1B50-\u1B59\u1B83-\u1BA0\u1BAE-\u1BE5\u1C00-\u1C23\u1C40-\u1C49\u1C4D-\u1C7D\u1C80-\u1C88\u1C90-\u1CBA\u1CBD-\u1CBF\u1CE9-\u1CEC\u1CEE-\u1CF3\u1CF5\u1CF6\u1CFA\u1D00-\u1DBF\u1E00-\u1F15\u1F18-\u1F1D\u1F20-\u1F45\u1F48-\u1F4D\u1F50-\u1F57\u1F59\u1F5B\u1F5D\u1F5F-\u1F7D\u1F80-\u1FB4\u1FB6-\u1FBC\u1FBE\u1FC2-\u1FC4\u1FC6-\u1FCC\u1FD0-\u1FD3\u1FD6-\u1FDB\u1FE0-\u1FEC\u1FF2-\u1FF4\u1FF6-\u1FFC\u2070\u2071\u2074-\u2079\u207F-\u2089\u2090-\u209C\u2102\u2107\u210A-\u2113\u2115\u2119-\u211D\u2124\u2126\u2128\u212A-\u212D\u212F-\u2139\u213C-\u213F\u2145-\u2149\u214E\u2150-\u2189\u2460-\u249B\u24EA-\u24FF\u2776-\u2793\u2C00-\u2C2E\u2C30-\u2C5E\u2C60-\u2CE4\u2CEB-\u2CEE\u2CF2\u2CF3\u2CFD\u2D00-\u2D25\u2D27\u2D2D\u2D30-\u2D67\u2D6F\u2D80-\u2D96\u2DA0-\u2DA6\u2DA8-\u2DAE\u2DB0-\u2DB6\u2DB8-\u2DBE\u2DC0-\u2DC6\u2DC8-\u2DCE\u2DD0-\u2DD6\u2DD8-\u2DDE\u2E2F\u3005-\u3007\u3021-\u3029\u3031-\u3035\u3038-\u303C\u3041-\u3096\u309D-\u309F\u30A1-\u30FA\u30FC-\u30FF\u3105-\u312F\u3131-\u318E\u3192-\u3195\u31A0-\u31BF\u31F0-\u31FF\u3220-\u3229\u3248-\u324F\u3251-\u325F\u3280-\u3289\u32B1-\u32BF\u3400-\u4DBF\u4E00-\u9FFC\uA000-\uA48C\uA4D0-\uA4FD\uA500-\uA60C\uA610-\uA62B\uA640-\uA66E\uA67F-\uA69D\uA6A0-\uA6EF\uA717-\uA71F\uA722-\uA788\uA78B-\uA7BF\uA7C2-\uA7CA\uA7F5-\uA801\uA803-\uA805\uA807-\uA80A\uA80C-\uA822\uA830-\uA835\uA840-\uA873\uA882-\uA8B3\uA8D0-\uA8D9\uA8F2-\uA8F7\uA8FB\uA8FD\uA8FE\uA900-\uA925\uA930-\uA946\uA960-\uA97C\uA984-\uA9B2\uA9CF-\uA9D9\uA9E0-\uA9E4\uA9E6-\uA9FE\uAA00-\uAA28\uAA40-\uAA42\uAA44-\uAA4B\uAA50-\uAA59\uAA60-\uAA76\uAA7A\uAA7E-\uAAAF\uAAB1\uAAB5\uAAB6\uAAB9-\uAABD\uAAC0\uAAC2\uAADB-\uAADD\uAAE0-\uAAEA\uAAF2-\uAAF4\uAB01-\uAB06\uAB09-\uAB0E\uAB11-\uAB16\uAB20-\uAB26\uAB28-\uAB2E\uAB30-\uAB5A\uAB5C-\uAB69\uAB70-\uABE2\uABF0-\uABF9\uAC00-\uD7A3\uD7B0-\uD7C6\uD7CB-\uD7FB\uF900-\uFA6D\uFA70-\uFAD9\uFB00-\uFB06\uFB13-\uFB17\uFB1D\uFB1F-\uFB28\uFB2A-\uFB36\uFB38-\uFB3C\uFB3E\uFB40\uFB41\uFB43\uFB44\uFB46-\uFBB1\uFBD3-\uFD3D\uFD50-\uFD8F\uFD92-\uFDC7\uFDF0-\uFDFB\uFE70-\uFE74\uFE76-\uFEFC\uFF10-\uFF19\uFF21-\uFF3A\uFF41-\uFF5A\uFF66-\uFFBE\uFFC2-\uFFC7\uFFCA-\uFFCF\uFFD2-\uFFD7\uFFDA-\uFFDC]|\uD800[\uDC00-\uDC0B\uDC0D-\uDC26\uDC28-\uDC3A\uDC3C\uDC3D\uDC3F-\uDC4D\uDC50-\uDC5D\uDC80-\uDCFA\uDD07-\uDD33\uDD40-\uDD78\uDD8A\uDD8B\uDE80-\uDE9C\uDEA0-\uDED0\uDEE1-\uDEFB\uDF00-\uDF23\uDF2D-\uDF4A\uDF50-\uDF75\uDF80-\uDF9D\uDFA0-\uDFC3\uDFC8-\uDFCF\uDFD1-\uDFD5]|\uD801[\uDC00-\uDC9D\uDCA0-\uDCA9\uDCB0-\uDCD3\uDCD8-\uDCFB\uDD00-\uDD27\uDD30-\uDD63\uDE00-\uDF36\uDF40-\uDF55\uDF60-\uDF67]|\uD802[\uDC00-\uDC05\uDC08\uDC0A-\uDC35\uDC37\uDC38\uDC3C\uDC3F-\uDC55\uDC58-\uDC76\uDC79-\uDC9E\uDCA7-\uDCAF\uDCE0-\uDCF2\uDCF4\uDCF5\uDCFB-\uDD1B\uDD20-\uDD39\uDD80-\uDDB7\uDDBC-\uDDCF\uDDD2-\uDE00\uDE10-\uDE13\uDE15-\uDE17\uDE19-\uDE35\uDE40-\uDE48\uDE60-\uDE7E\uDE80-\uDE9F\uDEC0-\uDEC7\uDEC9-\uDEE4\uDEEB-\uDEEF\uDF00-\uDF35\uDF40-\uDF55\uDF58-\uDF72\uDF78-\uDF91\uDFA9-\uDFAF]|\uD803[\uDC00-\uDC48\uDC80-\uDCB2\uDCC0-\uDCF2\uDCFA-\uDD23\uDD30-\uDD39\uDE60-\uDE7E\uDE80-\uDEA9\uDEB0\uDEB1\uDF00-\uDF27\uDF30-\uDF45\uDF51-\uDF54\uDFB0-\uDFCB\uDFE0-\uDFF6]|\uD804[\uDC03-\uDC37\uDC52-\uDC6F\uDC83-\uDCAF\uDCD0-\uDCE8\uDCF0-\uDCF9\uDD03-\uDD26\uDD36-\uDD3F\uDD44\uDD47\uDD50-\uDD72\uDD76\uDD83-\uDDB2\uDDC1-\uDDC4\uDDD0-\uDDDA\uDDDC\uDDE1-\uDDF4\uDE00-\uDE11\uDE13-\uDE2B\uDE80-\uDE86\uDE88\uDE8A-\uDE8D\uDE8F-\uDE9D\uDE9F-\uDEA8\uDEB0-\uDEDE\uDEF0-\uDEF9\uDF05-\uDF0C\uDF0F\uDF10\uDF13-\uDF28\uDF2A-\uDF30\uDF32\uDF33\uDF35-\uDF39\uDF3D\uDF50\uDF5D-\uDF61]|\uD805[\uDC00-\uDC34\uDC47-\uDC4A\uDC50-\uDC59\uDC5F-\uDC61\uDC80-\uDCAF\uDCC4\uDCC5\uDCC7\uDCD0-\uDCD9\uDD80-\uDDAE\uDDD8-\uDDDB\uDE00-\uDE2F\uDE44\uDE50-\uDE59\uDE80-\uDEAA\uDEB8\uDEC0-\uDEC9\uDF00-\uDF1A\uDF30-\uDF3B]|\uD806[\uDC00-\uDC2B\uDCA0-\uDCF2\uDCFF-\uDD06\uDD09\uDD0C-\uDD13\uDD15\uDD16\uDD18-\uDD2F\uDD3F\uDD41\uDD50-\uDD59\uDDA0-\uDDA7\uDDAA-\uDDD0\uDDE1\uDDE3\uDE00\uDE0B-\uDE32\uDE3A\uDE50\uDE5C-\uDE89\uDE9D\uDEC0-\uDEF8]|\uD807[\uDC00-\uDC08\uDC0A-\uDC2E\uDC40\uDC50-\uDC6C\uDC72-\uDC8F\uDD00-\uDD06\uDD08\uDD09\uDD0B-\uDD30\uDD46\uDD50-\uDD59\uDD60-\uDD65\uDD67\uDD68\uDD6A-\uDD89\uDD98\uDDA0-\uDDA9\uDEE0-\uDEF2\uDFB0\uDFC0-\uDFD4]|\uD808[\uDC00-\uDF99]|\uD809[\uDC00-\uDC6E\uDC80-\uDD43]|[\uD80C\uD81C-\uD820\uD822\uD840-\uD868\uD86A-\uD86C\uD86F-\uD872\uD874-\uD879\uD880-\uD883][\uDC00-\uDFFF]|\uD80D[\uDC00-\uDC2E]|\uD811[\uDC00-\uDE46]|\uD81A[\uDC00-\uDE38\uDE40-\uDE5E\uDE60-\uDE69\uDED0-\uDEED\uDF00-\uDF2F\uDF40-\uDF43\uDF50-\uDF59\uDF5B-\uDF61\uDF63-\uDF77\uDF7D-\uDF8F]|\uD81B[\uDE40-\uDE96\uDF00-\uDF4A\uDF50\uDF93-\uDF9F\uDFE0\uDFE1\uDFE3]|\uD821[\uDC00-\uDFF7]|\uD823[\uDC00-\uDCD5\uDD00-\uDD08]|\uD82C[\uDC00-\uDD1E\uDD50-\uDD52\uDD64-\uDD67\uDD70-\uDEFB]|\uD82F[\uDC00-\uDC6A\uDC70-\uDC7C\uDC80-\uDC88\uDC90-\uDC99]|\uD834[\uDEE0-\uDEF3\uDF60-\uDF78]|\uD835[\uDC00-\uDC54\uDC56-\uDC9C\uDC9E\uDC9F\uDCA2\uDCA5\uDCA6\uDCA9-\uDCAC\uDCAE-\uDCB9\uDCBB\uDCBD-\uDCC3\uDCC5-\uDD05\uDD07-\uDD0A\uDD0D-\uDD14\uDD16-\uDD1C\uDD1E-\uDD39\uDD3B-\uDD3E\uDD40-\uDD44\uDD46\uDD4A-\uDD50\uDD52-\uDEA5\uDEA8-\uDEC0\uDEC2-\uDEDA\uDEDC-\uDEFA\uDEFC-\uDF14\uDF16-\uDF34\uDF36-\uDF4E\uDF50-\uDF6E\uDF70-\uDF88\uDF8A-\uDFA8\uDFAA-\uDFC2\uDFC4-\uDFCB\uDFCE-\uDFFF]|\uD838[\uDD00-\uDD2C\uDD37-\uDD3D\uDD40-\uDD49\uDD4E\uDEC0-\uDEEB\uDEF0-\uDEF9]|\uD83A[\uDC00-\uDCC4\uDCC7-\uDCCF\uDD00-\uDD43\uDD4B\uDD50-\uDD59]|\uD83B[\uDC71-\uDCAB\uDCAD-\uDCAF\uDCB1-\uDCB4\uDD01-\uDD2D\uDD2F-\uDD3D\uDE00-\uDE03\uDE05-\uDE1F\uDE21\uDE22\uDE24\uDE27\uDE29-\uDE32\uDE34-\uDE37\uDE39\uDE3B\uDE42\uDE47\uDE49\uDE4B\uDE4D-\uDE4F\uDE51\uDE52\uDE54\uDE57\uDE59\uDE5B\uDE5D\uDE5F\uDE61\uDE62\uDE64\uDE67-\uDE6A\uDE6C-\uDE72\uDE74-\uDE77\uDE79-\uDE7C\uDE7E\uDE80-\uDE89\uDE8B-\uDE9B\uDEA1-\uDEA3\uDEA5-\uDEA9\uDEAB-\uDEBB]|\uD83C[\uDD00-\uDD0C]|\uD83E[\uDFF0-\uDFF9]|\uD869[\uDC00-\uDEDD\uDF00-\uDFFF]|\uD86D[\uDC00-\uDF34\uDF40-\uDFFF]|\uD86E[\uDC00-\uDC1D\uDC20-\uDFFF]|\uD873[\uDC00-\uDEA1\uDEB0-\uDFFF]|\uD87A[\uDC00-\uDFE0]|\uD87E[\uDC00-\uDE1D]|\uD884[\uDC00-\uDF4A])/)) return;
+             var nextChar = match[1] || match[2] || '';
 
-             throw new SyntaxError("JSON.parse");
-           };
-         }
-       })();
+             if (!nextChar || nextChar && (prevChar === '' || this.rules.inline.punctuation.exec(prevChar))) {
+               var lLength = match[0].length - 1;
+               var rDelim,
+                   rLength,
+                   delimTotal = lLength,
+                   midDelimTotal = 0;
+               var endReg = match[0][0] === '*' ? this.rules.inline.emStrong.rDelimAst : this.rules.inline.emStrong.rDelimUnd;
+               endReg.lastIndex = 0; // Clip maskedSrc to same section of string as src (move to lexer?)
 
-       var json2 = json2Plugin;
+               maskedSrc = maskedSrc.slice(-1 * src.length + lLength);
 
-       function json2Plugin() {
-         return {};
-       }
+               while ((match = endReg.exec(maskedSrc)) != null) {
+                 rDelim = match[1] || match[2] || match[3] || match[4] || match[5] || match[6];
+                 if (!rDelim) continue; // skip single * in __abc*abc__
 
-       var plugins = [json2];
-       var store_legacy = storeEngine.createStore(all, plugins);
+                 rLength = rDelim.length;
 
-       //
-       // This code is only compatible with IE10+ because the [XDomainRequest](http://bit.ly/LfO7xo)
-       // object, IE<10's idea of [CORS](http://en.wikipedia.org/wiki/Cross-origin_resource_sharing),
-       // does not support custom headers, which this uses everywhere.
+                 if (match[3] || match[4]) {
+                   // found another Left Delim
+                   delimTotal += rLength;
+                   continue;
+                 } else if (match[5] || match[6]) {
+                   // either Left or Right Delim
+                   if (lLength % 3 && !((lLength + rLength) % 3)) {
+                     midDelimTotal += rLength;
+                     continue; // CommonMark Emphasis Rules 9-10
+                   }
+                 }
 
+                 delimTotal -= rLength;
+                 if (delimTotal > 0) continue; // Haven't found enough closing delimiters
+                 // Remove extra characters. *a*** -> *a*
 
-       var osmAuth = function osmAuth(o) {
-         var oauth = {}; // authenticated users will also have a request token secret, but it's
-         // not used in transactions with the server
+                 rLength = Math.min(rLength, rLength + delimTotal + midDelimTotal); // Create `em` if smallest delimiter has odd char count. *a***
 
-         oauth.authenticated = function () {
-           return !!(token('oauth_token') && token('oauth_token_secret'));
-         };
+                 if (Math.min(lLength, rLength) % 2) {
+                   return {
+                     type: 'em',
+                     raw: src.slice(0, lLength + match.index + rLength + 1),
+                     text: src.slice(1, lLength + match.index + rLength)
+                   };
+                 } // Create 'strong' if smallest delimiter has even char count. **a***
 
-         oauth.logout = function () {
-           token('oauth_token', '');
-           token('oauth_token_secret', '');
-           token('oauth_request_token_secret', '');
-           return oauth;
-         }; // TODO: detect lack of click event
 
+                 return {
+                   type: 'strong',
+                   raw: src.slice(0, lLength + match.index + rLength + 1),
+                   text: src.slice(2, lLength + match.index + rLength - 1)
+                 };
+               }
+             }
+           }
+         }, {
+           key: "codespan",
+           value: function codespan(src) {
+             var cap = this.rules.inline.code.exec(src);
 
-         oauth.authenticate = function (callback) {
-           if (oauth.authenticated()) return callback();
-           oauth.logout(); // ## Getting a request token
+             if (cap) {
+               var text = cap[2].replace(/\n/g, ' ');
+               var hasNonSpaceChars = /[^ ]/.test(text);
+               var hasSpaceCharsOnBothEnds = /^ /.test(text) && / $/.test(text);
 
-           var params = timenonce(getAuth(o)),
-               url = o.url + '/oauth/request_token';
-           params.oauth_signature = ohauth_1.signature(o.oauth_secret, '', ohauth_1.baseString('POST', url, params));
+               if (hasNonSpaceChars && hasSpaceCharsOnBothEnds) {
+                 text = text.substring(1, text.length - 1);
+               }
 
-           if (!o.singlepage) {
-             // Create a 600x550 popup window in the center of the screen
-             var w = 600,
-                 h = 550,
-                 settings = [['width', w], ['height', h], ['left', screen.width / 2 - w / 2], ['top', screen.height / 2 - h / 2]].map(function (x) {
-               return x.join('=');
-             }).join(','),
-                 popup = window.open('about:blank', 'oauth_window', settings);
-             oauth.popupWindow = popup;
+               text = _escape(text, true);
+               return {
+                 type: 'codespan',
+                 raw: cap[0],
+                 text: text
+               };
+             }
+           }
+         }, {
+           key: "br",
+           value: function br(src) {
+             var cap = this.rules.inline.br.exec(src);
 
-             if (!popup) {
-               var error = new Error('Popup was blocked');
-               error.status = 'popup-blocked';
-               throw error;
+             if (cap) {
+               return {
+                 type: 'br',
+                 raw: cap[0]
+               };
              }
-           } // Request a request token. When this is complete, the popup
-           // window is redirected to OSM's authorization page.
+           }
+         }, {
+           key: "del",
+           value: function del(src) {
+             var cap = this.rules.inline.del.exec(src);
 
+             if (cap) {
+               return {
+                 type: 'del',
+                 raw: cap[0],
+                 text: cap[2]
+               };
+             }
+           }
+         }, {
+           key: "autolink",
+           value: function autolink(src, mangle) {
+             var cap = this.rules.inline.autolink.exec(src);
 
-           ohauth_1.xhr('POST', url, params, null, {}, reqTokenDone);
-           o.loading();
+             if (cap) {
+               var text, href;
 
-           function reqTokenDone(err, xhr) {
-             o.done();
-             if (err) return callback(err);
-             var resp = ohauth_1.stringQs(xhr.response);
-             token('oauth_request_token_secret', resp.oauth_token_secret);
-             var authorize_url = o.url + '/oauth/authorize?' + ohauth_1.qsString({
-               oauth_token: resp.oauth_token,
-               oauth_callback: resolveUrl$1(o.landing)
-             });
+               if (cap[2] === '@') {
+                 text = _escape(this.options.mangle ? mangle(cap[1]) : cap[1]);
+                 href = 'mailto:' + text;
+               } else {
+                 text = _escape(cap[1]);
+                 href = text;
+               }
 
-             if (o.singlepage) {
-               location.href = authorize_url;
-             } else {
-               popup.location = authorize_url;
+               return {
+                 type: 'link',
+                 raw: cap[0],
+                 text: text,
+                 href: href,
+                 tokens: [{
+                   type: 'text',
+                   raw: text,
+                   text: text
+                 }]
+               };
              }
-           } // Called by a function in a landing page, in the popup window. The
-           // window closes itself.
-
-
-           window.authComplete = function (token) {
-             var oauth_token = ohauth_1.stringQs(token.split('?')[1]);
-             get_access_token(oauth_token.oauth_token);
-             delete window.authComplete;
-           }; // ## Getting an request token
-           //
-           // At this point we have an `oauth_token`, brought in from a function
-           // call on a landing page popup.
+           }
+         }, {
+           key: "url",
+           value: function url(src, mangle) {
+             var cap;
 
+             if (cap = this.rules.inline.url.exec(src)) {
+               var text, href;
 
-           function get_access_token(oauth_token) {
-             var url = o.url + '/oauth/access_token',
-                 params = timenonce(getAuth(o)),
-                 request_token_secret = token('oauth_request_token_secret');
-             params.oauth_token = oauth_token;
-             params.oauth_signature = ohauth_1.signature(o.oauth_secret, request_token_secret, ohauth_1.baseString('POST', url, params)); // ## Getting an access token
-             //
-             // The final token required for authentication. At this point
-             // we have a `request token secret`
+               if (cap[2] === '@') {
+                 text = _escape(this.options.mangle ? mangle(cap[0]) : cap[0]);
+                 href = 'mailto:' + text;
+               } else {
+                 // do extended autolink path validation
+                 var prevCapZero;
 
-             ohauth_1.xhr('POST', url, params, null, {}, accessTokenDone);
-             o.loading();
-           }
+                 do {
+                   prevCapZero = cap[0];
+                   cap[0] = this.rules.inline._backpedal.exec(cap[0])[0];
+                 } while (prevCapZero !== cap[0]);
 
-           function accessTokenDone(err, xhr) {
-             o.done();
-             if (err) return callback(err);
-             var access_token = ohauth_1.stringQs(xhr.response);
-             token('oauth_token', access_token.oauth_token);
-             token('oauth_token_secret', access_token.oauth_token_secret);
-             callback(null, oauth);
-           }
-         };
+                 text = _escape(cap[0]);
 
-         oauth.bringPopupWindowToFront = function () {
-           var brougtPopupToFront = false;
+                 if (cap[1] === 'www.') {
+                   href = 'http://' + text;
+                 } else {
+                   href = text;
+                 }
+               }
 
-           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;
+               return {
+                 type: 'link',
+                 raw: cap[0],
+                 text: text,
+                 href: href,
+                 tokens: [{
+                   type: 'text',
+                   raw: text,
+                   text: text
+                 }]
+               };
              }
-           } catch (err) {// Bringing popup window to front failed (probably because of the cross-origin error mentioned above)
            }
+         }, {
+           key: "inlineText",
+           value: function inlineText(src, inRawBlock, smartypants) {
+             var cap = this.rules.inline.text.exec(src);
 
-           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
-           // call on a landing page popup.
-           function get_access_token(oauth_token) {
-             var url = o.url + '/oauth/access_token',
-                 params = timenonce(getAuth(o)),
-                 request_token_secret = token('oauth_request_token_secret');
-             params.oauth_token = oauth_token;
-             params.oauth_signature = ohauth_1.signature(o.oauth_secret, request_token_secret, ohauth_1.baseString('POST', url, params)); // ## Getting an access token
-             // The final token required for authentication. At this point
-             // we have a `request token secret`
+             if (cap) {
+               var text;
 
-             ohauth_1.xhr('POST', url, params, null, {}, accessTokenDone);
-             o.loading();
-           }
+               if (inRawBlock) {
+                 text = this.options.sanitize ? this.options.sanitizer ? this.options.sanitizer(cap[0]) : _escape(cap[0]) : cap[0];
+               } else {
+                 text = _escape(this.options.smartypants ? smartypants(cap[0]) : cap[0]);
+               }
 
-           function accessTokenDone(err, xhr) {
-             o.done();
-             if (err) return callback(err);
-             var access_token = ohauth_1.stringQs(xhr.response);
-             token('oauth_token', access_token.oauth_token);
-             token('oauth_token_secret', access_token.oauth_token_secret);
-             callback(null, oauth);
+               return {
+                 type: 'text',
+                 raw: cap[0],
+                 text: text
+               };
+             }
            }
+         }]);
 
-           get_access_token(oauth_token);
-         }; // # xhr
-         //
-         // A single XMLHttpRequest wrapper that does authenticated calls if the
-         // user has logged in.
+         return Tokenizer;
+       }();
 
+       var noopTest = helpers.noopTest,
+           edit = helpers.edit,
+           merge$1 = helpers.merge;
+       /**
+        * Block-Level Grammar
+        */
 
-         oauth.xhr = function (options, callback) {
-           if (!oauth.authenticated()) {
-             if (o.auto) {
-               return oauth.authenticate(run);
-             } else {
-               callback('not authenticated', null);
-               return;
-             }
-           } else {
-             return run();
-           }
+       var block$1 = {
+         newline: /^(?: *(?:\n|$))+/,
+         code: /^( {4}[^\n]+(?:\n(?: *(?:\n|$))*)?)+/,
+         fences: /^ {0,3}(`{3,}(?=[^`\n]*\n)|~{3,})([^\n]*)\n(?:|([\s\S]*?)\n)(?: {0,3}\1[~`]* *(?:\n+|$)|$)/,
+         hr: /^ {0,3}((?:- *){3,}|(?:_ *){3,}|(?:\* *){3,})(?:\n+|$)/,
+         heading: /^ {0,3}(#{1,6})(?=\s|$)(.*)(?:\n+|$)/,
+         blockquote: /^( {0,3}> ?(paragraph|[^\n]*)(?:\n|$))+/,
+         list: /^( {0,3})(bull) [\s\S]+?(?:hr|def|\n{2,}(?! )(?! {0,3}bull )\n*|\s*$)/,
+         html: '^ {0,3}(?:' // optional indentation
+         + '<(script|pre|style)[\\s>][\\s\\S]*?(?:</\\1>[^\\n]*\\n+|$)' // (1)
+         + '|comment[^\\n]*(\\n+|$)' // (2)
+         + '|<\\?[\\s\\S]*?(?:\\?>\\n*|$)' // (3)
+         + '|<![A-Z][\\s\\S]*?(?:>\\n*|$)' // (4)
+         + '|<!\\[CDATA\\[[\\s\\S]*?(?:\\]\\]>\\n*|$)' // (5)
+         + '|</?(tag)(?: +|\\n|/?>)[\\s\\S]*?(?:(?:\\n *)+\\n|$)' // (6)
+         + '|<(?!script|pre|style)([a-z][\\w-]*)(?:attribute)*? */?>(?=[ \\t]*(?:\\n|$))[\\s\\S]*?(?:(?:\\n *)+\\n|$)' // (7) open tag
+         + '|</(?!script|pre|style)[a-z][\\w-]*\\s*>(?=[ \\t]*(?:\\n|$))[\\s\\S]*?(?:(?:\\n *)+\\n|$)' // (7) closing tag
+         + ')',
+         def: /^ {0,3}\[(label)\]: *\n? *<?([^\s>]+)>?(?:(?: +\n? *| *\n *)(title))? *(?:\n+|$)/,
+         nptable: noopTest,
+         table: noopTest,
+         lheading: /^([^\n]+)\n {0,3}(=+|-+) *(?:\n+|$)/,
+         // regex template, placeholders will be replaced according to different paragraph
+         // interruption rules of commonmark and the original markdown spec:
+         _paragraph: /^([^\n]+(?:\n(?!hr|heading|lheading|blockquote|fences|list|html| +\n)[^\n]+)*)/,
+         text: /^[^\n]+/
+       };
+       block$1._label = /(?!\s*\])(?:\\[\[\]]|[^\[\]])+/;
+       block$1._title = /(?:"(?:\\"?|[^"\\])*"|'[^'\n]*(?:\n[^'\n]+)*\n?'|\([^()]*\))/;
+       block$1.def = edit(block$1.def).replace('label', block$1._label).replace('title', block$1._title).getRegex();
+       block$1.bullet = /(?:[*+-]|\d{1,9}[.)])/;
+       block$1.item = /^( *)(bull) ?[^\n]*(?:\n(?! *bull ?)[^\n]*)*/;
+       block$1.item = edit(block$1.item, 'gm').replace(/bull/g, block$1.bullet).getRegex();
+       block$1.listItemStart = edit(/^( *)(bull) */).replace('bull', block$1.bullet).getRegex();
+       block$1.list = edit(block$1.list).replace(/bull/g, block$1.bullet).replace('hr', '\\n+(?=\\1?(?:(?:- *){3,}|(?:_ *){3,}|(?:\\* *){3,})(?:\\n+|$))').replace('def', '\\n+(?=' + block$1.def.source + ')').getRegex();
+       block$1._tag = 'address|article|aside|base|basefont|blockquote|body|caption' + '|center|col|colgroup|dd|details|dialog|dir|div|dl|dt|fieldset|figcaption' + '|figure|footer|form|frame|frameset|h[1-6]|head|header|hr|html|iframe' + '|legend|li|link|main|menu|menuitem|meta|nav|noframes|ol|optgroup|option' + '|p|param|section|source|summary|table|tbody|td|tfoot|th|thead|title|tr' + '|track|ul';
+       block$1._comment = /<!--(?!-?>)[\s\S]*?(?:-->|$)/;
+       block$1.html = edit(block$1.html, 'i').replace('comment', block$1._comment).replace('tag', block$1._tag).replace('attribute', / +[a-zA-Z:_][\w.:-]*(?: *= *"[^"\n]*"| *= *'[^'\n]*'| *= *[^\s"'=<>`]+)?/).getRegex();
+       block$1.paragraph = edit(block$1._paragraph).replace('hr', block$1.hr).replace('heading', ' {0,3}#{1,6} ').replace('|lheading', '') // setex headings don't interrupt commonmark paragraphs
+       .replace('blockquote', ' {0,3}>').replace('fences', ' {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n').replace('list', ' {0,3}(?:[*+-]|1[.)]) ') // only lists starting from 1 can interrupt
+       .replace('html', '</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|!--)').replace('tag', block$1._tag) // pars can be interrupted by type (6) html blocks
+       .getRegex();
+       block$1.blockquote = edit(block$1.blockquote).replace('paragraph', block$1.paragraph).getRegex();
+       /**
+        * Normal Block Grammar
+        */
 
-           function run() {
-             var params = timenonce(getAuth(o)),
-                 oauth_token_secret = token('oauth_token_secret'),
-                 url = options.prefix !== false ? o.url + options.path : options.path,
-                 url_parts = url.replace(/#.*$/, '').split('?', 2),
-                 base_url = url_parts[0],
-                 query = url_parts.length === 2 ? url_parts[1] : ''; // https://tools.ietf.org/html/rfc5849#section-3.4.1.3.1
+       block$1.normal = merge$1({}, block$1);
+       /**
+        * GFM Block Grammar
+        */
 
-             if ((!options.options || !options.options.header || options.options.header['Content-Type'] === 'application/x-www-form-urlencoded') && options.content) {
-               params = immutable(params, ohauth_1.stringQs(options.content));
-             }
+       block$1.gfm = merge$1({}, block$1.normal, {
+         nptable: '^ *([^|\\n ].*\\|.*)\\n' // Header
+         + ' {0,3}([-:]+ *\\|[-| :]*)' // Align
+         + '(?:\\n((?:(?!\\n|hr|heading|blockquote|code|fences|list|html).*(?:\\n|$))*)\\n*|$)',
+         // Cells
+         table: '^ *\\|(.+)\\n' // Header
+         + ' {0,3}\\|?( *[-:]+[-| :]*)' // Align
+         + '(?:\\n *((?:(?!\\n|hr|heading|blockquote|code|fences|list|html).*(?:\\n|$))*)\\n*|$)' // Cells
 
-             params.oauth_token = token('oauth_token');
-             params.oauth_signature = ohauth_1.signature(o.oauth_secret, oauth_token_secret, ohauth_1.baseString(options.method, base_url, immutable(params, ohauth_1.stringQs(query))));
-             return ohauth_1.xhr(options.method, url, params, options.content, options.options, done);
-           }
+       });
+       block$1.gfm.nptable = edit(block$1.gfm.nptable).replace('hr', block$1.hr).replace('heading', ' {0,3}#{1,6} ').replace('blockquote', ' {0,3}>').replace('code', ' {4}[^\\n]').replace('fences', ' {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n').replace('list', ' {0,3}(?:[*+-]|1[.)]) ') // only lists starting from 1 can interrupt
+       .replace('html', '</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|!--)').replace('tag', block$1._tag) // tables can be interrupted by type (6) html blocks
+       .getRegex();
+       block$1.gfm.table = edit(block$1.gfm.table).replace('hr', block$1.hr).replace('heading', ' {0,3}#{1,6} ').replace('blockquote', ' {0,3}>').replace('code', ' {4}[^\\n]').replace('fences', ' {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n').replace('list', ' {0,3}(?:[*+-]|1[.)]) ') // only lists starting from 1 can interrupt
+       .replace('html', '</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|!--)').replace('tag', block$1._tag) // tables can be interrupted by type (6) html blocks
+       .getRegex();
+       /**
+        * Pedantic grammar (original John Gruber's loose markdown specification)
+        */
 
-           function done(err, xhr) {
-             if (err) return callback(err);else if (xhr.responseXML) return callback(err, xhr.responseXML);else return callback(err, xhr.response);
-           }
-         }; // pre-authorize this object, if we can just get a token and token_secret
-         // from the start
+       block$1.pedantic = merge$1({}, block$1.normal, {
+         html: edit('^ *(?:comment *(?:\\n|\\s*$)' + '|<(tag)[\\s\\S]+?</\\1> *(?:\\n{2,}|\\s*$)' // closed tag
+         + '|<tag(?:"[^"]*"|\'[^\']*\'|\\s[^\'"/>\\s]*)*?/?> *(?:\\n{2,}|\\s*$))').replace('comment', block$1._comment).replace(/tag/g, '(?!(?:' + 'a|em|strong|small|s|cite|q|dfn|abbr|data|time|code|var|samp|kbd|sub' + '|sup|i|b|u|mark|ruby|rt|rp|bdi|bdo|span|br|wbr|ins|del|img)' + '\\b)\\w+(?!:|[^\\w\\s@]*@)\\b').getRegex(),
+         def: /^ *\[([^\]]+)\]: *<?([^\s>]+)>?(?: +(["(][^\n]+[")]))? *(?:\n+|$)/,
+         heading: /^(#{1,6})(.*)(?:\n+|$)/,
+         fences: noopTest,
+         // fences not supported
+         paragraph: edit(block$1.normal._paragraph).replace('hr', block$1.hr).replace('heading', ' *#{1,6} *[^\n]').replace('lheading', block$1.lheading).replace('blockquote', ' {0,3}>').replace('|fences', '').replace('|list', '').replace('|html', '').getRegex()
+       });
+       /**
+        * Inline-Level Grammar
+        */
 
+       var inline$1 = {
+         escape: /^\\([!"#$%&'()*+,\-./:;<=>?@\[\]\\^_`{|}~])/,
+         autolink: /^<(scheme:[^\s\x00-\x1f<>]*|email)>/,
+         url: noopTest,
+         tag: '^comment' + '|^</[a-zA-Z][\\w:-]*\\s*>' // self-closing tag
+         + '|^<[a-zA-Z][\\w-]*(?:attribute)*?\\s*/?>' // open tag
+         + '|^<\\?[\\s\\S]*?\\?>' // processing instruction, e.g. <?php ?>
+         + '|^<![a-zA-Z]+\\s[\\s\\S]*?>' // declaration, e.g. <!DOCTYPE html>
+         + '|^<!\\[CDATA\\[[\\s\\S]*?\\]\\]>',
+         // CDATA section
+         link: /^!?\[(label)\]\(\s*(href)(?:\s+(title))?\s*\)/,
+         reflink: /^!?\[(label)\]\[(?!\s*\])((?:\\[\[\]]?|[^\[\]\\])+)\]/,
+         nolink: /^!?\[(?!\s*\])((?:\[[^\[\]]*\]|\\[\[\]]|[^\[\]])*)\](?:\[\])?/,
+         reflinkSearch: 'reflink|nolink(?!\\()',
+         emStrong: {
+           lDelim: /^(?:\*+(?:([punct_])|[^\s*]))|^_+(?:([punct*])|([^\s_]))/,
+           //        (1) and (2) can only be a Right Delimiter. (3) and (4) can only be Left.  (5) and (6) can be either Left or Right.
+           //        () Skip other delimiter (1) #***                   (2) a***#, a***                   (3) #***a, ***a                 (4) ***#              (5) #***#                 (6) a***a
+           rDelimAst: /\_\_[^_*]*?\*[^_*]*?\_\_|[punct_](\*+)(?=[\s]|$)|[^punct*_\s](\*+)(?=[punct_\s]|$)|[punct_\s](\*+)(?=[^punct*_\s])|[\s](\*+)(?=[punct_])|[punct_](\*+)(?=[punct_])|[^punct*_\s](\*+)(?=[^punct*_\s])/,
+           rDelimUnd: /\*\*[^_*]*?\_[^_*]*?\*\*|[punct*](\_+)(?=[\s]|$)|[^punct*_\s](\_+)(?=[punct*\s]|$)|[punct*\s](\_+)(?=[^punct*_\s])|[\s](\_+)(?=[punct*])|[punct*](\_+)(?=[punct*])/ // ^- Not allowed for _
 
-         oauth.preauth = function (c) {
-           if (!c) return;
-           if (c.oauth_token) token('oauth_token', c.oauth_token);
-           if (c.oauth_token_secret) token('oauth_token_secret', c.oauth_token_secret);
-           return oauth;
-         };
+         },
+         code: /^(`+)([^`]|[^`][\s\S]*?[^`])\1(?!`)/,
+         br: /^( {2,}|\\)\n(?!\s*$)/,
+         del: noopTest,
+         text: /^(`+|[^`])(?:(?= {2,}\n)|[\s\S]*?(?:(?=[\\<!\[`*_]|\b_|$)|[^ ](?= {2,}\n)))/,
+         punctuation: /^([\spunctuation])/
+       }; // list of punctuation marks from CommonMark spec
+       // without * and _ to handle the different emphasis markers * and _
+
+       inline$1._punctuation = '!"#$%&\'()+\\-.,/:;<=>?@\\[\\]`^{|}~';
+       inline$1.punctuation = edit(inline$1.punctuation).replace(/punctuation/g, inline$1._punctuation).getRegex(); // sequences em should skip over [title](link), `code`, <html>
+
+       inline$1.blockSkip = /\[[^\]]*?\]\([^\)]*?\)|`[^`]*?`|<[^>]*?>/g;
+       inline$1.escapedEmSt = /\\\*|\\_/g;
+       inline$1._comment = edit(block$1._comment).replace('(?:-->|$)', '-->').getRegex();
+       inline$1.emStrong.lDelim = edit(inline$1.emStrong.lDelim).replace(/punct/g, inline$1._punctuation).getRegex();
+       inline$1.emStrong.rDelimAst = edit(inline$1.emStrong.rDelimAst, 'g').replace(/punct/g, inline$1._punctuation).getRegex();
+       inline$1.emStrong.rDelimUnd = edit(inline$1.emStrong.rDelimUnd, 'g').replace(/punct/g, inline$1._punctuation).getRegex();
+       inline$1._escapes = /\\([!"#$%&'()*+,\-./:;<=>?@\[\]\\^_`{|}~])/g;
+       inline$1._scheme = /[a-zA-Z][a-zA-Z0-9+.-]{1,31}/;
+       inline$1._email = /[a-zA-Z0-9.!#$%&'*+/=?^_`{|}~-]+(@)[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)+(?![-_])/;
+       inline$1.autolink = edit(inline$1.autolink).replace('scheme', inline$1._scheme).replace('email', inline$1._email).getRegex();
+       inline$1._attribute = /\s+[a-zA-Z:_][\w.:-]*(?:\s*=\s*"[^"]*"|\s*=\s*'[^']*'|\s*=\s*[^\s"'=<>`]+)?/;
+       inline$1.tag = edit(inline$1.tag).replace('comment', inline$1._comment).replace('attribute', inline$1._attribute).getRegex();
+       inline$1._label = /(?:\[(?:\\.|[^\[\]\\])*\]|\\.|`[^`]*`|[^\[\]\\`])*?/;
+       inline$1._href = /<(?:\\.|[^\n<>\\])+>|[^\s\x00-\x1f]*/;
+       inline$1._title = /"(?:\\"?|[^"\\])*"|'(?:\\'?|[^'\\])*'|\((?:\\\)?|[^)\\])*\)/;
+       inline$1.link = edit(inline$1.link).replace('label', inline$1._label).replace('href', inline$1._href).replace('title', inline$1._title).getRegex();
+       inline$1.reflink = edit(inline$1.reflink).replace('label', inline$1._label).getRegex();
+       inline$1.reflinkSearch = edit(inline$1.reflinkSearch, 'g').replace('reflink', inline$1.reflink).replace('nolink', inline$1.nolink).getRegex();
+       /**
+        * Normal Inline Grammar
+        */
 
-         oauth.options = function (_) {
-           if (!arguments.length) return o;
-           o = _;
-           o.url = o.url || 'https://www.openstreetmap.org';
-           o.landing = o.landing || 'land.html';
-           o.singlepage = o.singlepage || false; // Optional loading and loading-done functions for nice UI feedback.
-           // by default, no-ops
+       inline$1.normal = merge$1({}, inline$1);
+       /**
+        * Pedantic Inline Grammar
+        */
 
-           o.loading = o.loading || function () {};
+       inline$1.pedantic = merge$1({}, inline$1.normal, {
+         strong: {
+           start: /^__|\*\*/,
+           middle: /^__(?=\S)([\s\S]*?\S)__(?!_)|^\*\*(?=\S)([\s\S]*?\S)\*\*(?!\*)/,
+           endAst: /\*\*(?!\*)/g,
+           endUnd: /__(?!_)/g
+         },
+         em: {
+           start: /^_|\*/,
+           middle: /^()\*(?=\S)([\s\S]*?\S)\*(?!\*)|^_(?=\S)([\s\S]*?\S)_(?!_)/,
+           endAst: /\*(?!\*)/g,
+           endUnd: /_(?!_)/g
+         },
+         link: edit(/^!?\[(label)\]\((.*?)\)/).replace('label', inline$1._label).getRegex(),
+         reflink: edit(/^!?\[(label)\]\s*\[([^\]]*)\]/).replace('label', inline$1._label).getRegex()
+       });
+       /**
+        * GFM Inline Grammar
+        */
 
-           o.done = o.done || function () {};
+       inline$1.gfm = merge$1({}, inline$1.normal, {
+         escape: edit(inline$1.escape).replace('])', '~|])').getRegex(),
+         _extended_email: /[A-Za-z0-9._+-]+(@)[a-zA-Z0-9-_]+(?:\.[a-zA-Z0-9-_]*[a-zA-Z0-9])+(?![-_])/,
+         url: /^((?:ftp|https?):\/\/|www\.)(?:[a-zA-Z0-9\-]+\.?)+[^\s<]*|^email/,
+         _backpedal: /(?:[^?!.,:;*_~()&]+|\([^)]*\)|&(?![a-zA-Z0-9]+;$)|[?!.,:;*_~)]+(?!$))+/,
+         del: /^(~~?)(?=[^\s~])([\s\S]*?[^\s~])\1(?=[^~]|$)/,
+         text: /^([`~]+|[^`~])(?:(?= {2,}\n)|(?=[a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-]+@)|[\s\S]*?(?:(?=[\\<!\[`*~_]|\b_|https?:\/\/|ftp:\/\/|www\.|$)|[^ ](?= {2,}\n)|[^a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-](?=[a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-]+@)))/
+       });
+       inline$1.gfm.url = edit(inline$1.gfm.url, 'i').replace('email', inline$1.gfm._extended_email).getRegex();
+       /**
+        * GFM + Line Breaks Inline Grammar
+        */
 
-           return oauth.preauth(o);
-         }; // 'stamp' an authentication object from `getAuth()`
-         // with a [nonce](http://en.wikipedia.org/wiki/Cryptographic_nonce)
-         // and timestamp
+       inline$1.breaks = merge$1({}, inline$1.gfm, {
+         br: edit(inline$1.br).replace('{2,}', '*').getRegex(),
+         text: edit(inline$1.gfm.text).replace('\\b_', '\\b_| {2,}\\n').replace(/\{2,\}/g, '*').getRegex()
+       });
+       var rules = {
+         block: block$1,
+         inline: inline$1
+       };
 
+       var defaults$3 = defaults$5.defaults;
+       var block = rules.block,
+           inline = rules.inline;
+       var repeatString = helpers.repeatString;
+       /**
+        * smartypants text replacement
+        */
 
-         function timenonce(o) {
-           o.oauth_timestamp = ohauth_1.timestamp();
-           o.oauth_nonce = ohauth_1.nonce();
-           return o;
-         } // get/set tokens. These are prefixed with the base URL so that `osm-auth`
-         // can be used with multiple APIs and the keys in `localStorage`
-         // will not clash
+       function smartypants(text) {
+         return text // em-dashes
+         .replace(/---/g, "\u2014") // en-dashes
+         .replace(/--/g, "\u2013") // opening singles
+         .replace(/(^|[-\u2014/(\[{"\s])'/g, "$1\u2018") // closing singles & apostrophes
+         .replace(/'/g, "\u2019") // opening doubles
+         .replace(/(^|[-\u2014/(\[{\u2018\s])"/g, "$1\u201C") // closing doubles
+         .replace(/"/g, "\u201D") // ellipses
+         .replace(/\.{3}/g, "\u2026");
+       }
+       /**
+        * mangle email addresses
+        */
 
 
-         var token;
+       function mangle(text) {
+         var out = '',
+             i,
+             ch;
+         var l = text.length;
 
-         if (store_legacy.enabled) {
-           token = function token(x, y) {
-             if (arguments.length === 1) return store_legacy.get(o.url + x);else if (arguments.length === 2) return store_legacy.set(o.url + x, y);
-           };
-         } else {
-           var storage = {};
+         for (i = 0; i < l; i++) {
+           ch = text.charCodeAt(i);
 
-           token = function token(x, y) {
-             if (arguments.length === 1) return storage[o.url + x];else if (arguments.length === 2) return storage[o.url + x] = y;
-           };
-         } // Get an authentication object. If you just add and remove properties
-         // from a single object, you'll need to use `delete` to make sure that
-         // it doesn't contain undesired properties for authentication
+           if (Math.random() > 0.5) {
+             ch = 'x' + ch.toString(16);
+           }
 
+           out += '&#' + ch + ';';
+         }
 
-         function getAuth(o) {
-           return {
-             oauth_consumer_key: o.oauth_consumer_key,
-             oauth_signature_method: 'HMAC-SHA1'
-           };
-         } // potentially pre-authorize
+         return out;
+       }
+       /**
+        * Block Lexer
+        */
 
 
-         oauth.options(o);
-         return oauth;
-       };
+       var Lexer_1 = /*#__PURE__*/function () {
+         function Lexer(options) {
+           _classCallCheck$1(this, Lexer);
 
-       var JXON = new function () {
-         var sValueProp = 'keyValue',
-             sAttributesProp = 'keyAttributes',
-             sAttrPref = '@',
+           this.tokens = [];
+           this.tokens.links = Object.create(null);
+           this.options = options || defaults$3;
+           this.options.tokenizer = this.options.tokenizer || new Tokenizer_1();
+           this.tokenizer = this.options.tokenizer;
+           this.tokenizer.options = this.options;
+           var rules = {
+             block: block.normal,
+             inline: inline.normal
+           };
 
-         /* you can customize these values */
-         aCache = [],
-             rIsNull = /^\s*$/,
-             rIsBool = /^(?:true|false)$/i;
+           if (this.options.pedantic) {
+             rules.block = block.pedantic;
+             rules.inline = inline.pedantic;
+           } else if (this.options.gfm) {
+             rules.block = block.gfm;
 
-         function parseText(sValue) {
-           if (rIsNull.test(sValue)) {
-             return null;
+             if (this.options.breaks) {
+               rules.inline = inline.breaks;
+             } else {
+               rules.inline = inline.gfm;
+             }
            }
 
-           if (rIsBool.test(sValue)) {
-             return sValue.toLowerCase() === 'true';
-           }
+           this.tokenizer.rules = rules;
+         }
+         /**
+          * Expose Rules
+          */
 
-           if (isFinite(sValue)) {
-             return parseFloat(sValue);
-           }
 
-           if (isFinite(Date.parse(sValue))) {
-             return new Date(sValue);
+         _createClass$1(Lexer, [{
+           key: "lex",
+           value:
+           /**
+            * Preprocessing
+            */
+           function lex(src) {
+             src = src.replace(/\r\n|\r/g, '\n').replace(/\t/g, '    ');
+             this.blockTokens(src, this.tokens, true);
+             this.inline(this.tokens);
+             return this.tokens;
            }
+           /**
+            * Lexing
+            */
 
-           return sValue;
-         }
+         }, {
+           key: "blockTokens",
+           value: function blockTokens(src) {
+             var tokens = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : [];
+             var top = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : true;
 
-         function EmptyTree() {}
+             if (this.options.pedantic) {
+               src = src.replace(/^ +$/gm, '');
+             }
 
-         EmptyTree.prototype.toString = function () {
-           return 'null';
-         };
+             var token, i, l, lastToken;
 
-         EmptyTree.prototype.valueOf = function () {
-           return null;
-         };
+             while (src) {
+               // newline
+               if (token = this.tokenizer.space(src)) {
+                 src = src.substring(token.raw.length);
 
-         function objectify(vValue) {
-           return vValue === null ? new EmptyTree() : vValue instanceof Object ? vValue : new vValue.constructor(vValue);
-         }
+                 if (token.type) {
+                   tokens.push(token);
+                 }
 
-         function createObjTree(oParentNode, nVerb, bFreeze, bNesteAttr) {
-           var nLevelStart = aCache.length,
-               bChildren = oParentNode.hasChildNodes(),
-               bAttributes = oParentNode.hasAttributes(),
-               bHighVerb = Boolean(nVerb & 2);
-           var sProp,
-               vContent,
-               nLength = 0,
-               sCollectedTxt = '',
-               vResult = bHighVerb ? {} :
-           /* put here the default value for empty nodes: */
-           true;
+                 continue;
+               } // code
 
-           if (bChildren) {
-             for (var oNode, nItem = 0; nItem < oParentNode.childNodes.length; nItem++) {
-               oNode = oParentNode.childNodes.item(nItem);
 
-               if (oNode.nodeType === 4) {
-                 sCollectedTxt += oNode.nodeValue;
-               }
-               /* nodeType is 'CDATASection' (4) */
-               else if (oNode.nodeType === 3) {
-                   sCollectedTxt += oNode.nodeValue.trim();
+               if (token = this.tokenizer.code(src)) {
+                 src = src.substring(token.raw.length);
+                 lastToken = tokens[tokens.length - 1]; // An indented code block cannot interrupt a paragraph.
+
+                 if (lastToken && lastToken.type === 'paragraph') {
+                   lastToken.raw += '\n' + token.raw;
+                   lastToken.text += '\n' + token.text;
+                 } else {
+                   tokens.push(token);
                  }
-                 /* nodeType is 'Text' (3) */
-                 else if (oNode.nodeType === 1 && !oNode.prefix) {
-                     aCache.push(oNode);
-                   }
-               /* nodeType is 'Element' (1) */
 
-             }
-           }
+                 continue;
+               } // fences
+
 
-           var nLevelEnd = aCache.length,
-               vBuiltVal = parseText(sCollectedTxt);
+               if (token = this.tokenizer.fences(src)) {
+                 src = src.substring(token.raw.length);
+                 tokens.push(token);
+                 continue;
+               } // heading
 
-           if (!bHighVerb && (bChildren || bAttributes)) {
-             vResult = nVerb === 0 ? objectify(vBuiltVal) : {};
-           }
 
-           for (var nElId = nLevelStart; nElId < nLevelEnd; nElId++) {
-             sProp = aCache[nElId].nodeName.toLowerCase();
-             vContent = createObjTree(aCache[nElId], nVerb, bFreeze, bNesteAttr);
+               if (token = this.tokenizer.heading(src)) {
+                 src = src.substring(token.raw.length);
+                 tokens.push(token);
+                 continue;
+               } // table no leading pipe (gfm)
 
-             if (vResult.hasOwnProperty(sProp)) {
-               if (vResult[sProp].constructor !== Array) {
-                 vResult[sProp] = [vResult[sProp]];
-               }
 
-               vResult[sProp].push(vContent);
-             } else {
-               vResult[sProp] = vContent;
-               nLength++;
-             }
-           }
+               if (token = this.tokenizer.nptable(src)) {
+                 src = src.substring(token.raw.length);
+                 tokens.push(token);
+                 continue;
+               } // hr
 
-           if (bAttributes) {
-             var nAttrLen = oParentNode.attributes.length,
-                 sAPrefix = bNesteAttr ? '' : sAttrPref,
-                 oAttrParent = bNesteAttr ? {} : vResult;
 
-             for (var oAttrib, nAttrib = 0; nAttrib < nAttrLen; nLength++, nAttrib++) {
-               oAttrib = oParentNode.attributes.item(nAttrib);
-               oAttrParent[sAPrefix + oAttrib.name.toLowerCase()] = parseText(oAttrib.value.trim());
-             }
+               if (token = this.tokenizer.hr(src)) {
+                 src = src.substring(token.raw.length);
+                 tokens.push(token);
+                 continue;
+               } // blockquote
 
-             if (bNesteAttr) {
-               if (bFreeze) {
-                 Object.freeze(oAttrParent);
-               }
 
-               vResult[sAttributesProp] = oAttrParent;
-               nLength -= nAttrLen - 1;
-             }
-           }
+               if (token = this.tokenizer.blockquote(src)) {
+                 src = src.substring(token.raw.length);
+                 token.tokens = this.blockTokens(token.text, [], top);
+                 tokens.push(token);
+                 continue;
+               } // list
 
-           if (nVerb === 3 || (nVerb === 2 || nVerb === 1 && nLength > 0) && sCollectedTxt) {
-             vResult[sValueProp] = vBuiltVal;
-           } else if (!bHighVerb && nLength === 0 && sCollectedTxt) {
-             vResult = vBuiltVal;
-           }
 
-           if (bFreeze && (bHighVerb || nLength > 0)) {
-             Object.freeze(vResult);
-           }
+               if (token = this.tokenizer.list(src)) {
+                 src = src.substring(token.raw.length);
+                 l = token.items.length;
 
-           aCache.length = nLevelStart;
-           return vResult;
-         }
+                 for (i = 0; i < l; i++) {
+                   token.items[i].tokens = this.blockTokens(token.items[i].text, [], false);
+                 }
 
-         function loadObjTree(oXMLDoc, oParentEl, oParentObj) {
-           var vValue, oChild;
+                 tokens.push(token);
+                 continue;
+               } // html
 
-           if (oParentObj instanceof String || oParentObj instanceof Number || oParentObj instanceof Boolean) {
-             oParentEl.appendChild(oXMLDoc.createTextNode(oParentObj.toString()));
-             /* verbosity level is 0 */
-           } else if (oParentObj.constructor === Date) {
-             oParentEl.appendChild(oXMLDoc.createTextNode(oParentObj.toGMTString()));
-           }
 
-           for (var sName in oParentObj) {
-             vValue = oParentObj[sName];
+               if (token = this.tokenizer.html(src)) {
+                 src = src.substring(token.raw.length);
+                 tokens.push(token);
+                 continue;
+               } // def
 
-             if (isFinite(sName) || vValue instanceof Function) {
-               continue;
-             }
-             /* verbosity level is 0 */
 
+               if (top && (token = this.tokenizer.def(src))) {
+                 src = src.substring(token.raw.length);
 
-             if (sName === sValueProp) {
-               if (vValue !== null && vValue !== true) {
-                 oParentEl.appendChild(oXMLDoc.createTextNode(vValue.constructor === Date ? vValue.toGMTString() : String(vValue)));
-               }
-             } else if (sName === sAttributesProp) {
-               /* verbosity level is 3 */
-               for (var sAttrib in vValue) {
-                 oParentEl.setAttribute(sAttrib, vValue[sAttrib]);
-               }
-             } else if (sName.charAt(0) === sAttrPref) {
-               oParentEl.setAttribute(sName.slice(1), vValue);
-             } else if (vValue.constructor === Array) {
-               for (var nItem = 0; nItem < vValue.length; nItem++) {
-                 oChild = oXMLDoc.createElement(sName);
-                 loadObjTree(oXMLDoc, oChild, vValue[nItem]);
-                 oParentEl.appendChild(oChild);
-               }
-             } else {
-               oChild = oXMLDoc.createElement(sName);
+                 if (!this.tokens.links[token.tag]) {
+                   this.tokens.links[token.tag] = {
+                     href: token.href,
+                     title: token.title
+                   };
+                 }
 
-               if (vValue instanceof Object) {
-                 loadObjTree(oXMLDoc, oChild, vValue);
-               } else if (vValue !== null && vValue !== true) {
-                 oChild.appendChild(oXMLDoc.createTextNode(vValue.toString()));
-               }
+                 continue;
+               } // table (gfm)
 
-               oParentEl.appendChild(oChild);
-             }
-           }
-         }
 
-         this.build = function (oXMLParent, nVerbosity
-         /* optional */
-         , bFreeze
-         /* optional */
-         , bNesteAttributes
-         /* optional */
-         ) {
-           var _nVerb = arguments.length > 1 && typeof nVerbosity === 'number' ? nVerbosity & 3 :
-           /* put here the default verbosity level: */
-           1;
+               if (token = this.tokenizer.table(src)) {
+                 src = src.substring(token.raw.length);
+                 tokens.push(token);
+                 continue;
+               } // lheading
 
-           return createObjTree(oXMLParent, _nVerb, bFreeze || false, arguments.length > 3 ? bNesteAttributes : _nVerb === 3);
-         };
 
-         this.unbuild = function (oObjTree) {
-           var oNewDoc = document.implementation.createDocument('', '', null);
-           loadObjTree(oNewDoc, oNewDoc, oObjTree);
-           return oNewDoc;
-         };
+               if (token = this.tokenizer.lheading(src)) {
+                 src = src.substring(token.raw.length);
+                 tokens.push(token);
+                 continue;
+               } // top-level paragraph
 
-         this.stringify = function (oObjTree) {
-           return new XMLSerializer().serializeToString(JXON.unbuild(oObjTree));
-         };
-       }(); // var myObject = JXON.build(doc);
-       // we got our javascript object! try: alert(JSON.stringify(myObject));
-       // var newDoc = JXON.unbuild(myObject);
-       // we got our Document instance! try: alert((new XMLSerializer()).serializeToString(newDoc));
 
-       var tiler$5 = utilTiler();
-       var dispatch$6 = dispatch('apiStatusChange', 'authLoading', 'authDone', 'change', 'loading', 'loaded', 'loadedNotes');
-       var urlroot = 'https://www.openstreetmap.org';
-       var oauth = osmAuth({
-         url: urlroot,
-         oauth_consumer_key: '5A043yRSEugj4DJ5TljuapfnrflWDte8jTOcWLlT',
-         oauth_secret: 'aB3jKq1TRsCOUrfOIZ6oQMEDmv2ptV76PA54NGLL',
-         loading: authLoading,
-         done: authDone
-       }); // hardcode default block of Google Maps
+               if (top && (token = this.tokenizer.paragraph(src))) {
+                 src = src.substring(token.raw.length);
+                 tokens.push(token);
+                 continue;
+               } // text
 
-       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: {}
-       };
 
-       var _cachedApiStatus;
+               if (token = this.tokenizer.text(src)) {
+                 src = src.substring(token.raw.length);
+                 lastToken = tokens[tokens.length - 1];
 
-       var _changeset = {};
+                 if (lastToken && lastToken.type === 'text') {
+                   lastToken.raw += '\n' + token.raw;
+                   lastToken.text += '\n' + token.text;
+                 } else {
+                   tokens.push(token);
+                 }
 
-       var _deferred = new Set();
+                 continue;
+               }
 
-       var _connectionID = 1;
-       var _tileZoom$3 = 16;
-       var _noteZoom = 12;
+               if (src) {
+                 var errMsg = 'Infinite loop on byte: ' + src.charCodeAt(0);
 
-       var _rateLimitError;
+                 if (this.options.silent) {
+                   console.error(errMsg);
+                   break;
+                 } else {
+                   throw new Error(errMsg);
+                 }
+               }
+             }
 
-       var _userChangesets;
+             return tokens;
+           }
+         }, {
+           key: "inline",
+           value: function inline(tokens) {
+             var i, j, k, l2, row, token;
+             var l = tokens.length;
 
-       var _userDetails;
+             for (i = 0; i < l; i++) {
+               token = tokens[i];
 
-       var _off; // set a default but also load this from the API status
+               switch (token.type) {
+                 case 'paragraph':
+                 case 'text':
+                 case 'heading':
+                   {
+                     token.tokens = [];
+                     this.inlineTokens(token.text, token.tokens);
+                     break;
+                   }
+
+                 case 'table':
+                   {
+                     token.tokens = {
+                       header: [],
+                       cells: []
+                     }; // header
 
+                     l2 = token.header.length;
 
-       var _maxWayNodes = 2000;
+                     for (j = 0; j < l2; j++) {
+                       token.tokens.header[j] = [];
+                       this.inlineTokens(token.header[j], token.tokens.header[j]);
+                     } // cells
 
-       function authLoading() {
-         dispatch$6.call('authLoading');
-       }
 
-       function authDone() {
-         dispatch$6.call('authDone');
-       }
+                     l2 = token.cells.length;
 
-       function abortRequest$5(controllerOrXHR) {
-         if (controllerOrXHR) {
-           controllerOrXHR.abort();
-         }
-       }
+                     for (j = 0; j < l2; j++) {
+                       row = token.cells[j];
+                       token.tokens.cells[j] = [];
 
-       function hasInflightRequests(cache) {
-         return Object.keys(cache.inflight).length;
-       }
+                       for (k = 0; k < row.length; k++) {
+                         token.tokens.cells[j][k] = [];
+                         this.inlineTokens(row[k], token.tokens.cells[j][k]);
+                       }
+                     }
 
-       function abortUnwantedRequests$3(cache, visibleTiles) {
-         Object.keys(cache.inflight).forEach(function (k) {
-           if (cache.toLoad[k]) return;
-           if (visibleTiles.find(function (tile) {
-             return k === tile.id;
-           })) return;
-           abortRequest$5(cache.inflight[k]);
-           delete cache.inflight[k];
-         });
-       }
+                     break;
+                   }
 
-       function getLoc(attrs) {
-         var lon = attrs.lon && attrs.lon.value;
-         var lat = attrs.lat && attrs.lat.value;
-         return [parseFloat(lon), parseFloat(lat)];
-       }
+                 case 'blockquote':
+                   {
+                     this.inline(token.tokens);
+                     break;
+                   }
 
-       function getNodes(obj) {
-         var elems = obj.getElementsByTagName('nd');
-         var nodes = new Array(elems.length);
+                 case 'list':
+                   {
+                     l2 = token.items.length;
 
-         for (var i = 0, l = elems.length; i < l; i++) {
-           nodes[i] = 'n' + elems[i].attributes.ref.value;
-         }
+                     for (j = 0; j < l2; j++) {
+                       this.inline(token.items[j].tokens);
+                     }
 
-         return nodes;
-       }
+                     break;
+                   }
+               }
+             }
 
-       function getNodesJSON(obj) {
-         var elems = obj.nodes;
-         var nodes = new Array(elems.length);
+             return tokens;
+           }
+           /**
+            * Lexing/Compiling
+            */
 
-         for (var i = 0, l = elems.length; i < l; i++) {
-           nodes[i] = 'n' + elems[i];
-         }
+         }, {
+           key: "inlineTokens",
+           value: function inlineTokens(src) {
+             var tokens = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : [];
+             var inLink = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false;
+             var inRawBlock = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : false;
+             var token, lastToken; // String with links masked to avoid interference with em and strong
 
-         return nodes;
-       }
+             var maskedSrc = src;
+             var match;
+             var keepPrevChar, prevChar; // Mask out reflinks
 
-       function getTags(obj) {
-         var elems = obj.getElementsByTagName('tag');
-         var tags = {};
+             if (this.tokens.links) {
+               var links = Object.keys(this.tokens.links);
 
-         for (var i = 0, l = elems.length; i < l; i++) {
-           var attrs = elems[i].attributes;
-           tags[attrs.k.value] = attrs.v.value;
-         }
+               if (links.length > 0) {
+                 while ((match = this.tokenizer.rules.inline.reflinkSearch.exec(maskedSrc)) != null) {
+                   if (links.includes(match[0].slice(match[0].lastIndexOf('[') + 1, -1))) {
+                     maskedSrc = maskedSrc.slice(0, match.index) + '[' + repeatString('a', match[0].length - 2) + ']' + maskedSrc.slice(this.tokenizer.rules.inline.reflinkSearch.lastIndex);
+                   }
+                 }
+               }
+             } // Mask out other blocks
 
-         return tags;
-       }
 
-       function getMembers(obj) {
-         var elems = obj.getElementsByTagName('member');
-         var members = new Array(elems.length);
+             while ((match = this.tokenizer.rules.inline.blockSkip.exec(maskedSrc)) != null) {
+               maskedSrc = maskedSrc.slice(0, match.index) + '[' + repeatString('a', match[0].length - 2) + ']' + maskedSrc.slice(this.tokenizer.rules.inline.blockSkip.lastIndex);
+             } // Mask out escaped em & strong delimiters
 
-         for (var i = 0, l = elems.length; i < l; i++) {
-           var attrs = elems[i].attributes;
-           members[i] = {
-             id: attrs.type.value[0] + attrs.ref.value,
-             type: attrs.type.value,
-             role: attrs.role.value
-           };
-         }
 
-         return members;
-       }
+             while ((match = this.tokenizer.rules.inline.escapedEmSt.exec(maskedSrc)) != null) {
+               maskedSrc = maskedSrc.slice(0, match.index) + '++' + maskedSrc.slice(this.tokenizer.rules.inline.escapedEmSt.lastIndex);
+             }
 
-       function getMembersJSON(obj) {
-         var elems = obj.members;
-         var members = new Array(elems.length);
+             while (src) {
+               if (!keepPrevChar) {
+                 prevChar = '';
+               }
 
-         for (var i = 0, l = elems.length; i < l; i++) {
-           var attrs = elems[i];
-           members[i] = {
-             id: attrs.type[0] + attrs.ref,
-             type: attrs.type,
-             role: attrs.role
-           };
-         }
+               keepPrevChar = false; // escape
 
-         return members;
-       }
+               if (token = this.tokenizer.escape(src)) {
+                 src = src.substring(token.raw.length);
+                 tokens.push(token);
+                 continue;
+               } // tag
 
-       function getVisible(attrs) {
-         return !attrs.visible || attrs.visible.value !== 'false';
-       }
 
-       function parseComments(comments) {
-         var parsedComments = []; // for each comment
+               if (token = this.tokenizer.tag(src, inLink, inRawBlock)) {
+                 src = src.substring(token.raw.length);
+                 inLink = token.inLink;
+                 inRawBlock = token.inRawBlock;
+                 var _lastToken = tokens[tokens.length - 1];
 
-         for (var i = 0; i < comments.length; i++) {
-           var comment = comments[i];
+                 if (_lastToken && token.type === 'text' && _lastToken.type === 'text') {
+                   _lastToken.raw += token.raw;
+                   _lastToken.text += token.text;
+                 } else {
+                   tokens.push(token);
+                 }
 
-           if (comment.nodeName === 'comment') {
-             var childNodes = comment.childNodes;
-             var parsedComment = {};
+                 continue;
+               } // link
 
-             for (var j = 0; j < childNodes.length; j++) {
-               var node = childNodes[j];
-               var nodeName = node.nodeName;
-               if (nodeName === '#text') continue;
-               parsedComment[nodeName] = node.textContent;
 
-               if (nodeName === 'uid') {
-                 var uid = node.textContent;
+               if (token = this.tokenizer.link(src)) {
+                 src = src.substring(token.raw.length);
 
-                 if (uid && !_userCache.user[uid]) {
-                   _userCache.toLoad[uid] = true;
+                 if (token.type === 'link') {
+                   token.tokens = this.inlineTokens(token.text, [], true, inRawBlock);
                  }
-               }
-             }
 
-             if (parsedComment) {
-               parsedComments.push(parsedComment);
-             }
-           }
-         }
+                 tokens.push(token);
+                 continue;
+               } // reflink, nolink
 
-         return parsedComments;
-       }
 
-       function encodeNoteRtree(note) {
-         return {
-           minX: note.loc[0],
-           minY: note.loc[1],
-           maxX: note.loc[0],
-           maxY: note.loc[1],
-           data: note
-         };
-       }
+               if (token = this.tokenizer.reflink(src, this.tokens.links)) {
+                 src = src.substring(token.raw.length);
+                 var _lastToken2 = tokens[tokens.length - 1];
 
-       var jsonparsers = {
-         node: function nodeData(obj, uid) {
-           return new osmNode({
-             id: uid,
-             visible: typeof obj.visible === 'boolean' ? obj.visible : true,
-             version: obj.version && obj.version.toString(),
-             changeset: obj.changeset && obj.changeset.toString(),
-             timestamp: obj.timestamp,
-             user: obj.user,
-             uid: obj.uid && obj.uid.toString(),
-             loc: [parseFloat(obj.lon), parseFloat(obj.lat)],
-             tags: obj.tags
-           });
-         },
-         way: function wayData(obj, uid) {
-           return new osmWay({
-             id: uid,
-             visible: typeof obj.visible === 'boolean' ? obj.visible : true,
-             version: obj.version && obj.version.toString(),
-             changeset: obj.changeset && obj.changeset.toString(),
-             timestamp: obj.timestamp,
-             user: obj.user,
-             uid: obj.uid && obj.uid.toString(),
-             tags: obj.tags,
-             nodes: getNodesJSON(obj)
-           });
-         },
-         relation: function relationData(obj, uid) {
-           return new osmRelation({
-             id: uid,
-             visible: typeof obj.visible === 'boolean' ? obj.visible : true,
-             version: obj.version && obj.version.toString(),
-             changeset: obj.changeset && obj.changeset.toString(),
-             timestamp: obj.timestamp,
-             user: obj.user,
-             uid: obj.uid && obj.uid.toString(),
-             tags: obj.tags,
-             members: getMembersJSON(obj)
-           });
-         }
-       };
+                 if (token.type === 'link') {
+                   token.tokens = this.inlineTokens(token.text, [], true, inRawBlock);
+                   tokens.push(token);
+                 } else if (_lastToken2 && token.type === 'text' && _lastToken2.type === 'text') {
+                   _lastToken2.raw += token.raw;
+                   _lastToken2.text += token.text;
+                 } else {
+                   tokens.push(token);
+                 }
+
+                 continue;
+               } // em & strong
+
+
+               if (token = this.tokenizer.emStrong(src, maskedSrc, prevChar)) {
+                 src = src.substring(token.raw.length);
+                 token.tokens = this.inlineTokens(token.text, [], inLink, inRawBlock);
+                 tokens.push(token);
+                 continue;
+               } // code
 
-       function parseJSON(payload, callback, options) {
-         options = Object.assign({
-           skipSeen: true
-         }, options);
 
-         if (!payload) {
-           return callback({
-             message: 'No JSON',
-             status: -1
-           });
-         }
+               if (token = this.tokenizer.codespan(src)) {
+                 src = src.substring(token.raw.length);
+                 tokens.push(token);
+                 continue;
+               } // br
 
-         var json = payload;
-         if (_typeof(json) !== 'object') json = JSON.parse(payload);
-         if (!json.elements) return callback({
-           message: 'No JSON',
-           status: -1
-         });
-         var children = json.elements;
-         var handle = window.requestIdleCallback(function () {
-           var results = [];
-           var result;
 
-           for (var i = 0; i < children.length; i++) {
-             result = parseChild(children[i]);
-             if (result) results.push(result);
-           }
+               if (token = this.tokenizer.br(src)) {
+                 src = src.substring(token.raw.length);
+                 tokens.push(token);
+                 continue;
+               } // del (gfm)
 
-           callback(null, results);
-         });
 
-         _deferred.add(handle);
+               if (token = this.tokenizer.del(src)) {
+                 src = src.substring(token.raw.length);
+                 token.tokens = this.inlineTokens(token.text, [], inLink, inRawBlock);
+                 tokens.push(token);
+                 continue;
+               } // autolink
 
-         function parseChild(child) {
-           var parser = jsonparsers[child.type];
-           if (!parser) return null;
-           var uid;
-           uid = osmEntity.id.fromOSM(child.type, child.id);
 
-           if (options.skipSeen) {
-             if (_tileCache.seen[uid]) return null; // avoid reparsing a "seen" entity
+               if (token = this.tokenizer.autolink(src, mangle)) {
+                 src = src.substring(token.raw.length);
+                 tokens.push(token);
+                 continue;
+               } // url (gfm)
 
-             _tileCache.seen[uid] = true;
-           }
 
-           return parser(child, uid);
-         }
-       }
+               if (!inLink && (token = this.tokenizer.url(src, mangle))) {
+                 src = src.substring(token.raw.length);
+                 tokens.push(token);
+                 continue;
+               } // text
 
-       var parsers = {
-         node: function nodeData(obj, uid) {
-           var attrs = obj.attributes;
-           return new osmNode({
-             id: uid,
-             visible: getVisible(attrs),
-             version: attrs.version.value,
-             changeset: attrs.changeset && attrs.changeset.value,
-             timestamp: attrs.timestamp && attrs.timestamp.value,
-             user: attrs.user && attrs.user.value,
-             uid: attrs.uid && attrs.uid.value,
-             loc: getLoc(attrs),
-             tags: getTags(obj)
-           });
-         },
-         way: function wayData(obj, uid) {
-           var attrs = obj.attributes;
-           return new osmWay({
-             id: uid,
-             visible: getVisible(attrs),
-             version: attrs.version.value,
-             changeset: attrs.changeset && attrs.changeset.value,
-             timestamp: attrs.timestamp && attrs.timestamp.value,
-             user: attrs.user && attrs.user.value,
-             uid: attrs.uid && attrs.uid.value,
-             tags: getTags(obj),
-             nodes: getNodes(obj)
-           });
-         },
-         relation: function relationData(obj, uid) {
-           var attrs = obj.attributes;
-           return new osmRelation({
-             id: uid,
-             visible: getVisible(attrs),
-             version: attrs.version.value,
-             changeset: attrs.changeset && attrs.changeset.value,
-             timestamp: attrs.timestamp && attrs.timestamp.value,
-             user: attrs.user && attrs.user.value,
-             uid: attrs.uid && attrs.uid.value,
-             tags: getTags(obj),
-             members: getMembers(obj)
-           });
-         },
-         note: function parseNote(obj, uid) {
-           var attrs = obj.attributes;
-           var childNodes = obj.childNodes;
-           var props = {};
-           props.id = uid;
-           props.loc = getLoc(attrs); // if notes are coincident, move them apart slightly
 
-           var coincident = false;
-           var epsilon = 0.00001;
+               if (token = this.tokenizer.inlineText(src, inRawBlock, smartypants)) {
+                 src = src.substring(token.raw.length);
 
-           do {
-             if (coincident) {
-               props.loc = geoVecAdd(props.loc, [epsilon, epsilon]);
-             }
+                 if (token.raw.slice(-1) !== '_') {
+                   // Track prevChar before string of ____ started
+                   prevChar = token.raw.slice(-1);
+                 }
 
-             var bbox = geoExtent(props.loc).bbox();
-             coincident = _noteCache.rtree.search(bbox).length;
-           } while (coincident); // parse note contents
+                 keepPrevChar = true;
+                 lastToken = tokens[tokens.length - 1];
 
+                 if (lastToken && lastToken.type === 'text') {
+                   lastToken.raw += token.raw;
+                   lastToken.text += token.text;
+                 } else {
+                   tokens.push(token);
+                 }
 
-           for (var i = 0; i < childNodes.length; i++) {
-             var node = childNodes[i];
-             var nodeName = node.nodeName;
-             if (nodeName === '#text') continue; // if the element is comments, parse the comments
+                 continue;
+               }
 
-             if (nodeName === 'comments') {
-               props[nodeName] = parseComments(node.childNodes);
-             } else {
-               props[nodeName] = node.textContent;
+               if (src) {
+                 var errMsg = 'Infinite loop on byte: ' + src.charCodeAt(0);
+
+                 if (this.options.silent) {
+                   console.error(errMsg);
+                   break;
+                 } else {
+                   throw new Error(errMsg);
+                 }
+               }
              }
+
+             return tokens;
+           }
+         }], [{
+           key: "rules",
+           get: function get() {
+             return {
+               block: block,
+               inline: inline
+             };
            }
+           /**
+            * Static Lex Method
+            */
 
-           var note = new osmNote(props);
-           var item = encodeNoteRtree(note);
-           _noteCache.note[note.id] = note;
+         }, {
+           key: "lex",
+           value: function lex(src, options) {
+             var lexer = new Lexer(options);
+             return lexer.lex(src);
+           }
+           /**
+            * Static Lex Inline Method
+            */
 
-           _noteCache.rtree.insert(item);
+         }, {
+           key: "lexInline",
+           value: function lexInline(src, options) {
+             var lexer = new Lexer(options);
+             return lexer.inlineTokens(src);
+           }
+         }]);
 
-           return note;
-         },
-         user: function parseUser(obj, uid) {
-           var attrs = obj.attributes;
-           var user = {
-             id: uid,
-             display_name: attrs.display_name && attrs.display_name.value,
-             account_created: attrs.account_created && attrs.account_created.value,
-             changesets_count: '0',
-             active_blocks: '0'
-           };
-           var img = obj.getElementsByTagName('img');
+         return Lexer;
+       }();
 
-           if (img && img[0] && img[0].getAttribute('href')) {
-             user.image_url = img[0].getAttribute('href');
-           }
+       var defaults$2 = defaults$5.defaults;
+       var cleanUrl = helpers.cleanUrl,
+           escape$2 = helpers.escape;
+       /**
+        * Renderer
+        */
 
-           var changesets = obj.getElementsByTagName('changesets');
+       var Renderer_1 = /*#__PURE__*/function () {
+         function Renderer(options) {
+           _classCallCheck$1(this, Renderer);
 
-           if (changesets && changesets[0] && changesets[0].getAttribute('count')) {
-             user.changesets_count = changesets[0].getAttribute('count');
-           }
+           this.options = options || defaults$2;
+         }
 
-           var blocks = obj.getElementsByTagName('blocks');
+         _createClass$1(Renderer, [{
+           key: "code",
+           value: function code(_code, infostring, escaped) {
+             var lang = (infostring || '').match(/\S*/)[0];
 
-           if (blocks && blocks[0]) {
-             var received = blocks[0].getElementsByTagName('received');
+             if (this.options.highlight) {
+               var out = this.options.highlight(_code, lang);
 
-             if (received && received[0] && received[0].getAttribute('active')) {
-               user.active_blocks = received[0].getAttribute('active');
+               if (out != null && out !== _code) {
+                 escaped = true;
+                 _code = out;
+               }
              }
-           }
 
-           _userCache.user[uid] = user;
-           delete _userCache.toLoad[uid];
-           return user;
-         }
-       };
+             _code = _code.replace(/\n$/, '') + '\n';
 
-       function parseXML(xml, callback, options) {
-         options = Object.assign({
-           skipSeen: true
-         }, options);
+             if (!lang) {
+               return '<pre><code>' + (escaped ? _code : escape$2(_code, true)) + '</code></pre>\n';
+             }
 
-         if (!xml || !xml.childNodes) {
-           return callback({
-             message: 'No XML',
-             status: -1
-           });
-         }
+             return '<pre><code class="' + this.options.langPrefix + escape$2(lang, true) + '">' + (escaped ? _code : escape$2(_code, true)) + '</code></pre>\n';
+           }
+         }, {
+           key: "blockquote",
+           value: function blockquote(quote) {
+             return '<blockquote>\n' + quote + '</blockquote>\n';
+           }
+         }, {
+           key: "html",
+           value: function html(_html) {
+             return _html;
+           }
+         }, {
+           key: "heading",
+           value: function heading(text, level, raw, slugger) {
+             if (this.options.headerIds) {
+               return '<h' + level + ' id="' + this.options.headerPrefix + slugger.slug(raw) + '">' + text + '</h' + level + '>\n';
+             } // ignore IDs
 
-         var root = xml.childNodes[0];
-         var children = root.childNodes;
-         var handle = window.requestIdleCallback(function () {
-           var results = [];
-           var result;
 
-           for (var i = 0; i < children.length; i++) {
-             result = parseChild(children[i]);
-             if (result) results.push(result);
+             return '<h' + level + '>' + text + '</h' + level + '>\n';
+           }
+         }, {
+           key: "hr",
+           value: function hr() {
+             return this.options.xhtml ? '<hr/>\n' : '<hr>\n';
+           }
+         }, {
+           key: "list",
+           value: function list(body, ordered, start) {
+             var type = ordered ? 'ol' : 'ul',
+                 startatt = ordered && start !== 1 ? ' start="' + start + '"' : '';
+             return '<' + type + startatt + '>\n' + body + '</' + type + '>\n';
+           }
+         }, {
+           key: "listitem",
+           value: function listitem(text) {
+             return '<li>' + text + '</li>\n';
+           }
+         }, {
+           key: "checkbox",
+           value: function checkbox(checked) {
+             return '<input ' + (checked ? 'checked="" ' : '') + 'disabled="" type="checkbox"' + (this.options.xhtml ? ' /' : '') + '> ';
+           }
+         }, {
+           key: "paragraph",
+           value: function paragraph(text) {
+             return '<p>' + text + '</p>\n';
+           }
+         }, {
+           key: "table",
+           value: function table(header, body) {
+             if (body) body = '<tbody>' + body + '</tbody>';
+             return '<table>\n' + '<thead>\n' + header + '</thead>\n' + body + '</table>\n';
+           }
+         }, {
+           key: "tablerow",
+           value: function tablerow(content) {
+             return '<tr>\n' + content + '</tr>\n';
            }
+         }, {
+           key: "tablecell",
+           value: function tablecell(content, flags) {
+             var type = flags.header ? 'th' : 'td';
+             var tag = flags.align ? '<' + type + ' align="' + flags.align + '">' : '<' + type + '>';
+             return tag + content + '</' + type + '>\n';
+           } // span level renderer
 
-           callback(null, results);
-         });
+         }, {
+           key: "strong",
+           value: function strong(text) {
+             return '<strong>' + text + '</strong>';
+           }
+         }, {
+           key: "em",
+           value: function em(text) {
+             return '<em>' + text + '</em>';
+           }
+         }, {
+           key: "codespan",
+           value: function codespan(text) {
+             return '<code>' + text + '</code>';
+           }
+         }, {
+           key: "br",
+           value: function br() {
+             return this.options.xhtml ? '<br/>' : '<br>';
+           }
+         }, {
+           key: "del",
+           value: function del(text) {
+             return '<del>' + text + '</del>';
+           }
+         }, {
+           key: "link",
+           value: function link(href, title, text) {
+             href = cleanUrl(this.options.sanitize, this.options.baseUrl, href);
 
-         _deferred.add(handle);
+             if (href === null) {
+               return text;
+             }
 
-         function parseChild(child) {
-           var parser = parsers[child.nodeName];
-           if (!parser) return null;
-           var uid;
+             var out = '<a href="' + escape$2(href) + '"';
 
-           if (child.nodeName === 'user') {
-             uid = child.attributes.id.value;
+             if (title) {
+               out += ' title="' + title + '"';
+             }
 
-             if (options.skipSeen && _userCache.user[uid]) {
-               delete _userCache.toLoad[uid];
-               return null;
+             out += '>' + text + '</a>';
+             return out;
+           }
+         }, {
+           key: "image",
+           value: function image(href, title, text) {
+             href = cleanUrl(this.options.sanitize, this.options.baseUrl, href);
+
+             if (href === null) {
+               return text;
              }
-           } else if (child.nodeName === 'note') {
-             uid = child.getElementsByTagName('id')[0].textContent;
-           } else {
-             uid = osmEntity.id.fromOSM(child.nodeName, child.attributes.id.value);
 
-             if (options.skipSeen) {
-               if (_tileCache.seen[uid]) return null; // avoid reparsing a "seen" entity
+             var out = '<img src="' + href + '" alt="' + text + '"';
 
-               _tileCache.seen[uid] = true;
+             if (title) {
+               out += ' title="' + title + '"';
              }
+
+             out += this.options.xhtml ? '/>' : '>';
+             return out;
+           }
+         }, {
+           key: "text",
+           value: function text(_text) {
+             return _text;
            }
+         }]);
 
-           return parser(child, uid);
+         return Renderer;
+       }();
+
+       /**
+        * TextRenderer
+        * returns only the textual part of the token
+        */
+       var TextRenderer_1 = /*#__PURE__*/function () {
+         function TextRenderer() {
+           _classCallCheck$1(this, TextRenderer);
          }
-       } // replace or remove note from rtree
 
+         _createClass$1(TextRenderer, [{
+           key: "strong",
+           value: // no need for block level renderers
+           function strong(text) {
+             return text;
+           }
+         }, {
+           key: "em",
+           value: function em(text) {
+             return text;
+           }
+         }, {
+           key: "codespan",
+           value: function codespan(text) {
+             return text;
+           }
+         }, {
+           key: "del",
+           value: function del(text) {
+             return text;
+           }
+         }, {
+           key: "html",
+           value: function html(text) {
+             return text;
+           }
+         }, {
+           key: "text",
+           value: function text(_text) {
+             return _text;
+           }
+         }, {
+           key: "link",
+           value: function link(href, title, text) {
+             return '' + text;
+           }
+         }, {
+           key: "image",
+           value: function image(href, title, text) {
+             return '' + text;
+           }
+         }, {
+           key: "br",
+           value: function br() {
+             return '';
+           }
+         }]);
+
+         return TextRenderer;
+       }();
 
-       function updateRtree$3(item, replace) {
-         _noteCache.rtree.remove(item, function isEql(a, b) {
-           return a.data.id === b.data.id;
-         });
+       /**
+        * Slugger generates header id
+        */
+       var Slugger_1 = /*#__PURE__*/function () {
+         function Slugger() {
+           _classCallCheck$1(this, Slugger);
 
-         if (replace) {
-           _noteCache.rtree.insert(item);
+           this.seen = {};
          }
-       }
 
-       function wrapcb(thisArg, callback, cid) {
-         return function (err, result) {
-           if (err) {
-             // 400 Bad Request, 401 Unauthorized, 403 Forbidden..
-             if (err.status === 400 || err.status === 401 || err.status === 403) {
-               thisArg.logout();
+         _createClass$1(Slugger, [{
+           key: "serialize",
+           value: function serialize(value) {
+             return value.toLowerCase().trim() // remove html tags
+             .replace(/<[!\/a-z].*?>/ig, '') // remove unwanted chars
+             .replace(/[\u2000-\u206F\u2E00-\u2E7F\\'!"#$%&()*+,./:;<=>?@[\]^`{|}~]/g, '').replace(/\s/g, '-');
+           }
+           /**
+            * Finds the next safe (unique) slug to use
+            */
+
+         }, {
+           key: "getNextSafeSlug",
+           value: function getNextSafeSlug(originalSlug, isDryRun) {
+             var slug = originalSlug;
+             var occurenceAccumulator = 0;
+
+             if (this.seen.hasOwnProperty(slug)) {
+               occurenceAccumulator = this.seen[originalSlug];
+
+               do {
+                 occurenceAccumulator++;
+                 slug = originalSlug + '-' + occurenceAccumulator;
+               } while (this.seen.hasOwnProperty(slug));
              }
 
-             return callback.call(thisArg, err);
-           } else if (thisArg.getConnectionId() !== cid) {
-             return callback.call(thisArg, {
-               message: 'Connection Switched',
-               status: -1
-             });
-           } else {
-             return callback.call(thisArg, err, result);
+             if (!isDryRun) {
+               this.seen[originalSlug] = occurenceAccumulator;
+               this.seen[slug] = 0;
+             }
+
+             return slug;
+           }
+           /**
+            * Convert string to unique id
+            * @param {object} options
+            * @param {boolean} options.dryrun Generates the next unique slug without updating the internal accumulator.
+            */
+
+         }, {
+           key: "slug",
+           value: function slug(value) {
+             var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
+             var slug = this.serialize(value);
+             return this.getNextSafeSlug(slug, options.dryrun);
            }
-         };
-       }
+         }]);
 
-       var serviceOsm = {
-         init: function init() {
-           utilRebind(this, dispatch$6, 'on');
-         },
-         reset: function reset() {
-           Array.from(_deferred).forEach(function (handle) {
-             window.cancelIdleCallback(handle);
+         return Slugger;
+       }();
 
-             _deferred["delete"](handle);
-           });
-           _connectionID++;
-           _userChangesets = undefined;
-           _userDetails = undefined;
-           _rateLimitError = undefined;
-           Object.values(_tileCache.inflight).forEach(abortRequest$5);
-           Object.values(_noteCache.inflight).forEach(abortRequest$5);
-           Object.values(_noteCache.inflightPost).forEach(abortRequest$5);
-           if (_changeset.inflight) abortRequest$5(_changeset.inflight);
-           _tileCache = {
-             toLoad: {},
-             loaded: {},
-             inflight: {},
-             seen: {},
-             rtree: new RBush()
-           };
-           _noteCache = {
-             toLoad: {},
-             loaded: {},
-             inflight: {},
-             inflightPost: {},
-             note: {},
-             closed: {},
-             rtree: new RBush()
-           };
-           _userCache = {
-             toLoad: {},
-             user: {}
-           };
-           _cachedApiStatus = undefined;
-           _changeset = {};
-           return this;
-         },
-         getConnectionId: function getConnectionId() {
-           return _connectionID;
-         },
-         changesetURL: function changesetURL(changesetID) {
-           return urlroot + '/changeset/' + changesetID;
-         },
-         changesetsURL: function changesetsURL(center, zoom) {
-           var precision = Math.max(0, Math.ceil(Math.log(zoom) / Math.LN2));
-           return urlroot + '/history#map=' + Math.floor(zoom) + '/' + center[1].toFixed(precision) + '/' + center[0].toFixed(precision);
-         },
-         entityURL: function entityURL(entity) {
-           return urlroot + '/' + entity.type + '/' + entity.osmId();
-         },
-         historyURL: function historyURL(entity) {
-           return urlroot + '/' + entity.type + '/' + entity.osmId() + '/history';
-         },
-         userURL: function userURL(username) {
-           return urlroot + '/user/' + username;
-         },
-         noteURL: function noteURL(note) {
-           return urlroot + '/note/' + note.id;
-         },
-         noteReportURL: function noteReportURL(note) {
-           return urlroot + '/reports/new?reportable_type=Note&reportable_id=' + note.id;
-         },
-         // Generic method to load data from the OSM API
-         // Can handle either auth or unauth calls.
-         loadFromAPI: function loadFromAPI(path, callback, options) {
-           options = Object.assign({
-             skipSeen: true
-           }, options);
-           var that = this;
-           var cid = _connectionID;
+       var defaults$1 = defaults$5.defaults;
+       var unescape$1 = helpers.unescape;
+       /**
+        * Parsing & Compiling
+        */
 
-           function done(err, payload) {
-             if (that.getConnectionId() !== cid) {
-               if (callback) callback({
-                 message: 'Connection Switched',
-                 status: -1
-               });
-               return;
-             }
+       var Parser_1 = /*#__PURE__*/function () {
+         function Parser(options) {
+           _classCallCheck$1(this, Parser);
 
-             var isAuthenticated = that.authenticated(); // 400 Bad Request, 401 Unauthorized, 403 Forbidden
-             // Logout and retry the request..
+           this.options = options || defaults$1;
+           this.options.renderer = this.options.renderer || new Renderer_1();
+           this.renderer = this.options.renderer;
+           this.renderer.options = this.options;
+           this.textRenderer = new TextRenderer_1();
+           this.slugger = new Slugger_1();
+         }
+         /**
+          * Static Parse Method
+          */
 
-             if (isAuthenticated && err && err.status && (err.status === 400 || err.status === 401 || err.status === 403)) {
-               that.logout();
-               that.loadFromAPI(path, callback, options); // else, no retry..
-             } else {
-               // 509 Bandwidth Limit Exceeded, 429 Too Many Requests
-               // Set the rateLimitError flag and trigger a warning..
-               if (!isAuthenticated && !_rateLimitError && err && err.status && (err.status === 509 || err.status === 429)) {
-                 _rateLimitError = err;
-                 dispatch$6.call('change');
-                 that.reloadApiStatus();
-               } else if (err && _cachedApiStatus === 'online' || !err && _cachedApiStatus !== 'online') {
-                 // If the response's error state doesn't match the status,
-                 // it's likely we lost or gained the connection so reload the status
-                 that.reloadApiStatus();
-               }
 
-               if (callback) {
-                 if (err) {
-                   return callback(err);
-                 } else {
-                   if (path.indexOf('.json') !== -1) {
-                     return parseJSON(payload, callback, options);
-                   } else {
-                     return parseXML(payload, callback, options);
-                   }
-                 }
-               }
-             }
-           }
+         _createClass$1(Parser, [{
+           key: "parse",
+           value:
+           /**
+            * Parse Loop
+            */
+           function parse(tokens) {
+             var top = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : true;
+             var out = '',
+                 i,
+                 j,
+                 k,
+                 l2,
+                 l3,
+                 row,
+                 cell,
+                 header,
+                 body,
+                 token,
+                 ordered,
+                 start,
+                 loose,
+                 itemBody,
+                 item,
+                 checked,
+                 task,
+                 checkbox;
+             var l = tokens.length;
 
-           if (this.authenticated()) {
-             return oauth.xhr({
-               method: 'GET',
-               path: path
-             }, done);
-           } else {
-             var url = urlroot + path;
-             var controller = new AbortController();
-             d3_json(url, {
-               signal: controller.signal
-             }).then(function (data) {
-               done(null, data);
-             })["catch"](function (err) {
-               if (err.name === 'AbortError') return; // d3-fetch includes status in the error message,
-               // but we can't access the response itself
-               // https://github.com/d3/d3-fetch/issues/27
+             for (i = 0; i < l; i++) {
+               token = tokens[i];
 
-               var match = err.message.match(/^\d{3}/);
+               switch (token.type) {
+                 case 'space':
+                   {
+                     continue;
+                   }
 
-               if (match) {
-                 done({
-                   status: +match[0],
-                   statusText: err.message
-                 });
-               } else {
-                 done(err.message);
-               }
-             });
-             return controller;
-           }
-         },
-         // Load a single entity by id (ways and relations use the `/full` call)
-         // GET /api/0.6/node/#id
-         // GET /api/0.6/[way|relation]/#id/full
-         loadEntity: function loadEntity(id, callback) {
-           var type = osmEntity.id.type(id);
-           var osmID = osmEntity.id.toOSM(id);
-           var options = {
-             skipSeen: false
-           };
-           this.loadFromAPI('/api/0.6/' + type + '/' + osmID + (type !== 'node' ? '/full' : '') + '.json', function (err, entities) {
-             if (callback) callback(err, {
-               data: entities
-             });
-           }, options);
-         },
-         // Load a single entity with a specific version
-         // GET /api/0.6/[node|way|relation]/#id/#version
-         loadEntityVersion: function loadEntityVersion(id, version, callback) {
-           var type = osmEntity.id.type(id);
-           var osmID = osmEntity.id.toOSM(id);
-           var options = {
-             skipSeen: false
-           };
-           this.loadFromAPI('/api/0.6/' + type + '/' + osmID + '/' + version + '.json', function (err, entities) {
-             if (callback) callback(err, {
-               data: entities
-             });
-           }, options);
-         },
-         // Load multiple entities in chunks
-         // (note: callback may be called multiple times)
-         // Unlike `loadEntity`, child nodes and members are not fetched
-         // GET /api/0.6/[nodes|ways|relations]?#parameters
-         loadMultiple: function loadMultiple(ids, callback) {
-           var that = this;
-           var groups = utilArrayGroupBy(utilArrayUniq(ids), osmEntity.id.type);
-           Object.keys(groups).forEach(function (k) {
-             var type = k + 's'; // nodes, ways, relations
+                 case 'hr':
+                   {
+                     out += this.renderer.hr();
+                     continue;
+                   }
 
-             var osmIDs = groups[k].map(function (id) {
-               return osmEntity.id.toOSM(id);
-             });
-             var options = {
-               skipSeen: false
-             };
-             utilArrayChunk(osmIDs, 150).forEach(function (arr) {
-               that.loadFromAPI('/api/0.6/' + type + '.json?' + type + '=' + arr.join(), function (err, entities) {
-                 if (callback) callback(err, {
-                   data: entities
-                 });
-               }, options);
-             });
-           });
-         },
-         // Create, upload, and close a changeset
-         // PUT /api/0.6/changeset/create
-         // POST /api/0.6/changeset/#id/upload
-         // PUT /api/0.6/changeset/#id/close
-         putChangeset: function putChangeset(changeset, changes, callback) {
-           var cid = _connectionID;
+                 case 'heading':
+                   {
+                     out += this.renderer.heading(this.parseInline(token.tokens), token.depth, unescape$1(this.parseInline(token.tokens, this.textRenderer)), this.slugger);
+                     continue;
+                   }
 
-           if (_changeset.inflight) {
-             return callback({
-               message: 'Changeset already inflight',
-               status: -2
-             }, changeset);
-           } else if (_changeset.open) {
-             // reuse existing open changeset..
-             return createdChangeset.call(this, null, _changeset.open);
-           } else {
-             // Open a new changeset..
-             var options = {
-               method: 'PUT',
-               path: '/api/0.6/changeset/create',
-               options: {
-                 header: {
-                   'Content-Type': 'text/xml'
-                 }
-               },
-               content: JXON.stringify(changeset.asJXON())
-             };
-             _changeset.inflight = oauth.xhr(options, wrapcb(this, createdChangeset, cid));
-           }
+                 case 'code':
+                   {
+                     out += this.renderer.code(token.text, token.lang, token.escaped);
+                     continue;
+                   }
 
-           function createdChangeset(err, changesetID) {
-             _changeset.inflight = null;
+                 case 'table':
+                   {
+                     header = ''; // header
 
-             if (err) {
-               return callback(err, changeset);
-             }
+                     cell = '';
+                     l2 = token.header.length;
 
-             _changeset.open = changesetID;
-             changeset = changeset.update({
-               id: changesetID
-             }); // Upload the changeset..
+                     for (j = 0; j < l2; j++) {
+                       cell += this.renderer.tablecell(this.parseInline(token.tokens.header[j]), {
+                         header: true,
+                         align: token.align[j]
+                       });
+                     }
 
-             var options = {
-               method: 'POST',
-               path: '/api/0.6/changeset/' + changesetID + '/upload',
-               options: {
-                 header: {
-                   'Content-Type': 'text/xml'
-                 }
-               },
-               content: JXON.stringify(changeset.osmChangeJXON(changes))
-             };
-             _changeset.inflight = oauth.xhr(options, wrapcb(this, uploadedChangeset, cid));
-           }
+                     header += this.renderer.tablerow(cell);
+                     body = '';
+                     l2 = token.cells.length;
 
-           function uploadedChangeset(err) {
-             _changeset.inflight = null;
-             if (err) return callback(err, changeset); // Upload was successful, safe to call the callback.
-             // Add delay to allow for postgres replication #1646 #2678
+                     for (j = 0; j < l2; j++) {
+                       row = token.tokens.cells[j];
+                       cell = '';
+                       l3 = row.length;
 
-             window.setTimeout(function () {
-               callback(null, changeset);
-             }, 2500);
-             _changeset.open = null; // At this point, we don't really care if the connection was switched..
-             // Only try to close the changeset if we're still talking to the same server.
+                       for (k = 0; k < l3; k++) {
+                         cell += this.renderer.tablecell(this.parseInline(row[k]), {
+                           header: false,
+                           align: token.align[k]
+                         });
+                       }
 
-             if (this.getConnectionId() === cid) {
-               // Still attempt to close changeset, but ignore response because #2667
-               oauth.xhr({
-                 method: 'PUT',
-                 path: '/api/0.6/changeset/' + changeset.id + '/close',
-                 options: {
-                   header: {
-                     'Content-Type': 'text/xml'
+                       body += this.renderer.tablerow(cell);
+                     }
+
+                     out += this.renderer.table(header, body);
+                     continue;
                    }
-                 }
-               }, function () {
-                 return true;
-               });
-             }
-           }
-         },
-         // Load multiple users in chunks
-         // (note: callback may be called multiple times)
-         // GET /api/0.6/users?users=#id1,#id2,...,#idn
-         loadUsers: function loadUsers(uids, callback) {
-           var toLoad = [];
-           var cached = [];
-           utilArrayUniq(uids).forEach(function (uid) {
-             if (_userCache.user[uid]) {
-               delete _userCache.toLoad[uid];
-               cached.push(_userCache.user[uid]);
-             } else {
-               toLoad.push(uid);
-             }
-           });
 
-           if (cached.length || !this.authenticated()) {
-             callback(undefined, cached);
-             if (!this.authenticated()) return; // require auth
-           }
+                 case 'blockquote':
+                   {
+                     body = this.parse(token.tokens);
+                     out += this.renderer.blockquote(body);
+                     continue;
+                   }
 
-           utilArrayChunk(toLoad, 150).forEach(function (arr) {
-             oauth.xhr({
-               method: 'GET',
-               path: '/api/0.6/users?users=' + arr.join()
-             }, wrapcb(this, done, _connectionID));
-           }.bind(this));
+                 case 'list':
+                   {
+                     ordered = token.ordered;
+                     start = token.start;
+                     loose = token.loose;
+                     l2 = token.items.length;
+                     body = '';
 
-           function done(err, xml) {
-             if (err) {
-               return callback(err);
-             }
+                     for (j = 0; j < l2; j++) {
+                       item = token.items[j];
+                       checked = item.checked;
+                       task = item.task;
+                       itemBody = '';
 
-             var options = {
-               skipSeen: true
-             };
-             return parseXML(xml, function (err, results) {
-               if (err) {
-                 return callback(err);
-               } else {
-                 return callback(undefined, results);
-               }
-             }, options);
-           }
-         },
-         // Load a given user by id
-         // GET /api/0.6/user/#id
-         loadUser: function loadUser(uid, callback) {
-           if (_userCache.user[uid] || !this.authenticated()) {
-             // require auth
-             delete _userCache.toLoad[uid];
-             return callback(undefined, _userCache.user[uid]);
-           }
+                       if (item.task) {
+                         checkbox = this.renderer.checkbox(checked);
 
-           oauth.xhr({
-             method: 'GET',
-             path: '/api/0.6/user/' + uid
-           }, wrapcb(this, done, _connectionID));
+                         if (loose) {
+                           if (item.tokens.length > 0 && item.tokens[0].type === 'text') {
+                             item.tokens[0].text = checkbox + ' ' + item.tokens[0].text;
 
-           function done(err, xml) {
-             if (err) {
-               return callback(err);
-             }
+                             if (item.tokens[0].tokens && item.tokens[0].tokens.length > 0 && item.tokens[0].tokens[0].type === 'text') {
+                               item.tokens[0].tokens[0].text = checkbox + ' ' + item.tokens[0].tokens[0].text;
+                             }
+                           } else {
+                             item.tokens.unshift({
+                               type: 'text',
+                               text: checkbox
+                             });
+                           }
+                         } else {
+                           itemBody += checkbox;
+                         }
+                       }
 
-             var options = {
-               skipSeen: true
-             };
-             return parseXML(xml, function (err, results) {
-               if (err) {
-                 return callback(err);
-               } else {
-                 return callback(undefined, results[0]);
-               }
-             }, options);
-           }
-         },
-         // Load the details of the logged-in user
-         // GET /api/0.6/user/details
-         userDetails: function userDetails(callback) {
-           if (_userDetails) {
-             // retrieve cached
-             return callback(undefined, _userDetails);
-           }
+                       itemBody += this.parse(item.tokens, loose);
+                       body += this.renderer.listitem(itemBody, task, checked);
+                     }
 
-           oauth.xhr({
-             method: 'GET',
-             path: '/api/0.6/user/details'
-           }, wrapcb(this, done, _connectionID));
+                     out += this.renderer.list(body, ordered, start);
+                     continue;
+                   }
 
-           function done(err, xml) {
-             if (err) {
-               return callback(err);
-             }
+                 case 'html':
+                   {
+                     // TODO parse inline content if parameter markdown=1
+                     out += this.renderer.html(token.text);
+                     continue;
+                   }
 
-             var options = {
-               skipSeen: false
-             };
-             return parseXML(xml, function (err, results) {
-               if (err) {
-                 return callback(err);
-               } else {
-                 _userDetails = results[0];
-                 return callback(undefined, _userDetails);
-               }
-             }, options);
-           }
-         },
-         // Load previous changesets for the logged in user
-         // GET /api/0.6/changesets?user=#id
-         userChangesets: function userChangesets(callback) {
-           if (_userChangesets) {
-             // retrieve cached
-             return callback(undefined, _userChangesets);
-           }
+                 case 'paragraph':
+                   {
+                     out += this.renderer.paragraph(this.parseInline(token.tokens));
+                     continue;
+                   }
 
-           this.userDetails(wrapcb(this, gotDetails, _connectionID));
+                 case 'text':
+                   {
+                     body = token.tokens ? this.parseInline(token.tokens) : token.text;
 
-           function gotDetails(err, user) {
-             if (err) {
-               return callback(err);
-             }
+                     while (i + 1 < l && tokens[i + 1].type === 'text') {
+                       token = tokens[++i];
+                       body += '\n' + (token.tokens ? this.parseInline(token.tokens) : token.text);
+                     }
 
-             oauth.xhr({
-               method: 'GET',
-               path: '/api/0.6/changesets?user=' + user.id
-             }, wrapcb(this, done, _connectionID));
-           }
+                     out += top ? this.renderer.paragraph(body) : body;
+                     continue;
+                   }
 
-           function done(err, xml) {
-             if (err) {
-               return callback(err);
+                 default:
+                   {
+                     var errMsg = 'Token with "' + token.type + '" type was not found.';
+
+                     if (this.options.silent) {
+                       console.error(errMsg);
+                       return;
+                     } else {
+                       throw new Error(errMsg);
+                     }
+                   }
+               }
              }
 
-             _userChangesets = Array.prototype.map.call(xml.getElementsByTagName('changeset'), function (changeset) {
-               return {
-                 tags: getTags(changeset)
-               };
-             }).filter(function (changeset) {
-               var comment = changeset.tags.comment;
-               return comment && comment !== '';
-             });
-             return callback(undefined, _userChangesets);
+             return out;
            }
-         },
-         // Fetch the status of the OSM API
-         // GET /api/capabilities
-         status: function status(callback) {
-           var url = urlroot + '/api/capabilities';
-           var errback = wrapcb(this, done, _connectionID);
-           d3_xml(url).then(function (data) {
-             errback(null, data);
-           })["catch"](function (err) {
-             errback(err.message);
-           });
+           /**
+            * Parse Inline Tokens
+            */
 
-           function done(err, xml) {
-             if (err) {
-               // the status is null if no response could be retrieved
-               return callback(err, null);
-             } // update blocklists
+         }, {
+           key: "parseInline",
+           value: function parseInline(tokens, renderer) {
+             renderer = renderer || this.renderer;
+             var out = '',
+                 i,
+                 token;
+             var l = tokens.length;
 
+             for (i = 0; i < l; i++) {
+               token = tokens[i];
 
-             var elements = xml.getElementsByTagName('blacklist');
-             var regexes = [];
+               switch (token.type) {
+                 case 'escape':
+                   {
+                     out += renderer.text(token.text);
+                     break;
+                   }
 
-             for (var i = 0; i < elements.length; i++) {
-               var regexString = elements[i].getAttribute('regex'); // needs unencode?
+                 case 'html':
+                   {
+                     out += renderer.html(token.text);
+                     break;
+                   }
 
-               if (regexString) {
-                 try {
-                   var regex = new RegExp(regexString);
-                   regexes.push(regex);
-                 } catch (e) {
-                   /* noop */
-                 }
-               }
-             }
+                 case 'link':
+                   {
+                     out += renderer.link(token.href, token.title, this.parseInline(token.tokens, renderer));
+                     break;
+                   }
 
-             if (regexes.length) {
-               _imageryBlocklists = regexes;
-             }
+                 case 'image':
+                   {
+                     out += renderer.image(token.href, token.title, token.text);
+                     break;
+                   }
 
-             if (_rateLimitError) {
-               return callback(_rateLimitError, 'rateLimited');
-             } else {
-               var waynodes = xml.getElementsByTagName('waynodes');
-               var maxWayNodes = waynodes.length && parseInt(waynodes[0].getAttribute('maximum'), 10);
-               if (maxWayNodes && isFinite(maxWayNodes)) _maxWayNodes = maxWayNodes;
-               var apiStatus = xml.getElementsByTagName('status');
-               var val = apiStatus[0].getAttribute('api');
-               return callback(undefined, val);
-             }
-           }
-         },
-         // Calls `status` and dispatches an `apiStatusChange` event if the returned
-         // status differs from the cached status.
-         reloadApiStatus: function reloadApiStatus() {
-           // throttle to avoid unnecessary API calls
-           if (!this.throttledReloadApiStatus) {
-             var that = this;
-             this.throttledReloadApiStatus = throttle(function () {
-               that.status(function (err, status) {
-                 if (status !== _cachedApiStatus) {
-                   _cachedApiStatus = status;
-                   dispatch$6.call('apiStatusChange', that, err, status);
-                 }
-               });
-             }, 500);
-           }
+                 case 'strong':
+                   {
+                     out += renderer.strong(this.parseInline(token.tokens, renderer));
+                     break;
+                   }
 
-           this.throttledReloadApiStatus();
-         },
-         // Returns the maximum number of nodes a single way can have
-         maxWayNodes: function maxWayNodes() {
-           return _maxWayNodes;
-         },
-         // Load data (entities) from the API in tiles
-         // GET /api/0.6/map?bbox=
-         loadTiles: function loadTiles(projection, callback) {
-           if (_off) return; // determine the needed tiles to cover the view
+                 case 'em':
+                   {
+                     out += renderer.em(this.parseInline(token.tokens, renderer));
+                     break;
+                   }
 
-           var tiles = tiler$5.zoomExtent([_tileZoom$3, _tileZoom$3]).getTiles(projection); // abort inflight requests that are no longer needed
+                 case 'codespan':
+                   {
+                     out += renderer.codespan(token.text);
+                     break;
+                   }
 
-           var hadRequests = hasInflightRequests(_tileCache);
-           abortUnwantedRequests$3(_tileCache, tiles);
+                 case 'br':
+                   {
+                     out += renderer.br();
+                     break;
+                   }
 
-           if (hadRequests && !hasInflightRequests(_tileCache)) {
-             dispatch$6.call('loaded'); // stop the spinner
-           } // issue new requests..
+                 case 'del':
+                   {
+                     out += renderer.del(this.parseInline(token.tokens, renderer));
+                     break;
+                   }
 
+                 case 'text':
+                   {
+                     out += renderer.text(token.text);
+                     break;
+                   }
 
-           tiles.forEach(function (tile) {
-             this.loadTile(tile, callback);
-           }, this);
-         },
-         // Load a single data tile
-         // GET /api/0.6/map?bbox=
-         loadTile: function loadTile(tile, callback) {
-           if (_off) return;
-           if (_tileCache.loaded[tile.id] || _tileCache.inflight[tile.id]) return;
+                 default:
+                   {
+                     var errMsg = 'Token with "' + token.type + '" type was not found.';
 
-           if (!hasInflightRequests(_tileCache)) {
-             dispatch$6.call('loading'); // start the spinner
+                     if (this.options.silent) {
+                       console.error(errMsg);
+                       return;
+                     } else {
+                       throw new Error(errMsg);
+                     }
+                   }
+               }
+             }
+
+             return out;
            }
+         }], [{
+           key: "parse",
+           value: function parse(tokens, options) {
+             var parser = new Parser(options);
+             return parser.parse(tokens);
+           }
+           /**
+            * Static Parse Inline Method
+            */
 
-           var path = '/api/0.6/map.json?bbox=';
-           var options = {
-             skipSeen: true
-           };
-           _tileCache.inflight[tile.id] = this.loadFromAPI(path + tile.extent.toParam(), tileCallback, options);
+         }, {
+           key: "parseInline",
+           value: function parseInline(tokens, options) {
+             var parser = new Parser(options);
+             return parser.parseInline(tokens);
+           }
+         }]);
 
-           function tileCallback(err, parsed) {
-             delete _tileCache.inflight[tile.id];
+         return Parser;
+       }();
 
-             if (!err) {
-               delete _tileCache.toLoad[tile.id];
-               _tileCache.loaded[tile.id] = true;
-               var bbox = tile.extent.bbox();
-               bbox.id = tile.id;
+       var merge = helpers.merge,
+           checkSanitizeDeprecation = helpers.checkSanitizeDeprecation,
+           escape$1 = helpers.escape;
+       var getDefaults = defaults$5.getDefaults,
+           changeDefaults = defaults$5.changeDefaults,
+           defaults = defaults$5.defaults;
+       /**
+        * Marked
+        */
 
-               _tileCache.rtree.insert(bbox);
-             }
+       function marked(src, opt, callback) {
+         // throw error in case of non string input
+         if (typeof src === 'undefined' || src === null) {
+           throw new Error('marked(): input parameter is undefined or null');
+         }
 
-             if (callback) {
-               callback(err, Object.assign({
-                 data: parsed
-               }, tile));
-             }
+         if (typeof src !== 'string') {
+           throw new Error('marked(): input parameter is of type ' + Object.prototype.toString.call(src) + ', string expected');
+         }
 
-             if (!hasInflightRequests(_tileCache)) {
-               dispatch$6.call('loaded'); // stop the spinner
-             }
-           }
-         },
-         isDataLoaded: function isDataLoaded(loc) {
-           var bbox = {
-             minX: loc[0],
-             minY: loc[1],
-             maxX: loc[0],
-             maxY: loc[1]
-           };
-           return _tileCache.rtree.collides(bbox);
-         },
-         // load the tile that covers the given `loc`
-         loadTileAtLoc: function loadTileAtLoc(loc, callback) {
-           // Back off if the toLoad queue is filling up.. re #6417
-           // (Currently `loadTileAtLoc` requests are considered low priority - used by operations to
-           // let users safely edit geometries which extend to unloaded tiles.  We can drop some.)
-           if (Object.keys(_tileCache.toLoad).length > 50) return;
-           var k = geoZoomToScale(_tileZoom$3 + 1);
-           var offset = geoRawMercator().scale(k)(loc);
-           var projection = geoRawMercator().transform({
-             k: k,
-             x: -offset[0],
-             y: -offset[1]
-           });
-           var tiles = tiler$5.zoomExtent([_tileZoom$3, _tileZoom$3]).getTiles(projection);
-           tiles.forEach(function (tile) {
-             if (_tileCache.toLoad[tile.id] || _tileCache.loaded[tile.id] || _tileCache.inflight[tile.id]) return;
-             _tileCache.toLoad[tile.id] = true;
-             this.loadTile(tile, callback);
-           }, this);
-         },
-         // Load notes from the API in tiles
-         // GET /api/0.6/notes?bbox=
-         loadNotes: function loadNotes(projection, noteOptions) {
-           noteOptions = Object.assign({
-             limit: 10000,
-             closed: 7
-           }, noteOptions);
-           if (_off) return;
-           var that = this;
-           var path = '/api/0.6/notes?limit=' + noteOptions.limit + '&closed=' + noteOptions.closed + '&bbox=';
+         if (typeof opt === 'function') {
+           callback = opt;
+           opt = null;
+         }
 
-           var throttleLoadUsers = throttle(function () {
-             var uids = Object.keys(_userCache.toLoad);
-             if (!uids.length) return;
-             that.loadUsers(uids, function () {}); // eagerly load user details
-           }, 750); // determine the needed tiles to cover the view
+         opt = merge({}, marked.defaults, opt || {});
+         checkSanitizeDeprecation(opt);
 
+         if (callback) {
+           var highlight = opt.highlight;
+           var tokens;
 
-           var tiles = tiler$5.zoomExtent([_noteZoom, _noteZoom]).getTiles(projection); // abort inflight requests that are no longer needed
+           try {
+             tokens = Lexer_1.lex(src, opt);
+           } catch (e) {
+             return callback(e);
+           }
 
-           abortUnwantedRequests$3(_noteCache, tiles); // issue new requests..
+           var done = function done(err) {
+             var out;
 
-           tiles.forEach(function (tile) {
-             if (_noteCache.loaded[tile.id] || _noteCache.inflight[tile.id]) return;
-             var options = {
-               skipSeen: false
-             };
-             _noteCache.inflight[tile.id] = that.loadFromAPI(path + tile.extent.toParam(), function (err) {
-               delete _noteCache.inflight[tile.id];
+             if (!err) {
+               try {
+                 if (opt.walkTokens) {
+                   marked.walkTokens(tokens, opt.walkTokens);
+                 }
 
-               if (!err) {
-                 _noteCache.loaded[tile.id] = true;
+                 out = Parser_1.parse(tokens, opt);
+               } catch (e) {
+                 err = e;
                }
+             }
 
-               throttleLoadUsers();
-               dispatch$6.call('loadedNotes');
-             }, options);
-           });
-         },
-         // Create a note
-         // POST /api/0.6/notes?params
-         postNoteCreate: function postNoteCreate(note, callback) {
-           if (!this.authenticated()) {
-             return callback({
-               message: 'Not Authenticated',
-               status: -3
-             }, note);
-           }
+             opt.highlight = highlight;
+             return err ? callback(err) : callback(null, out);
+           };
 
-           if (_noteCache.inflightPost[note.id]) {
-             return callback({
-               message: 'Note update already inflight',
-               status: -2
-             }, note);
+           if (!highlight || highlight.length < 3) {
+             return done();
            }
 
-           if (!note.loc[0] || !note.loc[1] || !note.newComment) return; // location & description required
+           delete opt.highlight;
+           if (!tokens.length) return done();
+           var pending = 0;
+           marked.walkTokens(tokens, function (token) {
+             if (token.type === 'code') {
+               pending++;
+               setTimeout(function () {
+                 highlight(token.text, token.lang, function (err, code) {
+                   if (err) {
+                     return done(err);
+                   }
 
-           var comment = note.newComment;
+                   if (code != null && code !== token.text) {
+                     token.text = code;
+                     token.escaped = true;
+                   }
 
-           if (note.newCategory && note.newCategory !== 'None') {
-             comment += ' #' + note.newCategory;
-           }
+                   pending--;
 
-           var path = '/api/0.6/notes?' + utilQsString({
-             lon: note.loc[0],
-             lat: note.loc[1],
-             text: comment
+                   if (pending === 0) {
+                     done();
+                   }
+                 });
+               }, 0);
+             }
            });
-           _noteCache.inflightPost[note.id] = oauth.xhr({
-             method: 'POST',
-             path: path
-           }, wrapcb(this, done, _connectionID));
-
-           function done(err, xml) {
-             delete _noteCache.inflightPost[note.id];
 
-             if (err) {
-               return callback(err);
-             } // we get the updated note back, remove from caches and reparse..
+           if (pending === 0) {
+             done();
+           }
 
+           return;
+         }
 
-             this.removeNote(note);
-             var options = {
-               skipSeen: false
-             };
-             return parseXML(xml, function (err, results) {
-               if (err) {
-                 return callback(err);
-               } else {
-                 return callback(undefined, results[0]);
-               }
-             }, options);
-           }
-         },
-         // Update a note
-         // POST /api/0.6/notes/#id/comment?text=comment
-         // POST /api/0.6/notes/#id/close?text=comment
-         // POST /api/0.6/notes/#id/reopen?text=comment
-         postNoteUpdate: function postNoteUpdate(note, newStatus, callback) {
-           if (!this.authenticated()) {
-             return callback({
-               message: 'Not Authenticated',
-               status: -3
-             }, note);
-           }
+         try {
+           var _tokens = Lexer_1.lex(src, opt);
 
-           if (_noteCache.inflightPost[note.id]) {
-             return callback({
-               message: 'Note update already inflight',
-               status: -2
-             }, note);
+           if (opt.walkTokens) {
+             marked.walkTokens(_tokens, opt.walkTokens);
            }
 
-           var action;
+           return Parser_1.parse(_tokens, opt);
+         } catch (e) {
+           e.message += '\nPlease report this to https://github.com/markedjs/marked.';
 
-           if (note.status !== 'closed' && newStatus === 'closed') {
-             action = 'close';
-           } else if (note.status !== 'open' && newStatus === 'open') {
-             action = 'reopen';
-           } else {
-             action = 'comment';
-             if (!note.newComment) return; // when commenting, comment required
+           if (opt.silent) {
+             return '<p>An error occurred:</p><pre>' + escape$1(e.message + '', true) + '</pre>';
            }
 
-           var path = '/api/0.6/notes/' + note.id + '/' + action;
+           throw e;
+         }
+       }
+       /**
+        * Options
+        */
 
-           if (note.newComment) {
-             path += '?' + utilQsString({
-               text: note.newComment
-             });
-           }
 
-           _noteCache.inflightPost[note.id] = oauth.xhr({
-             method: 'POST',
-             path: path
-           }, wrapcb(this, done, _connectionID));
+       marked.options = marked.setOptions = function (opt) {
+         merge(marked.defaults, opt);
+         changeDefaults(marked.defaults);
+         return marked;
+       };
 
-           function done(err, xml) {
-             delete _noteCache.inflightPost[note.id];
+       marked.getDefaults = getDefaults;
+       marked.defaults = defaults;
+       /**
+        * Use Extension
+        */
 
-             if (err) {
-               return callback(err);
-             } // we get the updated note back, remove from caches and reparse..
+       marked.use = function (extension) {
+         var opts = merge({}, extension);
 
+         if (extension.renderer) {
+           (function () {
+             var renderer = marked.defaults.renderer || new Renderer_1();
 
-             this.removeNote(note); // update closed note cache - used to populate `closed:note` changeset tag
+             var _loop = function _loop(prop) {
+               var prevRenderer = renderer[prop];
 
-             if (action === 'close') {
-               _noteCache.closed[note.id] = true;
-             } else if (action === 'reopen') {
-               delete _noteCache.closed[note.id];
-             }
+               renderer[prop] = function () {
+                 for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {
+                   args[_key] = arguments[_key];
+                 }
 
-             var options = {
-               skipSeen: false
-             };
-             return parseXML(xml, function (err, results) {
-               if (err) {
-                 return callback(err);
-               } else {
-                 return callback(undefined, results[0]);
-               }
-             }, options);
-           }
-         },
-         "switch": function _switch(options) {
-           urlroot = options.urlroot;
-           oauth.options(Object.assign({
-             url: urlroot,
-             loading: authLoading,
-             done: authDone
-           }, options));
-           this.reset();
-           this.userChangesets(function () {}); // eagerly load user details/changesets
+                 var ret = extension.renderer[prop].apply(renderer, args);
 
-           dispatch$6.call('change');
-           return this;
-         },
-         toggle: function toggle(val) {
-           _off = !val;
-           return this;
-         },
-         isChangesetInflight: function isChangesetInflight() {
-           return !!_changeset.inflight;
-         },
-         // get/set cached data
-         // This is used to save/restore the state when entering/exiting the walkthrough
-         // Also used for testing purposes.
-         caches: function caches(obj) {
-           function cloneCache(source) {
-             var target = {};
-             Object.keys(source).forEach(function (k) {
-               if (k === 'rtree') {
-                 target.rtree = new RBush().fromJSON(source.rtree.toJSON()); // clone rbush
-               } else if (k === 'note') {
-                 target.note = {};
-                 Object.keys(source.note).forEach(function (id) {
-                   target.note[id] = osmNote(source.note[id]); // copy notes
-                 });
-               } else {
-                 target[k] = JSON.parse(JSON.stringify(source[k])); // clone deep
-               }
-             });
-             return target;
-           }
+                 if (ret === false) {
+                   ret = prevRenderer.apply(renderer, args);
+                 }
 
-           if (!arguments.length) {
-             return {
-               tile: cloneCache(_tileCache),
-               note: cloneCache(_noteCache),
-               user: cloneCache(_userCache)
+                 return ret;
+               };
              };
-           } // access caches directly for testing (e.g., loading notes rtree)
 
+             for (var prop in extension.renderer) {
+               _loop(prop);
+             }
 
-           if (obj === 'get') {
-             return {
-               tile: _tileCache,
-               note: _noteCache,
-               user: _userCache
-             };
-           }
+             opts.renderer = renderer;
+           })();
+         }
 
-           if (obj.tile) {
-             _tileCache = obj.tile;
-             _tileCache.inflight = {};
-           }
+         if (extension.tokenizer) {
+           (function () {
+             var tokenizer = marked.defaults.tokenizer || new Tokenizer_1();
 
-           if (obj.note) {
-             _noteCache = obj.note;
-             _noteCache.inflight = {};
-             _noteCache.inflightPost = {};
-           }
+             var _loop2 = function _loop2(prop) {
+               var prevTokenizer = tokenizer[prop];
 
-           if (obj.user) {
-             _userCache = obj.user;
-           }
+               tokenizer[prop] = function () {
+                 for (var _len2 = arguments.length, args = new Array(_len2), _key2 = 0; _key2 < _len2; _key2++) {
+                   args[_key2] = arguments[_key2];
+                 }
 
-           return this;
-         },
-         logout: function logout() {
-           _userChangesets = undefined;
-           _userDetails = undefined;
-           oauth.logout();
-           dispatch$6.call('change');
-           return this;
-         },
-         authenticated: function authenticated() {
-           return oauth.authenticated();
-         },
-         authenticate: function authenticate(callback) {
-           var that = this;
-           var cid = _connectionID;
-           _userChangesets = undefined;
-           _userDetails = undefined;
+                 var ret = extension.tokenizer[prop].apply(tokenizer, args);
 
-           function done(err, res) {
-             if (err) {
-               if (callback) callback(err);
-               return;
-             }
+                 if (ret === false) {
+                   ret = prevTokenizer.apply(tokenizer, args);
+                 }
 
-             if (that.getConnectionId() !== cid) {
-               if (callback) callback({
-                 message: 'Connection Switched',
-                 status: -1
-               });
-               return;
+                 return ret;
+               };
+             };
+
+             for (var prop in extension.tokenizer) {
+               _loop2(prop);
              }
 
-             _rateLimitError = undefined;
-             dispatch$6.call('change');
-             if (callback) callback(err, res);
-             that.userChangesets(function () {}); // eagerly load user details/changesets
-           }
+             opts.tokenizer = tokenizer;
+           })();
+         }
 
-           return oauth.authenticate(done);
-         },
-         imageryBlocklists: function imageryBlocklists() {
-           return _imageryBlocklists;
-         },
-         tileZoom: function tileZoom(val) {
-           if (!arguments.length) return _tileZoom$3;
-           _tileZoom$3 = val;
-           return this;
-         },
-         // get all cached notes covering the viewport
-         notes: function notes(projection) {
-           var viewport = projection.clipExtent();
-           var min = [viewport[0][0], viewport[1][1]];
-           var max = [viewport[1][0], viewport[0][1]];
-           var bbox = geoExtent(projection.invert(min), projection.invert(max)).bbox();
-           return _noteCache.rtree.search(bbox).map(function (d) {
-             return d.data;
-           });
-         },
-         // get a single note from the cache
-         getNote: function getNote(id) {
-           return _noteCache.note[id];
-         },
-         // remove a single note from the cache
-         removeNote: function removeNote(note) {
-           if (!(note instanceof osmNote) || !note.id) return;
-           delete _noteCache.note[note.id];
-           updateRtree$3(encodeNoteRtree(note), false); // false = remove
-         },
-         // replace a single note in the cache
-         replaceNote: function replaceNote(note) {
-           if (!(note instanceof osmNote) || !note.id) return;
-           _noteCache.note[note.id] = note;
-           updateRtree$3(encodeNoteRtree(note), true); // true = replace
+         if (extension.walkTokens) {
+           var walkTokens = marked.defaults.walkTokens;
 
-           return note;
-         },
-         // Get an array of note IDs closed during this session.
-         // Used to populate `closed:note` changeset tag
-         getClosedIDs: function getClosedIDs() {
-           return Object.keys(_noteCache.closed).sort();
+           opts.walkTokens = function (token) {
+             extension.walkTokens(token);
+
+             if (walkTokens) {
+               walkTokens(token);
+             }
+           };
          }
-       };
 
-       var _apibase = 'https://wiki.openstreetmap.org/w/api.php';
-       var _inflight$1 = {};
-       var _wikibaseCache = {};
-       var _localeIDs = {
-         en: false
+         marked.setOptions(opts);
        };
+       /**
+        * Run callback for every token
+        */
 
-       var debouncedRequest = debounce(request, 500, {
-         leading: false
-       });
 
-       function request(url, callback) {
-         if (_inflight$1[url]) return;
-         var controller = new AbortController();
-         _inflight$1[url] = controller;
-         d3_json(url, {
-           signal: controller.signal
-         }).then(function (result) {
-           delete _inflight$1[url];
-           if (callback) callback(null, result);
-         })["catch"](function (err) {
-           delete _inflight$1[url];
-           if (err.name === 'AbortError') return;
-           if (callback) callback(err.message);
-         });
-       }
+       marked.walkTokens = function (tokens, callback) {
+         var _iterator = _createForOfIteratorHelper(tokens),
+             _step;
 
-       var serviceOsmWikibase = {
-         init: function init() {
-           _inflight$1 = {};
-           _wikibaseCache = {};
-           _localeIDs = {};
-         },
-         reset: function reset() {
-           Object.values(_inflight$1).forEach(function (controller) {
-             controller.abort();
-           });
-           _inflight$1 = {};
-         },
+         try {
+           for (_iterator.s(); !(_step = _iterator.n()).done;) {
+             var token = _step.value;
+             callback(token);
 
-         /**
-          * Get the best value for the property, or undefined if not found
-          * @param entity object from wikibase
-          * @param property string e.g. 'P4' for image
-          * @param langCode string e.g. 'fr' for French
-          */
-         claimToValue: function claimToValue(entity, property, langCode) {
-           if (!entity.claims[property]) return undefined;
-           var locale = _localeIDs[langCode];
-           var preferredPick, localePick;
-           entity.claims[property].forEach(function (stmt) {
-             // If exists, use value limited to the needed language (has a qualifier P26 = locale)
-             // Or if not found, use the first value with the "preferred" rank
-             if (!preferredPick && stmt.rank === 'preferred') {
-               preferredPick = stmt;
-             }
+             switch (token.type) {
+               case 'table':
+                 {
+                   var _iterator2 = _createForOfIteratorHelper(token.tokens.header),
+                       _step2;
 
-             if (locale && stmt.qualifiers && stmt.qualifiers.P26 && stmt.qualifiers.P26[0].datavalue.value.id === locale) {
-               localePick = stmt;
-             }
-           });
-           var result = localePick || preferredPick;
+                   try {
+                     for (_iterator2.s(); !(_step2 = _iterator2.n()).done;) {
+                       var cell = _step2.value;
+                       marked.walkTokens(cell, callback);
+                     }
+                   } catch (err) {
+                     _iterator2.e(err);
+                   } finally {
+                     _iterator2.f();
+                   }
 
-           if (result) {
-             var datavalue = result.mainsnak.datavalue;
-             return datavalue.type === 'wikibase-entityid' ? datavalue.value.id : datavalue.value;
-           } else {
-             return undefined;
-           }
-         },
+                   var _iterator3 = _createForOfIteratorHelper(token.tokens.cells),
+                       _step3;
 
-         /**
-          * Convert monolingual property into a key-value object (language -> value)
-          * @param entity object from wikibase
-          * @param property string e.g. 'P31' for monolingual wiki page title
-          */
-         monolingualClaimToValueObj: function monolingualClaimToValueObj(entity, property) {
-           if (!entity || !entity.claims[property]) return undefined;
-           return entity.claims[property].reduce(function (acc, obj) {
-             var value = obj.mainsnak.datavalue.value;
-             acc[value.language] = value.text;
-             return acc;
-           }, {});
-         },
-         toSitelink: function toSitelink(key, value) {
-           var result = value ? 'Tag:' + key + '=' + value : 'Key:' + key;
-           return result.replace(/_/g, ' ').trim();
-         },
-         //
-         // Pass params object of the form:
-         // {
-         //   key: 'string',
-         //   value: 'string',
-         //   langCode: 'string'
-         // }
-         //
-         getEntity: function getEntity(params, callback) {
-           var doRequest = params.debounce ? debouncedRequest : request;
-           var that = this;
-           var titles = [];
-           var result = {};
-           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;
+                   try {
+                     for (_iterator3.s(); !(_step3 = _iterator3.n()).done;) {
+                       var row = _step3.value;
 
-           if (params.langCodes) {
-             params.langCodes.forEach(function (langCode) {
-               if (_localeIDs[langCode] === undefined) {
-                 // If this is the first time we are asking about this locale,
-                 // fetch corresponding entity (if it exists), and cache it.
-                 // If there is no such entry, cache `false` value to avoid re-requesting it.
-                 localeSitelink = ('Locale:' + langCode).replace(/_/g, ' ').trim();
-                 titles.push(localeSitelink);
-               }
-             });
-           }
+                       var _iterator4 = _createForOfIteratorHelper(row),
+                           _step4;
 
-           if (rtypeSitelink) {
-             if (_wikibaseCache[rtypeSitelink]) {
-               result.rtype = _wikibaseCache[rtypeSitelink];
-             } else {
-               titles.push(rtypeSitelink);
+                       try {
+                         for (_iterator4.s(); !(_step4 = _iterator4.n()).done;) {
+                           var _cell = _step4.value;
+                           marked.walkTokens(_cell, callback);
+                         }
+                       } catch (err) {
+                         _iterator4.e(err);
+                       } finally {
+                         _iterator4.f();
+                       }
+                     }
+                   } catch (err) {
+                     _iterator3.e(err);
+                   } finally {
+                     _iterator3.f();
+                   }
+
+                   break;
+                 }
+
+               case 'list':
+                 {
+                   marked.walkTokens(token.items, callback);
+                   break;
+                 }
+
+               default:
+                 {
+                   if (token.tokens) {
+                     marked.walkTokens(token.tokens, callback);
+                   }
+                 }
              }
            }
+         } catch (err) {
+           _iterator.e(err);
+         } finally {
+           _iterator.f();
+         }
+       };
+       /**
+        * Parse Inline
+        */
 
-           if (keySitelink) {
-             if (_wikibaseCache[keySitelink]) {
-               result.key = _wikibaseCache[keySitelink];
-             } else {
-               titles.push(keySitelink);
-             }
+
+       marked.parseInline = function (src, opt) {
+         // throw error in case of non string input
+         if (typeof src === 'undefined' || src === null) {
+           throw new Error('marked.parseInline(): input parameter is undefined or null');
+         }
+
+         if (typeof src !== 'string') {
+           throw new Error('marked.parseInline(): input parameter is of type ' + Object.prototype.toString.call(src) + ', string expected');
+         }
+
+         opt = merge({}, marked.defaults, opt || {});
+         checkSanitizeDeprecation(opt);
+
+         try {
+           var tokens = Lexer_1.lexInline(src, opt);
+
+           if (opt.walkTokens) {
+             marked.walkTokens(tokens, opt.walkTokens);
            }
 
-           if (tagSitelink) {
-             if (_wikibaseCache[tagSitelink]) {
-               result.tag = _wikibaseCache[tagSitelink];
-             } else {
-               titles.push(tagSitelink);
-             }
+           return Parser_1.parseInline(tokens, opt);
+         } catch (e) {
+           e.message += '\nPlease report this to https://github.com/markedjs/marked.';
+
+           if (opt.silent) {
+             return '<p>An error occurred:</p><pre>' + escape$1(e.message + '', true) + '</pre>';
            }
 
-           if (!titles.length) {
-             // Nothing to do, we already had everything in the cache
-             return callback(null, result);
-           } // Requesting just the user language code
-           // If backend recognizes the code, it will perform proper fallbacks,
-           // and the result will contain the requested code. If not, all values are returned:
-           // {"zh-tw":{"value":"...","language":"zh-tw","source-language":"zh-hant"}
-           // {"pt-br":{"value":"...","language":"pt","for-language":"pt-br"}}
+           throw e;
+         }
+       };
+       /**
+        * Expose
+        */
 
 
-           var obj = {
-             action: 'wbgetentities',
-             sites: 'wiki',
-             titles: titles.join('|'),
-             languages: params.langCodes.join('|'),
-             languagefallback: 1,
-             origin: '*',
-             format: 'json' // There is an MW Wikibase API bug https://phabricator.wikimedia.org/T212069
-             // We shouldn't use v1 until it gets fixed, but should switch to it afterwards
-             // formatversion: 2,
+       marked.Parser = Parser_1;
+       marked.parser = Parser_1.parse;
+       marked.Renderer = Renderer_1;
+       marked.TextRenderer = TextRenderer_1;
+       marked.Lexer = Lexer_1;
+       marked.lexer = Lexer_1.lex;
+       marked.Tokenizer = Tokenizer_1;
+       marked.Slugger = Slugger_1;
+       marked.parse = marked;
+       var marked_1 = marked;
 
-           };
-           var url = _apibase + '?' + utilQsString(obj);
-           doRequest(url, function (err, d) {
-             if (err) {
-               callback(err);
-             } else if (!d.success || d.error) {
-               callback(d.error.messages.map(function (v) {
-                 return v.html['*'];
-               }).join('<br>'));
-             } else {
-               var localeID = false;
-               Object.values(d.entities).forEach(function (res) {
-                 if (res.missing !== '') {
-                   var title = res.sitelinks.wiki.title;
+       var tiler$4 = utilTiler();
+       var dispatch$5 = dispatch$8('loaded');
+       var _tileZoom$1 = 14;
+       var _osmoseUrlRoot = 'https://osmose.openstreetmap.fr/api/0.3';
+       var _osmoseData = {
+         icons: {},
+         items: []
+       }; // This gets reassigned if reset
 
-                   if (title === rtypeSitelink) {
-                     _wikibaseCache[rtypeSitelink] = res;
-                     result.rtype = res;
-                   } else if (title === keySitelink) {
-                     _wikibaseCache[keySitelink] = res;
-                     result.key = res;
-                   } else if (title === tagSitelink) {
-                     _wikibaseCache[tagSitelink] = res;
-                     result.tag = res;
-                   } else if (title === localeSitelink) {
-                     localeID = res.id;
-                   } else {
-                     console.log('Unexpected title ' + title); // eslint-disable-line no-console
-                   }
-                 }
-               });
+       var _cache;
 
-               if (localeSitelink) {
-                 // If locale ID is not found, store false to prevent repeated queries
-                 that.addLocale(params.langCodes[0], localeID);
-               }
+       function abortRequest$4(controller) {
+         if (controller) {
+           controller.abort();
+         }
+       }
 
-               callback(null, result);
-             }
-           });
-         },
-         //
-         // Pass params object of the form:
-         // {
-         //   key: 'string',     // required
-         //   value: 'string'    // optional
-         // }
-         //
-         // Get an result object used to display tag documentation
-         // {
-         //   title:        'string',
-         //   description:  'string',
-         //   editURL:      'string',
-         //   imageURL:     'string',
-         //   wiki:         { title: 'string', text: 'string', url: 'string' }
-         // }
-         //
-         getDocs: function getDocs(params, callback) {
-           var that = this;
-           var langCodes = _mainLocalizer.localeCodes().map(function (code) {
-             return code.toLowerCase();
+       function abortUnwantedRequests$1(cache, tiles) {
+         Object.keys(cache.inflightTile).forEach(function (k) {
+           var wanted = tiles.find(function (tile) {
+             return k === tile.id;
            });
-           params.langCodes = langCodes;
-           this.getEntity(params, function (err, data) {
-             if (err) {
-               callback(err);
-               return;
-             }
 
-             var entity = data.rtype || data.tag || data.key;
+           if (!wanted) {
+             abortRequest$4(cache.inflightTile[k]);
+             delete cache.inflightTile[k];
+           }
+         });
+       }
 
-             if (!entity) {
-               callback('No entity');
-               return;
-             }
+       function encodeIssueRtree(d) {
+         return {
+           minX: d.loc[0],
+           minY: d.loc[1],
+           maxX: d.loc[0],
+           maxY: d.loc[1],
+           data: d
+         };
+       } // Replace or remove QAItem from rtree
 
-             var i;
-             var description;
 
-             for (i in langCodes) {
-               var _code = langCodes[i];
+       function updateRtree$1(item, replace) {
+         _cache.rtree.remove(item, function (a, b) {
+           return a.data.id === b.data.id;
+         });
 
-               if (entity.descriptions[_code] && entity.descriptions[_code].language === _code) {
-                 description = entity.descriptions[_code];
-                 break;
-               }
-             }
+         if (replace) {
+           _cache.rtree.insert(item);
+         }
+       } // Issues shouldn't obscure each other
 
-             if (!description && Object.values(entity.descriptions).length) description = Object.values(entity.descriptions)[0]; // prepare result
 
-             var result = {
-               title: entity.title,
-               description: description ? description.value : '',
-               descriptionLocaleCode: description ? description.language : '',
-               editURL: 'https://wiki.openstreetmap.org/wiki/' + entity.title
-             }; // add image
+       function preventCoincident(loc) {
+         var coincident = false;
 
-             if (entity.claims) {
-               var imageroot;
-               var image = that.claimToValue(entity, 'P4', langCodes[0]);
+         do {
+           // first time, move marker up. after that, move marker right.
+           var delta = coincident ? [0.00001, 0] : [0, 0.00001];
+           loc = geoVecAdd(loc, delta);
+           var bbox = geoExtent(loc).bbox();
+           coincident = _cache.rtree.search(bbox).length;
+         } while (coincident);
 
-               if (image) {
-                 imageroot = 'https://commons.wikimedia.org/w/index.php';
-               } else {
-                 image = that.claimToValue(entity, 'P28', langCodes[0]);
+         return loc;
+       }
 
-                 if (image) {
-                   imageroot = 'https://wiki.openstreetmap.org/w/index.php';
-                 }
-               }
+       var serviceOsmose = {
+         title: 'osmose',
+         init: function init() {
+           _mainFileFetcher.get('qa_data').then(function (d) {
+             _osmoseData = d.osmose;
+             _osmoseData.items = Object.keys(d.osmose.icons).map(function (s) {
+               return s.split('-')[0];
+             }).reduce(function (unique, item) {
+               return unique.indexOf(item) !== -1 ? unique : [].concat(_toConsumableArray(unique), [item]);
+             }, []);
+           });
 
-               if (imageroot && image) {
-                 result.imageURL = imageroot + '?' + utilQsString({
-                   title: 'Special:Redirect/file/' + image,
-                   width: 400
-                 });
-               }
-             } // Try to get a wiki page from tag data item first, followed by the corresponding key data item.
-             // If neither tag nor key data item contain a wiki page in the needed language nor English,
-             // get the first found wiki page from either the tag or the key item.
+           if (!_cache) {
+             this.reset();
+           }
 
+           this.event = utilRebind(this, dispatch$5, 'on');
+         },
+         reset: function reset() {
+           var _strings = {};
+           var _colors = {};
 
-             var rtypeWiki = that.monolingualClaimToValueObj(data.rtype, 'P31');
-             var tagWiki = that.monolingualClaimToValueObj(data.tag, 'P31');
-             var keyWiki = that.monolingualClaimToValueObj(data.key, 'P31');
-             var wikis = [rtypeWiki, tagWiki, keyWiki];
+           if (_cache) {
+             Object.values(_cache.inflightTile).forEach(abortRequest$4); // Strings and colors are static and should not be re-populated
 
-             for (i in wikis) {
-               var wiki = wikis[i];
+             _strings = _cache.strings;
+             _colors = _cache.colors;
+           }
 
-               for (var j in langCodes) {
-                 var code = langCodes[j];
-                 var referenceId = langCodes[0].split('-')[0] !== 'en' && code.split('-')[0] === 'en' ? 'inspector.wiki_en_reference' : 'inspector.wiki_reference';
-                 var info = getWikiInfo(wiki, code, referenceId);
+           _cache = {
+             data: {},
+             loadedTile: {},
+             inflightTile: {},
+             inflightPost: {},
+             closed: {},
+             rtree: new RBush(),
+             strings: _strings,
+             colors: _colors
+           };
+         },
+         loadIssues: function loadIssues(projection) {
+           var _this = this;
 
-                 if (info) {
-                   result.wiki = info;
-                   break;
-                 }
-               }
+           var params = {
+             // Tiles return a maximum # of issues
+             // So we want to filter our request for only types iD supports
+             item: _osmoseData.items
+           }; // determine the needed tiles to cover the view
 
-               if (result.wiki) break;
-             }
+           var tiles = tiler$4.zoomExtent([_tileZoom$1, _tileZoom$1]).getTiles(projection); // abort inflight requests that are no longer needed
 
-             callback(null, result); // Helper method to get wiki info if a given language exists
+           abortUnwantedRequests$1(_cache, tiles); // issue new requests..
 
-             function getWikiInfo(wiki, langCode, tKey) {
-               if (wiki && wiki[langCode]) {
-                 return {
-                   title: wiki[langCode],
-                   text: tKey,
-                   url: 'https://wiki.openstreetmap.org/wiki/' + wiki[langCode]
-                 };
-               }
-             }
-           });
-         },
-         addLocale: function addLocale(langCode, qid) {
-           // Makes it easier to unit test
-           _localeIDs[langCode] = qid;
-         },
-         apibase: function apibase(val) {
-           if (!arguments.length) return _apibase;
-           _apibase = val;
-           return this;
-         }
-       };
+           tiles.forEach(function (tile) {
+             if (_cache.loadedTile[tile.id] || _cache.inflightTile[tile.id]) return;
 
-       var jsonpCache = {};
-       window.jsonpCache = jsonpCache;
-       function jsonpRequest(url, callback) {
-         var request = {
-           abort: function abort() {}
-         };
+             var _tile$xyz = _slicedToArray(tile.xyz, 3),
+                 x = _tile$xyz[0],
+                 y = _tile$xyz[1],
+                 z = _tile$xyz[2];
 
-         if (window.JSONP_FIX) {
-           if (window.JSONP_DELAY === 0) {
-             callback(window.JSONP_FIX);
-           } else {
-             var t = window.setTimeout(function () {
-               callback(window.JSONP_FIX);
-             }, window.JSONP_DELAY || 0);
+             var url = "".concat(_osmoseUrlRoot, "/issues/").concat(z, "/").concat(x, "/").concat(y, ".json?") + utilQsString(params);
+             var controller = new AbortController();
+             _cache.inflightTile[tile.id] = controller;
+             d3_json(url, {
+               signal: controller.signal
+             }).then(function (data) {
+               delete _cache.inflightTile[tile.id];
+               _cache.loadedTile[tile.id] = true;
 
-             request.abort = function () {
-               window.clearTimeout(t);
-             };
-           }
+               if (data.features) {
+                 data.features.forEach(function (issue) {
+                   var _issue$properties = issue.properties,
+                       item = _issue$properties.item,
+                       cl = _issue$properties["class"],
+                       id = _issue$properties.uuid;
+                   /* Osmose issues are uniquely identified by a unique
+                     `item` and `class` combination (both integer values) */
 
-           return request;
-         }
+                   var itemType = "".concat(item, "-").concat(cl); // Filter out unsupported issue types (some are too specific or advanced)
 
-         function rand() {
-           var chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz';
-           var c = '';
-           var i = -1;
+                   if (itemType in _osmoseData.icons) {
+                     var loc = issue.geometry.coordinates; // lon, lat
 
-           while (++i < 15) {
-             c += chars.charAt(Math.floor(Math.random() * 52));
-           }
+                     loc = preventCoincident(loc);
+                     var d = new QAItem(loc, _this, itemType, id, {
+                       item: item
+                     }); // Setting elems here prevents UI detail requests
 
-           return c;
-         }
+                     if (item === 8300 || item === 8360) {
+                       d.elems = [];
+                     }
 
-         function create(url) {
-           var e = url.match(/callback=(\w+)/);
-           var c = e ? e[1] : rand();
+                     _cache.data[d.id] = d;
 
-           jsonpCache[c] = function (data) {
-             if (jsonpCache[c]) {
-               callback(data);
-             }
+                     _cache.rtree.insert(encodeIssueRtree(d));
+                   }
+                 });
+               }
 
-             finalize();
-           };
+               dispatch$5.call('loaded');
+             })["catch"](function () {
+               delete _cache.inflightTile[tile.id];
+               _cache.loadedTile[tile.id] = true;
+             });
+           });
+         },
+         loadIssueDetail: function loadIssueDetail(issue) {
+           var _this2 = this;
 
-           function finalize() {
-             delete jsonpCache[c];
-             script.remove();
+           // Issue details only need to be fetched once
+           if (issue.elems !== undefined) {
+             return Promise.resolve(issue);
            }
 
-           request.abort = finalize;
-           return 'jsonpCache.' + c;
-         }
+           var url = "".concat(_osmoseUrlRoot, "/issue/").concat(issue.id, "?langs=").concat(_mainLocalizer.localeCode());
 
-         var cb = create(url);
-         var script = select('head').append('script').attr('type', 'text/javascript').attr('src', url.replace(/(\{|%7B)callback(\}|%7D)/, cb));
-         return request;
-       }
+           var cacheDetails = function cacheDetails(data) {
+             // Associated elements used for highlighting
+             // Assign directly for immediate use in the callback
+             issue.elems = data.elems.map(function (e) {
+               return e.type.substring(0, 1) + e.id;
+             }); // Some issues have instance specific detail in a subtitle
 
-       var bubbleApi = 'https://dev.virtualearth.net/mapcontrol/HumanScaleServices/GetBubbles.ashx?';
-       var streetsideImagesApi = 'https://t.ssl.ak.tiles.virtualearth.net/tiles/';
-       var bubbleAppKey = 'AuftgJsO0Xs8Ts4M1xZUQJQXJNsvmh3IV8DkNieCiy3tCwCUMq76-WpkrBtNAuEm';
-       var pannellumViewerCSS = 'pannellum-streetside/pannellum.css';
-       var pannellumViewerJS = 'pannellum-streetside/pannellum.js';
-       var maxResults$2 = 2000;
-       var tileZoom$2 = 16.5;
-       var tiler$6 = utilTiler().zoomExtent([tileZoom$2, tileZoom$2]).skipNullIsland(true);
-       var dispatch$7 = dispatch('loadedImages', 'viewerChanged');
-       var minHfov = 10; // zoom in degrees:  20, 10, 5
+             issue.detail = data.subtitle ? marked_1(data.subtitle.auto) : '';
 
-       var maxHfov = 90; // zoom out degrees
+             _this2.replaceItem(issue);
+           };
 
-       var defaultHfov = 45;
-       var _hires = false;
-       var _resolution = 512; // higher numbers are slower - 512, 1024, 2048, 4096
+           return d3_json(url).then(cacheDetails).then(function () {
+             return issue;
+           });
+         },
+         loadStrings: function loadStrings() {
+           var locale = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : _mainLocalizer.localeCode();
+           var items = Object.keys(_osmoseData.icons);
 
-       var _currScene = 0;
+           if (locale in _cache.strings && Object.keys(_cache.strings[locale]).length === items.length) {
+             return Promise.resolve(_cache.strings[locale]);
+           } // May be partially populated already if some requests were successful
 
-       var _ssCache;
 
-       var _pannellumViewer;
+           if (!(locale in _cache.strings)) {
+             _cache.strings[locale] = {};
+           } // Only need to cache strings for supported issue types
+           // Using multiple individual item + class requests to reduce fetched data size
 
-       var _sceneOptions = {
-         showFullscreenCtrl: false,
-         autoLoad: true,
-         compass: true,
-         yaw: 0,
-         minHfov: minHfov,
-         maxHfov: maxHfov,
-         hfov: defaultHfov,
-         type: 'cubemap',
-         cubeMap: []
-       };
 
-       var _loadViewerPromise$2;
-       /**
-        * abortRequest().
-        */
+           var allRequests = items.map(function (itemType) {
+             // No need to request data we already have
+             if (itemType in _cache.strings[locale]) return null;
 
+             var cacheData = function cacheData(data) {
+               // Bunch of nested single value arrays of objects
+               var _data$categories = _slicedToArray(data.categories, 1),
+                   _data$categories$ = _data$categories[0],
+                   cat = _data$categories$ === void 0 ? {
+                 items: []
+               } : _data$categories$;
 
-       function abortRequest$6(i) {
-         i.abort();
-       }
-       /**
-        * localeTimeStamp().
-        */
+               var _cat$items = _slicedToArray(cat.items, 1),
+                   _cat$items$ = _cat$items[0],
+                   item = _cat$items$ === void 0 ? {
+                 "class": []
+               } : _cat$items$;
 
+               var _item$class = _slicedToArray(item["class"], 1),
+                   _item$class$ = _item$class[0],
+                   cl = _item$class$ === void 0 ? null : _item$class$; // If null default value is reached, data wasn't as expected (or was empty)
 
-       function localeTimestamp(s) {
-         if (!s) return null;
-         var options = {
-           day: 'numeric',
-           month: 'short',
-           year: 'numeric'
-         };
-         var d = new Date(s);
-         if (isNaN(d.getTime())) return null;
-         return d.toLocaleString(_mainLocalizer.localeCode(), options);
-       }
-       /**
-        * loadTiles() wraps the process of generating tiles and then fetching image points for each tile.
-        */
 
+               if (!cl) {
+                 /* eslint-disable no-console */
+                 console.log("Osmose strings request (".concat(itemType, ") had unexpected data"));
+                 /* eslint-enable no-console */
 
-       function loadTiles$2(which, url, projection, margin) {
-         var tiles = tiler$6.margin(margin).getTiles(projection); // abort inflight requests that are no longer needed
+                 return;
+               } // Cache served item colors to automatically style issue markers later
 
-         var cache = _ssCache[which];
-         Object.keys(cache.inflight).forEach(function (k) {
-           var wanted = tiles.find(function (tile) {
-             return k.indexOf(tile.id + ',') === 0;
-           });
 
-           if (!wanted) {
-             abortRequest$6(cache.inflight[k]);
-             delete cache.inflight[k];
-           }
-         });
-         tiles.forEach(function (tile) {
-           return loadNextTilePage$2(which, url, tile);
-         });
-       }
-       /**
-        * loadNextTilePage() load data for the next tile page in line.
-        */
+               var itemInt = item.item,
+                   color = item.color;
 
+               if (/^#[A-Fa-f0-9]{6}|[A-Fa-f0-9]{3}/.test(color)) {
+                 _cache.colors[itemInt] = color;
+               } // Value of root key will be null if no string exists
+               // If string exists, value is an object with key 'auto' for string
 
-       function loadNextTilePage$2(which, url, tile) {
-         var cache = _ssCache[which];
-         var nextPage = cache.nextPage[tile.id] || 0;
-         var id = tile.id + ',' + String(nextPage);
-         if (cache.loaded[id] || cache.inflight[id]) return;
-         cache.inflight[id] = getBubbles(url, tile, function (bubbles) {
-           cache.loaded[id] = true;
-           delete cache.inflight[id];
-           if (!bubbles) return; // [].shift() removes the first element, some statistics info, not a bubble point
 
-           bubbles.shift();
-           var features = bubbles.map(function (bubble) {
-             if (cache.points[bubble.id]) return null; // skip duplicates
+               var title = cl.title,
+                   detail = cl.detail,
+                   fix = cl.fix,
+                   trap = cl.trap; // Osmose titles shouldn't contain markdown
 
-             var loc = [bubble.lo, bubble.la];
-             var d = {
-               loc: loc,
-               key: bubble.id,
-               ca: bubble.he,
-               captured_at: bubble.cd,
-               captured_by: 'microsoft',
-               // nbn: bubble.nbn,
-               // pbn: bubble.pbn,
-               // ad: bubble.ad,
-               // rn: bubble.rn,
-               pr: bubble.pr,
-               // previous
-               ne: bubble.ne,
-               // next
-               pano: true,
-               sequenceKey: null
+               var issueStrings = {};
+               if (title) issueStrings.title = title.auto;
+               if (detail) issueStrings.detail = marked_1(detail.auto);
+               if (trap) issueStrings.trap = marked_1(trap.auto);
+               if (fix) issueStrings.fix = marked_1(fix.auto);
+               _cache.strings[locale][itemType] = issueStrings;
              };
-             cache.points[bubble.id] = d; // a sequence starts here
 
-             if (bubble.pr === undefined) {
-               cache.leaders.push(bubble.id);
-             }
+             var _itemType$split = itemType.split('-'),
+                 _itemType$split2 = _slicedToArray(_itemType$split, 2),
+                 item = _itemType$split2[0],
+                 cl = _itemType$split2[1]; // Osmose API falls back to English strings where untranslated or if locale doesn't exist
 
-             return {
-               minX: loc[0],
-               minY: loc[1],
-               maxX: loc[0],
-               maxY: loc[1],
-               data: d
-             };
+
+             var url = "".concat(_osmoseUrlRoot, "/items/").concat(item, "/class/").concat(cl, "?langs=").concat(locale);
+             return d3_json(url).then(cacheData);
            }).filter(Boolean);
-           cache.rtree.load(features);
-           connectSequences();
+           return Promise.all(allRequests).then(function () {
+             return _cache.strings[locale];
+           });
+         },
+         getStrings: function getStrings(itemType) {
+           var locale = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : _mainLocalizer.localeCode();
+           // No need to fallback to English, Osmose API handles this for us
+           return locale in _cache.strings ? _cache.strings[locale][itemType] : {};
+         },
+         getColor: function getColor(itemType) {
+           return itemType in _cache.colors ? _cache.colors[itemType] : '#FFFFFF';
+         },
+         postUpdate: function postUpdate(issue, callback) {
+           var _this3 = this;
 
-           if (which === 'bubbles') {
-             dispatch$7.call('loadedImages');
-           }
-         });
-       } // call this sometimes to connect the bubbles into sequences
+           if (_cache.inflightPost[issue.id]) {
+             return callback({
+               message: 'Issue update already inflight',
+               status: -2
+             }, issue);
+           } // UI sets the status to either 'done' or 'false'
 
 
-       function connectSequences() {
-         var cache = _ssCache.bubbles;
-         var keepLeaders = [];
+           var url = "".concat(_osmoseUrlRoot, "/issue/").concat(issue.id, "/").concat(issue.newStatus);
+           var controller = new AbortController();
 
-         for (var i = 0; i < cache.leaders.length; i++) {
-           var bubble = cache.points[cache.leaders[i]];
-           var seen = {}; // try to make a sequence.. use the key of the leader bubble.
+           var after = function after() {
+             delete _cache.inflightPost[issue.id];
 
-           var sequence = {
-             key: bubble.key,
-             bubbles: []
-           };
-           var complete = false;
+             _this3.removeItem(issue);
 
-           do {
-             sequence.bubbles.push(bubble);
-             seen[bubble.key] = true;
+             if (issue.newStatus === 'done') {
+               // Keep track of the number of issues closed per `item` to tag the changeset
+               if (!(issue.item in _cache.closed)) {
+                 _cache.closed[issue.item] = 0;
+               }
 
-             if (bubble.ne === undefined) {
-               complete = true;
-             } else {
-               bubble = cache.points[bubble.ne]; // advance to next
+               _cache.closed[issue.item] += 1;
              }
-           } while (bubble && !seen[bubble.key] && !complete);
 
-           if (complete) {
-             _ssCache.sequences[sequence.key] = sequence; // assign bubbles to the sequence
+             if (callback) callback(null, issue);
+           };
 
-             for (var j = 0; j < sequence.bubbles.length; j++) {
-               sequence.bubbles[j].sequenceKey = sequence.key;
-             } // create a GeoJSON LineString
+           _cache.inflightPost[issue.id] = controller;
+           fetch(url, {
+             signal: controller.signal
+           }).then(after)["catch"](function (err) {
+             delete _cache.inflightPost[issue.id];
+             if (callback) callback(err.message);
+           });
+         },
+         // Get all cached QAItems covering the viewport
+         getItems: function getItems(projection) {
+           var viewport = projection.clipExtent();
+           var min = [viewport[0][0], viewport[1][1]];
+           var max = [viewport[1][0], viewport[0][1]];
+           var bbox = geoExtent(projection.invert(min), projection.invert(max)).bbox();
+           return _cache.rtree.search(bbox).map(function (d) {
+             return d.data;
+           });
+         },
+         // Get a QAItem from cache
+         // NOTE: Don't change method name until UI v3 is merged
+         getError: function getError(id) {
+           return _cache.data[id];
+         },
+         // get the name of the icon to display for this item
+         getIcon: function getIcon(itemType) {
+           return _osmoseData.icons[itemType];
+         },
+         // Replace a single QAItem in the cache
+         replaceItem: function replaceItem(item) {
+           if (!(item instanceof QAItem) || !item.id) return;
+           _cache.data[item.id] = item;
+           updateRtree$1(encodeIssueRtree(item), true); // true = replace
+
+           return item;
+         },
+         // Remove a single QAItem from the cache
+         removeItem: function removeItem(item) {
+           if (!(item instanceof QAItem) || !item.id) return;
+           delete _cache.data[item.id];
+           updateRtree$1(encodeIssueRtree(item), false); // false = remove
+         },
+         // Used to populate `closed:osmose:*` changeset tags
+         getClosedCounts: function getClosedCounts() {
+           return _cache.closed;
+         },
+         itemURL: function itemURL(item) {
+           return "https://osmose.openstreetmap.fr/en/error/".concat(item.id);
+         }
+       };
+
+       /*! ieee754. BSD-3-Clause License. Feross Aboukhadijeh <https://feross.org/opensource> */
+       var read$6 = function read(buffer, offset, isLE, mLen, nBytes) {
+         var e, m;
+         var eLen = nBytes * 8 - mLen - 1;
+         var eMax = (1 << eLen) - 1;
+         var eBias = eMax >> 1;
+         var nBits = -7;
+         var i = isLE ? nBytes - 1 : 0;
+         var d = isLE ? -1 : 1;
+         var s = buffer[offset + i];
+         i += d;
+         e = s & (1 << -nBits) - 1;
+         s >>= -nBits;
+         nBits += eLen;
+
+         for (; nBits > 0; e = e * 256 + buffer[offset + i], i += d, nBits -= 8) {}
+
+         m = e & (1 << -nBits) - 1;
+         e >>= -nBits;
+         nBits += mLen;
+
+         for (; nBits > 0; m = m * 256 + buffer[offset + i], i += d, nBits -= 8) {}
 
+         if (e === 0) {
+           e = 1 - eBias;
+         } else if (e === eMax) {
+           return m ? NaN : (s ? -1 : 1) * Infinity;
+         } else {
+           m = m + Math.pow(2, mLen);
+           e = e - eBias;
+         }
 
-             sequence.geojson = {
-               type: 'LineString',
-               properties: {
-                 captured_at: sequence.bubbles[0] ? sequence.bubbles[0].captured_at : null,
-                 captured_by: sequence.bubbles[0] ? sequence.bubbles[0].captured_by : null,
-                 key: sequence.key
-               },
-               coordinates: sequence.bubbles.map(function (d) {
-                 return d.loc;
-               })
-             };
-           } else {
-             keepLeaders.push(cache.leaders[i]);
-           }
-         } // couldn't complete these, save for later
+         return (s ? -1 : 1) * m * Math.pow(2, e - mLen);
+       };
 
+       var write$6 = function write(buffer, value, offset, isLE, mLen, nBytes) {
+         var e, m, c;
+         var eLen = nBytes * 8 - mLen - 1;
+         var eMax = (1 << eLen) - 1;
+         var eBias = eMax >> 1;
+         var rt = mLen === 23 ? Math.pow(2, -24) - Math.pow(2, -77) : 0;
+         var i = isLE ? 0 : nBytes - 1;
+         var d = isLE ? 1 : -1;
+         var s = value < 0 || value === 0 && 1 / value < 0 ? 1 : 0;
+         value = Math.abs(value);
 
-         cache.leaders = keepLeaders;
-       }
-       /**
-        * getBubbles() handles the request to the server for a tile extent of 'bubbles' (streetside image locations).
-        */
+         if (isNaN(value) || value === Infinity) {
+           m = isNaN(value) ? 1 : 0;
+           e = eMax;
+         } else {
+           e = Math.floor(Math.log(value) / Math.LN2);
 
+           if (value * (c = Math.pow(2, -e)) < 1) {
+             e--;
+             c *= 2;
+           }
 
-       function getBubbles(url, tile, callback) {
-         var rect = tile.extent.rectangle();
-         var urlForRequest = url + utilQsString({
-           n: rect[3],
-           s: rect[1],
-           e: rect[2],
-           w: rect[0],
-           c: maxResults$2,
-           appkey: bubbleAppKey,
-           jsCallback: '{callback}'
-         });
-         return jsonpRequest(urlForRequest, function (data) {
-           if (!data || data.error) {
-             callback(null);
+           if (e + eBias >= 1) {
+             value += rt / c;
            } else {
-             callback(data);
+             value += rt * Math.pow(2, 1 - eBias);
            }
-         });
-       } // partition viewport into higher zoom tiles
-
 
-       function partitionViewport$2(projection) {
-         var z = geoScaleToZoom(projection.scale());
-         var z2 = Math.ceil(z * 2) / 2 + 2.5; // round to next 0.5 and add 2.5
+           if (value * c >= 2) {
+             e++;
+             c /= 2;
+           }
 
-         var tiler = utilTiler().zoomExtent([z2, z2]);
-         return tiler.getTiles(projection).map(function (tile) {
-           return tile.extent;
-         });
-       } // no more than `limit` results per partition.
+           if (e + eBias >= eMax) {
+             m = 0;
+             e = eMax;
+           } else if (e + eBias >= 1) {
+             m = (value * c - 1) * Math.pow(2, mLen);
+             e = e + eBias;
+           } else {
+             m = value * Math.pow(2, eBias - 1) * Math.pow(2, mLen);
+             e = 0;
+           }
+         }
 
+         for (; mLen >= 8; buffer[offset + i] = m & 0xff, i += d, m /= 256, mLen -= 8) {}
 
-       function searchLimited$2(limit, projection, rtree) {
-         limit = limit || 5;
-         return partitionViewport$2(projection).reduce(function (result, extent) {
-           var found = rtree.search(extent.bbox()).slice(0, limit).map(function (d) {
-             return d.data;
-           });
-           return found.length ? result.concat(found) : result;
-         }, []);
-       }
-       /**
-        * loadImage()
-        */
+         e = e << mLen | m;
+         eLen += mLen;
 
+         for (; eLen > 0; buffer[offset + i] = e & 0xff, i += d, e /= 256, eLen -= 8) {}
 
-       function loadImage(imgInfo) {
-         return new Promise(function (resolve) {
-           var img = new Image();
+         buffer[offset + i - d] |= s * 128;
+       };
 
-           img.onload = function () {
-             var canvas = document.getElementById('ideditor-canvas' + imgInfo.face);
-             var ctx = canvas.getContext('2d');
-             ctx.drawImage(img, imgInfo.x, imgInfo.y);
-             resolve({
-               imgInfo: imgInfo,
-               status: 'ok'
-             });
-           };
+       var ieee754 = {
+         read: read$6,
+         write: write$6
+       };
 
-           img.onerror = function () {
-             resolve({
-               data: imgInfo,
-               status: 'error'
-             });
-           };
+       var pbf = Pbf;
 
-           img.setAttribute('crossorigin', '');
-           img.src = imgInfo.url;
-         });
+       function Pbf(buf) {
+         this.buf = ArrayBuffer.isView && ArrayBuffer.isView(buf) ? buf : new Uint8Array(buf || 0);
+         this.pos = 0;
+         this.type = 0;
+         this.length = this.buf.length;
        }
-       /**
-        * loadCanvas()
-        */
 
+       Pbf.Varint = 0; // varint: int32, int64, uint32, uint64, sint32, sint64, bool, enum
 
-       function loadCanvas(imageGroup) {
-         return Promise.all(imageGroup.map(loadImage)).then(function (data) {
-           var canvas = document.getElementById('ideditor-canvas' + data[0].imgInfo.face);
-           var which = {
-             '01': 0,
-             '02': 1,
-             '03': 2,
-             '10': 3,
-             '11': 4,
-             '12': 5
-           };
-           var face = data[0].imgInfo.face;
-           _sceneOptions.cubeMap[which[face]] = canvas.toDataURL('image/jpeg', 1.0);
-           return {
-             status: 'loadCanvas for face ' + data[0].imgInfo.face + 'ok'
-           };
-         });
-       }
-       /**
-        * loadFaces()
-        */
+       Pbf.Fixed64 = 1; // 64-bit: double, fixed64, sfixed64
 
+       Pbf.Bytes = 2; // length-delimited: string, bytes, embedded messages, packed repeated fields
 
-       function loadFaces(faceGroup) {
-         return Promise.all(faceGroup.map(loadCanvas)).then(function () {
-           return {
-             status: 'loadFaces done'
-           };
-         });
-       }
+       Pbf.Fixed32 = 5; // 32-bit: float, fixed32, sfixed32
 
-       function setupCanvas(selection, reset) {
-         if (reset) {
-           selection.selectAll('#ideditor-stitcher-canvases').remove();
-         } // Add the Streetside working canvases. These are used for 'stitching', or combining,
-         // multiple images for each of the six faces, before passing to the Pannellum control as DataUrls
+       var SHIFT_LEFT_32 = (1 << 16) * (1 << 16),
+           SHIFT_RIGHT_32 = 1 / SHIFT_LEFT_32; // Threshold chosen based on both benchmarking and knowledge about browser string
+       // data structures (which currently switch structure types at 12 bytes or more)
 
+       var TEXT_DECODER_MIN_LENGTH = 12;
+       var utf8TextDecoder = typeof TextDecoder === 'undefined' ? null : new TextDecoder('utf8');
+       Pbf.prototype = {
+         destroy: function destroy() {
+           this.buf = null;
+         },
+         // === READING =================================================================
+         readFields: function readFields(readField, result, end) {
+           end = end || this.length;
 
-         selection.selectAll('#ideditor-stitcher-canvases').data([0]).enter().append('div').attr('id', 'ideditor-stitcher-canvases').attr('display', 'none').selectAll('canvas').data(['canvas01', 'canvas02', 'canvas03', 'canvas10', 'canvas11', 'canvas12']).enter().append('canvas').attr('id', function (d) {
-           return 'ideditor-' + d;
-         }).attr('width', _resolution).attr('height', _resolution);
-       }
+           while (this.pos < end) {
+             var val = this.readVarint(),
+                 tag = val >> 3,
+                 startPos = this.pos;
+             this.type = val & 0x7;
+             readField(tag, result, this);
+             if (this.pos === startPos) this.skip(val);
+           }
 
-       function qkToXY(qk) {
-         var x = 0;
-         var y = 0;
-         var scale = 256;
+           return result;
+         },
+         readMessage: function readMessage(readField, result) {
+           return this.readFields(readField, result, this.readVarint() + this.pos);
+         },
+         readFixed32: function readFixed32() {
+           var val = readUInt32(this.buf, this.pos);
+           this.pos += 4;
+           return val;
+         },
+         readSFixed32: function readSFixed32() {
+           var val = readInt32(this.buf, this.pos);
+           this.pos += 4;
+           return val;
+         },
+         // 64-bit int handling is based on github.com/dpw/node-buffer-more-ints (MIT-licensed)
+         readFixed64: function readFixed64() {
+           var val = readUInt32(this.buf, this.pos) + readUInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32;
+           this.pos += 8;
+           return val;
+         },
+         readSFixed64: function readSFixed64() {
+           var val = readUInt32(this.buf, this.pos) + readInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32;
+           this.pos += 8;
+           return val;
+         },
+         readFloat: function readFloat() {
+           var val = ieee754.read(this.buf, this.pos, true, 23, 4);
+           this.pos += 4;
+           return val;
+         },
+         readDouble: function readDouble() {
+           var val = ieee754.read(this.buf, this.pos, true, 52, 8);
+           this.pos += 8;
+           return val;
+         },
+         readVarint: function readVarint(isSigned) {
+           var buf = this.buf,
+               val,
+               b;
+           b = buf[this.pos++];
+           val = b & 0x7f;
+           if (b < 0x80) return val;
+           b = buf[this.pos++];
+           val |= (b & 0x7f) << 7;
+           if (b < 0x80) return val;
+           b = buf[this.pos++];
+           val |= (b & 0x7f) << 14;
+           if (b < 0x80) return val;
+           b = buf[this.pos++];
+           val |= (b & 0x7f) << 21;
+           if (b < 0x80) return val;
+           b = buf[this.pos];
+           val |= (b & 0x0f) << 28;
+           return readVarintRemainder(val, isSigned, this);
+         },
+         readVarint64: function readVarint64() {
+           // for compatibility with v2.0.1
+           return this.readVarint(true);
+         },
+         readSVarint: function readSVarint() {
+           var num = this.readVarint();
+           return num % 2 === 1 ? (num + 1) / -2 : num / 2; // zigzag encoding
+         },
+         readBoolean: function readBoolean() {
+           return Boolean(this.readVarint());
+         },
+         readString: function readString() {
+           var end = this.readVarint() + this.pos;
+           var pos = this.pos;
+           this.pos = end;
 
-         for (var i = qk.length; i > 0; i--) {
-           var key = qk[i - 1];
-           x += +(key === '1' || key === '3') * scale;
-           y += +(key === '2' || key === '3') * scale;
-           scale *= 2;
-         }
+           if (end - pos >= TEXT_DECODER_MIN_LENGTH && utf8TextDecoder) {
+             // longer strings are fast with the built-in browser TextDecoder API
+             return readUtf8TextDecoder(this.buf, pos, end);
+           } // short strings are fast with our custom implementation
 
-         return [x, y];
-       }
 
-       function getQuadKeys() {
-         var dim = _resolution / 256;
-         var quadKeys;
+           return readUtf8(this.buf, pos, end);
+         },
+         readBytes: function readBytes() {
+           var end = this.readVarint() + this.pos,
+               buffer = this.buf.subarray(this.pos, end);
+           this.pos = end;
+           return buffer;
+         },
+         // verbose for performance reasons; doesn't affect gzipped size
+         readPackedVarint: function readPackedVarint(arr, isSigned) {
+           if (this.type !== Pbf.Bytes) return arr.push(this.readVarint(isSigned));
+           var end = readPackedEnd(this);
+           arr = arr || [];
 
-         if (dim === 16) {
-           quadKeys = ['0000', '0001', '0010', '0011', '0100', '0101', '0110', '0111', '1000', '1001', '1010', '1011', '1100', '1101', '1110', '1111', '0002', '0003', '0012', '0013', '0102', '0103', '0112', '0113', '1002', '1003', '1012', '1013', '1102', '1103', '1112', '1113', '0020', '0021', '0030', '0031', '0120', '0121', '0130', '0131', '1020', '1021', '1030', '1031', '1120', '1121', '1130', '1131', '0022', '0023', '0032', '0033', '0122', '0123', '0132', '0133', '1022', '1023', '1032', '1033', '1122', '1123', '1132', '1133', '0200', '0201', '0210', '0211', '0300', '0301', '0310', '0311', '1200', '1201', '1210', '1211', '1300', '1301', '1310', '1311', '0202', '0203', '0212', '0213', '0302', '0303', '0312', '0313', '1202', '1203', '1212', '1213', '1302', '1303', '1312', '1313', '0220', '0221', '0230', '0231', '0320', '0321', '0330', '0331', '1220', '1221', '1230', '1231', '1320', '1321', '1330', '1331', '0222', '0223', '0232', '0233', '0322', '0323', '0332', '0333', '1222', '1223', '1232', '1233', '1322', '1323', '1332', '1333', '2000', '2001', '2010', '2011', '2100', '2101', '2110', '2111', '3000', '3001', '3010', '3011', '3100', '3101', '3110', '3111', '2002', '2003', '2012', '2013', '2102', '2103', '2112', '2113', '3002', '3003', '3012', '3013', '3102', '3103', '3112', '3113', '2020', '2021', '2030', '2031', '2120', '2121', '2130', '2131', '3020', '3021', '3030', '3031', '3120', '3121', '3130', '3131', '2022', '2023', '2032', '2033', '2122', '2123', '2132', '2133', '3022', '3023', '3032', '3033', '3122', '3123', '3132', '3133', '2200', '2201', '2210', '2211', '2300', '2301', '2310', '2311', '3200', '3201', '3210', '3211', '3300', '3301', '3310', '3311', '2202', '2203', '2212', '2213', '2302', '2303', '2312', '2313', '3202', '3203', '3212', '3213', '3302', '3303', '3312', '3313', '2220', '2221', '2230', '2231', '2320', '2321', '2330', '2331', '3220', '3221', '3230', '3231', '3320', '3321', '3330', '3331', '2222', '2223', '2232', '2233', '2322', '2323', '2332', '2333', '3222', '3223', '3232', '3233', '3322', '3323', '3332', '3333'];
-         } else if (dim === 8) {
-           quadKeys = ['000', '001', '010', '011', '100', '101', '110', '111', '002', '003', '012', '013', '102', '103', '112', '113', '020', '021', '030', '031', '120', '121', '130', '131', '022', '023', '032', '033', '122', '123', '132', '133', '200', '201', '210', '211', '300', '301', '310', '311', '202', '203', '212', '213', '302', '303', '312', '313', '220', '221', '230', '231', '320', '321', '330', '331', '222', '223', '232', '233', '322', '323', '332', '333'];
-         } else if (dim === 4) {
-           quadKeys = ['00', '01', '10', '11', '02', '03', '12', '13', '20', '21', '30', '31', '22', '23', '32', '33'];
-         } else {
-           // dim === 2
-           quadKeys = ['0', '1', '2', '3'];
-         }
+           while (this.pos < end) {
+             arr.push(this.readVarint(isSigned));
+           }
 
-         return quadKeys;
-       }
+           return arr;
+         },
+         readPackedSVarint: function readPackedSVarint(arr) {
+           if (this.type !== Pbf.Bytes) return arr.push(this.readSVarint());
+           var end = readPackedEnd(this);
+           arr = arr || [];
 
-       var serviceStreetside = {
-         /**
-          * init() initialize streetside.
-          */
-         init: function init() {
-           if (!_ssCache) {
-             this.reset();
+           while (this.pos < end) {
+             arr.push(this.readSVarint());
            }
 
-           this.event = utilRebind(this, dispatch$7, 'on');
+           return arr;
          },
+         readPackedBoolean: function readPackedBoolean(arr) {
+           if (this.type !== Pbf.Bytes) return arr.push(this.readBoolean());
+           var end = readPackedEnd(this);
+           arr = arr || [];
 
-         /**
-          * reset() reset the cache.
-          */
-         reset: function reset() {
-           if (_ssCache) {
-             Object.values(_ssCache.bubbles.inflight).forEach(abortRequest$6);
+           while (this.pos < end) {
+             arr.push(this.readBoolean());
            }
 
-           _ssCache = {
-             bubbles: {
-               inflight: {},
-               loaded: {},
-               nextPage: {},
-               rtree: new RBush(),
-               points: {},
-               leaders: []
-             },
-             sequences: {}
-           };
+           return arr;
          },
+         readPackedFloat: function readPackedFloat(arr) {
+           if (this.type !== Pbf.Bytes) return arr.push(this.readFloat());
+           var end = readPackedEnd(this);
+           arr = arr || [];
 
-         /**
-          * bubbles()
-          */
-         bubbles: function bubbles(projection) {
-           var limit = 5;
-           return searchLimited$2(limit, projection, _ssCache.bubbles.rtree);
-         },
-         cachedImage: function cachedImage(imageKey) {
-           return _ssCache.bubbles.points[imageKey];
-         },
-         sequences: function sequences(projection) {
-           var viewport = projection.clipExtent();
-           var min = [viewport[0][0], viewport[1][1]];
-           var max = [viewport[1][0], viewport[0][1]];
-           var bbox = geoExtent(projection.invert(min), projection.invert(max)).bbox();
-           var seen = {};
-           var results = []; // all sequences for bubbles in viewport
+           while (this.pos < end) {
+             arr.push(this.readFloat());
+           }
 
-           _ssCache.bubbles.rtree.search(bbox).forEach(function (d) {
-             var key = d.data.sequenceKey;
+           return arr;
+         },
+         readPackedDouble: function readPackedDouble(arr) {
+           if (this.type !== Pbf.Bytes) return arr.push(this.readDouble());
+           var end = readPackedEnd(this);
+           arr = arr || [];
 
-             if (key && !seen[key]) {
-               seen[key] = true;
-               results.push(_ssCache.sequences[key].geojson);
-             }
-           });
+           while (this.pos < end) {
+             arr.push(this.readDouble());
+           }
 
-           return results;
+           return arr;
          },
+         readPackedFixed32: function readPackedFixed32(arr) {
+           if (this.type !== Pbf.Bytes) return arr.push(this.readFixed32());
+           var end = readPackedEnd(this);
+           arr = arr || [];
 
-         /**
-          * loadBubbles()
-          */
-         loadBubbles: function loadBubbles(projection, margin) {
-           // by default: request 2 nearby tiles so we can connect sequences.
-           if (margin === undefined) margin = 2;
-           loadTiles$2('bubbles', bubbleApi, projection, margin);
-         },
-         viewer: function viewer() {
-           return _pannellumViewer;
+           while (this.pos < end) {
+             arr.push(this.readFixed32());
+           }
+
+           return arr;
          },
-         initViewer: function initViewer() {
-           if (!window.pannellum) return;
-           if (_pannellumViewer) return;
-           _currScene += 1;
+         readPackedSFixed32: function readPackedSFixed32(arr) {
+           if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed32());
+           var end = readPackedEnd(this);
+           arr = arr || [];
 
-           var sceneID = _currScene.toString();
+           while (this.pos < end) {
+             arr.push(this.readSFixed32());
+           }
 
-           var options = {
-             'default': {
-               firstScene: sceneID
-             },
-             scenes: {}
-           };
-           options.scenes[sceneID] = _sceneOptions;
-           _pannellumViewer = window.pannellum.viewer('ideditor-viewer-streetside', options);
+           return arr;
          },
-         ensureViewerLoaded: function ensureViewerLoaded(context) {
-           if (_loadViewerPromise$2) return _loadViewerPromise$2; // create ms-wrapper, a photo wrapper class
-
-           var wrap = context.container().select('.photoviewer').selectAll('.ms-wrapper').data([0]); // inject ms-wrapper into the photoviewer div
-           // (used by all to house each custom photo viewer)
+         readPackedFixed64: function readPackedFixed64(arr) {
+           if (this.type !== Pbf.Bytes) return arr.push(this.readFixed64());
+           var end = readPackedEnd(this);
+           arr = arr || [];
 
-           var wrapEnter = wrap.enter().append('div').attr('class', 'photo-wrapper ms-wrapper').classed('hide', true);
-           var that = this;
-           var pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse'; // inject div to support streetside viewer (pannellum) and attribution line
+           while (this.pos < end) {
+             arr.push(this.readFixed64());
+           }
 
-           wrapEnter.append('div').attr('id', 'ideditor-viewer-streetside').on(pointerPrefix + 'down.streetside', function () {
-             select(window).on(pointerPrefix + 'move.streetside', function () {
-               dispatch$7.call('viewerChanged');
-             }, true);
-           }).on(pointerPrefix + 'up.streetside pointercancel.streetside', function () {
-             select(window).on(pointerPrefix + 'move.streetside', null); // continue dispatching events for a few seconds, in case viewer has inertia.
+           return arr;
+         },
+         readPackedSFixed64: function readPackedSFixed64(arr) {
+           if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed64());
+           var end = readPackedEnd(this);
+           arr = arr || [];
 
-             var t = timer(function (elapsed) {
-               dispatch$7.call('viewerChanged');
+           while (this.pos < end) {
+             arr.push(this.readSFixed64());
+           }
 
-               if (elapsed > 2000) {
-                 t.stop();
-               }
-             });
-           }).append('div').attr('class', 'photo-attribution fillD');
-           var controlsEnter = wrapEnter.append('div').attr('class', 'photo-controls-wrap').append('div').attr('class', 'photo-controls');
-           controlsEnter.append('button').on('click.back', step(-1)).html('◄');
-           controlsEnter.append('button').on('click.forward', step(1)).html('►'); // create working canvas for stitching together images
+           return arr;
+         },
+         skip: function skip(val) {
+           var type = val & 0x7;
+           if (type === Pbf.Varint) while (this.buf[this.pos++] > 0x7f) {} else if (type === Pbf.Bytes) this.pos = this.readVarint() + this.pos;else if (type === Pbf.Fixed32) this.pos += 4;else if (type === Pbf.Fixed64) this.pos += 8;else throw new Error('Unimplemented type: ' + type);
+         },
+         // === WRITING =================================================================
+         writeTag: function writeTag(tag, type) {
+           this.writeVarint(tag << 3 | type);
+         },
+         realloc: function realloc(min) {
+           var length = this.length || 16;
 
-           wrap = wrap.merge(wrapEnter).call(setupCanvas, true); // Register viewer resize handler
+           while (length < this.pos + min) {
+             length *= 2;
+           }
 
-           context.ui().photoviewer.on('resize.streetside', function () {
-             if (_pannellumViewer) {
-               _pannellumViewer.resize();
-             }
-           });
-           _loadViewerPromise$2 = new Promise(function (resolve, reject) {
-             var loadedCount = 0;
+           if (length !== this.length) {
+             var buf = new Uint8Array(length);
+             buf.set(this.buf);
+             this.buf = buf;
+             this.length = length;
+           }
+         },
+         finish: function finish() {
+           this.length = this.pos;
+           this.pos = 0;
+           return this.buf.subarray(0, this.length);
+         },
+         writeFixed32: function writeFixed32(val) {
+           this.realloc(4);
+           writeInt32(this.buf, val, this.pos);
+           this.pos += 4;
+         },
+         writeSFixed32: function writeSFixed32(val) {
+           this.realloc(4);
+           writeInt32(this.buf, val, this.pos);
+           this.pos += 4;
+         },
+         writeFixed64: function writeFixed64(val) {
+           this.realloc(8);
+           writeInt32(this.buf, val & -1, this.pos);
+           writeInt32(this.buf, Math.floor(val * SHIFT_RIGHT_32), this.pos + 4);
+           this.pos += 8;
+         },
+         writeSFixed64: function writeSFixed64(val) {
+           this.realloc(8);
+           writeInt32(this.buf, val & -1, this.pos);
+           writeInt32(this.buf, Math.floor(val * SHIFT_RIGHT_32), this.pos + 4);
+           this.pos += 8;
+         },
+         writeVarint: function writeVarint(val) {
+           val = +val || 0;
 
-             function loaded() {
-               loadedCount += 1; // wait until both files are loaded
+           if (val > 0xfffffff || val < 0) {
+             writeBigVarint(val, this);
+             return;
+           }
 
-               if (loadedCount === 2) resolve();
-             }
+           this.realloc(4);
+           this.buf[this.pos++] = val & 0x7f | (val > 0x7f ? 0x80 : 0);
+           if (val <= 0x7f) return;
+           this.buf[this.pos++] = (val >>>= 7) & 0x7f | (val > 0x7f ? 0x80 : 0);
+           if (val <= 0x7f) return;
+           this.buf[this.pos++] = (val >>>= 7) & 0x7f | (val > 0x7f ? 0x80 : 0);
+           if (val <= 0x7f) return;
+           this.buf[this.pos++] = val >>> 7 & 0x7f;
+         },
+         writeSVarint: function writeSVarint(val) {
+           this.writeVarint(val < 0 ? -val * 2 - 1 : val * 2);
+         },
+         writeBoolean: function writeBoolean(val) {
+           this.writeVarint(Boolean(val));
+         },
+         writeString: function writeString(str) {
+           str = String(str);
+           this.realloc(str.length * 4);
+           this.pos++; // reserve 1 byte for short string length
 
-             var head = select('head'); // load streetside pannellum viewer css
+           var startPos = this.pos; // write the string directly to the buffer and see how much was written
 
-             head.selectAll('#ideditor-streetside-viewercss').data([0]).enter().append('link').attr('id', 'ideditor-streetside-viewercss').attr('rel', 'stylesheet').attr('crossorigin', 'anonymous').attr('href', context.asset(pannellumViewerCSS)).on('load.serviceStreetside', loaded).on('error.serviceStreetside', function () {
-               reject();
-             }); // load streetside pannellum viewer js
+           this.pos = writeUtf8(this.buf, str, this.pos);
+           var len = this.pos - startPos;
+           if (len >= 0x80) makeRoomForExtraLength(startPos, len, this); // finally, write the message length in the reserved place and restore the position
 
-             head.selectAll('#ideditor-streetside-viewerjs').data([0]).enter().append('script').attr('id', 'ideditor-streetside-viewerjs').attr('crossorigin', 'anonymous').attr('src', context.asset(pannellumViewerJS)).on('load.serviceStreetside', loaded).on('error.serviceStreetside', function () {
-               reject();
-             });
-           })["catch"](function () {
-             _loadViewerPromise$2 = null;
-           });
-           return _loadViewerPromise$2;
+           this.pos = startPos - 1;
+           this.writeVarint(len);
+           this.pos += len;
+         },
+         writeFloat: function writeFloat(val) {
+           this.realloc(4);
+           ieee754.write(this.buf, val, this.pos, true, 23, 4);
+           this.pos += 4;
+         },
+         writeDouble: function writeDouble(val) {
+           this.realloc(8);
+           ieee754.write(this.buf, val, this.pos, true, 52, 8);
+           this.pos += 8;
+         },
+         writeBytes: function writeBytes(buffer) {
+           var len = buffer.length;
+           this.writeVarint(len);
+           this.realloc(len);
 
-           function step(stepBy) {
-             return function () {
-               var viewer = context.container().select('.photoviewer');
-               var selected = viewer.empty() ? undefined : viewer.datum();
-               if (!selected) return;
-               var nextID = stepBy === 1 ? selected.ne : selected.pr;
+           for (var i = 0; i < len; i++) {
+             this.buf[this.pos++] = buffer[i];
+           }
+         },
+         writeRawMessage: function writeRawMessage(fn, obj) {
+           this.pos++; // reserve 1 byte for short message length
+           // write the message directly to the buffer and see how much was written
 
-               var yaw = _pannellumViewer.getYaw();
+           var startPos = this.pos;
+           fn(obj, this);
+           var len = this.pos - startPos;
+           if (len >= 0x80) makeRoomForExtraLength(startPos, len, this); // finally, write the message length in the reserved place and restore the position
 
-               var ca = selected.ca + yaw;
-               var origin = selected.loc; // construct a search trapezoid pointing out from current bubble
+           this.pos = startPos - 1;
+           this.writeVarint(len);
+           this.pos += len;
+         },
+         writeMessage: function writeMessage(tag, fn, obj) {
+           this.writeTag(tag, Pbf.Bytes);
+           this.writeRawMessage(fn, obj);
+         },
+         writePackedVarint: function writePackedVarint(tag, arr) {
+           if (arr.length) this.writeMessage(tag, _writePackedVarint, arr);
+         },
+         writePackedSVarint: function writePackedSVarint(tag, arr) {
+           if (arr.length) this.writeMessage(tag, _writePackedSVarint, arr);
+         },
+         writePackedBoolean: function writePackedBoolean(tag, arr) {
+           if (arr.length) this.writeMessage(tag, _writePackedBoolean, arr);
+         },
+         writePackedFloat: function writePackedFloat(tag, arr) {
+           if (arr.length) this.writeMessage(tag, _writePackedFloat, arr);
+         },
+         writePackedDouble: function writePackedDouble(tag, arr) {
+           if (arr.length) this.writeMessage(tag, _writePackedDouble, arr);
+         },
+         writePackedFixed32: function writePackedFixed32(tag, arr) {
+           if (arr.length) this.writeMessage(tag, _writePackedFixed, arr);
+         },
+         writePackedSFixed32: function writePackedSFixed32(tag, arr) {
+           if (arr.length) this.writeMessage(tag, _writePackedSFixed, arr);
+         },
+         writePackedFixed64: function writePackedFixed64(tag, arr) {
+           if (arr.length) this.writeMessage(tag, _writePackedFixed2, arr);
+         },
+         writePackedSFixed64: function writePackedSFixed64(tag, arr) {
+           if (arr.length) this.writeMessage(tag, _writePackedSFixed2, arr);
+         },
+         writeBytesField: function writeBytesField(tag, buffer) {
+           this.writeTag(tag, Pbf.Bytes);
+           this.writeBytes(buffer);
+         },
+         writeFixed32Field: function writeFixed32Field(tag, val) {
+           this.writeTag(tag, Pbf.Fixed32);
+           this.writeFixed32(val);
+         },
+         writeSFixed32Field: function writeSFixed32Field(tag, val) {
+           this.writeTag(tag, Pbf.Fixed32);
+           this.writeSFixed32(val);
+         },
+         writeFixed64Field: function writeFixed64Field(tag, val) {
+           this.writeTag(tag, Pbf.Fixed64);
+           this.writeFixed64(val);
+         },
+         writeSFixed64Field: function writeSFixed64Field(tag, val) {
+           this.writeTag(tag, Pbf.Fixed64);
+           this.writeSFixed64(val);
+         },
+         writeVarintField: function writeVarintField(tag, val) {
+           this.writeTag(tag, Pbf.Varint);
+           this.writeVarint(val);
+         },
+         writeSVarintField: function writeSVarintField(tag, val) {
+           this.writeTag(tag, Pbf.Varint);
+           this.writeSVarint(val);
+         },
+         writeStringField: function writeStringField(tag, str) {
+           this.writeTag(tag, Pbf.Bytes);
+           this.writeString(str);
+         },
+         writeFloatField: function writeFloatField(tag, val) {
+           this.writeTag(tag, Pbf.Fixed32);
+           this.writeFloat(val);
+         },
+         writeDoubleField: function writeDoubleField(tag, val) {
+           this.writeTag(tag, Pbf.Fixed64);
+           this.writeDouble(val);
+         },
+         writeBooleanField: function writeBooleanField(tag, val) {
+           this.writeVarintField(tag, Boolean(val));
+         }
+       };
 
-               var meters = 35;
-               var p1 = [origin[0] + geoMetersToLon(meters / 5, origin[1]), origin[1]];
-               var p2 = [origin[0] + geoMetersToLon(meters / 2, origin[1]), origin[1] + geoMetersToLat(meters)];
-               var p3 = [origin[0] - geoMetersToLon(meters / 2, origin[1]), origin[1] + geoMetersToLat(meters)];
-               var p4 = [origin[0] - geoMetersToLon(meters / 5, origin[1]), origin[1]];
-               var poly = [p1, p2, p3, p4, p1]; // rotate it to face forward/backward
+       function readVarintRemainder(l, s, p) {
+         var buf = p.buf,
+             h,
+             b;
+         b = buf[p.pos++];
+         h = (b & 0x70) >> 4;
+         if (b < 0x80) return toNum(l, h, s);
+         b = buf[p.pos++];
+         h |= (b & 0x7f) << 3;
+         if (b < 0x80) return toNum(l, h, s);
+         b = buf[p.pos++];
+         h |= (b & 0x7f) << 10;
+         if (b < 0x80) return toNum(l, h, s);
+         b = buf[p.pos++];
+         h |= (b & 0x7f) << 17;
+         if (b < 0x80) return toNum(l, h, s);
+         b = buf[p.pos++];
+         h |= (b & 0x7f) << 24;
+         if (b < 0x80) return toNum(l, h, s);
+         b = buf[p.pos++];
+         h |= (b & 0x01) << 31;
+         if (b < 0x80) return toNum(l, h, s);
+         throw new Error('Expected varint not more than 10 bytes');
+       }
 
-               var angle = (stepBy === 1 ? ca : ca + 180) * (Math.PI / 180);
-               poly = geoRotate(poly, -angle, origin);
-               var extent = poly.reduce(function (extent, point) {
-                 return extent.extend(geoExtent(point));
-               }, geoExtent()); // find nearest other bubble in the search polygon
+       function readPackedEnd(pbf) {
+         return pbf.type === Pbf.Bytes ? pbf.readVarint() + pbf.pos : pbf.pos + 1;
+       }
 
-               var minDist = Infinity;
+       function toNum(low, high, isSigned) {
+         if (isSigned) {
+           return high * 0x100000000 + (low >>> 0);
+         }
 
-               _ssCache.bubbles.rtree.search(extent.bbox()).forEach(function (d) {
-                 if (d.data.key === selected.key) return;
-                 if (!geoPointInPolygon(d.data.loc, poly)) return;
-                 var dist = geoVecLength(d.data.loc, selected.loc);
-                 var theta = selected.ca - d.data.ca;
-                 var minTheta = Math.min(Math.abs(theta), 360 - Math.abs(theta));
+         return (high >>> 0) * 0x100000000 + (low >>> 0);
+       }
 
-                 if (minTheta > 20) {
-                   dist += 5; // penalize distance if camera angles don't match
-                 }
+       function writeBigVarint(val, pbf) {
+         var low, high;
 
-                 if (dist < minDist) {
-                   nextID = d.data.key;
-                   minDist = dist;
-                 }
-               });
+         if (val >= 0) {
+           low = val % 0x100000000 | 0;
+           high = val / 0x100000000 | 0;
+         } else {
+           low = ~(-val % 0x100000000);
+           high = ~(-val / 0x100000000);
 
-               var nextBubble = nextID && that.cachedImage(nextID);
-               if (!nextBubble) return;
-               context.map().centerEase(nextBubble.loc);
-               that.selectImage(context, nextBubble.key).yaw(yaw).showViewer(context);
-             };
+           if (low ^ 0xffffffff) {
+             low = low + 1 | 0;
+           } else {
+             low = 0;
+             high = high + 1 | 0;
            }
-         },
-         yaw: function yaw(_yaw) {
-           if (typeof _yaw !== 'number') return _yaw;
-           _sceneOptions.yaw = _yaw;
-           return this;
-         },
+         }
 
-         /**
-          * showViewer()
-          */
-         showViewer: function showViewer(context) {
-           var wrap = context.container().select('.photoviewer').classed('hide', false);
-           var isHidden = wrap.selectAll('.photo-wrapper.ms-wrapper.hide').size();
+         if (val >= 0x10000000000000000 || val < -0x10000000000000000) {
+           throw new Error('Given varint doesn\'t fit into 10 bytes');
+         }
 
-           if (isHidden) {
-             wrap.selectAll('.photo-wrapper:not(.ms-wrapper)').classed('hide', true);
-             wrap.selectAll('.photo-wrapper.ms-wrapper').classed('hide', false);
-           }
+         pbf.realloc(10);
+         writeBigVarintLow(low, high, pbf);
+         writeBigVarintHigh(high, pbf);
+       }
 
-           return this;
-         },
+       function writeBigVarintLow(low, high, pbf) {
+         pbf.buf[pbf.pos++] = low & 0x7f | 0x80;
+         low >>>= 7;
+         pbf.buf[pbf.pos++] = low & 0x7f | 0x80;
+         low >>>= 7;
+         pbf.buf[pbf.pos++] = low & 0x7f | 0x80;
+         low >>>= 7;
+         pbf.buf[pbf.pos++] = low & 0x7f | 0x80;
+         low >>>= 7;
+         pbf.buf[pbf.pos] = low & 0x7f;
+       }
 
-         /**
-          * hideViewer()
-          */
-         hideViewer: function hideViewer(context) {
-           var viewer = context.container().select('.photoviewer');
-           if (!viewer.empty()) viewer.datum(null);
-           viewer.classed('hide', true).selectAll('.photo-wrapper').classed('hide', true);
-           context.container().selectAll('.viewfield-group, .sequence, .icon-sign').classed('currentView', false);
-           this.updateUrlImage(null);
-           return this.setStyles(context, null, true);
-         },
+       function writeBigVarintHigh(high, pbf) {
+         var lsb = (high & 0x07) << 4;
+         pbf.buf[pbf.pos++] |= lsb | ((high >>>= 3) ? 0x80 : 0);
+         if (!high) return;
+         pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0);
+         if (!high) return;
+         pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0);
+         if (!high) return;
+         pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0);
+         if (!high) return;
+         pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0);
+         if (!high) return;
+         pbf.buf[pbf.pos++] = high & 0x7f;
+       }
 
-         /**
-          * selectImage().
-          */
-         selectImage: function selectImage(context, key) {
-           var that = this;
-           var d = this.cachedImage(key);
-           var viewer = context.container().select('.photoviewer');
-           if (!viewer.empty()) viewer.datum(d);
-           this.setStyles(context, null, true);
-           var wrap = context.container().select('.photoviewer .ms-wrapper');
-           var attribution = wrap.selectAll('.photo-attribution').html('');
-           wrap.selectAll('.pnlm-load-box') // display "loading.."
-           .style('display', 'block');
-           if (!d) return this;
-           this.updateUrlImage(key);
-           _sceneOptions.northOffset = d.ca;
-           var line1 = attribution.append('div').attr('class', 'attribution-row');
-           var hiresDomId = utilUniqueDomId('streetside-hires'); // Add hires checkbox
+       function makeRoomForExtraLength(startPos, len, pbf) {
+         var extraLen = len <= 0x3fff ? 1 : len <= 0x1fffff ? 2 : len <= 0xfffffff ? 3 : Math.floor(Math.log(len) / (Math.LN2 * 7)); // if 1 byte isn't enough for encoding message length, shift the data to the right
 
-           var label = line1.append('label').attr('for', hiresDomId).attr('class', 'streetside-hires');
-           label.append('input').attr('type', 'checkbox').attr('id', hiresDomId).property('checked', _hires).on('click', function (d3_event) {
-             d3_event.stopPropagation();
-             _hires = !_hires;
-             _resolution = _hires ? 1024 : 512;
-             wrap.call(setupCanvas, true);
-             var viewstate = {
-               yaw: _pannellumViewer.getYaw(),
-               pitch: _pannellumViewer.getPitch(),
-               hfov: _pannellumViewer.getHfov()
-             };
-             _sceneOptions = Object.assign(_sceneOptions, viewstate);
-             that.selectImage(context, d.key).showViewer(context);
-           });
-           label.append('span').html(_t.html('streetside.hires'));
-           var captureInfo = line1.append('div').attr('class', 'attribution-capture-info'); // Add capture date
+         pbf.realloc(extraLen);
 
-           if (d.captured_by) {
-             var yyyy = new Date().getFullYear();
-             captureInfo.append('a').attr('class', 'captured_by').attr('target', '_blank').attr('href', 'https://www.microsoft.com/en-us/maps/streetside').html('©' + yyyy + ' Microsoft');
-             captureInfo.append('span').html('|');
-           }
+         for (var i = pbf.pos - 1; i >= startPos; i--) {
+           pbf.buf[i + extraLen] = pbf.buf[i];
+         }
+       }
 
-           if (d.captured_at) {
-             captureInfo.append('span').attr('class', 'captured_at').html(localeTimestamp(d.captured_at));
-           } // Add image links
+       function _writePackedVarint(arr, pbf) {
+         for (var i = 0; i < arr.length; i++) {
+           pbf.writeVarint(arr[i]);
+         }
+       }
 
+       function _writePackedSVarint(arr, pbf) {
+         for (var i = 0; i < arr.length; i++) {
+           pbf.writeSVarint(arr[i]);
+         }
+       }
 
-           var line2 = attribution.append('div').attr('class', 'attribution-row');
-           line2.append('a').attr('class', 'image-view-link').attr('target', '_blank').attr('href', 'https://www.bing.com/maps?cp=' + d.loc[1] + '~' + d.loc[0] + '&lvl=17&dir=' + d.ca + '&style=x&v=2&sV=1').html(_t.html('streetside.view_on_bing'));
-           line2.append('a').attr('class', 'image-report-link').attr('target', '_blank').attr('href', 'https://www.bing.com/maps/privacyreport/streetsideprivacyreport?bubbleid=' + encodeURIComponent(d.key) + '&focus=photo&lat=' + d.loc[1] + '&lng=' + d.loc[0] + '&z=17').html(_t.html('streetside.report'));
-           var bubbleIdQuadKey = d.key.toString(4);
-           var paddingNeeded = 16 - bubbleIdQuadKey.length;
+       function _writePackedFloat(arr, pbf) {
+         for (var i = 0; i < arr.length; i++) {
+           pbf.writeFloat(arr[i]);
+         }
+       }
 
-           for (var i = 0; i < paddingNeeded; i++) {
-             bubbleIdQuadKey = '0' + bubbleIdQuadKey;
-           }
+       function _writePackedDouble(arr, pbf) {
+         for (var i = 0; i < arr.length; i++) {
+           pbf.writeDouble(arr[i]);
+         }
+       }
 
-           var imgUrlPrefix = streetsideImagesApi + 'hs' + bubbleIdQuadKey;
-           var imgUrlSuffix = '.jpg?g=6338&n=z'; // Cubemap face code order matters here: front=01, right=02, back=03, left=10, up=11, down=12
+       function _writePackedBoolean(arr, pbf) {
+         for (var i = 0; i < arr.length; i++) {
+           pbf.writeBoolean(arr[i]);
+         }
+       }
 
-           var faceKeys = ['01', '02', '03', '10', '11', '12']; // Map images to cube faces
+       function _writePackedFixed(arr, pbf) {
+         for (var i = 0; i < arr.length; i++) {
+           pbf.writeFixed32(arr[i]);
+         }
+       }
 
-           var quadKeys = getQuadKeys();
-           var faces = faceKeys.map(function (faceKey) {
-             return quadKeys.map(function (quadKey) {
-               var xy = qkToXY(quadKey);
-               return {
-                 face: faceKey,
-                 url: imgUrlPrefix + faceKey + quadKey + imgUrlSuffix,
-                 x: xy[0],
-                 y: xy[1]
-               };
-             });
-           });
-           loadFaces(faces).then(function () {
-             if (!_pannellumViewer) {
-               that.initViewer();
-             } else {
-               // make a new scene
-               _currScene += 1;
+       function _writePackedSFixed(arr, pbf) {
+         for (var i = 0; i < arr.length; i++) {
+           pbf.writeSFixed32(arr[i]);
+         }
+       }
 
-               var sceneID = _currScene.toString();
+       function _writePackedFixed2(arr, pbf) {
+         for (var i = 0; i < arr.length; i++) {
+           pbf.writeFixed64(arr[i]);
+         }
+       }
 
-               _pannellumViewer.addScene(sceneID, _sceneOptions).loadScene(sceneID); // remove previous scene
+       function _writePackedSFixed2(arr, pbf) {
+         for (var i = 0; i < arr.length; i++) {
+           pbf.writeSFixed64(arr[i]);
+         }
+       } // Buffer code below from https://github.com/feross/buffer, MIT-licensed
 
 
-               if (_currScene > 2) {
-                 sceneID = (_currScene - 1).toString();
+       function readUInt32(buf, pos) {
+         return (buf[pos] | buf[pos + 1] << 8 | buf[pos + 2] << 16) + buf[pos + 3] * 0x1000000;
+       }
 
-                 _pannellumViewer.removeScene(sceneID);
-               }
-             }
-           });
-           return this;
-         },
-         getSequenceKeyForBubble: function getSequenceKeyForBubble(d) {
-           return d && d.sequenceKey;
-         },
-         // Updates the currently highlighted sequence and selected bubble.
-         // Reset is only necessary when interacting with the viewport because
-         // this implicitly changes the currently selected bubble/sequence
-         setStyles: function setStyles(context, hovered, reset) {
-           if (reset) {
-             // reset all layers
-             context.container().selectAll('.viewfield-group').classed('highlighted', false).classed('hovered', false).classed('currentView', false);
-             context.container().selectAll('.sequence').classed('highlighted', false).classed('currentView', false);
-           }
+       function writeInt32(buf, val, pos) {
+         buf[pos] = val;
+         buf[pos + 1] = val >>> 8;
+         buf[pos + 2] = val >>> 16;
+         buf[pos + 3] = val >>> 24;
+       }
 
-           var hoveredBubbleKey = hovered && hovered.key;
-           var hoveredSequenceKey = this.getSequenceKeyForBubble(hovered);
-           var hoveredSequence = hoveredSequenceKey && _ssCache.sequences[hoveredSequenceKey];
-           var hoveredBubbleKeys = hoveredSequence && hoveredSequence.bubbles.map(function (d) {
-             return d.key;
-           }) || [];
-           var viewer = context.container().select('.photoviewer');
-           var selected = viewer.empty() ? undefined : viewer.datum();
-           var selectedBubbleKey = selected && selected.key;
-           var selectedSequenceKey = this.getSequenceKeyForBubble(selected);
-           var selectedSequence = selectedSequenceKey && _ssCache.sequences[selectedSequenceKey];
-           var selectedBubbleKeys = selectedSequence && selectedSequence.bubbles.map(function (d) {
-             return d.key;
-           }) || []; // highlight sibling viewfields on either the selected or the hovered sequences
+       function readInt32(buf, pos) {
+         return (buf[pos] | buf[pos + 1] << 8 | buf[pos + 2] << 16) + (buf[pos + 3] << 24);
+       }
 
-           var highlightedBubbleKeys = utilArrayUnion(hoveredBubbleKeys, selectedBubbleKeys);
-           context.container().selectAll('.layer-streetside-images .viewfield-group').classed('highlighted', function (d) {
-             return highlightedBubbleKeys.indexOf(d.key) !== -1;
-           }).classed('hovered', function (d) {
-             return d.key === hoveredBubbleKey;
-           }).classed('currentView', function (d) {
-             return d.key === selectedBubbleKey;
-           });
-           context.container().selectAll('.layer-streetside-images .sequence').classed('highlighted', function (d) {
-             return d.properties.key === hoveredSequenceKey;
-           }).classed('currentView', function (d) {
-             return d.properties.key === selectedSequenceKey;
-           }); // update viewfields if needed
+       function readUtf8(buf, pos, end) {
+         var str = '';
+         var i = pos;
+
+         while (i < end) {
+           var b0 = buf[i];
+           var c = null; // codepoint
 
-           context.container().selectAll('.viewfield-group .viewfield').attr('d', viewfieldPath);
+           var bytesPerSequence = b0 > 0xEF ? 4 : b0 > 0xDF ? 3 : b0 > 0xBF ? 2 : 1;
+           if (i + bytesPerSequence > end) break;
+           var b1, b2, b3;
 
-           function viewfieldPath() {
-             var d = this.parentNode.__data__;
+           if (bytesPerSequence === 1) {
+             if (b0 < 0x80) {
+               c = b0;
+             }
+           } else if (bytesPerSequence === 2) {
+             b1 = buf[i + 1];
 
-             if (d.pano && d.key !== selectedBubbleKey) {
-               return 'M 8,13 m -10,0 a 10,10 0 1,0 20,0 a 10,10 0 1,0 -20,0';
-             } else {
-               return 'M 6,9 C 8,8.4 8,8.4 10,9 L 16,-2 C 12,-5 4,-5 0,-2 z';
+             if ((b1 & 0xC0) === 0x80) {
+               c = (b0 & 0x1F) << 0x6 | b1 & 0x3F;
+
+               if (c <= 0x7F) {
+                 c = null;
+               }
              }
-           }
+           } else if (bytesPerSequence === 3) {
+             b1 = buf[i + 1];
+             b2 = buf[i + 2];
 
-           return this;
-         },
-         updateUrlImage: function updateUrlImage(imageKey) {
-           if (!window.mocha) {
-             var hash = utilStringQs(window.location.hash);
+             if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80) {
+               c = (b0 & 0xF) << 0xC | (b1 & 0x3F) << 0x6 | b2 & 0x3F;
 
-             if (imageKey) {
-               hash.photo = 'streetside/' + imageKey;
-             } else {
-               delete hash.photo;
+               if (c <= 0x7FF || c >= 0xD800 && c <= 0xDFFF) {
+                 c = null;
+               }
              }
+           } else if (bytesPerSequence === 4) {
+             b1 = buf[i + 1];
+             b2 = buf[i + 2];
+             b3 = buf[i + 3];
 
-             window.location.replace('#' + utilQsString(hash, true));
-           }
-         },
+             if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80 && (b3 & 0xC0) === 0x80) {
+               c = (b0 & 0xF) << 0x12 | (b1 & 0x3F) << 0xC | (b2 & 0x3F) << 0x6 | b3 & 0x3F;
 
-         /**
-          * cache().
-          */
-         cache: function cache() {
-           return _ssCache;
-         }
-       };
+               if (c <= 0xFFFF || c >= 0x110000) {
+                 c = null;
+               }
+             }
+           }
 
-       var _apibase$1 = 'https://taginfo.openstreetmap.org/api/4/';
-       var _inflight$2 = {};
-       var _popularKeys = {};
-       var _taginfoCache = {};
-       var tag_sorts = {
-         point: 'count_nodes',
-         vertex: 'count_nodes',
-         area: 'count_ways',
-         line: 'count_ways'
-       };
-       var tag_sort_members = {
-         point: 'count_node_members',
-         vertex: 'count_node_members',
-         area: 'count_way_members',
-         line: 'count_way_members',
-         relation: 'count_relation_members'
-       };
-       var tag_filters = {
-         point: 'nodes',
-         vertex: 'nodes',
-         area: 'ways',
-         line: 'ways'
-       };
-       var tag_members_fractions = {
-         point: 'count_node_members_fraction',
-         vertex: 'count_node_members_fraction',
-         area: 'count_way_members_fraction',
-         line: 'count_way_members_fraction',
-         relation: 'count_relation_members_fraction'
-       };
+           if (c === null) {
+             c = 0xFFFD;
+             bytesPerSequence = 1;
+           } else if (c > 0xFFFF) {
+             c -= 0x10000;
+             str += String.fromCharCode(c >>> 10 & 0x3FF | 0xD800);
+             c = 0xDC00 | c & 0x3FF;
+           }
 
-       function sets(params, n, o) {
-         if (params.geometry && o[params.geometry]) {
-           params[n] = o[params.geometry];
+           str += String.fromCharCode(c);
+           i += bytesPerSequence;
          }
 
-         return params;
-       }
-
-       function setFilter(params) {
-         return sets(params, 'filter', tag_filters);
+         return str;
        }
 
-       function setSort(params) {
-         return sets(params, 'sortname', tag_sorts);
+       function readUtf8TextDecoder(buf, pos, end) {
+         return utf8TextDecoder.decode(buf.subarray(pos, end));
        }
 
-       function setSortMembers(params) {
-         return sets(params, 'sortname', tag_sort_members);
-       }
+       function writeUtf8(buf, str, pos) {
+         for (var i = 0, c, lead; i < str.length; i++) {
+           c = str.charCodeAt(i); // code point
 
-       function clean(params) {
-         return utilObjectOmit(params, ['geometry', 'debounce']);
-       }
+           if (c > 0xD7FF && c < 0xE000) {
+             if (lead) {
+               if (c < 0xDC00) {
+                 buf[pos++] = 0xEF;
+                 buf[pos++] = 0xBF;
+                 buf[pos++] = 0xBD;
+                 lead = c;
+                 continue;
+               } else {
+                 c = lead - 0xD800 << 10 | c - 0xDC00 | 0x10000;
+                 lead = null;
+               }
+             } else {
+               if (c > 0xDBFF || i + 1 === str.length) {
+                 buf[pos++] = 0xEF;
+                 buf[pos++] = 0xBF;
+                 buf[pos++] = 0xBD;
+               } else {
+                 lead = c;
+               }
 
-       function filterKeys(type) {
-         var count_type = type ? 'count_' + type : 'count_all';
-         return function (d) {
-           return parseFloat(d[count_type]) > 2500 || d.in_wiki;
-         };
-       }
+               continue;
+             }
+           } else if (lead) {
+             buf[pos++] = 0xEF;
+             buf[pos++] = 0xBF;
+             buf[pos++] = 0xBD;
+             lead = null;
+           }
 
-       function filterMultikeys(prefix) {
-         return function (d) {
-           // d.key begins with prefix, and d.key contains no additional ':'s
-           var re = new RegExp('^' + prefix + '(.*)$');
-           var matches = d.key.match(re) || [];
-           return matches.length === 2 && matches[1].indexOf(':') === -1;
-         };
-       }
+           if (c < 0x80) {
+             buf[pos++] = c;
+           } else {
+             if (c < 0x800) {
+               buf[pos++] = c >> 0x6 | 0xC0;
+             } else {
+               if (c < 0x10000) {
+                 buf[pos++] = c >> 0xC | 0xE0;
+               } else {
+                 buf[pos++] = c >> 0x12 | 0xF0;
+                 buf[pos++] = c >> 0xC & 0x3F | 0x80;
+               }
 
-       function filterValues(allowUpperCase) {
-         return function (d) {
-           if (d.value.match(/[;,]/) !== null) return false; // exclude some punctuation
+               buf[pos++] = c >> 0x6 & 0x3F | 0x80;
+             }
 
-           if (!allowUpperCase && d.value.match(/[A-Z*]/) !== null) return false; // exclude uppercase letters
+             buf[pos++] = c & 0x3F | 0x80;
+           }
+         }
 
-           return parseFloat(d.fraction) > 0.0;
-         };
+         return pos;
        }
 
-       function filterRoles(geometry) {
-         return function (d) {
-           if (d.role === '') return false; // exclude empty role
-
-           if (d.role.match(/[A-Z*;,]/) !== null) return false; // exclude uppercase letters and some punctuation
+       var pointGeometry = Point;
+       /**
+        * A standalone point geometry with useful accessor, comparison, and
+        * modification methods.
+        *
+        * @class Point
+        * @param {Number} x the x-coordinate. this could be longitude or screen
+        * pixels, or any other sort of unit.
+        * @param {Number} y the y-coordinate. this could be latitude or screen
+        * pixels, or any other sort of unit.
+        * @example
+        * var point = new Point(-77, 38);
+        */
 
-           return parseFloat(d[tag_members_fractions[geometry]]) > 0.0;
-         };
+       function Point(x, y) {
+         this.x = x;
+         this.y = y;
        }
 
-       function valKey(d) {
-         return {
-           value: d.key,
-           title: d.key
-         };
-       }
+       Point.prototype = {
+         /**
+          * Clone this point, returning a new point that can be modified
+          * without affecting the old one.
+          * @return {Point} the clone
+          */
+         clone: function clone() {
+           return new Point(this.x, this.y);
+         },
 
-       function valKeyDescription(d) {
-         var obj = {
-           value: d.value,
-           title: d.description || d.value
-         };
+         /**
+          * Add this point's x & y coordinates to another point,
+          * yielding a new point.
+          * @param {Point} p the other point
+          * @return {Point} output point
+          */
+         add: function add(p) {
+           return this.clone()._add(p);
+         },
 
-         if (d.count) {
-           obj.count = d.count;
-         }
+         /**
+          * Subtract this point's x & y coordinates to from point,
+          * yielding a new point.
+          * @param {Point} p the other point
+          * @return {Point} output point
+          */
+         sub: function sub(p) {
+           return this.clone()._sub(p);
+         },
 
-         return obj;
-       }
+         /**
+          * Multiply this point's x & y coordinates by point,
+          * yielding a new point.
+          * @param {Point} p the other point
+          * @return {Point} output point
+          */
+         multByPoint: function multByPoint(p) {
+           return this.clone()._multByPoint(p);
+         },
 
-       function roleKey(d) {
-         return {
-           value: d.role,
-           title: d.role
-         };
-       } // sort keys with ':' lower than keys without ':'
+         /**
+          * Divide this point's x & y coordinates by point,
+          * yielding a new point.
+          * @param {Point} p the other point
+          * @return {Point} output point
+          */
+         divByPoint: function divByPoint(p) {
+           return this.clone()._divByPoint(p);
+         },
 
+         /**
+          * Multiply this point's x & y coordinates by a factor,
+          * yielding a new point.
+          * @param {Point} k factor
+          * @return {Point} output point
+          */
+         mult: function mult(k) {
+           return this.clone()._mult(k);
+         },
 
-       function sortKeys(a, b) {
-         return a.key.indexOf(':') === -1 && b.key.indexOf(':') !== -1 ? -1 : a.key.indexOf(':') !== -1 && b.key.indexOf(':') === -1 ? 1 : 0;
-       }
+         /**
+          * Divide this point's x & y coordinates by a factor,
+          * yielding a new point.
+          * @param {Point} k factor
+          * @return {Point} output point
+          */
+         div: function div(k) {
+           return this.clone()._div(k);
+         },
 
-       var debouncedRequest$1 = debounce(request$1, 300, {
-         leading: false
-       });
+         /**
+          * Rotate this point around the 0, 0 origin by an angle a,
+          * given in radians
+          * @param {Number} a angle to rotate around, in radians
+          * @return {Point} output point
+          */
+         rotate: function rotate(a) {
+           return this.clone()._rotate(a);
+         },
 
-       function request$1(url, params, exactMatch, callback, loaded) {
-         if (_inflight$2[url]) return;
-         if (checkCache(url, params, exactMatch, callback)) return;
-         var controller = new AbortController();
-         _inflight$2[url] = controller;
-         d3_json(url, {
-           signal: controller.signal
-         }).then(function (result) {
-           delete _inflight$2[url];
-           if (loaded) loaded(null, result);
-         })["catch"](function (err) {
-           delete _inflight$2[url];
-           if (err.name === 'AbortError') return;
-           if (loaded) loaded(err.message);
-         });
-       }
+         /**
+          * Rotate this point around p point by an angle a,
+          * given in radians
+          * @param {Number} a angle to rotate around, in radians
+          * @param {Point} p Point to rotate around
+          * @return {Point} output point
+          */
+         rotateAround: function rotateAround(a, p) {
+           return this.clone()._rotateAround(a, p);
+         },
 
-       function checkCache(url, params, exactMatch, callback) {
-         var rp = params.rp || 25;
-         var testQuery = params.query || '';
-         var testUrl = url;
+         /**
+          * Multiply this point by a 4x1 transformation matrix
+          * @param {Array<Number>} m transformation matrix
+          * @return {Point} output point
+          */
+         matMult: function matMult(m) {
+           return this.clone()._matMult(m);
+         },
 
-         do {
-           var hit = _taginfoCache[testUrl]; // exact match, or shorter match yielding fewer than max results (rp)
+         /**
+          * Calculate this point but as a unit vector from 0, 0, meaning
+          * that the distance from the resulting point to the 0, 0
+          * coordinate will be equal to 1 and the angle from the resulting
+          * point to the 0, 0 coordinate will be the same as before.
+          * @return {Point} unit vector point
+          */
+         unit: function unit() {
+           return this.clone()._unit();
+         },
 
-           if (hit && (url === testUrl || hit.length < rp)) {
-             callback(null, hit);
-             return true;
-           } // don't try to shorten the query
+         /**
+          * Compute a perpendicular point, where the new y coordinate
+          * is the old x coordinate and the new x coordinate is the old y
+          * coordinate multiplied by -1
+          * @return {Point} perpendicular point
+          */
+         perp: function perp() {
+           return this.clone()._perp();
+         },
 
+         /**
+          * Return a version of this point with the x & y coordinates
+          * rounded to integers.
+          * @return {Point} rounded point
+          */
+         round: function round() {
+           return this.clone()._round();
+         },
 
-           if (exactMatch || !testQuery.length) return false; // do shorten the query to see if we already have a cached result
-           // that has returned fewer than max results (rp)
+         /**
+          * Return the magitude of this point: this is the Euclidean
+          * distance from the 0, 0 coordinate to this point's x and y
+          * coordinates.
+          * @return {Number} magnitude
+          */
+         mag: function mag() {
+           return Math.sqrt(this.x * this.x + this.y * this.y);
+         },
 
-           testQuery = testQuery.slice(0, -1);
-           testUrl = url.replace(/&query=(.*?)&/, '&query=' + testQuery + '&');
-         } while (testQuery.length >= 0);
+         /**
+          * Judge whether this point is equal to another point, returning
+          * true or false.
+          * @param {Point} other the other point
+          * @return {boolean} whether the points are equal
+          */
+         equals: function equals(other) {
+           return this.x === other.x && this.y === other.y;
+         },
 
-         return false;
-       }
+         /**
+          * Calculate the distance from this point to another point
+          * @param {Point} p the other point
+          * @return {Number} distance
+          */
+         dist: function dist(p) {
+           return Math.sqrt(this.distSqr(p));
+         },
 
-       var serviceTaginfo = {
-         init: function init() {
-           _inflight$2 = {};
-           _taginfoCache = {};
-           _popularKeys = {
-             // manually exclude some keys – #5377, #7485
-             postal_code: true,
-             full_name: true,
-             loc_name: true,
-             reg_name: true,
-             short_name: true,
-             sorting_name: true,
-             artist_name: true,
-             nat_name: true,
-             long_name: true,
-             'bridge:name': true
-           }; // Fetch popular keys.  We'll exclude these from `values`
-           // lookups because they stress taginfo, and they aren't likely
-           // to yield meaningful autocomplete results.. see #3955
+         /**
+          * Calculate the distance from this point to another point,
+          * without the square root step. Useful if you're comparing
+          * relative distances.
+          * @param {Point} p the other point
+          * @return {Number} distance
+          */
+         distSqr: function distSqr(p) {
+           var dx = p.x - this.x,
+               dy = p.y - this.y;
+           return dx * dx + dy * dy;
+         },
 
-           var params = {
-             rp: 100,
-             sortname: 'values_all',
-             sortorder: 'desc',
-             page: 1,
-             debounce: false,
-             lang: _mainLocalizer.languageCode()
-           };
-           this.keys(params, function (err, data) {
-             if (err) return;
-             data.forEach(function (d) {
-               if (d.value === 'opening_hours') return; // exception
+         /**
+          * Get the angle from the 0, 0 coordinate to this point, in radians
+          * coordinates.
+          * @return {Number} angle
+          */
+         angle: function angle() {
+           return Math.atan2(this.y, this.x);
+         },
 
-               _popularKeys[d.value] = true;
-             });
-           });
+         /**
+          * Get the angle from this point to another point, in radians
+          * @param {Point} b the other point
+          * @return {Number} angle
+          */
+         angleTo: function angleTo(b) {
+           return Math.atan2(this.y - b.y, this.x - b.x);
+         },
+
+         /**
+          * Get the angle between this point and another point, in radians
+          * @param {Point} b the other point
+          * @return {Number} angle
+          */
+         angleWith: function angleWith(b) {
+           return this.angleWithSep(b.x, b.y);
+         },
+
+         /*
+          * Find the angle of the two vectors, solving the formula for
+          * the cross product a x b = |a||b|sin(θ) for θ.
+          * @param {Number} x the x-coordinate
+          * @param {Number} y the y-coordinate
+          * @return {Number} the angle in radians
+          */
+         angleWithSep: function angleWithSep(x, y) {
+           return Math.atan2(this.x * y - this.y * x, this.x * x + this.y * y);
+         },
+         _matMult: function _matMult(m) {
+           var x = m[0] * this.x + m[1] * this.y,
+               y = m[2] * this.x + m[3] * this.y;
+           this.x = x;
+           this.y = y;
+           return this;
+         },
+         _add: function _add(p) {
+           this.x += p.x;
+           this.y += p.y;
+           return this;
+         },
+         _sub: function _sub(p) {
+           this.x -= p.x;
+           this.y -= p.y;
+           return this;
+         },
+         _mult: function _mult(k) {
+           this.x *= k;
+           this.y *= k;
+           return this;
          },
-         reset: function reset() {
-           Object.values(_inflight$2).forEach(function (controller) {
-             controller.abort();
-           });
-           _inflight$2 = {};
+         _div: function _div(k) {
+           this.x /= k;
+           this.y /= k;
+           return this;
          },
-         keys: function keys(params, callback) {
-           var doRequest = params.debounce ? debouncedRequest$1 : request$1;
-           params = clean(setSort(params));
-           params = Object.assign({
-             rp: 10,
-             sortname: 'count_all',
-             sortorder: 'desc',
-             page: 1,
-             lang: _mainLocalizer.languageCode()
-           }, params);
-           var url = _apibase$1 + 'keys/all?' + utilQsString(params);
-           doRequest(url, params, false, callback, function (err, d) {
-             if (err) {
-               callback(err);
-             } else {
-               var f = filterKeys(params.filter);
-               var result = d.data.filter(f).sort(sortKeys).map(valKey);
-               _taginfoCache[url] = result;
-               callback(null, result);
-             }
-           });
+         _multByPoint: function _multByPoint(p) {
+           this.x *= p.x;
+           this.y *= p.y;
+           return this;
          },
-         multikeys: function multikeys(params, callback) {
-           var doRequest = params.debounce ? debouncedRequest$1 : request$1;
-           params = clean(setSort(params));
-           params = Object.assign({
-             rp: 25,
-             sortname: 'count_all',
-             sortorder: 'desc',
-             page: 1,
-             lang: _mainLocalizer.languageCode()
-           }, params);
-           var prefix = params.query;
-           var url = _apibase$1 + 'keys/all?' + utilQsString(params);
-           doRequest(url, params, true, callback, function (err, d) {
-             if (err) {
-               callback(err);
-             } else {
-               var f = filterMultikeys(prefix);
-               var result = d.data.filter(f).map(valKey);
-               _taginfoCache[url] = result;
-               callback(null, result);
-             }
-           });
+         _divByPoint: function _divByPoint(p) {
+           this.x /= p.x;
+           this.y /= p.y;
+           return this;
          },
-         values: function values(params, callback) {
-           // Exclude popular keys from values lookups.. see #3955
-           var key = params.key;
-
-           if (key && _popularKeys[key]) {
-             callback(null, []);
-             return;
-           }
+         _unit: function _unit() {
+           this._div(this.mag());
 
-           var doRequest = params.debounce ? debouncedRequest$1 : request$1;
-           params = clean(setSort(setFilter(params)));
-           params = Object.assign({
-             rp: 25,
-             sortname: 'count_all',
-             sortorder: 'desc',
-             page: 1,
-             lang: _mainLocalizer.languageCode()
-           }, params);
-           var url = _apibase$1 + 'key/values?' + utilQsString(params);
-           doRequest(url, params, false, callback, function (err, d) {
-             if (err) {
-               callback(err);
-             } else {
-               // In most cases we prefer taginfo value results with lowercase letters.
-               // A few OSM keys expect values to contain uppercase values (see #3377).
-               // This is not an exhaustive list (e.g. `name` also has uppercase values)
-               // but these are the fields where taginfo value lookup is most useful.
-               var re = /network|taxon|genus|species|brand|grape_variety|royal_cypher|listed_status|booth|rating|stars|:output|_hours|_times|_ref|manufacturer|country|target|brewery/;
-               var allowUpperCase = re.test(params.key);
-               var f = filterValues(allowUpperCase);
-               var result = d.data.filter(f).map(valKeyDescription);
-               _taginfoCache[url] = result;
-               callback(null, result);
-             }
-           });
+           return this;
          },
-         roles: function roles(params, callback) {
-           var doRequest = params.debounce ? debouncedRequest$1 : request$1;
-           var geometry = params.geometry;
-           params = clean(setSortMembers(params));
-           params = Object.assign({
-             rp: 25,
-             sortname: 'count_all_members',
-             sortorder: 'desc',
-             page: 1,
-             lang: _mainLocalizer.languageCode()
-           }, params);
-           var url = _apibase$1 + 'relation/roles?' + utilQsString(params);
-           doRequest(url, params, true, callback, function (err, d) {
-             if (err) {
-               callback(err);
-             } else {
-               var f = filterRoles(geometry);
-               var result = d.data.filter(f).map(roleKey);
-               _taginfoCache[url] = result;
-               callback(null, result);
-             }
-           });
+         _perp: function _perp() {
+           var y = this.y;
+           this.y = this.x;
+           this.x = -y;
+           return this;
          },
-         docs: function docs(params, callback) {
-           var doRequest = params.debounce ? debouncedRequest$1 : request$1;
-           params = clean(setSort(params));
-           var path = 'key/wiki_pages?';
-
-           if (params.value) {
-             path = 'tag/wiki_pages?';
-           } else if (params.rtype) {
-             path = 'relation/wiki_pages?';
-           }
-
-           var url = _apibase$1 + path + utilQsString(params);
-           doRequest(url, params, true, callback, function (err, d) {
-             if (err) {
-               callback(err);
-             } else {
-               _taginfoCache[url] = d.data;
-               callback(null, d.data);
-             }
-           });
+         _rotate: function _rotate(angle) {
+           var cos = Math.cos(angle),
+               sin = Math.sin(angle),
+               x = cos * this.x - sin * this.y,
+               y = sin * this.x + cos * this.y;
+           this.x = x;
+           this.y = y;
+           return this;
          },
-         apibase: function apibase(_) {
-           if (!arguments.length) return _apibase$1;
-           _apibase$1 = _;
+         _rotateAround: function _rotateAround(angle, p) {
+           var cos = Math.cos(angle),
+               sin = Math.sin(angle),
+               x = p.x + cos * (this.x - p.x) - sin * (this.y - p.y),
+               y = p.y + sin * (this.x - p.x) + cos * (this.y - p.y);
+           this.x = x;
+           this.y = y;
+           return this;
+         },
+         _round: function _round() {
+           this.x = Math.round(this.x);
+           this.y = Math.round(this.y);
            return this;
          }
        };
-
        /**
-        * Wraps a GeoJSON {@link Geometry} in a GeoJSON {@link Feature}.
-        *
-        * @name feature
-        * @param {Geometry} geometry input geometry
-        * @param {Object} [properties={}] an Object of key-value pairs to add as properties
-        * @param {Object} [options={}] Optional Parameters
-        * @param {Array<number>} [options.bbox] Bounding Box Array [west, south, east, north] associated with the Feature
-        * @param {string|number} [options.id] Identifier associated with the Feature
-        * @returns {Feature} a GeoJSON Feature
+        * Construct a point from an array if necessary, otherwise if the input
+        * is already a Point, or an unknown type, return it unchanged
+        * @param {Array<Number>|Point|*} a any kind of input value
+        * @return {Point} constructed point, or passed-through value.
         * @example
-        * var geometry = {
-        *   "type": "Point",
-        *   "coordinates": [110, 50]
-        * };
-        *
-        * var feature = turf.feature(geometry);
-        *
-        * //=feature
+        * // this
+        * var point = Point.convert([0, 1]);
+        * // is equivalent to
+        * var point = new Point(0, 1);
         */
 
-       function feature(geom, properties, options) {
-         if (options === void 0) {
-           options = {};
+       Point.convert = function (a) {
+         if (a instanceof Point) {
+           return a;
          }
 
-         var feat = {
-           type: "Feature"
-         };
-
-         if (options.id === 0 || options.id) {
-           feat.id = options.id;
+         if (Array.isArray(a)) {
+           return new Point(a[0], a[1]);
          }
 
-         if (options.bbox) {
-           feat.bbox = options.bbox;
-         }
+         return a;
+       };
 
-         feat.properties = properties || {};
-         feat.geometry = geom;
-         return feat;
+       var vectortilefeature = VectorTileFeature$1;
+
+       function VectorTileFeature$1(pbf, end, extent, keys, values) {
+         // Public
+         this.properties = {};
+         this.extent = extent;
+         this.type = 0; // Private
+
+         this._pbf = pbf;
+         this._geometry = -1;
+         this._keys = keys;
+         this._values = values;
+         pbf.readFields(readFeature, this, end);
        }
-       /**
-        * Creates a {@link Polygon} {@link Feature} from an Array of LinearRings.
-        *
-        * @name polygon
-        * @param {Array<Array<Array<number>>>} coordinates an array of LinearRings
-        * @param {Object} [properties={}] an Object of key-value pairs to add as properties
-        * @param {Object} [options={}] Optional Parameters
-        * @param {Array<number>} [options.bbox] Bounding Box Array [west, south, east, north] associated with the Feature
-        * @param {string|number} [options.id] Identifier associated with the Feature
-        * @returns {Feature<Polygon>} Polygon Feature
-        * @example
-        * var polygon = turf.polygon([[[-5, 52], [-4, 56], [-2, 51], [-7, 54], [-5, 52]]], { name: 'poly1' });
-        *
-        * //=polygon
-        */
 
-       function polygon(coordinates, properties, options) {
-         if (options === void 0) {
-           options = {};
+       function readFeature(tag, feature, pbf) {
+         if (tag == 1) feature.id = pbf.readVarint();else if (tag == 2) readTag(pbf, feature);else if (tag == 3) feature.type = pbf.readVarint();else if (tag == 4) feature._geometry = pbf.pos;
+       }
+
+       function readTag(pbf, feature) {
+         var end = pbf.readVarint() + pbf.pos;
+
+         while (pbf.pos < end) {
+           var key = feature._keys[pbf.readVarint()],
+               value = feature._values[pbf.readVarint()];
+
+           feature.properties[key] = value;
          }
+       }
 
-         for (var _i = 0, coordinates_1 = coordinates; _i < coordinates_1.length; _i++) {
-           var ring = coordinates_1[_i];
+       VectorTileFeature$1.types = ['Unknown', 'Point', 'LineString', 'Polygon'];
 
-           if (ring.length < 4) {
-             throw new Error("Each LinearRing of a Polygon must have 4 or more Positions.");
-           }
+       VectorTileFeature$1.prototype.loadGeometry = function () {
+         var pbf = this._pbf;
+         pbf.pos = this._geometry;
+         var end = pbf.readVarint() + pbf.pos,
+             cmd = 1,
+             length = 0,
+             x = 0,
+             y = 0,
+             lines = [],
+             line;
 
-           for (var j = 0; j < ring[ring.length - 1].length; j++) {
-             // Check if first point of Polygon contains two numbers
-             if (ring[ring.length - 1][j] !== ring[0][j]) {
-               throw new Error("First and last Position are not equivalent.");
-             }
+         while (pbf.pos < end) {
+           if (length <= 0) {
+             var cmdLen = pbf.readVarint();
+             cmd = cmdLen & 0x7;
+             length = cmdLen >> 3;
            }
-         }
 
-         var geom = {
-           type: "Polygon",
-           coordinates: coordinates
-         };
-         return feature(geom, properties, options);
-       }
-       /**
-        * Creates a {@link LineString} {@link Feature} from an Array of Positions.
-        *
-        * @name lineString
-        * @param {Array<Array<number>>} coordinates an array of Positions
-        * @param {Object} [properties={}] an Object of key-value pairs to add as properties
-        * @param {Object} [options={}] Optional Parameters
-        * @param {Array<number>} [options.bbox] Bounding Box Array [west, south, east, north] associated with the Feature
-        * @param {string|number} [options.id] Identifier associated with the Feature
-        * @returns {Feature<LineString>} LineString Feature
-        * @example
-        * var linestring1 = turf.lineString([[-24, 63], [-23, 60], [-25, 65], [-20, 69]], {name: 'line 1'});
-        * var linestring2 = turf.lineString([[-14, 43], [-13, 40], [-15, 45], [-10, 49]], {name: 'line 2'});
-        *
-        * //=linestring1
-        * //=linestring2
-        */
+           length--;
 
-       function lineString(coordinates, properties, options) {
-         if (options === void 0) {
-           options = {};
-         }
+           if (cmd === 1 || cmd === 2) {
+             x += pbf.readSVarint();
+             y += pbf.readSVarint();
 
-         if (coordinates.length < 2) {
-           throw new Error("coordinates must be an array of two or more positions");
+             if (cmd === 1) {
+               // moveTo
+               if (line) lines.push(line);
+               line = [];
+             }
+
+             line.push(new pointGeometry(x, y));
+           } else if (cmd === 7) {
+             // Workaround for https://github.com/mapbox/mapnik-vector-tile/issues/90
+             if (line) {
+               line.push(line[0].clone()); // closePolygon
+             }
+           } else {
+             throw new Error('unknown command ' + cmd);
+           }
          }
 
-         var geom = {
-           type: "LineString",
-           coordinates: coordinates
-         };
-         return feature(geom, properties, options);
-       }
-       /**
-        * Creates a {@link Feature<MultiLineString>} based on a
-        * coordinate array. Properties can be added optionally.
-        *
-        * @name multiLineString
-        * @param {Array<Array<Array<number>>>} coordinates an array of LineStrings
-        * @param {Object} [properties={}] an Object of key-value pairs to add as properties
-        * @param {Object} [options={}] Optional Parameters
-        * @param {Array<number>} [options.bbox] Bounding Box Array [west, south, east, north] associated with the Feature
-        * @param {string|number} [options.id] Identifier associated with the Feature
-        * @returns {Feature<MultiLineString>} a MultiLineString feature
-        * @throws {Error} if no coordinates are passed
-        * @example
-        * var multiLine = turf.multiLineString([[[0,0],[10,10]]]);
-        *
-        * //=multiLine
-        */
+         if (line) lines.push(line);
+         return lines;
+       };
 
-       function multiLineString(coordinates, properties, options) {
-         if (options === void 0) {
-           options = {};
-         }
+       VectorTileFeature$1.prototype.bbox = function () {
+         var pbf = this._pbf;
+         pbf.pos = this._geometry;
+         var end = pbf.readVarint() + pbf.pos,
+             cmd = 1,
+             length = 0,
+             x = 0,
+             y = 0,
+             x1 = Infinity,
+             x2 = -Infinity,
+             y1 = Infinity,
+             y2 = -Infinity;
 
-         var geom = {
-           type: "MultiLineString",
-           coordinates: coordinates
-         };
-         return feature(geom, properties, options);
-       }
-       /**
-        * Creates a {@link Feature<MultiPolygon>} based on a
-        * coordinate array. Properties can be added optionally.
-        *
-        * @name multiPolygon
-        * @param {Array<Array<Array<Array<number>>>>} coordinates an array of Polygons
-        * @param {Object} [properties={}] an Object of key-value pairs to add as properties
-        * @param {Object} [options={}] Optional Parameters
-        * @param {Array<number>} [options.bbox] Bounding Box Array [west, south, east, north] associated with the Feature
-        * @param {string|number} [options.id] Identifier associated with the Feature
-        * @returns {Feature<MultiPolygon>} a multipolygon feature
-        * @throws {Error} if no coordinates are passed
-        * @example
-        * var multiPoly = turf.multiPolygon([[[[0,0],[0,10],[10,10],[10,0],[0,0]]]]);
-        *
-        * //=multiPoly
-        *
-        */
+         while (pbf.pos < end) {
+           if (length <= 0) {
+             var cmdLen = pbf.readVarint();
+             cmd = cmdLen & 0x7;
+             length = cmdLen >> 3;
+           }
 
-       function multiPolygon(coordinates, properties, options) {
-         if (options === void 0) {
-           options = {};
+           length--;
+
+           if (cmd === 1 || cmd === 2) {
+             x += pbf.readSVarint();
+             y += pbf.readSVarint();
+             if (x < x1) x1 = x;
+             if (x > x2) x2 = x;
+             if (y < y1) y1 = y;
+             if (y > y2) y2 = y;
+           } else if (cmd !== 7) {
+             throw new Error('unknown command ' + cmd);
+           }
          }
 
-         var geom = {
-           type: "MultiPolygon",
-           coordinates: coordinates
-         };
-         return feature(geom, properties, options);
-       }
+         return [x1, y1, x2, y2];
+       };
 
-       /**
-        * Get Geometry from Feature or Geometry Object
-        *
-        * @param {Feature|Geometry} geojson GeoJSON Feature or Geometry Object
-        * @returns {Geometry|null} GeoJSON Geometry Object
-        * @throws {Error} if geojson is not a Feature or Geometry Object
-        * @example
-        * var point = {
-        *   "type": "Feature",
-        *   "properties": {},
-        *   "geometry": {
-        *     "type": "Point",
-        *     "coordinates": [110, 40]
-        *   }
-        * }
-        * var geom = turf.getGeom(point)
-        * //={"type": "Point", "coordinates": [110, 40]}
-        */
+       VectorTileFeature$1.prototype.toGeoJSON = function (x, y, z) {
+         var size = this.extent * Math.pow(2, z),
+             x0 = this.extent * x,
+             y0 = this.extent * y,
+             coords = this.loadGeometry(),
+             type = VectorTileFeature$1.types[this.type],
+             i,
+             j;
 
-       function getGeom(geojson) {
-         if (geojson.type === "Feature") {
-           return geojson.geometry;
+         function project(line) {
+           for (var j = 0; j < line.length; j++) {
+             var p = line[j],
+                 y2 = 180 - (p.y + y0) * 360 / size;
+             line[j] = [(p.x + x0) * 360 / size - 180, 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90];
+           }
          }
 
-         return geojson;
-       }
+         switch (this.type) {
+           case 1:
+             var points = [];
 
-       // Cohen-Sutherland line clippign algorithm, adapted to efficiently
-       // handle polylines rather than just segments
-       function lineclip(points, bbox, result) {
-         var len = points.length,
-             codeA = bitCode(points[0], bbox),
-             part = [],
-             i,
-             a,
-             b,
-             codeB,
-             lastCode;
-         if (!result) result = [];
+             for (i = 0; i < coords.length; i++) {
+               points[i] = coords[i][0];
+             }
 
-         for (i = 1; i < len; i++) {
-           a = points[i - 1];
-           b = points[i];
-           codeB = lastCode = bitCode(b, bbox);
+             coords = points;
+             project(coords);
+             break;
 
-           while (true) {
-             if (!(codeA | codeB)) {
-               // accept
-               part.push(a);
+           case 2:
+             for (i = 0; i < coords.length; i++) {
+               project(coords[i]);
+             }
 
-               if (codeB !== lastCode) {
-                 // segment went outside
-                 part.push(b);
+             break;
 
-                 if (i < len - 1) {
-                   // start a new line
-                   result.push(part);
-                   part = [];
-                 }
-               } else if (i === len - 1) {
-                 part.push(b);
-               }
+           case 3:
+             coords = classifyRings(coords);
 
-               break;
-             } else if (codeA & codeB) {
-               // trivial reject
-               break;
-             } else if (codeA) {
-               // a outside, intersect with clip edge
-               a = intersect(a, b, codeA, bbox);
-               codeA = bitCode(a, bbox);
-             } else {
-               // b outside
-               b = intersect(a, b, codeB, bbox);
-               codeB = bitCode(b, bbox);
+             for (i = 0; i < coords.length; i++) {
+               for (j = 0; j < coords[i].length; j++) {
+                 project(coords[i][j]);
+               }
              }
-           }
 
-           codeA = lastCode;
+             break;
          }
 
-         if (part.length) result.push(part);
-         return result;
-       } // Sutherland-Hodgeman polygon clipping algorithm
+         if (coords.length === 1) {
+           coords = coords[0];
+         } else {
+           type = 'Multi' + type;
+         }
 
-       function polygonclip(points, bbox) {
-         var result, edge, prev, prevInside, i, p, inside; // clip against each side of the clip rectangle
+         var result = {
+           type: "Feature",
+           geometry: {
+             type: type,
+             coordinates: coords
+           },
+           properties: this.properties
+         };
 
-         for (edge = 1; edge <= 8; edge *= 2) {
-           result = [];
-           prev = points[points.length - 1];
-           prevInside = !(bitCode(prev, bbox) & edge);
+         if ('id' in this) {
+           result.id = this.id;
+         }
 
-           for (i = 0; i < points.length; i++) {
-             p = points[i];
-             inside = !(bitCode(p, bbox) & edge); // if segment goes through the clip window, add an intersection
+         return result;
+       }; // classifies an array of rings into polygons with outer rings and holes
 
-             if (inside !== prevInside) result.push(intersect(prev, p, edge, bbox));
-             if (inside) result.push(p); // add a point if it's inside
 
-             prev = p;
-             prevInside = inside;
-           }
+       function classifyRings(rings) {
+         var len = rings.length;
+         if (len <= 1) return [rings];
+         var polygons = [],
+             polygon,
+             ccw;
 
-           points = result;
-           if (!points.length) break;
+         for (var i = 0; i < len; i++) {
+           var area = signedArea(rings[i]);
+           if (area === 0) continue;
+           if (ccw === undefined) ccw = area < 0;
+
+           if (ccw === area < 0) {
+             if (polygon) polygons.push(polygon);
+             polygon = [rings[i]];
+           } else {
+             polygon.push(rings[i]);
+           }
          }
 
-         return result;
-       } // intersect a segment against one of the 4 lines that make up the bbox
+         if (polygon) polygons.push(polygon);
+         return polygons;
+       }
 
-       function intersect(a, b, edge, bbox) {
-         return edge & 8 ? [a[0] + (b[0] - a[0]) * (bbox[3] - a[1]) / (b[1] - a[1]), bbox[3]] // top
-         : edge & 4 ? [a[0] + (b[0] - a[0]) * (bbox[1] - a[1]) / (b[1] - a[1]), bbox[1]] // bottom
-         : edge & 2 ? [bbox[2], a[1] + (b[1] - a[1]) * (bbox[2] - a[0]) / (b[0] - a[0])] // right
-         : edge & 1 ? [bbox[0], a[1] + (b[1] - a[1]) * (bbox[0] - a[0]) / (b[0] - a[0])] // left
-         : null;
-       } // bit code reflects the point position relative to the bbox:
-       //         left  mid  right
-       //    top  1001  1000  1010
-       //    mid  0001  0000  0010
-       // bottom  0101  0100  0110
+       function signedArea(ring) {
+         var sum = 0;
+
+         for (var i = 0, len = ring.length, j = len - 1, p1, p2; i < len; j = i++) {
+           p1 = ring[i];
+           p2 = ring[j];
+           sum += (p2.x - p1.x) * (p1.y + p2.y);
+         }
 
+         return sum;
+       }
 
-       function bitCode(p, bbox) {
-         var code = 0;
-         if (p[0] < bbox[0]) code |= 1; // left
-         else if (p[0] > bbox[2]) code |= 2; // right
+       var vectortilelayer = VectorTileLayer$1;
 
-         if (p[1] < bbox[1]) code |= 4; // bottom
-         else if (p[1] > bbox[3]) code |= 8; // top
+       function VectorTileLayer$1(pbf, end) {
+         // Public
+         this.version = 1;
+         this.name = null;
+         this.extent = 4096;
+         this.length = 0; // Private
 
-         return code;
+         this._pbf = pbf;
+         this._keys = [];
+         this._values = [];
+         this._features = [];
+         pbf.readFields(readLayer, this, end);
+         this.length = this._features.length;
        }
 
-       /**
-        * Takes a {@link Feature} and a bbox and clips the feature to the bbox using
-        * [lineclip](https://github.com/mapbox/lineclip).
-        * May result in degenerate edges when clipping Polygons.
-        *
-        * @name bboxClip
-        * @param {Feature<LineString|MultiLineString|Polygon|MultiPolygon>} feature feature to clip to the bbox
-        * @param {BBox} bbox extent in [minX, minY, maxX, maxY] order
-        * @returns {Feature<LineString|MultiLineString|Polygon|MultiPolygon>} clipped Feature
-        * @example
-        * var bbox = [0, 0, 10, 10];
-        * var poly = turf.polygon([[[2, 2], [8, 4], [12, 8], [3, 7], [2, 2]]]);
-        *
-        * var clipped = turf.bboxClip(poly, bbox);
-        *
-        * //addToMap
-        * var addToMap = [bbox, poly, clipped]
-        */
+       function readLayer(tag, layer, pbf) {
+         if (tag === 15) layer.version = pbf.readVarint();else if (tag === 1) layer.name = pbf.readString();else if (tag === 5) layer.extent = pbf.readVarint();else if (tag === 2) layer._features.push(pbf.pos);else if (tag === 3) layer._keys.push(pbf.readString());else if (tag === 4) layer._values.push(readValueMessage(pbf));
+       }
 
-       function bboxClip(feature, bbox) {
-         var geom = getGeom(feature);
-         var type = geom.type;
-         var properties = feature.type === "Feature" ? feature.properties : {};
-         var coords = geom.coordinates;
+       function readValueMessage(pbf) {
+         var value = null,
+             end = pbf.readVarint() + pbf.pos;
 
-         switch (type) {
-           case "LineString":
-           case "MultiLineString":
-             var lines_1 = [];
+         while (pbf.pos < end) {
+           var tag = pbf.readVarint() >> 3;
+           value = tag === 1 ? pbf.readString() : tag === 2 ? pbf.readFloat() : tag === 3 ? pbf.readDouble() : tag === 4 ? pbf.readVarint64() : tag === 5 ? pbf.readVarint() : tag === 6 ? pbf.readSVarint() : tag === 7 ? pbf.readBoolean() : null;
+         }
 
-             if (type === "LineString") {
-               coords = [coords];
-             }
+         return value;
+       } // return feature `i` from this layer as a `VectorTileFeature`
 
-             coords.forEach(function (line) {
-               lineclip(line, bbox, lines_1);
-             });
 
-             if (lines_1.length === 1) {
-               return lineString(lines_1[0], properties);
-             }
+       VectorTileLayer$1.prototype.feature = function (i) {
+         if (i < 0 || i >= this._features.length) throw new Error('feature index out of bounds');
+         this._pbf.pos = this._features[i];
 
-             return multiLineString(lines_1, properties);
+         var end = this._pbf.readVarint() + this._pbf.pos;
 
-           case "Polygon":
-             return polygon(clipPolygon(coords, bbox), properties);
+         return new vectortilefeature(this._pbf, end, this.extent, this._keys, this._values);
+       };
 
-           case "MultiPolygon":
-             return multiPolygon(coords.map(function (poly) {
-               return clipPolygon(poly, bbox);
-             }), properties);
+       var vectortile = VectorTile$1;
 
-           default:
-             throw new Error("geometry " + type + " not supported");
+       function VectorTile$1(pbf, end) {
+         this.layers = pbf.readFields(readTile, {}, end);
+       }
+
+       function readTile(tag, layers, pbf) {
+         if (tag === 3) {
+           var layer = new vectortilelayer(pbf, pbf.readVarint() + pbf.pos);
+           if (layer.length) layers[layer.name] = layer;
          }
        }
 
-       function clipPolygon(rings, bbox) {
-         var outRings = [];
+       var VectorTile = vectortile;
+       var VectorTileFeature = vectortilefeature;
+       var VectorTileLayer = vectortilelayer;
+       var vectorTile = {
+         VectorTile: VectorTile,
+         VectorTileFeature: VectorTileFeature,
+         VectorTileLayer: VectorTileLayer
+       };
 
-         for (var _i = 0, rings_1 = rings; _i < rings_1.length; _i++) {
-           var ring = rings_1[_i];
-           var clipped = polygonclip(ring, bbox);
+       var accessToken = 'MLY|4100327730013843|5bb78b81720791946a9a7b956c57b7cf';
+       var apiUrl = 'https://graph.mapillary.com/';
+       var baseTileUrl = 'https://tiles.mapillary.com/maps/vtp';
+       var mapFeatureTileUrl = "".concat(baseTileUrl, "/mly_map_feature_point/2/{z}/{x}/{y}?access_token=").concat(accessToken);
+       var tileUrl = "".concat(baseTileUrl, "/mly1_public/2/{z}/{x}/{y}?access_token=").concat(accessToken);
+       var trafficSignTileUrl = "".concat(baseTileUrl, "/mly_map_feature_traffic_sign/2/{z}/{x}/{y}?access_token=").concat(accessToken);
+       var viewercss = 'mapillary-js/mapillary.css';
+       var viewerjs = 'mapillary-js/mapillary.js';
+       var minZoom$1 = 14;
+       var dispatch$4 = dispatch$8('change', 'loadedImages', 'loadedSigns', 'loadedMapFeatures', 'bearingChanged', 'imageChanged');
 
-           if (clipped.length > 0) {
-             if (clipped[0][0] !== clipped[clipped.length - 1][0] || clipped[0][1] !== clipped[clipped.length - 1][1]) {
-               clipped.push(clipped[0]);
-             }
+       var _loadViewerPromise$2;
 
-             if (clipped.length >= 4) {
-               outRings.push(clipped);
-             }
-           }
-         }
+       var _mlyActiveImage;
 
-         return outRings;
-       }
+       var _mlyCache;
 
-       var fastJsonStableStringify = function fastJsonStableStringify(data, opts) {
-         if (!opts) opts = {};
-         if (typeof opts === 'function') opts = {
-           cmp: opts
-         };
-         var cycles = typeof opts.cycles === 'boolean' ? opts.cycles : false;
+       var _mlyFallback = false;
 
-         var cmp = opts.cmp && function (f) {
-           return function (node) {
-             return function (a, b) {
-               var aobj = {
-                 key: a,
-                 value: node[a]
-               };
-               var bobj = {
-                 key: b,
-                 value: node[b]
-               };
-               return f(aobj, bobj);
-             };
-           };
-         }(opts.cmp);
+       var _mlyHighlightedDetection;
 
-         var seen = [];
-         return function stringify(node) {
-           if (node && node.toJSON && typeof node.toJSON === 'function') {
-             node = node.toJSON();
-           }
+       var _mlyShowFeatureDetections = false;
+       var _mlyShowSignDetections = false;
 
-           if (node === undefined) return;
-           if (typeof node == 'number') return isFinite(node) ? '' + node : 'null';
-           if (_typeof(node) !== 'object') return JSON.stringify(node);
-           var i, out;
+       var _mlyViewer;
 
-           if (Array.isArray(node)) {
-             out = '[';
+       var _mlyViewerFilter = ['all']; // Load all data for the specified type from Mapillary vector tiles
 
-             for (i = 0; i < node.length; i++) {
-               if (i) out += ',';
-               out += stringify(node[i]) || 'null';
-             }
+       function loadTiles$2(which, url, maxZoom, projection) {
+         var tiler = utilTiler().zoomExtent([minZoom$1, maxZoom]).skipNullIsland(true);
+         var tiles = tiler.getTiles(projection);
+         tiles.forEach(function (tile) {
+           loadTile$1(which, url, tile);
+         });
+       } // Load all data for the specified type from one vector tile
 
-             return out + ']';
-           }
 
-           if (node === null) return 'null';
+       function loadTile$1(which, url, tile) {
+         var cache = _mlyCache.requests;
+         var tileId = "".concat(tile.id, "-").concat(which);
+         if (cache.loaded[tileId] || cache.inflight[tileId]) return;
+         var controller = new AbortController();
+         cache.inflight[tileId] = controller;
+         var requestUrl = url.replace('{x}', tile.xyz[0]).replace('{y}', tile.xyz[1]).replace('{z}', tile.xyz[2]);
+         fetch(requestUrl, {
+           signal: controller.signal
+         }).then(function (response) {
+           if (!response.ok) {
+             throw new Error(response.status + ' ' + response.statusText);
+           }
 
-           if (seen.indexOf(node) !== -1) {
-             if (cycles) return JSON.stringify('__cycle__');
-             throw new TypeError('Converting circular structure to JSON');
+           cache.loaded[tileId] = true;
+           delete cache.inflight[tileId];
+           return response.arrayBuffer();
+         }).then(function (data) {
+           if (!data) {
+             throw new Error('No Data');
            }
 
-           var seenIndex = seen.push(node) - 1;
-           var keys = Object.keys(node).sort(cmp && cmp(node));
-           out = '';
+           loadTileDataToCache(data, tile, which);
 
-           for (i = 0; i < keys.length; i++) {
-             var key = keys[i];
-             var value = stringify(node[key]);
-             if (!value) continue;
-             if (out) out += ',';
-             out += JSON.stringify(key) + ':' + value;
+           if (which === 'images') {
+             dispatch$4.call('loadedImages');
+           } else if (which === 'signs') {
+             dispatch$4.call('loadedSigns');
+           } else if (which === 'points') {
+             dispatch$4.call('loadedMapFeatures');
            }
+         })["catch"](function () {
+           cache.loaded[tileId] = true;
+           delete cache.inflight[tileId];
+         });
+       } // Load the data from the vector tile into cache
 
-           seen.splice(seenIndex, 1);
-           return '{' + out + '}';
-         }(data);
-       };
 
-       function DEFAULT_COMPARE(a, b) {
-         return a > b ? 1 : a < b ? -1 : 0;
-       }
+       function loadTileDataToCache(data, tile, which) {
+         var vectorTile = new VectorTile(new pbf(data));
+         var features, cache, layer, i, feature, loc, d;
 
-       var SplayTree = /*#__PURE__*/function () {
-         function SplayTree() {
-           var compare = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : DEFAULT_COMPARE;
-           var noDuplicates = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false;
+         if (vectorTile.layers.hasOwnProperty('image')) {
+           features = [];
+           cache = _mlyCache.images;
+           layer = vectorTile.layers.image;
 
-           _classCallCheck(this, SplayTree);
+           for (i = 0; i < layer.length; i++) {
+             feature = layer.feature(i).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
+             loc = feature.geometry.coordinates;
+             d = {
+               loc: loc,
+               captured_at: feature.properties.captured_at,
+               ca: feature.properties.compass_angle,
+               id: feature.properties.id,
+               is_pano: feature.properties.is_pano,
+               sequence_id: feature.properties.sequence_id
+             };
+             cache.forImageId[d.id] = d;
+             features.push({
+               minX: loc[0],
+               minY: loc[1],
+               maxX: loc[0],
+               maxY: loc[1],
+               data: d
+             });
+           }
 
-           this._compare = compare;
-           this._root = null;
-           this._size = 0;
-           this._noDuplicates = !!noDuplicates;
+           if (cache.rtree) {
+             cache.rtree.load(features);
+           }
          }
 
-         _createClass(SplayTree, [{
-           key: "rotateLeft",
-           value: function rotateLeft(x) {
-             var y = x.right;
+         if (vectorTile.layers.hasOwnProperty('sequence')) {
+           features = [];
+           cache = _mlyCache.sequences;
+           layer = vectorTile.layers.sequence;
+
+           for (i = 0; i < layer.length; i++) {
+             feature = layer.feature(i).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
 
-             if (y) {
-               x.right = y.left;
-               if (y.left) y.left.parent = x;
-               y.parent = x.parent;
+             if (cache.lineString[feature.properties.id]) {
+               cache.lineString[feature.properties.id].push(feature);
+             } else {
+               cache.lineString[feature.properties.id] = [feature];
              }
+           }
+         }
+
+         if (vectorTile.layers.hasOwnProperty('point')) {
+           features = [];
+           cache = _mlyCache[which];
+           layer = vectorTile.layers.point;
 
-             if (!x.parent) this._root = y;else if (x === x.parent.left) x.parent.left = y;else x.parent.right = y;
-             if (y) y.left = x;
-             x.parent = y;
+           for (i = 0; i < layer.length; i++) {
+             feature = layer.feature(i).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
+             loc = feature.geometry.coordinates;
+             d = {
+               loc: loc,
+               id: feature.properties.id,
+               first_seen_at: feature.properties.first_seen_at,
+               last_seen_at: feature.properties.last_seen_at,
+               value: feature.properties.value
+             };
+             features.push({
+               minX: loc[0],
+               minY: loc[1],
+               maxX: loc[0],
+               maxY: loc[1],
+               data: d
+             });
            }
-         }, {
-           key: "rotateRight",
-           value: function rotateRight(x) {
-             var y = x.left;
 
-             if (y) {
-               x.left = y.right;
-               if (y.right) y.right.parent = x;
-               y.parent = x.parent;
-             }
+           if (cache.rtree) {
+             cache.rtree.load(features);
+           }
+         }
+
+         if (vectorTile.layers.hasOwnProperty('traffic_sign')) {
+           features = [];
+           cache = _mlyCache[which];
+           layer = vectorTile.layers.traffic_sign;
 
-             if (!x.parent) this._root = y;else if (x === x.parent.left) x.parent.left = y;else x.parent.right = y;
-             if (y) y.right = x;
-             x.parent = y;
+           for (i = 0; i < layer.length; i++) {
+             feature = layer.feature(i).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
+             loc = feature.geometry.coordinates;
+             d = {
+               loc: loc,
+               id: feature.properties.id,
+               first_seen_at: feature.properties.first_seen_at,
+               last_seen_at: feature.properties.last_seen_at,
+               value: feature.properties.value
+             };
+             features.push({
+               minX: loc[0],
+               minY: loc[1],
+               maxX: loc[0],
+               maxY: loc[1],
+               data: d
+             });
            }
-         }, {
-           key: "_splay",
-           value: function _splay(x) {
-             while (x.parent) {
-               var p = x.parent;
-
-               if (!p.parent) {
-                 if (p.left === x) this.rotateRight(p);else this.rotateLeft(p);
-               } else if (p.left === x && p.parent.left === p) {
-                 this.rotateRight(p.parent);
-                 this.rotateRight(p);
-               } else if (p.right === x && p.parent.right === p) {
-                 this.rotateLeft(p.parent);
-                 this.rotateLeft(p);
-               } else if (p.left === x && p.parent.right === p) {
-                 this.rotateRight(p);
-                 this.rotateLeft(p);
-               } else {
-                 this.rotateLeft(p);
-                 this.rotateRight(p);
-               }
-             }
+
+           if (cache.rtree) {
+             cache.rtree.load(features);
            }
-         }, {
-           key: "splay",
-           value: function splay(x) {
-             var p, gp, ggp, l, r;
-
-             while (x.parent) {
-               p = x.parent;
-               gp = p.parent;
-
-               if (gp && gp.parent) {
-                 ggp = gp.parent;
-                 if (ggp.left === gp) ggp.left = x;else ggp.right = x;
-                 x.parent = ggp;
-               } else {
-                 x.parent = null;
-                 this._root = x;
-               }
+         }
+       } // Get data from the API
 
-               l = x.left;
-               r = x.right;
 
-               if (x === p.left) {
-                 // left
-                 if (gp) {
-                   if (gp.left === p) {
-                     /* zig-zig */
-                     if (p.right) {
-                       gp.left = p.right;
-                       gp.left.parent = gp;
-                     } else gp.left = null;
+       function loadData(url) {
+         return fetch(url).then(function (response) {
+           if (!response.ok) {
+             throw new Error(response.status + ' ' + response.statusText);
+           }
 
-                     p.right = gp;
-                     gp.parent = p;
-                   } else {
-                     /* zig-zag */
-                     if (l) {
-                       gp.right = l;
-                       l.parent = gp;
-                     } else gp.right = null;
-
-                     x.left = gp;
-                     gp.parent = x;
-                   }
-                 }
+           return response.json();
+         }).then(function (result) {
+           if (!result) {
+             return [];
+           }
 
-                 if (r) {
-                   p.left = r;
-                   r.parent = p;
-                 } else p.left = null;
+           return result.data || [];
+         });
+       } // Partition viewport into higher zoom tiles
 
-                 x.right = p;
-                 p.parent = x;
-               } else {
-                 // right
-                 if (gp) {
-                   if (gp.right === p) {
-                     /* zig-zig */
-                     if (p.left) {
-                       gp.right = p.left;
-                       gp.right.parent = gp;
-                     } else gp.right = null;
-
-                     p.left = gp;
-                     gp.parent = p;
-                   } else {
-                     /* zig-zag */
-                     if (r) {
-                       gp.left = r;
-                       r.parent = gp;
-                     } else gp.left = null;
-
-                     x.right = gp;
-                     gp.parent = x;
-                   }
-                 }
 
-                 if (l) {
-                   p.right = l;
-                   l.parent = p;
-                 } else p.right = null;
+       function partitionViewport$2(projection) {
+         var z = geoScaleToZoom(projection.scale());
+         var z2 = Math.ceil(z * 2) / 2 + 2.5; // round to next 0.5 and add 2.5
 
-                 x.left = p;
-                 p.parent = x;
-               }
-             }
-           }
-         }, {
-           key: "replace",
-           value: function replace(u, v) {
-             if (!u.parent) this._root = v;else if (u === u.parent.left) u.parent.left = v;else u.parent.right = v;
-             if (v) v.parent = u.parent;
-           }
-         }, {
-           key: "minNode",
-           value: function minNode() {
-             var u = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : this._root;
-             if (u) while (u.left) {
-               u = u.left;
-             }
-             return u;
+         var tiler = utilTiler().zoomExtent([z2, z2]);
+         return tiler.getTiles(projection).map(function (tile) {
+           return tile.extent;
+         });
+       } // Return no more than `limit` results per partition.
+
+
+       function searchLimited$2(limit, projection, rtree) {
+         limit = limit || 5;
+         return partitionViewport$2(projection).reduce(function (result, extent) {
+           var found = rtree.search(extent.bbox()).slice(0, limit).map(function (d) {
+             return d.data;
+           });
+           return found.length ? result.concat(found) : result;
+         }, []);
+       }
+
+       var serviceMapillary = {
+         // Initialize Mapillary
+         init: function init() {
+           if (!_mlyCache) {
+             this.reset();
            }
-         }, {
-           key: "maxNode",
-           value: function maxNode() {
-             var u = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : this._root;
-             if (u) while (u.right) {
-               u = u.right;
-             }
-             return u;
+
+           this.event = utilRebind(this, dispatch$4, 'on');
+         },
+         // Reset cache and state
+         reset: function reset() {
+           if (_mlyCache) {
+             Object.values(_mlyCache.requests.inflight).forEach(function (request) {
+               request.abort();
+             });
            }
-         }, {
-           key: "insert",
-           value: function insert(key, data) {
-             var z = this._root;
-             var p = null;
-             var comp = this._compare;
-             var cmp;
-
-             if (this._noDuplicates) {
-               while (z) {
-                 p = z;
-                 cmp = comp(z.key, key);
-                 if (cmp === 0) return;else if (comp(z.key, key) < 0) z = z.right;else z = z.left;
-               }
-             } else {
-               while (z) {
-                 p = z;
-                 if (comp(z.key, key) < 0) z = z.right;else z = z.left;
-               }
+
+           _mlyCache = {
+             images: {
+               rtree: new RBush(),
+               forImageId: {}
+             },
+             image_detections: {
+               forImageId: {}
+             },
+             signs: {
+               rtree: new RBush()
+             },
+             points: {
+               rtree: new RBush()
+             },
+             sequences: {
+               rtree: new RBush(),
+               lineString: {}
+             },
+             requests: {
+               loaded: {},
+               inflight: {}
              }
+           };
+           _mlyActiveImage = null;
+         },
+         // Get visible images
+         images: function images(projection) {
+           var limit = 5;
+           return searchLimited$2(limit, projection, _mlyCache.images.rtree);
+         },
+         // Get visible traffic signs
+         signs: function signs(projection) {
+           var limit = 5;
+           return searchLimited$2(limit, projection, _mlyCache.signs.rtree);
+         },
+         // Get visible map (point) features
+         mapFeatures: function mapFeatures(projection) {
+           var limit = 5;
+           return searchLimited$2(limit, projection, _mlyCache.points.rtree);
+         },
+         // Get cached image by id
+         cachedImage: function cachedImage(imageId) {
+           return _mlyCache.images.forImageId[imageId];
+         },
+         // Get visible sequences
+         sequences: function sequences(projection) {
+           var viewport = projection.clipExtent();
+           var min = [viewport[0][0], viewport[1][1]];
+           var max = [viewport[1][0], viewport[0][1]];
+           var bbox = geoExtent(projection.invert(min), projection.invert(max)).bbox();
+           var sequenceIds = {};
+           var lineStrings = [];
 
-             z = {
-               key: key,
-               data: data,
-               left: null,
-               right: null,
-               parent: p
-             };
-             if (!p) this._root = z;else if (comp(p.key, z.key) < 0) p.right = z;else p.left = z;
-             this.splay(z);
-             this._size++;
-             return z;
-           }
-         }, {
-           key: "find",
-           value: function find(key) {
-             var z = this._root;
-             var comp = this._compare;
+           _mlyCache.images.rtree.search(bbox).forEach(function (d) {
+             if (d.data.sequence_id) {
+               sequenceIds[d.data.sequence_id] = true;
+             }
+           });
 
-             while (z) {
-               var cmp = comp(z.key, key);
-               if (cmp < 0) z = z.right;else if (cmp > 0) z = z.left;else return z;
+           Object.keys(sequenceIds).forEach(function (sequenceId) {
+             if (_mlyCache.sequences.lineString[sequenceId]) {
+               lineStrings = lineStrings.concat(_mlyCache.sequences.lineString[sequenceId]);
              }
+           });
+           return lineStrings;
+         },
+         // Load images in the visible area
+         loadImages: function loadImages(projection) {
+           loadTiles$2('images', tileUrl, 14, projection);
+         },
+         // Load traffic signs in the visible area
+         loadSigns: function loadSigns(projection) {
+           loadTiles$2('signs', trafficSignTileUrl, 14, projection);
+         },
+         // Load map (point) features in the visible area
+         loadMapFeatures: function loadMapFeatures(projection) {
+           loadTiles$2('points', mapFeatureTileUrl, 14, projection);
+         },
+         // Return a promise that resolves when the image viewer (Mapillary JS) library has finished loading
+         ensureViewerLoaded: function ensureViewerLoaded(context) {
+           if (_loadViewerPromise$2) return _loadViewerPromise$2; // add mly-wrapper
 
-             return null;
-           }
-           /**
-            * Whether the tree contains a node with the given key
-            * @param  {Key} key
-            * @return {boolean} true/false
-            */
+           var wrap = context.container().select('.photoviewer').selectAll('.mly-wrapper').data([0]);
+           wrap.enter().append('div').attr('id', 'ideditor-mly').attr('class', 'photo-wrapper mly-wrapper').classed('hide', true);
+           var that = this;
+           _loadViewerPromise$2 = new Promise(function (resolve, reject) {
+             var loadedCount = 0;
 
-         }, {
-           key: "contains",
-           value: function contains(key) {
-             var node = this._root;
-             var comparator = this._compare;
+             function loaded() {
+               loadedCount += 1; // wait until both files are loaded
 
-             while (node) {
-               var cmp = comparator(key, node.key);
-               if (cmp === 0) return true;else if (cmp < 0) node = node.left;else node = node.right;
+               if (loadedCount === 2) resolve();
              }
 
-             return false;
-           }
-         }, {
-           key: "remove",
-           value: function remove(key) {
-             var z = this.find(key);
-             if (!z) return false;
-             this.splay(z);
-             if (!z.left) this.replace(z, z.right);else if (!z.right) this.replace(z, z.left);else {
-               var y = this.minNode(z.right);
+             var head = select('head'); // load mapillary-viewercss
 
-               if (y.parent !== z) {
-                 this.replace(y, y.right);
-                 y.right = z.right;
-                 y.right.parent = y;
-               }
+             head.selectAll('#ideditor-mapillary-viewercss').data([0]).enter().append('link').attr('id', 'ideditor-mapillary-viewercss').attr('rel', 'stylesheet').attr('crossorigin', 'anonymous').attr('href', context.asset(viewercss)).on('load.serviceMapillary', loaded).on('error.serviceMapillary', function () {
+               reject();
+             }); // load mapillary-viewerjs
 
-               this.replace(z, y);
-               y.left = z.left;
-               y.left.parent = y;
-             }
-             this._size--;
-             return true;
+             head.selectAll('#ideditor-mapillary-viewerjs').data([0]).enter().append('script').attr('id', 'ideditor-mapillary-viewerjs').attr('crossorigin', 'anonymous').attr('src', context.asset(viewerjs)).on('load.serviceMapillary', loaded).on('error.serviceMapillary', function () {
+               reject();
+             });
+           })["catch"](function () {
+             _loadViewerPromise$2 = null;
+           }).then(function () {
+             that.initViewer(context);
+           });
+           return _loadViewerPromise$2;
+         },
+         // Load traffic sign image sprites
+         loadSignResources: function loadSignResources(context) {
+           context.ui().svgDefs.addSprites(['mapillary-sprite'], false
+           /* don't override colors */
+           );
+           return this;
+         },
+         // Load map (point) feature image sprites
+         loadObjectResources: function loadObjectResources(context) {
+           context.ui().svgDefs.addSprites(['mapillary-object-sprite'], false
+           /* don't override colors */
+           );
+           return this;
+         },
+         // Remove previous detections in image viewer
+         resetTags: function resetTags() {
+           if (_mlyViewer && !_mlyFallback) {
+             _mlyViewer.getComponent('tag').removeAll();
            }
-         }, {
-           key: "removeNode",
-           value: function removeNode(z) {
-             if (!z) return false;
-             this.splay(z);
-             if (!z.left) this.replace(z, z.right);else if (!z.right) this.replace(z, z.left);else {
-               var y = this.minNode(z.right);
-
-               if (y.parent !== z) {
-                 this.replace(y, y.right);
-                 y.right = z.right;
-                 y.right.parent = y;
-               }
+         },
+         // Show map feature detections in image viewer
+         showFeatureDetections: function showFeatureDetections(value) {
+           _mlyShowFeatureDetections = value;
 
-               this.replace(z, y);
-               y.left = z.left;
-               y.left.parent = y;
-             }
-             this._size--;
-             return true;
+           if (!_mlyShowFeatureDetections && !_mlyShowSignDetections) {
+             this.resetTags();
            }
-         }, {
-           key: "erase",
-           value: function erase(key) {
-             var z = this.find(key);
-             if (!z) return;
-             this.splay(z);
-             var s = z.left;
-             var t = z.right;
-             var sMax = null;
+         },
+         // Show traffic sign detections in image viewer
+         showSignDetections: function showSignDetections(value) {
+           _mlyShowSignDetections = value;
 
-             if (s) {
-               s.parent = null;
-               sMax = this.maxNode(s);
-               this.splay(sMax);
-               this._root = sMax;
-             }
+           if (!_mlyShowFeatureDetections && !_mlyShowSignDetections) {
+             this.resetTags();
+           }
+         },
+         // Apply filter to image viewer
+         filterViewer: function filterViewer(context) {
+           var showsPano = context.photos().showsPanoramic();
+           var showsFlat = context.photos().showsFlat();
+           var fromDate = context.photos().fromDate();
+           var toDate = context.photos().toDate();
+           var filter = ['all'];
+           if (!showsPano) filter.push(['!=', 'cameraType', 'spherical']);
+           if (!showsFlat && showsPano) filter.push(['==', 'pano', true]);
 
-             if (t) {
-               if (s) sMax.right = t;else this._root = t;
-               t.parent = sMax;
-             }
+           if (fromDate) {
+             filter.push(['>=', 'capturedAt', new Date(fromDate).getTime()]);
+           }
 
-             this._size--;
+           if (toDate) {
+             filter.push(['>=', 'capturedAt', new Date(toDate).getTime()]);
            }
-           /**
-            * Removes and returns the node with smallest key
-            * @return {?Node}
-            */
 
-         }, {
-           key: "pop",
-           value: function pop() {
-             var node = this._root,
-                 returnValue = null;
+           if (_mlyViewer) {
+             _mlyViewer.setFilter(filter);
+           }
 
-             if (node) {
-               while (node.left) {
-                 node = node.left;
-               }
+           _mlyViewerFilter = filter;
+           return filter;
+         },
+         // Make the image viewer visible
+         showViewer: function showViewer(context) {
+           var wrap = context.container().select('.photoviewer').classed('hide', false);
+           var isHidden = wrap.selectAll('.photo-wrapper.mly-wrapper.hide').size();
 
-               returnValue = {
-                 key: node.key,
-                 data: node.data
-               };
-               this.remove(node.key);
-             }
+           if (isHidden && _mlyViewer) {
+             wrap.selectAll('.photo-wrapper:not(.mly-wrapper)').classed('hide', true);
+             wrap.selectAll('.photo-wrapper.mly-wrapper').classed('hide', false);
 
-             return returnValue;
+             _mlyViewer.resize();
            }
-           /* eslint-disable class-methods-use-this */
-
-           /**
-            * Successor node
-            * @param  {Node} node
-            * @return {?Node}
-            */
 
-         }, {
-           key: "next",
-           value: function next(node) {
-             var successor = node;
+           return this;
+         },
+         // Hide the image viewer and resets map markers
+         hideViewer: function hideViewer(context) {
+           _mlyActiveImage = null;
 
-             if (successor) {
-               if (successor.right) {
-                 successor = successor.right;
+           if (!_mlyFallback && _mlyViewer) {
+             _mlyViewer.getComponent('sequence').stop();
+           }
 
-                 while (successor && successor.left) {
-                   successor = successor.left;
-                 }
-               } else {
-                 successor = node.parent;
+           var viewer = context.container().select('.photoviewer');
+           if (!viewer.empty()) viewer.datum(null);
+           viewer.classed('hide', true).selectAll('.photo-wrapper').classed('hide', true);
+           this.updateUrlImage(null);
+           dispatch$4.call('imageChanged');
+           dispatch$4.call('loadedMapFeatures');
+           dispatch$4.call('loadedSigns');
+           return this.setStyles(context, null);
+         },
+         // Update the URL with current image id
+         updateUrlImage: function updateUrlImage(imageId) {
+           if (!window.mocha) {
+             var hash = utilStringQs(window.location.hash);
 
-                 while (successor && successor.right === node) {
-                   node = successor;
-                   successor = successor.parent;
-                 }
-               }
+             if (imageId) {
+               hash.photo = 'mapillary/' + imageId;
+             } else {
+               delete hash.photo;
              }
 
-             return successor;
+             window.location.replace('#' + utilQsString(hash, true));
+           }
+         },
+         // Highlight the detection in the viewer that is related to the clicked map feature
+         highlightDetection: function highlightDetection(detection) {
+           if (detection) {
+             _mlyHighlightedDetection = detection.id;
            }
-           /**
-            * Predecessor node
-            * @param  {Node} node
-            * @return {?Node}
-            */
-
-         }, {
-           key: "prev",
-           value: function prev(node) {
-             var predecessor = node;
-
-             if (predecessor) {
-               if (predecessor.left) {
-                 predecessor = predecessor.left;
 
-                 while (predecessor && predecessor.right) {
-                   predecessor = predecessor.right;
-                 }
-               } else {
-                 predecessor = node.parent;
+           return this;
+         },
+         // Initialize image viewer (Mapillar JS)
+         initViewer: function initViewer(context) {
+           var that = this;
+           if (!window.mapillary) return;
+           var opts = {
+             accessToken: accessToken,
+             component: {
+               cover: false,
+               keyboard: false,
+               tag: true
+             },
+             container: 'ideditor-mly'
+           }; // Disable components requiring WebGL support
 
-                 while (predecessor && predecessor.left === node) {
-                   node = predecessor;
-                   predecessor = predecessor.parent;
-                 }
-               }
-             }
+           if (!mapillary.isSupported() && mapillary.isFallbackSupported()) {
+             _mlyFallback = true;
+             opts.component = {
+               cover: false,
+               direction: false,
+               imagePlane: false,
+               keyboard: false,
+               mouse: false,
+               sequence: false,
+               tag: false,
+               image: true,
+               // fallback
+               navigation: true // fallback
 
-             return predecessor;
+             };
            }
-           /* eslint-enable class-methods-use-this */
 
-           /**
-            * @param  {forEachCallback} callback
-            * @return {SplayTree}
-            */
+           _mlyViewer = new mapillary.Viewer(opts);
 
-         }, {
-           key: "forEach",
-           value: function forEach(callback) {
-             var current = this._root;
-             var s = [],
-                 done = false,
-                 i = 0;
+           _mlyViewer.on('image', imageChanged);
 
-             while (!done) {
-               // Reach the left most Node of the current Node
-               if (current) {
-                 // Place pointer to a tree node on the stack
-                 // before traversing the node's left subtree
-                 s.push(current);
-                 current = current.left;
-               } else {
-                 // BackTrack from the empty subtree and visit the Node
-                 // at the top of the stack; however, if the stack is
-                 // empty you are done
-                 if (s.length > 0) {
-                   current = s.pop();
-                   callback(current, i++); // We have visited the node and its left
-                   // subtree. Now, it's right subtree's turn
+           _mlyViewer.on('bearing', bearingChanged);
 
-                   current = current.right;
-                 } else done = true;
-               }
-             }
+           if (_mlyViewerFilter) {
+             _mlyViewer.setFilter(_mlyViewerFilter);
+           } // Register viewer resize handler
 
-             return this;
-           }
-           /**
-            * Walk key range from `low` to `high`. Stops if `fn` returns a value.
-            * @param  {Key}      low
-            * @param  {Key}      high
-            * @param  {Function} fn
-            * @param  {*?}       ctx
-            * @return {SplayTree}
-            */
 
-         }, {
-           key: "range",
-           value: function range(low, high, fn, ctx) {
-             var Q = [];
-             var compare = this._compare;
-             var node = this._root,
-                 cmp;
-
-             while (Q.length !== 0 || node) {
-               if (node) {
-                 Q.push(node);
-                 node = node.left;
-               } else {
-                 node = Q.pop();
-                 cmp = compare(node.key, high);
+           context.ui().photoviewer.on('resize.mapillary', function () {
+             if (_mlyViewer) _mlyViewer.resize();
+           }); // imageChanged: called after the viewer has changed images and is ready.
 
-                 if (cmp > 0) {
-                   break;
-                 } else if (compare(node.key, low) >= 0) {
-                   if (fn.call(ctx, node)) return this; // stop if smth is returned
-                 }
+           function imageChanged(node) {
+             that.resetTags();
+             var image = node.image;
+             that.setActiveImage(image);
+             that.setStyles(context, null);
+             var loc = [image.originalLngLat.lng, image.originalLngLat.lat];
+             context.map().centerEase(loc);
+             that.updateUrlImage(image.id);
 
-                 node = node.right;
-               }
+             if (_mlyShowFeatureDetections || _mlyShowSignDetections) {
+               that.updateDetections(image.id, "".concat(apiUrl, "/").concat(image.id, "/detections?access_token=").concat(accessToken, "&fields=id,image,geometry,value"));
              }
 
-             return this;
-           }
-           /**
-            * Returns all keys in order
-            * @return {Array<Key>}
-            */
+             dispatch$4.call('imageChanged');
+           } // bearingChanged: called when the bearing changes in the image viewer.
 
-         }, {
-           key: "keys",
-           value: function keys() {
-             var current = this._root;
-             var s = [],
-                 r = [],
-                 done = false;
-
-             while (!done) {
-               if (current) {
-                 s.push(current);
-                 current = current.left;
-               } else {
-                 if (s.length > 0) {
-                   current = s.pop();
-                   r.push(current.key);
-                   current = current.right;
-                 } else done = true;
-               }
-             }
 
-             return r;
+           function bearingChanged(e) {
+             dispatch$4.call('bearingChanged', undefined, e);
+           }
+         },
+         // Move to an image
+         selectImage: function selectImage(context, imageId) {
+           if (_mlyViewer && imageId) {
+             _mlyViewer.moveTo(imageId)["catch"](function (e) {
+               console.error('mly3', e); // eslint-disable-line no-console
+             });
            }
-           /**
-            * Returns `data` fields of all nodes in order.
-            * @return {Array<Value>}
-            */
-
-         }, {
-           key: "values",
-           value: function values() {
-             var current = this._root;
-             var s = [],
-                 r = [],
-                 done = false;
-
-             while (!done) {
-               if (current) {
-                 s.push(current);
-                 current = current.left;
-               } else {
-                 if (s.length > 0) {
-                   current = s.pop();
-                   r.push(current.data);
-                   current = current.right;
-                 } else done = true;
-               }
-             }
 
-             return r;
+           return this;
+         },
+         // Return the currently displayed image
+         getActiveImage: function getActiveImage() {
+           return _mlyActiveImage;
+         },
+         // Return a list of detection objects for the given id
+         getDetections: function getDetections(id) {
+           return loadData("".concat(apiUrl, "/").concat(id, "/detections?access_token=").concat(accessToken, "&fields=id,value,image"));
+         },
+         // Set the currently visible image
+         setActiveImage: function setActiveImage(image) {
+           if (image) {
+             _mlyActiveImage = {
+               ca: image.originalCompassAngle,
+               id: image.id,
+               loc: [image.originalLngLat.lng, image.originalLngLat.lat],
+               is_pano: image.cameraType === 'spherical',
+               sequence_id: image.sequenceId
+             };
+           } else {
+             _mlyActiveImage = null;
            }
-           /**
-            * Returns node at given index
-            * @param  {number} index
-            * @return {?Node}
-            */
+         },
+         // Update the currently highlighted sequence and selected bubble.
+         setStyles: function setStyles(context, hovered) {
+           var hoveredImageId = hovered && hovered.id;
+           var hoveredSequenceId = hovered && hovered.sequence_id;
+           var selectedSequenceId = _mlyActiveImage && _mlyActiveImage.sequence_id;
+           context.container().selectAll('.layer-mapillary .viewfield-group').classed('highlighted', function (d) {
+             return d.sequence_id === selectedSequenceId || d.id === hoveredImageId;
+           }).classed('hovered', function (d) {
+             return d.id === hoveredImageId;
+           });
+           context.container().selectAll('.layer-mapillary .sequence').classed('highlighted', function (d) {
+             return d.properties.id === hoveredSequenceId;
+           }).classed('currentView', function (d) {
+             return d.properties.id === selectedSequenceId;
+           });
+           return this;
+         },
+         // Get detections for the current image and shows them in the image viewer
+         updateDetections: function updateDetections(imageId, url) {
+           if (!_mlyViewer || _mlyFallback) return;
+           if (!imageId) return;
+           var cache = _mlyCache.image_detections;
 
-         }, {
-           key: "at",
-           value: function at(index) {
-             // removed after a consideration, more misleading than useful
-             // index = index % this.size;
-             // if (index < 0) index = this.size - index;
-             var current = this._root;
-             var s = [],
-                 done = false,
-                 i = 0;
+           if (cache.forImageId[imageId]) {
+             showDetections(_mlyCache.image_detections.forImageId[imageId]);
+           } else {
+             loadData(url).then(function (detections) {
+               detections.forEach(function (detection) {
+                 if (!cache.forImageId[imageId]) {
+                   cache.forImageId[imageId] = [];
+                 }
 
-             while (!done) {
-               if (current) {
-                 s.push(current);
-                 current = current.left;
-               } else {
-                 if (s.length > 0) {
-                   current = s.pop();
-                   if (i === index) return current;
-                   i++;
-                   current = current.right;
-                 } else done = true;
-               }
-             }
+                 cache.forImageId[imageId].push({
+                   geometry: detection.geometry,
+                   id: detection.id,
+                   image_id: imageId,
+                   value: detection.value
+                 });
+               });
+               showDetections(_mlyCache.image_detections.forImageId[imageId] || []);
+             });
+           } // Create a tag for each detection and shows it in the image viewer
 
-             return null;
-           }
-           /**
-            * Bulk-load items. Both array have to be same size
-            * @param  {Array<Key>}    keys
-            * @param  {Array<Value>}  [values]
-            * @param  {Boolean}       [presort=false] Pre-sort keys and values, using
-            *                                         tree's comparator. Sorting is done
-            *                                         in-place
-            * @return {AVLTree}
-            */
 
-         }, {
-           key: "load",
-           value: function load() {
-             var keys = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
-             var values = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : [];
-             var presort = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false;
-             if (this._size !== 0) throw new Error('bulk-load: tree is not empty');
-             var size = keys.length;
-             if (presort) sort(keys, values, 0, size - 1, this._compare);
-             this._root = loadRecursive(null, keys, values, 0, size);
-             this._size = size;
-             return this;
-           }
-         }, {
-           key: "min",
-           value: function min() {
-             var node = this.minNode(this._root);
-             if (node) return node.key;else return null;
-           }
-         }, {
-           key: "max",
-           value: function max() {
-             var node = this.maxNode(this._root);
-             if (node) return node.key;else return null;
-           }
-         }, {
-           key: "isEmpty",
-           value: function isEmpty() {
-             return this._root === null;
-           }
-         }, {
-           key: "size",
-           get: function get() {
-             return this._size;
-           }
-           /**
-            * Create a tree and load it with items
-            * @param  {Array<Key>}          keys
-            * @param  {Array<Value>?}        [values]
-             * @param  {Function?}            [comparator]
-            * @param  {Boolean?}             [presort=false] Pre-sort keys and values, using
-            *                                               tree's comparator. Sorting is done
-            *                                               in-place
-            * @param  {Boolean?}             [noDuplicates=false]   Allow duplicates
-            * @return {SplayTree}
-            */
+           function showDetections(detections) {
+             var tagComponent = _mlyViewer.getComponent('tag');
 
-         }], [{
-           key: "createTree",
-           value: function createTree(keys, values, comparator, presort, noDuplicates) {
-             return new SplayTree(comparator, noDuplicates).load(keys, values, presort);
-           }
-         }]);
+             detections.forEach(function (data) {
+               var tag = makeTag(data);
 
-         return SplayTree;
-       }();
+               if (tag) {
+                 tagComponent.add([tag]);
+               }
+             });
+           } // Create a Mapillary JS tag object
 
-       function loadRecursive(parent, keys, values, start, end) {
-         var size = end - start;
 
-         if (size > 0) {
-           var middle = start + Math.floor(size / 2);
-           var key = keys[middle];
-           var data = values[middle];
-           var node = {
-             key: key,
-             data: data,
-             parent: parent
-           };
-           node.left = loadRecursive(node, keys, values, start, middle);
-           node.right = loadRecursive(node, keys, values, middle + 1, end);
-           return node;
-         }
+           function makeTag(data) {
+             var valueParts = data.value.split('--');
+             if (!valueParts.length) return;
+             var tag;
+             var text;
+             var color = 0xffffff;
 
-         return null;
-       }
+             if (_mlyHighlightedDetection === data.id) {
+               color = 0xffff00;
+               text = valueParts[1];
 
-       function sort(keys, values, left, right, compare) {
-         if (left >= right) return;
-         var pivot = keys[left + right >> 1];
-         var i = left - 1;
-         var j = right + 1;
+               if (text === 'flat' || text === 'discrete' || text === 'sign') {
+                 text = valueParts[2];
+               }
 
-         while (true) {
-           do {
-             i++;
-           } while (compare(keys[i], pivot) < 0);
+               text = text.replace(/-/g, ' ');
+               text = text.charAt(0).toUpperCase() + text.slice(1);
+               _mlyHighlightedDetection = null;
+             }
 
-           do {
-             j--;
-           } while (compare(keys[j], pivot) > 0);
+             var decodedGeometry = window.atob(data.geometry);
+             var uintArray = new Uint8Array(decodedGeometry.length);
 
-           if (i >= j) break;
-           var tmp = keys[i];
-           keys[i] = keys[j];
-           keys[j] = tmp;
-           tmp = values[i];
-           values[i] = values[j];
-           values[j] = tmp;
+             for (var i = 0; i < decodedGeometry.length; i++) {
+               uintArray[i] = decodedGeometry.charCodeAt(i);
+             }
+
+             var tile = new VectorTile(new pbf(uintArray.buffer));
+             var layer = tile.layers['mpy-or'];
+             var geometries = layer.feature(0).loadGeometry();
+             var polygon = geometries.map(function (ring) {
+               return ring.map(function (point) {
+                 return [point.x / layer.extent, point.y / layer.extent];
+               });
+             });
+             tag = new mapillary.OutlineTag(data.id, new mapillary.PolygonGeometry(polygon[0]), {
+               text: text,
+               textColor: color,
+               lineColor: color,
+               lineWidth: 2,
+               fillColor: color,
+               fillOpacity: 0.3
+             });
+             return tag;
+           }
+         },
+         // Return the current cache
+         cache: function cache() {
+           return _mlyCache;
          }
+       };
 
-         sort(keys, values, left, j, compare);
-         sort(keys, values, j + 1, right, compare);
-       }
+       function validationIssue(attrs) {
+         this.type = attrs.type; // required - name of rule that created the issue (e.g. 'missing_tag')
 
-       var NORMAL = 0;
-       var NON_CONTRIBUTING = 1;
-       var SAME_TRANSITION = 2;
-       var DIFFERENT_TRANSITION = 3;
+         this.subtype = attrs.subtype; // optional - category of the issue within the type (e.g. 'relation_type' under 'missing_tag')
 
-       var INTERSECTION = 0;
-       var UNION = 1;
-       var DIFFERENCE = 2;
-       var XOR = 3;
+         this.severity = attrs.severity; // required - 'warning' or 'error'
 
-       /**
-        * @param  {SweepEvent} event
-        * @param  {SweepEvent} prev
-        * @param  {Operation} operation
-        */
+         this.message = attrs.message; // required - function returning localized string
 
-       function computeFields(event, prev, operation) {
-         // compute inOut and otherInOut fields
-         if (prev === null) {
-           event.inOut = false;
-           event.otherInOut = true; // previous line segment in sweepline belongs to the same polygon
-         } else {
-           if (event.isSubject === prev.isSubject) {
-             event.inOut = !prev.inOut;
-             event.otherInOut = prev.otherInOut; // previous line segment in sweepline belongs to the clipping polygon
-           } else {
-             event.inOut = !prev.otherInOut;
-             event.otherInOut = prev.isVertical() ? !prev.inOut : prev.inOut;
-           } // compute prevInResult field
+         this.reference = attrs.reference; // optional - function(selection) to render reference information
 
+         this.entityIds = attrs.entityIds; // optional - array of IDs of entities involved in the issue
 
-           if (prev) {
-             event.prevInResult = !inResult(prev, operation) || prev.isVertical() ? prev.prevInResult : prev;
-           }
-         } // check if the line segment belongs to the Boolean operation
+         this.loc = attrs.loc; // optional - [lon, lat] to zoom in on to see the issue
 
+         this.data = attrs.data; // optional - object containing extra data for the fixes
 
-         var isInResult = inResult(event, operation);
+         this.dynamicFixes = attrs.dynamicFixes; // optional - function(context) returning fixes
 
-         if (isInResult) {
-           event.resultTransition = determineResultTransition(event, operation);
-         } else {
-           event.resultTransition = 0;
-         }
-       }
-       /* eslint-disable indent */
+         this.hash = attrs.hash; // optional - string to further differentiate the issue
 
-       function inResult(event, operation) {
-         switch (event.type) {
-           case NORMAL:
-             switch (operation) {
-               case INTERSECTION:
-                 return !event.otherInOut;
+         this.id = generateID.apply(this); // generated - see below
 
-               case UNION:
-                 return event.otherInOut;
+         this.autoFix = null; // generated - if autofix exists, will be set below
+         // A unique, deterministic string hash.
+         // Issues with identical id values are considered identical.
 
-               case DIFFERENCE:
-                 // return (event.isSubject && !event.otherInOut) ||
-                 //         (!event.isSubject && event.otherInOut);
-                 return event.isSubject && event.otherInOut || !event.isSubject && !event.otherInOut;
+         function generateID() {
+           var parts = [this.type];
 
-               case XOR:
-                 return true;
-             }
+           if (this.hash) {
+             // subclasses can pass in their own differentiator
+             parts.push(this.hash);
+           }
 
-             break;
+           if (this.subtype) {
+             parts.push(this.subtype);
+           } // include the entities this issue is for
+           // (sort them so the id is deterministic)
 
-           case SAME_TRANSITION:
-             return operation === INTERSECTION || operation === UNION;
 
-           case DIFFERENT_TRANSITION:
-             return operation === DIFFERENCE;
+           if (this.entityIds) {
+             var entityKeys = this.entityIds.slice().sort();
+             parts.push.apply(parts, entityKeys);
+           }
 
-           case NON_CONTRIBUTING:
-             return false;
+           return parts.join(':');
          }
 
-         return false;
-       }
-       /* eslint-enable indent */
-
+         this.extent = function (resolver) {
+           if (this.loc) {
+             return geoExtent(this.loc);
+           }
 
-       function determineResultTransition(event, operation) {
-         var thisIn = !event.inOut;
-         var thatIn = !event.otherInOut;
-         var isIn;
+           if (this.entityIds && this.entityIds.length) {
+             return this.entityIds.reduce(function (extent, entityId) {
+               return extent.extend(resolver.entity(entityId).extent(resolver));
+             }, geoExtent());
+           }
 
-         switch (operation) {
-           case INTERSECTION:
-             isIn = thisIn && thatIn;
-             break;
+           return null;
+         };
 
-           case UNION:
-             isIn = thisIn || thatIn;
-             break;
+         this.fixes = function (context) {
+           var fixes = this.dynamicFixes ? this.dynamicFixes(context) : [];
+           var issue = this;
 
-           case XOR:
-             isIn = thisIn ^ thatIn;
-             break;
+           if (issue.severity === 'warning') {
+             // allow ignoring any issue that's not an error
+             fixes.push(new validationIssueFix({
+               title: _t.html('issues.fix.ignore_issue.title'),
+               icon: 'iD-icon-close',
+               onClick: function onClick() {
+                 context.validator().ignoreIssue(this.issue.id);
+               }
+             }));
+           }
 
-           case DIFFERENCE:
-             if (event.isSubject) {
-               isIn = thisIn && !thatIn;
-             } else {
-               isIn = thatIn && !thisIn;
-             }
+           fixes.forEach(function (fix) {
+             // the id doesn't matter as long as it's unique to this issue/fix
+             fix.id = fix.title; // add a reference to the issue for use in actions
 
-             break;
-         }
+             fix.issue = issue;
 
-         return isIn ? +1 : -1;
+             if (fix.autoArgs) {
+               issue.autoFix = fix;
+             }
+           });
+           return fixes;
+         };
        }
+       function validationIssueFix(attrs) {
+         this.title = attrs.title; // Required
 
-       var SweepEvent = /*#__PURE__*/function () {
-         /**
-          * Sweepline event
-          *
-          * @class {SweepEvent}
-          * @param {Array.<Number>}  point
-          * @param {Boolean}         left
-          * @param {SweepEvent=}     otherEvent
-          * @param {Boolean}         isSubject
-          * @param {Number}          edgeType
-          */
-         function SweepEvent(point, left, otherEvent, isSubject, edgeType) {
-           _classCallCheck(this, SweepEvent);
-
-           /**
-            * Is left endpoint?
-            * @type {Boolean}
-            */
-           this.left = left;
-           /**
-            * @type {Array.<Number>}
-            */
-
-           this.point = point;
-           /**
-            * Other edge reference
-            * @type {SweepEvent}
-            */
-
-           this.otherEvent = otherEvent;
-           /**
-            * Belongs to source or clipping polygon
-            * @type {Boolean}
-            */
-
-           this.isSubject = isSubject;
-           /**
-            * Edge contribution type
-            * @type {Number}
-            */
-
-           this.type = edgeType || NORMAL;
-           /**
-            * In-out transition for the sweepline crossing polygon
-            * @type {Boolean}
-            */
+         this.onClick = attrs.onClick; // Optional - the function to run to apply the fix
 
-           this.inOut = false;
-           /**
-            * @type {Boolean}
-            */
+         this.disabledReason = attrs.disabledReason; // Optional - a string explaining why the fix is unavailable, if any
 
-           this.otherInOut = false;
-           /**
-            * Previous event in result?
-            * @type {SweepEvent}
-            */
+         this.icon = attrs.icon; // Optional - shows 'iD-icon-wrench' if not set
 
-           this.prevInResult = null;
-           /**
-            * Type of result transition (0 = not in result, +1 = out-in, -1, in-out)
-            * @type {Number}
-            */
+         this.entityIds = attrs.entityIds || []; // Optional - used for hover-higlighting.
 
-           this.resultTransition = 0; // connection step
+         this.autoArgs = attrs.autoArgs; // Optional - pass [actions, annotation] arglist if this fix can automatically run
 
-           /**
-            * @type {Number}
-            */
+         this.issue = null; // Generated link - added by validationIssue
+       }
 
-           this.otherPos = -1;
-           /**
-            * @type {Number}
-            */
+       var buildRuleChecks = function buildRuleChecks() {
+         return {
+           equals: function equals(_equals) {
+             return function (tags) {
+               return Object.keys(_equals).every(function (k) {
+                 return _equals[k] === tags[k];
+               });
+             };
+           },
+           notEquals: function notEquals(_notEquals) {
+             return function (tags) {
+               return Object.keys(_notEquals).some(function (k) {
+                 return _notEquals[k] !== tags[k];
+               });
+             };
+           },
+           absence: function absence(_absence) {
+             return function (tags) {
+               return Object.keys(tags).indexOf(_absence) === -1;
+             };
+           },
+           presence: function presence(_presence) {
+             return function (tags) {
+               return Object.keys(tags).indexOf(_presence) > -1;
+             };
+           },
+           greaterThan: function greaterThan(_greaterThan) {
+             var key = Object.keys(_greaterThan)[0];
+             var value = _greaterThan[key];
+             return function (tags) {
+               return tags[key] > value;
+             };
+           },
+           greaterThanEqual: function greaterThanEqual(_greaterThanEqual) {
+             var key = Object.keys(_greaterThanEqual)[0];
+             var value = _greaterThanEqual[key];
+             return function (tags) {
+               return tags[key] >= value;
+             };
+           },
+           lessThan: function lessThan(_lessThan) {
+             var key = Object.keys(_lessThan)[0];
+             var value = _lessThan[key];
+             return function (tags) {
+               return tags[key] < value;
+             };
+           },
+           lessThanEqual: function lessThanEqual(_lessThanEqual) {
+             var key = Object.keys(_lessThanEqual)[0];
+             var value = _lessThanEqual[key];
+             return function (tags) {
+               return tags[key] <= value;
+             };
+           },
+           positiveRegex: function positiveRegex(_positiveRegex) {
+             var tagKey = Object.keys(_positiveRegex)[0];
 
-           this.outputContourId = -1;
-           this.isExteriorRing = true; // TODO: Looks unused, remove?
-         }
-         /**
-          * @param  {Array.<Number>}  p
-          * @return {Boolean}
-          */
+             var expression = _positiveRegex[tagKey].join('|');
 
+             var regex = new RegExp(expression);
+             return function (tags) {
+               return regex.test(tags[tagKey]);
+             };
+           },
+           negativeRegex: function negativeRegex(_negativeRegex) {
+             var tagKey = Object.keys(_negativeRegex)[0];
 
-         _createClass(SweepEvent, [{
-           key: "isBelow",
-           value: function isBelow(p) {
-             var p0 = this.point,
-                 p1 = this.otherEvent.point;
-             return this.left ? (p0[0] - p[0]) * (p1[1] - p[1]) - (p1[0] - p[0]) * (p0[1] - p[1]) > 0 // signedArea(this.point, this.otherEvent.point, p) > 0 :
-             : (p1[0] - p[0]) * (p0[1] - p[1]) - (p0[0] - p[0]) * (p1[1] - p[1]) > 0; //signedArea(this.otherEvent.point, this.point, p) > 0;
-           }
-           /**
-            * @param  {Array.<Number>}  p
-            * @return {Boolean}
-            */
+             var expression = _negativeRegex[tagKey].join('|');
 
-         }, {
-           key: "isAbove",
-           value: function isAbove(p) {
-             return !this.isBelow(p);
+             var regex = new RegExp(expression);
+             return function (tags) {
+               return !regex.test(tags[tagKey]);
+             };
            }
-           /**
-            * @return {Boolean}
-            */
+         };
+       };
 
-         }, {
-           key: "isVertical",
-           value: function isVertical() {
-             return this.point[0] === this.otherEvent.point[0];
+       var buildLineKeys = function buildLineKeys() {
+         return {
+           highway: {
+             rest_area: true,
+             services: true
+           },
+           railway: {
+             roundhouse: true,
+             station: true,
+             traverser: true,
+             turntable: true,
+             wash: true
            }
-           /**
-            * Does event belong to result?
-            * @return {Boolean}
-            */
+         };
+       };
 
-         }, {
-           key: "inResult",
-           get: function get() {
-             return this.resultTransition !== 0;
-           }
-         }, {
-           key: "clone",
-           value: function clone() {
-             var copy = new SweepEvent(this.point, this.left, this.otherEvent, this.isSubject, this.type);
-             copy.contourId = this.contourId;
-             copy.resultTransition = this.resultTransition;
-             copy.prevInResult = this.prevInResult;
-             copy.isExteriorRing = this.isExteriorRing;
-             copy.inOut = this.inOut;
-             copy.otherInOut = this.otherInOut;
-             return copy;
-           }
-         }]);
+       var serviceMapRules = {
+         init: function init() {
+           this._ruleChecks = buildRuleChecks();
+           this._validationRules = [];
+           this._areaKeys = osmAreaKeys;
+           this._lineKeys = buildLineKeys();
+         },
+         // list of rules only relevant to tag checks...
+         filterRuleChecks: function filterRuleChecks(selector) {
+           var _ruleChecks = this._ruleChecks;
+           return Object.keys(selector).reduce(function (rules, key) {
+             if (['geometry', 'error', 'warning'].indexOf(key) === -1) {
+               rules.push(_ruleChecks[key](selector[key]));
+             }
 
-         return SweepEvent;
-       }();
+             return rules;
+           }, []);
+         },
+         // builds tagMap from mapcss-parse selector object...
+         buildTagMap: function buildTagMap(selector) {
+           var getRegexValues = function getRegexValues(regexes) {
+             return regexes.map(function (regex) {
+               return regex.replace(/\$|\^/g, '');
+             });
+           };
 
-       function equals(p1, p2) {
-         if (p1[0] === p2[0]) {
-           if (p1[1] === p2[1]) {
-             return true;
-           } else {
-             return false;
-           }
-         }
+           var tagMap = Object.keys(selector).reduce(function (expectedTags, key) {
+             var values;
+             var isRegex = /regex/gi.test(key);
+             var isEqual = /equals/gi.test(key);
 
-         return false;
-       } // const EPSILON = 1e-9;
-       // const abs = Math.abs;
-       // TODO https://github.com/w8r/martinez/issues/6#issuecomment-262847164
-       // Precision problem.
-       //
-       // module.exports = function equals(p1, p2) {
-       //   return abs(p1[0] - p2[0]) <= EPSILON && abs(p1[1] - p2[1]) <= EPSILON;
-       // };
+             if (isRegex || isEqual) {
+               Object.keys(selector[key]).forEach(function (selectorKey) {
+                 values = isEqual ? [selector[key][selectorKey]] : getRegexValues(selector[key][selectorKey]);
 
-       var epsilon$1 = 1.1102230246251565e-16;
-       var splitter = 134217729;
-       var resulterrbound = (3 + 8 * epsilon$1) * epsilon$1; // fast_expansion_sum_zeroelim routine from oritinal code
+                 if (expectedTags.hasOwnProperty(selectorKey)) {
+                   values = values.concat(expectedTags[selectorKey]);
+                 }
 
-       function sum(elen, e, flen, f, h) {
-         var Q, Qnew, hh, bvirt;
-         var enow = e[0];
-         var fnow = f[0];
-         var eindex = 0;
-         var findex = 0;
+                 expectedTags[selectorKey] = values;
+               });
+             } else if (/(greater|less)Than(Equal)?|presence/g.test(key)) {
+               var tagKey = /presence/.test(key) ? selector[key] : Object.keys(selector[key])[0];
+               values = [selector[key][tagKey]];
 
-         if (fnow > enow === fnow > -enow) {
-           Q = enow;
-           enow = e[++eindex];
-         } else {
-           Q = fnow;
-           fnow = f[++findex];
-         }
+               if (expectedTags.hasOwnProperty(tagKey)) {
+                 values = values.concat(expectedTags[tagKey]);
+               }
 
-         var hindex = 0;
+               expectedTags[tagKey] = values;
+             }
 
-         if (eindex < elen && findex < flen) {
-           if (fnow > enow === fnow > -enow) {
-             Qnew = enow + Q;
-             hh = Q - (Qnew - enow);
-             enow = e[++eindex];
-           } else {
-             Qnew = fnow + Q;
-             hh = Q - (Qnew - fnow);
-             fnow = f[++findex];
-           }
+             return expectedTags;
+           }, {});
+           return tagMap;
+         },
+         // inspired by osmWay#isArea()
+         inferGeometry: function inferGeometry(tagMap) {
+           var _lineKeys = this._lineKeys;
+           var _areaKeys = this._areaKeys;
 
-           Q = Qnew;
+           var keyValueDoesNotImplyArea = function keyValueDoesNotImplyArea(key) {
+             return utilArrayIntersection(tagMap[key], Object.keys(_areaKeys[key])).length > 0;
+           };
 
-           if (hh !== 0) {
-             h[hindex++] = hh;
-           }
+           var keyValueImpliesLine = function keyValueImpliesLine(key) {
+             return utilArrayIntersection(tagMap[key], Object.keys(_lineKeys[key])).length > 0;
+           };
 
-           while (eindex < elen && findex < flen) {
-             if (fnow > enow === fnow > -enow) {
-               Qnew = Q + enow;
-               bvirt = Qnew - Q;
-               hh = Q - (Qnew - bvirt) + (enow - bvirt);
-               enow = e[++eindex];
-             } else {
-               Qnew = Q + fnow;
-               bvirt = Qnew - Q;
-               hh = Q - (Qnew - bvirt) + (fnow - bvirt);
-               fnow = f[++findex];
+           if (tagMap.hasOwnProperty('area')) {
+             if (tagMap.area.indexOf('yes') > -1) {
+               return 'area';
              }
 
-             Q = Qnew;
-
-             if (hh !== 0) {
-               h[hindex++] = hh;
+             if (tagMap.area.indexOf('no') > -1) {
+               return 'line';
              }
            }
-         }
 
-         while (eindex < elen) {
-           Qnew = Q + enow;
-           bvirt = Qnew - Q;
-           hh = Q - (Qnew - bvirt) + (enow - bvirt);
-           enow = e[++eindex];
-           Q = Qnew;
+           for (var key in tagMap) {
+             if (key in _areaKeys && !keyValueDoesNotImplyArea(key)) {
+               return 'area';
+             }
 
-           if (hh !== 0) {
-             h[hindex++] = hh;
+             if (key in _lineKeys && keyValueImpliesLine(key)) {
+               return 'area';
+             }
            }
-         }
-
-         while (findex < flen) {
-           Qnew = Q + fnow;
-           bvirt = Qnew - Q;
-           hh = Q - (Qnew - bvirt) + (fnow - bvirt);
-           fnow = f[++findex];
-           Q = Qnew;
 
-           if (hh !== 0) {
-             h[hindex++] = hh;
-           }
-         }
-
-         if (Q !== 0 || hindex === 0) {
-           h[hindex++] = Q;
-         }
-
-         return hindex;
-       }
-       function estimate(elen, e) {
-         var Q = e[0];
-
-         for (var i = 1; i < elen; i++) {
-           Q += e[i];
-         }
-
-         return Q;
-       }
-       function vec(n) {
-         return new Float64Array(n);
-       }
-
-       var ccwerrboundA = (3 + 16 * epsilon$1) * epsilon$1;
-       var ccwerrboundB = (2 + 12 * epsilon$1) * epsilon$1;
-       var ccwerrboundC = (9 + 64 * epsilon$1) * epsilon$1 * epsilon$1;
-       var B = vec(4);
-       var C1 = vec(8);
-       var C2 = vec(12);
-       var D = vec(16);
-       var u = vec(4);
-
-       function orient2dadapt(ax, ay, bx, by, cx, cy, detsum) {
-         var acxtail, acytail, bcxtail, bcytail;
-
-         var bvirt, c, ahi, alo, bhi, blo, _i, _j, _0, s1, s0, t1, t0, u3;
-
-         var acx = ax - cx;
-         var bcx = bx - cx;
-         var acy = ay - cy;
-         var bcy = by - cy;
-         s1 = acx * bcy;
-         c = splitter * acx;
-         ahi = c - (c - acx);
-         alo = acx - ahi;
-         c = splitter * bcy;
-         bhi = c - (c - bcy);
-         blo = bcy - bhi;
-         s0 = alo * blo - (s1 - ahi * bhi - alo * bhi - ahi * blo);
-         t1 = acy * bcx;
-         c = splitter * acy;
-         ahi = c - (c - acy);
-         alo = acy - ahi;
-         c = splitter * bcx;
-         bhi = c - (c - bcx);
-         blo = bcx - bhi;
-         t0 = alo * blo - (t1 - ahi * bhi - alo * bhi - ahi * blo);
-         _i = s0 - t0;
-         bvirt = s0 - _i;
-         B[0] = s0 - (_i + bvirt) + (bvirt - t0);
-         _j = s1 + _i;
-         bvirt = _j - s1;
-         _0 = s1 - (_j - bvirt) + (_i - bvirt);
-         _i = _0 - t1;
-         bvirt = _0 - _i;
-         B[1] = _0 - (_i + bvirt) + (bvirt - t1);
-         u3 = _j + _i;
-         bvirt = u3 - _j;
-         B[2] = _j - (u3 - bvirt) + (_i - bvirt);
-         B[3] = u3;
-         var det = estimate(4, B);
-         var errbound = ccwerrboundB * detsum;
-
-         if (det >= errbound || -det >= errbound) {
-           return det;
-         }
-
-         bvirt = ax - acx;
-         acxtail = ax - (acx + bvirt) + (bvirt - cx);
-         bvirt = bx - bcx;
-         bcxtail = bx - (bcx + bvirt) + (bvirt - cx);
-         bvirt = ay - acy;
-         acytail = ay - (acy + bvirt) + (bvirt - cy);
-         bvirt = by - bcy;
-         bcytail = by - (bcy + bvirt) + (bvirt - cy);
-
-         if (acxtail === 0 && acytail === 0 && bcxtail === 0 && bcytail === 0) {
-           return det;
-         }
-
-         errbound = ccwerrboundC * detsum + resulterrbound * Math.abs(det);
-         det += acx * bcytail + bcy * acxtail - (acy * bcxtail + bcx * acytail);
-         if (det >= errbound || -det >= errbound) return det;
-         s1 = acxtail * bcy;
-         c = splitter * acxtail;
-         ahi = c - (c - acxtail);
-         alo = acxtail - ahi;
-         c = splitter * bcy;
-         bhi = c - (c - bcy);
-         blo = bcy - bhi;
-         s0 = alo * blo - (s1 - ahi * bhi - alo * bhi - ahi * blo);
-         t1 = acytail * bcx;
-         c = splitter * acytail;
-         ahi = c - (c - acytail);
-         alo = acytail - ahi;
-         c = splitter * bcx;
-         bhi = c - (c - bcx);
-         blo = bcx - bhi;
-         t0 = alo * blo - (t1 - ahi * bhi - alo * bhi - ahi * blo);
-         _i = s0 - t0;
-         bvirt = s0 - _i;
-         u[0] = s0 - (_i + bvirt) + (bvirt - t0);
-         _j = s1 + _i;
-         bvirt = _j - s1;
-         _0 = s1 - (_j - bvirt) + (_i - bvirt);
-         _i = _0 - t1;
-         bvirt = _0 - _i;
-         u[1] = _0 - (_i + bvirt) + (bvirt - t1);
-         u3 = _j + _i;
-         bvirt = u3 - _j;
-         u[2] = _j - (u3 - bvirt) + (_i - bvirt);
-         u[3] = u3;
-         var C1len = sum(4, B, 4, u, C1);
-         s1 = acx * bcytail;
-         c = splitter * acx;
-         ahi = c - (c - acx);
-         alo = acx - ahi;
-         c = splitter * bcytail;
-         bhi = c - (c - bcytail);
-         blo = bcytail - bhi;
-         s0 = alo * blo - (s1 - ahi * bhi - alo * bhi - ahi * blo);
-         t1 = acy * bcxtail;
-         c = splitter * acy;
-         ahi = c - (c - acy);
-         alo = acy - ahi;
-         c = splitter * bcxtail;
-         bhi = c - (c - bcxtail);
-         blo = bcxtail - bhi;
-         t0 = alo * blo - (t1 - ahi * bhi - alo * bhi - ahi * blo);
-         _i = s0 - t0;
-         bvirt = s0 - _i;
-         u[0] = s0 - (_i + bvirt) + (bvirt - t0);
-         _j = s1 + _i;
-         bvirt = _j - s1;
-         _0 = s1 - (_j - bvirt) + (_i - bvirt);
-         _i = _0 - t1;
-         bvirt = _0 - _i;
-         u[1] = _0 - (_i + bvirt) + (bvirt - t1);
-         u3 = _j + _i;
-         bvirt = u3 - _j;
-         u[2] = _j - (u3 - bvirt) + (_i - bvirt);
-         u[3] = u3;
-         var C2len = sum(C1len, C1, 4, u, C2);
-         s1 = acxtail * bcytail;
-         c = splitter * acxtail;
-         ahi = c - (c - acxtail);
-         alo = acxtail - ahi;
-         c = splitter * bcytail;
-         bhi = c - (c - bcytail);
-         blo = bcytail - bhi;
-         s0 = alo * blo - (s1 - ahi * bhi - alo * bhi - ahi * blo);
-         t1 = acytail * bcxtail;
-         c = splitter * acytail;
-         ahi = c - (c - acytail);
-         alo = acytail - ahi;
-         c = splitter * bcxtail;
-         bhi = c - (c - bcxtail);
-         blo = bcxtail - bhi;
-         t0 = alo * blo - (t1 - ahi * bhi - alo * bhi - ahi * blo);
-         _i = s0 - t0;
-         bvirt = s0 - _i;
-         u[0] = s0 - (_i + bvirt) + (bvirt - t0);
-         _j = s1 + _i;
-         bvirt = _j - s1;
-         _0 = s1 - (_j - bvirt) + (_i - bvirt);
-         _i = _0 - t1;
-         bvirt = _0 - _i;
-         u[1] = _0 - (_i + bvirt) + (bvirt - t1);
-         u3 = _j + _i;
-         bvirt = u3 - _j;
-         u[2] = _j - (u3 - bvirt) + (_i - bvirt);
-         u[3] = u3;
-         var Dlen = sum(C2len, C2, 4, u, D);
-         return D[Dlen - 1];
-       }
-
-       function orient2d(ax, ay, bx, by, cx, cy) {
-         var detleft = (ay - cy) * (bx - cx);
-         var detright = (ax - cx) * (by - cy);
-         var det = detleft - detright;
-         if (detleft === 0 || detright === 0 || detleft > 0 !== detright > 0) return det;
-         var detsum = Math.abs(detleft + detright);
-         if (Math.abs(det) >= ccwerrboundA * detsum) return det;
-         return -orient2dadapt(ax, ay, bx, by, cx, cy, detsum);
-       }
+           return 'line';
+         },
+         // adds from mapcss-parse selector check...
+         addRule: function addRule(selector) {
+           var rule = {
+             // checks relevant to mapcss-selector
+             checks: this.filterRuleChecks(selector),
+             // true if all conditions for a tag error are true..
+             matches: function matches(entity) {
+               return this.checks.every(function (check) {
+                 return check(entity.tags);
+               });
+             },
+             // borrowed from Way#isArea()
+             inferredGeometry: this.inferGeometry(this.buildTagMap(selector), this._areaKeys),
+             geometryMatches: function geometryMatches(entity, graph) {
+               if (entity.type === 'node' || entity.type === 'relation') {
+                 return selector.geometry === entity.type;
+               } else if (entity.type === 'way') {
+                 return this.inferredGeometry === entity.geometry(graph);
+               }
+             },
+             // when geometries match and tag matches are present, return a warning...
+             findIssues: function findIssues(entity, graph, issues) {
+               if (this.geometryMatches(entity, graph) && this.matches(entity)) {
+                 var severity = Object.keys(selector).indexOf('error') > -1 ? 'error' : 'warning';
+                 var _message = selector[severity];
+                 issues.push(new validationIssue({
+                   type: 'maprules',
+                   severity: severity,
+                   message: function message() {
+                     return _message;
+                   },
+                   entityIds: [entity.id]
+                 }));
+               }
+             }
+           };
 
-       /**
-        * Signed area of the triangle (p0, p1, p2)
-        * @param  {Array.<Number>} p0
-        * @param  {Array.<Number>} p1
-        * @param  {Array.<Number>} p2
-        * @return {Number}
-        */
+           this._validationRules.push(rule);
+         },
+         clearRules: function clearRules() {
+           this._validationRules = [];
+         },
+         // returns validationRules...
+         validationRules: function validationRules() {
+           return this._validationRules;
+         },
+         // returns ruleChecks
+         ruleChecks: function ruleChecks() {
+           return this._ruleChecks;
+         }
+       };
 
-       function signedArea(p0, p1, p2) {
-         var res = orient2d(p0[0], p0[1], p1[0], p1[1], p2[0], p2[1]);
-         if (res > 0) return -1;
-         if (res < 0) return 1;
-         return 0;
-       }
+       var apibase$2 = 'https://nominatim.openstreetmap.org/';
+       var _inflight$2 = {};
 
-       /**
-        * @param  {SweepEvent} e1
-        * @param  {SweepEvent} e2
-        * @return {Number}
-        */
+       var _nominatimCache;
 
-       function compareEvents(e1, e2) {
-         var p1 = e1.point;
-         var p2 = e2.point; // Different x-coordinate
+       var serviceNominatim = {
+         init: function init() {
+           _inflight$2 = {};
+           _nominatimCache = new RBush();
+         },
+         reset: function reset() {
+           Object.values(_inflight$2).forEach(function (controller) {
+             controller.abort();
+           });
+           _inflight$2 = {};
+           _nominatimCache = new RBush();
+         },
+         countryCode: function countryCode(location, callback) {
+           this.reverse(location, function (err, result) {
+             if (err) {
+               return callback(err);
+             } else if (result.address) {
+               return callback(null, result.address.country_code);
+             } else {
+               return callback('Unable to geocode', null);
+             }
+           });
+         },
+         reverse: function reverse(loc, callback) {
+           var cached = _nominatimCache.search({
+             minX: loc[0],
+             minY: loc[1],
+             maxX: loc[0],
+             maxY: loc[1]
+           });
 
-         if (p1[0] > p2[0]) return 1;
-         if (p1[0] < p2[0]) return -1; // Different points, but same x-coordinate
-         // Event with lower y-coordinate is processed first
+           if (cached.length > 0) {
+             if (callback) callback(null, cached[0].data);
+             return;
+           }
 
-         if (p1[1] !== p2[1]) return p1[1] > p2[1] ? 1 : -1;
-         return specialCases(e1, e2, p1);
-       }
-       /* eslint-disable no-unused-vars */
+           var params = {
+             zoom: 13,
+             format: 'json',
+             addressdetails: 1,
+             lat: loc[1],
+             lon: loc[0]
+           };
+           var url = apibase$2 + 'reverse?' + utilQsString(params);
+           if (_inflight$2[url]) return;
+           var controller = new AbortController();
+           _inflight$2[url] = controller;
+           d3_json(url, {
+             signal: controller.signal
+           }).then(function (result) {
+             delete _inflight$2[url];
 
-       function specialCases(e1, e2, p1, p2) {
-         // Same coordinates, but one is a left endpoint and the other is
-         // a right endpoint. The right endpoint is processed first
-         if (e1.left !== e2.left) return e1.left ? 1 : -1; // const p2 = e1.otherEvent.point, p3 = e2.otherEvent.point;
-         // const sa = (p1[0] - p3[0]) * (p2[1] - p3[1]) - (p2[0] - p3[0]) * (p1[1] - p3[1])
-         // Same coordinates, both events
-         // are left endpoints or right endpoints.
-         // not collinear
+             if (result && result.error) {
+               throw new Error(result.error);
+             }
 
-         if (signedArea(p1, e1.otherEvent.point, e2.otherEvent.point) !== 0) {
-           // the event associate to the bottom segment is processed first
-           return !e1.isBelow(e2.otherEvent.point) ? 1 : -1;
-         }
+             var extent = geoExtent(loc).padByMeters(200);
 
-         return !e1.isSubject && e2.isSubject ? 1 : -1;
-       }
-       /* eslint-enable no-unused-vars */
+             _nominatimCache.insert(Object.assign(extent.bbox(), {
+               data: result
+             }));
 
-       /**
-        * @param  {SweepEvent} se
-        * @param  {Array.<Number>} p
-        * @param  {Queue} queue
-        * @return {Queue}
-        */
+             if (callback) callback(null, result);
+           })["catch"](function (err) {
+             delete _inflight$2[url];
+             if (err.name === 'AbortError') return;
+             if (callback) callback(err.message);
+           });
+         },
+         search: function search(val, callback) {
+           var searchVal = encodeURIComponent(val);
+           var url = apibase$2 + 'search/' + searchVal + '?limit=10&format=json';
+           if (_inflight$2[url]) return;
+           var controller = new AbortController();
+           _inflight$2[url] = controller;
+           d3_json(url, {
+             signal: controller.signal
+           }).then(function (result) {
+             delete _inflight$2[url];
 
-       function divideSegment(se, p, queue) {
-         var r = new SweepEvent(p, false, se, se.isSubject);
-         var l = new SweepEvent(p, true, se.otherEvent, se.isSubject);
-         /* eslint-disable no-console */
+             if (result && result.error) {
+               throw new Error(result.error);
+             }
 
-         if (equals(se.point, se.otherEvent.point)) {
-           console.warn('what is that, a collapsed segment?', se);
+             if (callback) callback(null, result);
+           })["catch"](function (err) {
+             delete _inflight$2[url];
+             if (err.name === 'AbortError') return;
+             if (callback) callback(err.message);
+           });
          }
-         /* eslint-enable no-console */
-
-
-         r.contourId = l.contourId = se.contourId; // avoid a rounding error. The left event would be processed after the right event
-
-         if (compareEvents(l, se.otherEvent) > 0) {
-           se.otherEvent.left = true;
-           l.left = false;
-         } // avoid a rounding error. The left event would be processed after the right event
-         // if (compareEvents(se, r) > 0) {}
+       };
 
+       // for punction see https://stackoverflow.com/a/21224179
 
-         se.otherEvent.otherEvent = l;
-         se.otherEvent = r;
-         queue.push(l);
-         queue.push(r);
-         return queue;
+       function simplify$1(str) {
+         if (typeof str !== 'string') return '';
+         return diacritics.remove(str.replace(/&/g, 'and').replace(/İ/ig, 'i') // for BİM, İşbank - #5017
+         .replace(/[\s\-=_!"#%'*{},.\/:;?\(\)\[\]@\\$\^*+<>«»~`’\u00a1\u00a7\u00b6\u00b7\u00bf\u037e\u0387\u055a-\u055f\u0589\u05c0\u05c3\u05c6\u05f3\u05f4\u0609\u060a\u060c\u060d\u061b\u061e\u061f\u066a-\u066d\u06d4\u0700-\u070d\u07f7-\u07f9\u0830-\u083e\u085e\u0964\u0965\u0970\u0af0\u0df4\u0e4f\u0e5a\u0e5b\u0f04-\u0f12\u0f14\u0f85\u0fd0-\u0fd4\u0fd9\u0fda\u104a-\u104f\u10fb\u1360-\u1368\u166d\u166e\u16eb-\u16ed\u1735\u1736\u17d4-\u17d6\u17d8-\u17da\u1800-\u1805\u1807-\u180a\u1944\u1945\u1a1e\u1a1f\u1aa0-\u1aa6\u1aa8-\u1aad\u1b5a-\u1b60\u1bfc-\u1bff\u1c3b-\u1c3f\u1c7e\u1c7f\u1cc0-\u1cc7\u1cd3\u2000-\u206f\u2cf9-\u2cfc\u2cfe\u2cff\u2d70\u2e00-\u2e7f\u3001-\u3003\u303d\u30fb\ua4fe\ua4ff\ua60d-\ua60f\ua673\ua67e\ua6f2-\ua6f7\ua874-\ua877\ua8ce\ua8cf\ua8f8-\ua8fa\ua92e\ua92f\ua95f\ua9c1-\ua9cd\ua9de\ua9df\uaa5c-\uaa5f\uaade\uaadf\uaaf0\uaaf1\uabeb\ufe10-\ufe16\ufe19\ufe30\ufe45\ufe46\ufe49-\ufe4c\ufe50-\ufe52\ufe54-\ufe57\ufe5f-\ufe61\ufe68\ufe6a\ufe6b\ufeff\uff01-\uff03\uff05-\uff07\uff0a\uff0c\uff0e\uff0f\uff1a\uff1b\uff1f\uff20\uff3c\uff61\uff64\uff65]+/g, '').toLowerCase());
        }
 
-       //const EPS = 1e-9;
+       var matchGroups$1 = {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/beer","shop/beverages","shop/wine"],camping:["leisure/park","tourism/camp_site","tourism/caravan_site"],car_parts:["shop/car_parts","shop/car_repair","shop/tires","shop/tyres"],clinic:["amenity/clinic","amenity/doctors","healthcare/clinic","healthcare/dialysis"],confectionery:["shop/candy","shop/chocolate","shop/confectionery"],convenience:["shop/beauty","shop/chemist","shop/convenience","shop/cosmetics","shop/grocery","shop/newsagent"],coworking:["amenity/coworking_space","office/coworking","office/coworking_space"],dentist:["amenity/dentist","amenity/doctors","healthcare/dentist"],electronics:["office/telecommunication","shop/computer","shop/electronics","shop/hifi","shop/mobile","shop/mobile_phone","shop/telecommunication"],fabric:["shop/fabric","shop/haberdashery","shop/sewing"],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/pub","amenity/bar","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/bathroom_furnishing","shop/carpet","shop/diy","shop/doityourself","shop/doors","shop/electrical","shop/flooring","shop/hardware","shop/hardware_store","shop/power_tools","shop/tool_hire","shop/tools","shop/trade"],health_food:["shop/health","shop/health_food","shop/herbalist","shop/nutrition_supplements"],hobby:["shop/electronics","shop/hobby","shop/books","shop/games","shop/collector","shop/toys","shop/model","shop/video_games","shop/anime"],hospital:["amenity/doctors","amenity/hospital","healthcare/hospital"],houseware:["shop/houseware","shop/interior_decoration"],lifeboat_station:["amenity/lifeboat_station","emergency/lifeboat_station","emergency/marine_rescue"],lodging:["tourism/hotel","tourism/motel"],money_transfer:["amenity/money_transfer","shop/money_transfer"],office_supplies:["shop/office_supplies","shop/stationary","shop/stationery"],outdoor:["shop/outdoor","shop/sports"],pharmacy:["amenity/doctors","amenity/pharmacy","healthcare/pharmacy"],playground:["amenity/theme_park","leisure/amusement_arcade","leisure/playground"],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/discount","shop/convenience"],vending:["amenity/vending_machine","shop/vending_machine"],storage:["shop/storage_units","shop/storage_rental"],weight_loss:["amenity/doctors","amenity/weight_clinic","healthcare/counselling","leisure/fitness_centre","office/therapist","shop/beauty","shop/diet","shop/food","shop/health_food","shop/herbalist","shop/nutrition","shop/nutrition_supplements","shop/weight_loss"],wholesale:["shop/wholesale","shop/supermarket","shop/department_store"]};
+       var matchGroupsJSON = {
+       matchGroups: matchGroups$1
+       };
 
-       /**
-        * Finds the magnitude of the cross product of two vectors (if we pretend
-        * they're in three dimensions)
-        *
-        * @param {Object} a First vector
-        * @param {Object} b Second vector
-        * @private
-        * @returns {Number} The magnitude of the cross product
-        */
-       function crossProduct(a, b) {
-         return a[0] * b[1] - a[1] * b[0];
-       }
-       /**
-        * Finds the dot product of two vectors.
-        *
-        * @param {Object} a First vector
-        * @param {Object} b Second vector
-        * @private
-        * @returns {Number} The dot product
-        */
+       var genericWords = ["^(barn|bazaa?r|bench|bou?tique|building|casa|church)$","^(baseball|basketball|football|soccer|softball|tennis(halle)?)\\s?(field|court)?$","^(club|green|out|ware)\\s?house$","^(driveway|el árbol|fountain|golf|government|graveyard)$","^(hofladen|librairie|magazine?|maison)$","^(mobile home|skate)?\\s?park$","^(n\\s?\\/?\\s?a|name|no\\s?name|none|null|temporary|test|unknown)$","^(obuwie|pond|pool|sale|shops?|sklep|stores?)$","^\\?+$","^tattoo( studio)?$","^windmill$","^церковная( лавка)?$"];
+       var genericWordsJSON = {
+       genericWords: genericWords
+       };
 
+       var trees$1 = {brands:{emoji:"🍔",mainTag:"brand:wikidata",sourceTags:["brand","name"],nameTags:{primary:"^(name|name:\\w+)$",alternate:"^(brand|brand:\\w+|operator|operator:\\w+|\\w+_name|\\w+_name:\\w+)$"}},flags:{emoji:"🚩",mainTag:"flag:wikidata",nameTags:{primary:"^(flag:name|flag:name:\\w+)$",alternate:"^(country|country:\\w+|flag|flag:\\w+|subject|subject:\\w+)$"}},operators:{emoji:"💼",mainTag:"operator:wikidata",sourceTags:["operator"],nameTags:{primary:"^(name|name:\\w+|operator|operator:\\w+)$",alternate:"^(brand|brand:\\w+|\\w+_name|\\w+_name:\\w+)$"}},transit:{emoji:"🚇",mainTag:"network:wikidata",sourceTags:["network"],nameTags:{primary:"^network$",alternate:"^(operator|operator:\\w+|network:\\w+|\\w+_name|\\w+_name:\\w+)$"}}};
+       var treesJSON = {
+       trees: trees$1
+       };
 
-       function dotProduct(a, b) {
-         return a[0] * b[0] + a[1] * b[1];
-       }
-       /**
-        * Finds the intersection (if any) between two line segments a and b, given the
-        * line segments' end points a1, a2 and b1, b2.
-        *
-        * This algorithm is based on Schneider and Eberly.
-        * http://www.cimec.org.ar/~ncalvo/Schneider_Eberly.pdf
-        * Page 244.
-        *
-        * @param {Array.<Number>} a1 point of first line
-        * @param {Array.<Number>} a2 point of first line
-        * @param {Array.<Number>} b1 point of second line
-        * @param {Array.<Number>} b2 point of second line
-        * @param {Boolean=}       noEndpointTouch whether to skip single touchpoints
-        *                                         (meaning connected segments) as
-        *                                         intersections
-        * @returns {Array.<Array.<Number>>|Null} If the lines intersect, the point of
-        * intersection. If they overlap, the two end points of the overlapping segment.
-        * Otherwise, null.
-        */
+       var matchGroups = matchGroupsJSON.matchGroups;
+       var trees = treesJSON.trees;
+       var Matcher = /*#__PURE__*/function () {
+         //
+         // `constructor`
+         // initialize the genericWords regexes
+         function Matcher() {
+           var _this = this;
 
+           _classCallCheck$1(this, Matcher);
 
-       function intersection (a1, a2, b1, b2, noEndpointTouch) {
-         // The algorithm expects our lines in the form P + sd, where P is a point,
-         // s is on the interval [0, 1], and d is a vector.
-         // We are passed two points. P can be the first point of each pair. The
-         // vector, then, could be thought of as the distance (in x and y components)
-         // from the first point to the second point.
-         // So first, let's make our vectors:
-         var va = [a2[0] - a1[0], a2[1] - a1[1]];
-         var vb = [b2[0] - b1[0], b2[1] - b1[1]]; // We also define a function to convert back to regular point form:
+           // The `matchIndex` is a specialized structure that allows us to quickly answer
+           //   _"Given a [key/value tagpair, name, location], what canonical items (brands etc) can match it?"_
+           //
+           // The index contains all valid combinations of k/v tagpairs and names
+           // matchIndex:
+           // {
+           //   'k/v': {
+           //     'primary':         Map (String 'nsimple' -> Set (itemIDs…),   // matches for tags like `name`, `name:xx`, etc.
+           //     'alternate':       Map (String 'nsimple' -> Set (itemIDs…),   // matches for tags like `alt_name`, `brand`, etc.
+           //     'excludeNamed':    Map (String 'pattern' -> RegExp),
+           //     'excludeGeneric':  Map (String 'pattern' -> RegExp)
+           //   },
+           // }
+           //
+           // {
+           //   'amenity/bank': {
+           //     'primary': {
+           //       'firstbank':              Set ("firstbank-978cca", "firstbank-9794e6", "firstbank-f17495", …),
+           //       …
+           //     },
+           //     'alternate': {
+           //       '1stbank':                Set ("firstbank-f17495"),
+           //       …
+           //     }
+           //   },
+           //   'shop/supermarket': {
+           //     'primary': {
+           //       'coop':                   Set ("coop-76454b", "coop-ebf2d9", "coop-36e991", …),
+           //       'coopfood':               Set ("coopfood-a8278b", …),
+           //       …
+           //     },
+           //     'alternate': {
+           //       'coop':                   Set ("coopfood-a8278b", …),
+           //       'federatedcooperatives':  Set ("coop-76454b", …),
+           //       'thecooperative':         Set ("coopfood-a8278b", …),
+           //       …
+           //     }
+           //   }
+           // }
+           //
+           this.matchIndex = undefined; // The `genericWords` structure matches the contents of genericWords.json to instantiated RegExp objects
+           // Map (String 'pattern' -> RegExp),
+
+           this.genericWords = new Map();
+           (genericWordsJSON.genericWords || []).forEach(function (s) {
+             return _this.genericWords.set(s, new RegExp(s, 'i'));
+           }); // The `itemLocation` structure maps itemIDs to locationSetIDs:
+           // {
+           //   'firstbank-f17495':  '+[first_bank_western_us.geojson]',
+           //   'firstbank-978cca':  '+[first_bank_carolinas.geojson]',
+           //   'coop-76454b':       '+[Q16]',
+           //   'coopfood-a8278b':   '+[Q23666]',
+           //   …
+           // }
 
-         /* eslint-disable arrow-body-style */
+           this.itemLocation = undefined; // The `locationSets` structure maps locationSetIDs to *resolved* locationSets:
+           // {
+           //   '+[first_bank_western_us.geojson]':  GeoJSON {…},
+           //   '+[first_bank_carolinas.geojson]':   GeoJSON {…},
+           //   '+[Q16]':                            GeoJSON {…},
+           //   '+[Q23666]':                         GeoJSON {…},
+           //   …
+           // }
 
-         function toPoint(p, s, d) {
-           return [p[0] + s * d[0], p[1] + s * d[1]];
-         }
-         /* eslint-enable arrow-body-style */
-         // The rest is pretty much a straight port of the algorithm.
+           this.locationSets = undefined; // The `locationIndex` is an instance of which-polygon spatial index for the locationSets.
 
+           this.locationIndex = undefined; // Array of match conflict pairs (currently unused)
 
-         var e = [b1[0] - a1[0], b1[1] - a1[1]];
-         var kross = crossProduct(va, vb);
-         var sqrKross = kross * kross;
-         var sqrLenA = dotProduct(va, va); //const sqrLenB  = dotProduct(vb, vb);
-         // Check for line intersection. This works because of the properties of the
-         // cross product -- specifically, two vectors are parallel if and only if the
-         // cross product is the 0 vector. The full calculation involves relative error
-         // to account for possible very small line segments. See Schneider & Eberly
-         // for details.
+           this.warnings = [];
+         } //
+         // `buildMatchIndex()`
+         // Call this to prepare the matcher for use
+         //
+         // `data` needs to be an Object indexed on a 'tree/key/value' path.
+         // (e.g. cache filled by `fileTree.read` or data found in `dist/nsi.json`)
+         // {
+         //    'brands/amenity/bank': { properties: {}, items: [ {}, {}, … ] },
+         //    'brands/amenity/bar':  { properties: {}, items: [ {}, {}, … ] },
+         //    …
+         // }
+         //
 
-         if (sqrKross > 0
-         /* EPS * sqrLenB * sqLenA */
-         ) {
-             // If they're not parallel, then (because these are line segments) they
-             // still might not actually intersect. This code checks that the
-             // intersection point of the lines is actually on both line segments.
-             var s = crossProduct(e, vb) / kross;
 
-             if (s < 0 || s > 1) {
-               // not on line segment a
-               return null;
-             }
+         _createClass$1(Matcher, [{
+           key: "buildMatchIndex",
+           value: function buildMatchIndex(data) {
+             var that = this;
+             if (that.matchIndex) return; // it was built already
+
+             that.matchIndex = new Map();
+             Object.keys(data).forEach(function (tkv) {
+               var category = data[tkv];
+               var parts = tkv.split('/', 3); // tkv = "tree/key/value"
+
+               var t = parts[0];
+               var k = parts[1];
+               var v = parts[2];
+               var thiskv = "".concat(k, "/").concat(v);
+               var tree = trees[t];
+               var branch = that.matchIndex.get(thiskv);
+
+               if (!branch) {
+                 branch = {
+                   primary: new Map(),
+                   alternate: new Map(),
+                   excludeGeneric: new Map(),
+                   excludeNamed: new Map()
+                 };
+                 that.matchIndex.set(thiskv, branch);
+               } // ADD EXCLUSIONS
 
-             var t = crossProduct(e, va) / kross;
 
-             if (t < 0 || t > 1) {
-               // not on line segment b
-               return null;
-             }
+               var properties = category.properties || {};
+               var exclude = properties.exclude || {};
+               (exclude.generic || []).forEach(function (s) {
+                 return branch.excludeGeneric.set(s, new RegExp(s, 'i'));
+               });
+               (exclude.named || []).forEach(function (s) {
+                 return branch.excludeNamed.set(s, new RegExp(s, 'i'));
+               });
+               var excludeRegexes = [].concat(_toConsumableArray(branch.excludeGeneric.values()), _toConsumableArray(branch.excludeNamed.values())); // ADD ITEMS
 
-             if (s === 0 || s === 1) {
-               // on an endpoint of line segment a
-               return noEndpointTouch ? null : [toPoint(a1, s, va)];
-             }
+               var items = category.items;
+               if (!Array.isArray(items) || !items.length) return; // Primary name patterns, match tags to take first
+               //  e.g. `name`, `name:ru`
 
-             if (t === 0 || t === 1) {
-               // on an endpoint of line segment b
-               return noEndpointTouch ? null : [toPoint(b1, t, vb)];
-             }
+               var primaryName = new RegExp(tree.nameTags.primary, 'i'); // Alternate name patterns, match tags to consider after primary
+               //  e.g. `alt_name`, `short_name`, `brand`, `brand:ru`, etc..
 
-             return [toPoint(a1, s, va)];
-           } // If we've reached this point, then the lines are either parallel or the
-         // same, but the segments could overlap partially or fully, or not at all.
-         // So we need to find the overlap, if any. To do that, we can use e, which is
-         // the (vector) difference between the two initial points. If this is parallel
-         // with the line itself, then the two lines are the same line, and there will
-         // be overlap.
-         //const sqrLenE = dotProduct(e, e);
+               var alternateName = new RegExp(tree.nameTags.alternate, 'i'); // There are a few exceptions to the name matching regexes.
+               // Usually a tag suffix contains a language code like `name:en`, `name:ru`
+               // but we want to exclude things like `operator:type`, `name:etymology`, etc..
 
+               var notName = /:(colou?r|type|forward|backward|left|right|etymology|pronunciation|wikipedia)$/i; // For certain categories we do not want to match generic KV pairs like `building/yes` or `amenity/yes`
 
-         kross = crossProduct(e, va);
-         sqrKross = kross * kross;
+               var skipGenericKV = skipGenericKVMatches(t, k, v); // We will collect the generic KV pairs anyway (for the purpose of filtering them out of matchTags)
 
-         if (sqrKross > 0
-         /* EPS * sqLenB * sqLenE */
-         ) {
-             // Lines are just parallel, not the same. No overlap.
-             return null;
-           }
+               var genericKV = new Set(["".concat(k, "/yes"), "building/yes"]); // Collect alternate tagpairs for this kv category from matchGroups.
+               // We might also pick up a few more generic KVs (like `shop/yes`)
 
-         var sa = dotProduct(va, e) / sqrLenA;
-         var sb = sa + dotProduct(va, vb) / sqrLenA;
-         var smin = Math.min(sa, sb);
-         var smax = Math.max(sa, sb); // this is, essentially, the FindIntersection acting on floats from
-         // Schneider & Eberly, just inlined into this function.
+               var matchGroupKV = new Set();
+               Object.values(matchGroups).forEach(function (matchGroup) {
+                 var inGroup = matchGroup.some(function (otherkv) {
+                   return otherkv === thiskv;
+                 });
+                 if (!inGroup) return;
+                 matchGroup.forEach(function (otherkv) {
+                   if (otherkv === thiskv) return; // skip self
 
-         if (smin <= 1 && smax >= 0) {
-           // overlap on an end point
-           if (smin === 1) {
-             return noEndpointTouch ? null : [toPoint(a1, smin > 0 ? smin : 0, va)];
-           }
+                   matchGroupKV.add(otherkv);
+                   var otherk = otherkv.split('/', 2)[0]; // we might pick up a `shop/yes`
 
-           if (smax === 0) {
-             return noEndpointTouch ? null : [toPoint(a1, smax < 1 ? smax : 1, va)];
-           }
+                   genericKV.add("".concat(otherk, "/yes"));
+                 });
+               }); // For each item, insert all [key, value, name] combinations into the match index
 
-           if (noEndpointTouch && smin === 0 && smax === 1) return null; // There's overlap on a segment -- two points of intersection. Return both.
+               items.forEach(function (item) {
+                 if (!item.id) return; // Automatically remove redundant `matchTags` - #3417
+                 // (i.e. This kv is already covered by matchGroups, so it doesn't need to be in `item.matchTags`)
 
-           return [toPoint(a1, smin > 0 ? smin : 0, va), toPoint(a1, smax < 1 ? smax : 1, va)];
-         }
+                 if (Array.isArray(item.matchTags) && item.matchTags.length) {
+                   item.matchTags = item.matchTags.filter(function (matchTag) {
+                     return !matchGroupKV.has(matchTag) && !genericKV.has(matchTag);
+                   });
+                   if (!item.matchTags.length) delete item.matchTags;
+                 } // key/value tagpairs to insert into the match index..
 
-         return null;
-       }
 
-       /**
-        * @param  {SweepEvent} se1
-        * @param  {SweepEvent} se2
-        * @param  {Queue}      queue
-        * @return {Number}
-        */
+                 var kvTags = ["".concat(thiskv)].concat(item.matchTags || []);
 
-       function possibleIntersection(se1, se2, queue) {
-         // that disallows self-intersecting polygons,
-         // did cost us half a day, so I'll leave it
-         // out of respect
-         // if (se1.isSubject === se2.isSubject) return;
-         var inter = intersection(se1.point, se1.otherEvent.point, se2.point, se2.otherEvent.point);
-         var nintersections = inter ? inter.length : 0;
-         if (nintersections === 0) return 0; // no intersection
-         // the line segments intersect at an endpoint of both line segments
-
-         if (nintersections === 1 && (equals(se1.point, se2.point) || equals(se1.otherEvent.point, se2.otherEvent.point))) {
-           return 0;
-         }
+                 if (!skipGenericKV) {
+                   kvTags = kvTags.concat(Array.from(genericKV)); // #3454 - match some generic tags
+                 } // Index all the namelike tag values
 
-         if (nintersections === 2 && se1.isSubject === se2.isSubject) {
-           // if(se1.contourId === se2.contourId){
-           // console.warn('Edges of the same polygon overlap',
-           //   se1.point, se1.otherEvent.point, se2.point, se2.otherEvent.point);
-           // }
-           //throw new Error('Edges of the same polygon overlap');
-           return 0;
-         } // The line segments associated to se1 and se2 intersect
 
+                 Object.keys(item.tags).forEach(function (osmkey) {
+                   if (notName.test(osmkey)) return; // osmkey is not a namelike tag, skip
 
-         if (nintersections === 1) {
-           // if the intersection point is not an endpoint of se1
-           if (!equals(se1.point, inter[0]) && !equals(se1.otherEvent.point, inter[0])) {
-             divideSegment(se1, inter[0], queue);
-           } // if the intersection point is not an endpoint of se2
+                   var osmvalue = item.tags[osmkey];
+                   if (!osmvalue || excludeRegexes.some(function (regex) {
+                     return regex.test(osmvalue);
+                   })) return; // osmvalue missing or excluded
 
+                   if (primaryName.test(osmkey)) {
+                     kvTags.forEach(function (kv) {
+                       return insertName('primary', kv, simplify$1(osmvalue), item.id);
+                     });
+                   } else if (alternateName.test(osmkey)) {
+                     kvTags.forEach(function (kv) {
+                       return insertName('alternate', kv, simplify$1(osmvalue), item.id);
+                     });
+                   }
+                 }); // Index `matchNames` after indexing all other names..
+
+                 var keepMatchNames = new Set();
+                 (item.matchNames || []).forEach(function (matchName) {
+                   // If this matchname isn't already indexed, add it to the alternate index
+                   var nsimple = simplify$1(matchName);
+                   kvTags.forEach(function (kv) {
+                     var branch = that.matchIndex.get(kv);
+                     var primaryLeaf = branch && branch.primary.get(nsimple);
+                     var alternateLeaf = branch && branch.alternate.get(nsimple);
+                     var inPrimary = primaryLeaf && primaryLeaf.has(item.id);
+                     var inAlternate = alternateLeaf && alternateLeaf.has(item.id);
+
+                     if (!inPrimary && !inAlternate) {
+                       insertName('alternate', kv, nsimple, item.id);
+                       keepMatchNames.add(matchName);
+                     }
+                   });
+                 }); // Automatically remove redundant `matchNames` - #3417
+                 // (i.e. This name got indexed some other way, so it doesn't need to be in `item.matchNames`)
 
-           if (!equals(se2.point, inter[0]) && !equals(se2.otherEvent.point, inter[0])) {
-             divideSegment(se2, inter[0], queue);
-           }
+                 if (keepMatchNames.size) {
+                   item.matchNames = Array.from(keepMatchNames);
+                 } else {
+                   delete item.matchNames;
+                 }
+               }); // each item
+             }); // each tkv
+             // Insert this item into the matchIndex
+
+             function insertName(which, kv, nsimple, itemID) {
+               if (!nsimple) return;
+               var branch = that.matchIndex.get(kv);
+
+               if (!branch) {
+                 branch = {
+                   primary: new Map(),
+                   alternate: new Map(),
+                   excludeGeneric: new Map(),
+                   excludeNamed: new Map()
+                 };
+                 that.matchIndex.set(kv, branch);
+               }
 
-           return 1;
-         } // The line segments associated to se1 and se2 overlap
+               var leaf = branch[which].get(nsimple);
 
+               if (!leaf) {
+                 leaf = new Set();
+                 branch[which].set(nsimple, leaf);
+               }
 
-         var events = [];
-         var leftCoincide = false;
-         var rightCoincide = false;
+               leaf.add(itemID); // insert
+             } // For certain categories we do not want to match generic KV pairs like `building/yes` or `amenity/yes`
 
-         if (equals(se1.point, se2.point)) {
-           leftCoincide = true; // linked
-         } else if (compareEvents(se1, se2) === 1) {
-           events.push(se2, se1);
-         } else {
-           events.push(se1, se2);
-         }
 
-         if (equals(se1.otherEvent.point, se2.otherEvent.point)) {
-           rightCoincide = true;
-         } else if (compareEvents(se1.otherEvent, se2.otherEvent) === 1) {
-           events.push(se2.otherEvent, se1.otherEvent);
-         } else {
-           events.push(se1.otherEvent, se2.otherEvent);
-         }
+             function skipGenericKVMatches(t, k, v) {
+               return t === 'flags' || t === 'transit' || k === 'landuse' || v === 'atm' || v === 'bicycle_parking' || v === 'car_sharing' || v === 'caravan_site' || v === 'charging_station' || v === 'dog_park' || v === 'parking' || v === 'phone' || v === 'playground' || v === 'post_box' || v === 'public_bookcase' || v === 'recycling' || v === 'vending_machine';
+             }
+           } //
+           // `buildLocationIndex()`
+           // Call this to prepare a which-polygon location index.
+           // This *resolves* all the locationSets into GeoJSON, which takes some time.
+           // You can skip this step if you don't care about matching within a location.
+           //
+           // `data` needs to be an Object indexed on a 'tree/key/value' path.
+           // (e.g. cache filled by `fileTree.read` or data found in `dist/nsi.json`)
+           // {
+           //    'brands/amenity/bank': { properties: {}, items: [ {}, {}, … ] },
+           //    'brands/amenity/bar':  { properties: {}, items: [ {}, {}, … ] },
+           //    …
+           // }
+           //
 
-         if (leftCoincide && rightCoincide || leftCoincide) {
-           // both line segments are equal or share the left endpoint
-           se2.type = NON_CONTRIBUTING;
-           se1.type = se2.inOut === se1.inOut ? SAME_TRANSITION : DIFFERENT_TRANSITION;
+         }, {
+           key: "buildLocationIndex",
+           value: function buildLocationIndex(data, loco) {
+             var that = this;
+             if (that.locationIndex) return; // it was built already
 
-           if (leftCoincide && !rightCoincide) {
-             // honestly no idea, but changing events selection from [2, 1]
-             // to [0, 1] fixes the overlapping self-intersecting polygons issue
-             divideSegment(events[1].otherEvent, events[0].point, queue);
-           }
+             that.itemLocation = new Map();
+             that.locationSets = new Map();
+             Object.keys(data).forEach(function (tkv) {
+               var items = data[tkv].items;
+               if (!Array.isArray(items) || !items.length) return;
+               items.forEach(function (item) {
+                 if (that.itemLocation.has(item.id)) return; // we've seen item id already - shouldn't be possible?
 
-           return 2;
-         } // the line segments share the right endpoint
+                 var resolved;
 
+                 try {
+                   resolved = loco.resolveLocationSet(item.locationSet); // resolve a feature for this locationSet
+                 } catch (err) {
+                   console.warn("buildLocationIndex: ".concat(err.message)); // couldn't resolve
+                 }
 
-         if (rightCoincide) {
-           divideSegment(events[0], events[1].point, queue);
-           return 3;
-         } // no line segment includes totally the other one
+                 if (!resolved || !resolved.id) return;
+                 that.itemLocation.set(item.id, resolved.id); // link it to the item
 
+                 if (that.locationSets.has(resolved.id)) return; // we've seen this locationSet feature before..
+                 // First time seeing this locationSet feature, make a copy and add to locationSet cache..
 
-         if (events[0] !== events[3].otherEvent) {
-           divideSegment(events[0], events[1].point, queue);
-           divideSegment(events[1], events[2].point, queue);
-           return 3;
-         } // one line segment includes the other one
+                 var feature = _cloneDeep(resolved.feature);
 
+                 feature.id = resolved.id; // Important: always use the locationSet `id` (`+[Q30]`), not the feature `id` (`Q30`)
 
-         divideSegment(events[0], events[1].point, queue);
-         divideSegment(events[3].otherEvent, events[2].point, queue);
-         return 3;
-       }
+                 feature.properties.id = resolved.id;
 
-       /**
-        * @param  {SweepEvent} le1
-        * @param  {SweepEvent} le2
-        * @return {Number}
-        */
+                 if (!feature.geometry.coordinates.length || !feature.properties.area) {
+                   console.warn("buildLocationIndex: locationSet ".concat(resolved.id, " for ").concat(item.id, " resolves to an empty feature:"));
+                   console.warn(JSON.stringify(feature));
+                   return;
+                 }
 
-       function compareSegments(le1, le2) {
-         if (le1 === le2) return 0; // Segments are not collinear
+                 that.locationSets.set(resolved.id, feature);
+               });
+             });
+             that.locationIndex = whichPolygon_1({
+               type: 'FeatureCollection',
+               features: _toConsumableArray(that.locationSets.values())
+             });
 
-         if (signedArea(le1.point, le1.otherEvent.point, le2.point) !== 0 || signedArea(le1.point, le1.otherEvent.point, le2.otherEvent.point) !== 0) {
-           // If they share their left endpoint use the right endpoint to sort
-           if (equals(le1.point, le2.point)) return le1.isBelow(le2.otherEvent.point) ? -1 : 1; // Different left endpoint: use the left endpoint to sort
+             function _cloneDeep(obj) {
+               return JSON.parse(JSON.stringify(obj));
+             }
+           } //
+           // `match()`
+           // Pass parts and return an Array of matches.
+           // `k` - key
+           // `v` - value
+           // `n` - namelike
+           // `loc` - optional - [lon,lat] location to search
+           //
+           // 1. If the [k,v,n] tuple matches a canonical item…
+           // Return an Array of match results.
+           // Each result will include the area in km² that the item is valid.
+           //
+           // Order of results:
+           // Primary ordering will be on the "match" column:
+           //   "primary" - where the query matches the `name` tag, followed by
+           //   "alternate" - where the query matches an alternate name tag (e.g. short_name, brand, operator, etc)
+           // Secondary ordering will be on the "area" column:
+           //   "area descending" if no location was provided, (worldwide before local)
+           //   "area ascending" if location was provided (local before worldwide)
+           //
+           // [
+           //   { match: 'primary',   itemID: String,  area: Number,  kv: String,  nsimple: String },
+           //   { match: 'primary',   itemID: String,  area: Number,  kv: String,  nsimple: String },
+           //   { match: 'alternate', itemID: String,  area: Number,  kv: String,  nsimple: String },
+           //   { match: 'alternate', itemID: String,  area: Number,  kv: String,  nsimple: String },
+           //   …
+           // ]
+           //
+           // -or-
+           //
+           // 2. If the [k,v,n] tuple matches an exclude pattern…
+           // Return an Array with a single exclude result, either
+           //
+           // [ { match: 'excludeGeneric', pattern: String,  kv: String } ]  // "generic" e.g. "Food Court"
+           //   or
+           // [ { match: 'excludeNamed', pattern: String,  kv: String } ]    // "named", e.g. "Kebabai"
+           //
+           // About results
+           //   "generic" - a generic word that is probably not really a name.
+           //     For these, iD should warn the user "Hey don't put 'food court' in the name tag".
+           //   "named" - a real name like "Kebabai" that is just common, but not a brand.
+           //     For these, iD should just let it be. We don't include these in NSI, but we don't want to nag users about it either.
+           //
+           // -or-
+           //
+           // 3. If the [k,v,n] tuple matches nothing of any kind, return `null`
+           //
+           //
 
-           if (le1.point[0] === le2.point[0]) return le1.point[1] < le2.point[1] ? -1 : 1; // has the line segment associated to e1 been inserted
-           // into S after the line segment associated to e2 ?
+         }, {
+           key: "match",
+           value: function match(k, v, n, loc) {
+             var that = this;
 
-           if (compareEvents(le1, le2) === 1) return le2.isAbove(le1.point) ? -1 : 1; // The line segment associated to e2 has been inserted
-           // into S after the line segment associated to e1
+             if (!that.matchIndex) {
+               throw new Error('match:  matchIndex not built.');
+             } // If we were supplied a location, and a that.locationIndex has been set up,
+             // get the locationSets that are valid there so we can filter results.
 
-           return le1.isBelow(le2.point) ? -1 : 1;
-         }
 
-         if (le1.isSubject === le2.isSubject) {
-           // same polygon
-           var p1 = le1.point,
-               p2 = le2.point;
+             var matchLocations;
 
-           if (p1[0] === p2[0] && p1[1] === p2[1]
-           /*equals(le1.point, le2.point)*/
-           ) {
-               p1 = le1.otherEvent.point;
-               p2 = le2.otherEvent.point;
-               if (p1[0] === p2[0] && p1[1] === p2[1]) return 0;else return le1.contourId > le2.contourId ? 1 : -1;
+             if (Array.isArray(loc) && that.locationIndex) {
+               // which-polygon query returns an array of GeoJSON properties, pass true to return all results
+               matchLocations = that.locationIndex([loc[0], loc[1], loc[0], loc[1]], true);
              }
-         } else {
-           // Segments are collinear, but belong to separate polygons
-           return le1.isSubject ? -1 : 1;
-         }
 
-         return compareEvents(le1, le2) === 1 ? 1 : -1;
-       }
+             var nsimple = simplify$1(n);
+             var seen = new Set();
+             var results = [];
+             gatherResults('primary');
+             gatherResults('alternate');
+             if (results.length) return results;
+             gatherResults('exclude');
+             return results.length ? results : null;
+
+             function gatherResults(which) {
+               // First try an exact match on k/v
+               var kv = "".concat(k, "/").concat(v);
+               var didMatch = tryMatch(which, kv);
+               if (didMatch) return; // If that didn't work, look in match groups for other pairs considered equivalent to k/v..
+
+               for (var mg in matchGroups) {
+                 var matchGroup = matchGroups[mg];
+                 var inGroup = matchGroup.some(function (otherkv) {
+                   return otherkv === kv;
+                 });
+                 if (!inGroup) continue;
 
-       function subdivide(eventQueue, subject, clipping, sbbox, cbbox, operation) {
-         var sweepLine = new SplayTree(compareSegments);
-         var sortedEvents = [];
-         var rightbound = Math.min(sbbox[2], cbbox[2]);
-         var prev, next, begin;
+                 for (var i = 0; i < matchGroup.length; i++) {
+                   var otherkv = matchGroup[i];
+                   if (otherkv === kv) continue; // skip self
 
-         while (eventQueue.length !== 0) {
-           var event = eventQueue.pop();
-           sortedEvents.push(event); // optimization by bboxes for intersection and difference goes here
+                   didMatch = tryMatch(which, otherkv);
+                   if (didMatch) return;
+                 }
+               } // If finished 'exclude' pass and still haven't matched anything, try the global `genericWords.json` patterns
 
-           if (operation === INTERSECTION && event.point[0] > rightbound || operation === DIFFERENCE && event.point[0] > sbbox[2]) {
-             break;
-           }
 
-           if (event.left) {
-             next = prev = sweepLine.insert(event);
-             begin = sweepLine.minNode();
-             if (prev !== begin) prev = sweepLine.prev(prev);else prev = null;
-             next = sweepLine.next(next);
-             var prevEvent = prev ? prev.key : null;
-             var prevprevEvent = void 0;
-             computeFields(event, prevEvent, operation);
+               if (which === 'exclude') {
+                 var regex = _toConsumableArray(that.genericWords.values()).find(function (regex) {
+                   return regex.test(n);
+                 });
 
-             if (next) {
-               if (possibleIntersection(event, next.key, eventQueue) === 2) {
-                 computeFields(event, prevEvent, operation);
-                 computeFields(event, next.key, operation);
-               }
-             }
+                 if (regex) {
+                   results.push({
+                     match: 'excludeGeneric',
+                     pattern: String(regex)
+                   }); // note no `branch`, no `kv`
 
-             if (prev) {
-               if (possibleIntersection(prev.key, event, eventQueue) === 2) {
-                 var prevprev = prev;
-                 if (prevprev !== begin) prevprev = sweepLine.prev(prevprev);else prevprev = null;
-                 prevprevEvent = prevprev ? prevprev.key : null;
-                 computeFields(prevEvent, prevprevEvent, operation);
-                 computeFields(event, prevEvent, operation);
+                   return;
+                 }
                }
              }
-           } else {
-             event = event.otherEvent;
-             next = prev = sweepLine.find(event);
 
-             if (prev && next) {
-               if (prev !== begin) prev = sweepLine.prev(prev);else prev = null;
-               next = sweepLine.next(next);
-               sweepLine.remove(event);
+             function tryMatch(which, kv) {
+               var branch = that.matchIndex.get(kv);
+               if (!branch) return;
 
-               if (next && prev) {
-                 possibleIntersection(prev.key, next.key, eventQueue);
-               }
-             }
-           }
-         }
+               if (which === 'exclude') {
+                 // Test name `n` against named and generic exclude patterns
+                 var regex = _toConsumableArray(branch.excludeNamed.values()).find(function (regex) {
+                   return regex.test(n);
+                 });
 
-         return sortedEvents;
-       }
+                 if (regex) {
+                   results.push({
+                     match: 'excludeNamed',
+                     pattern: String(regex),
+                     kv: kv
+                   });
+                   return;
+                 }
 
-       var Contour = /*#__PURE__*/function () {
-         /**
-          * Contour
-          *
-          * @class {Contour}
-          */
-         function Contour() {
-           _classCallCheck(this, Contour);
+                 regex = _toConsumableArray(branch.excludeGeneric.values()).find(function (regex) {
+                   return regex.test(n);
+                 });
 
-           this.points = [];
-           this.holeIds = [];
-           this.holeOf = null;
-           this.depth = null;
-         }
+                 if (regex) {
+                   results.push({
+                     match: 'excludeGeneric',
+                     pattern: String(regex),
+                     kv: kv
+                   });
+                   return;
+                 }
 
-         _createClass(Contour, [{
-           key: "isExterior",
-           value: function isExterior() {
-             return this.holeOf == null;
-           }
-         }]);
+                 return;
+               }
 
-         return Contour;
-       }();
+               var leaf = branch[which].get(nsimple);
+               if (!leaf || !leaf.size) return; // If we get here, we matched something..
+               // Prepare the results, calculate areas (if location index was set up)
 
-       /**
-        * @param  {Array.<SweepEvent>} sortedEvents
-        * @return {Array.<SweepEvent>}
-        */
+               var hits = Array.from(leaf).map(function (itemID) {
+                 var area = Infinity;
 
-       function orderEvents(sortedEvents) {
-         var event, i, len, tmp;
-         var resultEvents = [];
+                 if (that.itemLocation && that.locationSets) {
+                   var location = that.locationSets.get(that.itemLocation.get(itemID));
+                   area = location && location.properties.area || Infinity;
+                 }
 
-         for (i = 0, len = sortedEvents.length; i < len; i++) {
-           event = sortedEvents[i];
+                 return {
+                   match: which,
+                   itemID: itemID,
+                   area: area,
+                   kv: kv,
+                   nsimple: nsimple
+                 };
+               });
+               var sortFn = byAreaDescending; // Filter the match to include only results valid in the requested `loc`..
 
-           if (event.left && event.inResult || !event.left && event.otherEvent.inResult) {
-             resultEvents.push(event);
-           }
-         } // Due to overlapping edges the resultEvents array can be not wholly sorted
+               if (matchLocations) {
+                 hits = hits.filter(isValidLocation);
+                 sortFn = byAreaAscending;
+               }
 
+               if (!hits.length) return; // push results
 
-         var sorted = false;
+               hits.sort(sortFn).forEach(function (hit) {
+                 if (seen.has(hit.itemID)) return;
+                 seen.add(hit.itemID);
+                 results.push(hit);
+               });
+               return true;
 
-         while (!sorted) {
-           sorted = true;
+               function isValidLocation(hit) {
+                 if (!that.itemLocation) return true;
+                 return matchLocations.find(function (props) {
+                   return props.id === that.itemLocation.get(hit.itemID);
+                 });
+               } // Sort smaller (more local) locations first.
 
-           for (i = 0, len = resultEvents.length; i < len; i++) {
-             if (i + 1 < len && compareEvents(resultEvents[i], resultEvents[i + 1]) === 1) {
-               tmp = resultEvents[i];
-               resultEvents[i] = resultEvents[i + 1];
-               resultEvents[i + 1] = tmp;
-               sorted = false;
-             }
-           }
-         }
 
-         for (i = 0, len = resultEvents.length; i < len; i++) {
-           event = resultEvents[i];
-           event.otherPos = i;
-         } // imagine, the right event is found in the beginning of the queue,
-         // when his left counterpart is not marked yet
+               function byAreaAscending(hitA, hitB) {
+                 return hitA.area - hitB.area;
+               } // Sort larger (more worldwide) locations first.
 
 
-         for (i = 0, len = resultEvents.length; i < len; i++) {
-           event = resultEvents[i];
+               function byAreaDescending(hitA, hitB) {
+                 return hitB.area - hitA.area;
+               }
+             }
+           } //
+           // `getWarnings()`
+           // Return any warnings discovered when buiding the index.
+           // (currently this does nothing)
+           //
 
-           if (!event.left) {
-             tmp = event.otherPos;
-             event.otherPos = event.otherEvent.otherPos;
-             event.otherEvent.otherPos = tmp;
+         }, {
+           key: "getWarnings",
+           value: function getWarnings() {
+             return this.warnings;
            }
-         }
+         }]);
+
+         return Matcher;
+       }();
 
-         return resultEvents;
-       }
        /**
-        * @param  {Number} pos
-        * @param  {Array.<SweepEvent>} resultEvents
-        * @param  {Object>}    processed
-        * @return {Number}
+        * Checks if `value` is the
+        * [language type](http://www.ecma-international.org/ecma-262/7.0/#sec-ecmascript-language-types)
+        * of `Object`. (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`)
+        *
+        * @static
+        * @memberOf _
+        * @since 0.1.0
+        * @category Lang
+        * @param {*} value The value to check.
+        * @returns {boolean} Returns `true` if `value` is an object, else `false`.
+        * @example
+        *
+        * _.isObject({});
+        * // => true
+        *
+        * _.isObject([1, 2, 3]);
+        * // => true
+        *
+        * _.isObject(_.noop);
+        * // => true
+        *
+        * _.isObject(null);
+        * // => false
         */
+       function isObject$2(value) {
+         var type = _typeof(value);
 
-
-       function nextPos(pos, resultEvents, processed, origPos) {
-         var newPos = pos + 1,
-             p = resultEvents[pos].point,
-             p1;
-         var length = resultEvents.length;
-         if (newPos < length) p1 = resultEvents[newPos].point;
-
-         while (newPos < length && p1[0] === p[0] && p1[1] === p[1]) {
-           if (!processed[newPos]) {
-             return newPos;
-           } else {
-             newPos++;
-           }
-
-           p1 = resultEvents[newPos].point;
-         }
-
-         newPos = pos - 1;
-
-         while (processed[newPos] && newPos > origPos) {
-           newPos--;
-         }
-
-         return newPos;
+         return value != null && (type == 'object' || type == 'function');
        }
 
-       function initializeContourFromContext(event, contours, contourId) {
-         var contour = new Contour();
-
-         if (event.prevInResult != null) {
-           var prevInResult = event.prevInResult; // Note that it is valid to query the "previous in result" for its output contour id,
-           // because we must have already processed it (i.e., assigned an output contour id)
-           // in an earlier iteration, otherwise it wouldn't be possible that it is "previous in
-           // result".
+       /** Detect free variable `global` from Node.js. */
+       var freeGlobal = (typeof global === "undefined" ? "undefined" : _typeof(global)) == 'object' && global && global.Object === Object && global;
 
-           var lowerContourId = prevInResult.outputContourId;
-           var lowerResultTransition = prevInResult.resultTransition;
+       /** Detect free variable `self`. */
 
-           if (lowerResultTransition > 0) {
-             // We are inside. Now we have to check if the thing below us is another hole or
-             // an exterior contour.
-             var lowerContour = contours[lowerContourId];
+       var freeSelf = (typeof self === "undefined" ? "undefined" : _typeof(self)) == 'object' && self && self.Object === Object && self;
+       /** Used as a reference to the global object. */
 
-             if (lowerContour.holeOf != null) {
-               // The lower contour is a hole => Connect the new contour as a hole to its parent,
-               // and use same depth.
-               var parentContourId = lowerContour.holeOf;
-               contours[parentContourId].holeIds.push(contourId);
-               contour.holeOf = parentContourId;
-               contour.depth = contours[lowerContourId].depth;
-             } else {
-               // The lower contour is an exterior contour => Connect the new contour as a hole,
-               // and increment depth.
-               contours[lowerContourId].holeIds.push(contourId);
-               contour.holeOf = lowerContourId;
-               contour.depth = contours[lowerContourId].depth + 1;
-             }
-           } else {
-             // We are outside => this contour is an exterior contour of same depth.
-             contour.holeOf = null;
-             contour.depth = contours[lowerContourId].depth;
-           }
-         } else {
-           // There is no lower/previous contour => this contour is an exterior contour of depth 0.
-           contour.holeOf = null;
-           contour.depth = 0;
-         }
+       var root = freeGlobal || freeSelf || Function('return this')();
 
-         return contour;
-       }
        /**
-        * @param  {Array.<SweepEvent>} sortedEvents
-        * @return {Array.<*>} polygons
+        * Gets the timestamp of the number of milliseconds that have elapsed since
+        * the Unix epoch (1 January 1970 00:00:00 UTC).
+        *
+        * @static
+        * @memberOf _
+        * @since 2.4.0
+        * @category Date
+        * @returns {number} Returns the timestamp.
+        * @example
+        *
+        * _.defer(function(stamp) {
+        *   console.log(_.now() - stamp);
+        * }, _.now());
+        * // => Logs the number of milliseconds it took for the deferred invocation.
         */
 
+       var now = function now() {
+         return root.Date.now();
+       };
 
-       function connectEdges(sortedEvents) {
-         var i, len;
-         var resultEvents = orderEvents(sortedEvents); // "false"-filled array
-
-         var processed = {};
-         var contours = [];
-
-         var _loop = function _loop() {
-           if (processed[i]) {
-             return "continue";
-           }
-
-           var contourId = contours.length;
-           var contour = initializeContourFromContext(resultEvents[i], contours, contourId); // Helper function that combines marking an event as processed with assigning its output contour ID
-
-           var markAsProcessed = function markAsProcessed(pos) {
-             processed[pos] = true;
-             resultEvents[pos].outputContourId = contourId;
-           };
-
-           var pos = i;
-           var origPos = i;
-           var initial = resultEvents[i].point;
-           contour.points.push(initial);
-           /* eslint no-constant-condition: "off" */
-
-           while (true) {
-             markAsProcessed(pos);
-             pos = resultEvents[pos].otherPos;
-             markAsProcessed(pos);
-             contour.points.push(resultEvents[pos].point);
-             pos = nextPos(pos, resultEvents, processed, origPos);
-
-             if (pos == origPos) {
-               break;
-             }
-           }
-
-           contours.push(contour);
-         };
+       /** Used to match a single whitespace character. */
+       var reWhitespace = /\s/;
+       /**
+        * Used by `_.trim` and `_.trimEnd` to get the index of the last non-whitespace
+        * character of `string`.
+        *
+        * @private
+        * @param {string} string The string to inspect.
+        * @returns {number} Returns the index of the last non-whitespace character.
+        */
 
-         for (i = 0, len = resultEvents.length; i < len; i++) {
-           var _ret = _loop();
+       function trimmedEndIndex(string) {
+         var index = string.length;
 
-           if (_ret === "continue") continue;
-         }
+         while (index-- && reWhitespace.test(string.charAt(index))) {}
 
-         return contours;
+         return index;
        }
 
-       var tinyqueue = TinyQueue;
-       var _default = TinyQueue;
-
-       function TinyQueue(data, compare) {
-         if (!(this instanceof TinyQueue)) return new TinyQueue(data, compare);
-         this.data = data || [];
-         this.length = this.data.length;
-         this.compare = compare || defaultCompare$1;
+       /** Used to match leading whitespace. */
 
-         if (this.length > 0) {
-           for (var i = (this.length >> 1) - 1; i >= 0; i--) {
-             this._down(i);
-           }
-         }
-       }
+       var reTrimStart = /^\s+/;
+       /**
+        * The base implementation of `_.trim`.
+        *
+        * @private
+        * @param {string} string The string to trim.
+        * @returns {string} Returns the trimmed string.
+        */
 
-       function defaultCompare$1(a, b) {
-         return a < b ? -1 : a > b ? 1 : 0;
+       function baseTrim(string) {
+         return string ? string.slice(0, trimmedEndIndex(string) + 1).replace(reTrimStart, '') : string;
        }
 
-       TinyQueue.prototype = {
-         push: function push(item) {
-           this.data.push(item);
-           this.length++;
-
-           this._up(this.length - 1);
-         },
-         pop: function pop() {
-           if (this.length === 0) return undefined;
-           var top = this.data[0];
-           this.length--;
-
-           if (this.length > 0) {
-             this.data[0] = this.data[this.length];
-
-             this._down(0);
-           }
-
-           this.data.pop();
-           return top;
-         },
-         peek: function peek() {
-           return this.data[0];
-         },
-         _up: function _up(pos) {
-           var data = this.data;
-           var compare = this.compare;
-           var item = data[pos];
-
-           while (pos > 0) {
-             var parent = pos - 1 >> 1;
-             var current = data[parent];
-             if (compare(item, current) >= 0) break;
-             data[pos] = current;
-             pos = parent;
-           }
-
-           data[pos] = item;
-         },
-         _down: function _down(pos) {
-           var data = this.data;
-           var compare = this.compare;
-           var halfLength = this.length >> 1;
-           var item = data[pos];
-
-           while (pos < halfLength) {
-             var left = (pos << 1) + 1;
-             var right = left + 1;
-             var best = data[left];
+       /** Built-in value references. */
 
-             if (right < this.length && compare(data[right], best) < 0) {
-               left = right;
-               best = data[right];
-             }
+       var _Symbol = root.Symbol;
 
-             if (compare(best, item) >= 0) break;
-             data[pos] = best;
-             pos = left;
-           }
+       /** Used for built-in method references. */
 
-           data[pos] = item;
-         }
-       };
-       tinyqueue["default"] = _default;
+       var objectProto$1 = Object.prototype;
+       /** Used to check objects for own properties. */
 
-       var max$5 = Math.max;
-       var min$8 = Math.min;
-       var contourId = 0;
+       var hasOwnProperty$2 = objectProto$1.hasOwnProperty;
+       /**
+        * Used to resolve the
+        * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)
+        * of values.
+        */
 
-       function processPolygon(contourOrHole, isSubject, depth, Q, bbox, isExteriorRing) {
-         var i, len, s1, s2, e1, e2;
+       var nativeObjectToString$1 = objectProto$1.toString;
+       /** Built-in value references. */
 
-         for (i = 0, len = contourOrHole.length - 1; i < len; i++) {
-           s1 = contourOrHole[i];
-           s2 = contourOrHole[i + 1];
-           e1 = new SweepEvent(s1, false, undefined, isSubject);
-           e2 = new SweepEvent(s2, false, e1, isSubject);
-           e1.otherEvent = e2;
+       var symToStringTag$1 = _Symbol ? _Symbol.toStringTag : undefined;
+       /**
+        * A specialized version of `baseGetTag` which ignores `Symbol.toStringTag` values.
+        *
+        * @private
+        * @param {*} value The value to query.
+        * @returns {string} Returns the raw `toStringTag`.
+        */
 
-           if (s1[0] === s2[0] && s1[1] === s2[1]) {
-             continue; // skip collapsed edges, or it breaks
-           }
+       function getRawTag(value) {
+         var isOwn = hasOwnProperty$2.call(value, symToStringTag$1),
+             tag = value[symToStringTag$1];
 
-           e1.contourId = e2.contourId = depth;
+         try {
+           value[symToStringTag$1] = undefined;
+           var unmasked = true;
+         } catch (e) {}
 
-           if (!isExteriorRing) {
-             e1.isExteriorRing = false;
-             e2.isExteriorRing = false;
-           }
+         var result = nativeObjectToString$1.call(value);
 
-           if (compareEvents(e1, e2) > 0) {
-             e2.left = true;
+         if (unmasked) {
+           if (isOwn) {
+             value[symToStringTag$1] = tag;
            } else {
-             e1.left = true;
+             delete value[symToStringTag$1];
            }
-
-           var x = s1[0],
-               y = s1[1];
-           bbox[0] = min$8(bbox[0], x);
-           bbox[1] = min$8(bbox[1], y);
-           bbox[2] = max$5(bbox[2], x);
-           bbox[3] = max$5(bbox[3], y); // Pushing it so the queue is sorted from left to right,
-           // with object on the left having the highest priority.
-
-           Q.push(e1);
-           Q.push(e2);
          }
+
+         return result;
        }
 
-       function fillQueue(subject, clipping, sbbox, cbbox, operation) {
-         var eventQueue = new tinyqueue(null, compareEvents);
-         var polygonSet, isExteriorRing, i, ii, j, jj; //, k, kk;
+       /** Used for built-in method references. */
+       var objectProto = Object.prototype;
+       /**
+        * Used to resolve the
+        * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)
+        * of values.
+        */
+
+       var nativeObjectToString = objectProto.toString;
+       /**
+        * Converts `value` to a string using `Object.prototype.toString`.
+        *
+        * @private
+        * @param {*} value The value to convert.
+        * @returns {string} Returns the converted string.
+        */
 
-         for (i = 0, ii = subject.length; i < ii; i++) {
-           polygonSet = subject[i];
+       function objectToString(value) {
+         return nativeObjectToString.call(value);
+       }
 
-           for (j = 0, jj = polygonSet.length; j < jj; j++) {
-             isExteriorRing = j === 0;
-             if (isExteriorRing) contourId++;
-             processPolygon(polygonSet[j], true, contourId, eventQueue, sbbox, isExteriorRing);
-           }
-         }
+       /** `Object#toString` result references. */
 
-         for (i = 0, ii = clipping.length; i < ii; i++) {
-           polygonSet = clipping[i];
+       var nullTag = '[object Null]',
+           undefinedTag = '[object Undefined]';
+       /** Built-in value references. */
 
-           for (j = 0, jj = polygonSet.length; j < jj; j++) {
-             isExteriorRing = j === 0;
-             if (operation === DIFFERENCE) isExteriorRing = false;
-             if (isExteriorRing) contourId++;
-             processPolygon(polygonSet[j], false, contourId, eventQueue, cbbox, isExteriorRing);
-           }
+       var symToStringTag = _Symbol ? _Symbol.toStringTag : undefined;
+       /**
+        * The base implementation of `getTag` without fallbacks for buggy environments.
+        *
+        * @private
+        * @param {*} value The value to query.
+        * @returns {string} Returns the `toStringTag`.
+        */
+
+       function baseGetTag(value) {
+         if (value == null) {
+           return value === undefined ? undefinedTag : nullTag;
          }
 
-         return eventQueue;
+         return symToStringTag && symToStringTag in Object(value) ? getRawTag(value) : objectToString(value);
        }
 
-       var EMPTY = [];
+       /**
+        * Checks if `value` is object-like. A value is object-like if it's not `null`
+        * and has a `typeof` result of "object".
+        *
+        * @static
+        * @memberOf _
+        * @since 4.0.0
+        * @category Lang
+        * @param {*} value The value to check.
+        * @returns {boolean} Returns `true` if `value` is object-like, else `false`.
+        * @example
+        *
+        * _.isObjectLike({});
+        * // => true
+        *
+        * _.isObjectLike([1, 2, 3]);
+        * // => true
+        *
+        * _.isObjectLike(_.noop);
+        * // => false
+        *
+        * _.isObjectLike(null);
+        * // => false
+        */
+       function isObjectLike(value) {
+         return value != null && _typeof(value) == 'object';
+       }
 
-       function trivialOperation(subject, clipping, operation) {
-         var result = null;
+       /** `Object#toString` result references. */
 
-         if (subject.length * clipping.length === 0) {
-           if (operation === INTERSECTION) {
-             result = EMPTY;
-           } else if (operation === DIFFERENCE) {
-             result = subject;
-           } else if (operation === UNION || operation === XOR) {
-             result = subject.length === 0 ? clipping : subject;
-           }
-         }
+       var symbolTag = '[object Symbol]';
+       /**
+        * Checks if `value` is classified as a `Symbol` primitive or object.
+        *
+        * @static
+        * @memberOf _
+        * @since 4.0.0
+        * @category Lang
+        * @param {*} value The value to check.
+        * @returns {boolean} Returns `true` if `value` is a symbol, else `false`.
+        * @example
+        *
+        * _.isSymbol(Symbol.iterator);
+        * // => true
+        *
+        * _.isSymbol('abc');
+        * // => false
+        */
 
-         return result;
+       function isSymbol(value) {
+         return _typeof(value) == 'symbol' || isObjectLike(value) && baseGetTag(value) == symbolTag;
        }
 
-       function compareBBoxes(subject, clipping, sbbox, cbbox, operation) {
-         var result = null;
+       /** Used as references for various `Number` constants. */
 
-         if (sbbox[0] > cbbox[2] || cbbox[0] > sbbox[2] || sbbox[1] > cbbox[3] || cbbox[1] > sbbox[3]) {
-           if (operation === INTERSECTION) {
-             result = EMPTY;
-           } else if (operation === DIFFERENCE) {
-             result = subject;
-           } else if (operation === UNION || operation === XOR) {
-             result = subject.concat(clipping);
-           }
-         }
+       var NAN = 0 / 0;
+       /** Used to detect bad signed hexadecimal string values. */
 
-         return result;
-       }
+       var reIsBadHex = /^[-+]0x[0-9a-f]+$/i;
+       /** Used to detect binary string values. */
+
+       var reIsBinary = /^0b[01]+$/i;
+       /** Used to detect octal string values. */
+
+       var reIsOctal = /^0o[0-7]+$/i;
+       /** Built-in method references without a dependency on `root`. */
+
+       var freeParseInt = parseInt;
+       /**
+        * Converts `value` to a number.
+        *
+        * @static
+        * @memberOf _
+        * @since 4.0.0
+        * @category Lang
+        * @param {*} value The value to process.
+        * @returns {number} Returns the number.
+        * @example
+        *
+        * _.toNumber(3.2);
+        * // => 3.2
+        *
+        * _.toNumber(Number.MIN_VALUE);
+        * // => 5e-324
+        *
+        * _.toNumber(Infinity);
+        * // => Infinity
+        *
+        * _.toNumber('3.2');
+        * // => 3.2
+        */
 
-       function _boolean(subject, clipping, operation) {
-         if (typeof subject[0][0][0] === 'number') {
-           subject = [subject];
+       function toNumber(value) {
+         if (typeof value == 'number') {
+           return value;
          }
 
-         if (typeof clipping[0][0][0] === 'number') {
-           clipping = [clipping];
+         if (isSymbol(value)) {
+           return NAN;
          }
 
-         var trivial = trivialOperation(subject, clipping, operation);
+         if (isObject$2(value)) {
+           var other = typeof value.valueOf == 'function' ? value.valueOf() : value;
+           value = isObject$2(other) ? other + '' : other;
+         }
 
-         if (trivial) {
-           return trivial === EMPTY ? null : trivial;
+         if (typeof value != 'string') {
+           return value === 0 ? value : +value;
          }
 
-         var sbbox = [Infinity, Infinity, -Infinity, -Infinity];
-         var cbbox = [Infinity, Infinity, -Infinity, -Infinity]; // console.time('fill queue');
+         value = baseTrim(value);
+         var isBinary = reIsBinary.test(value);
+         return isBinary || reIsOctal.test(value) ? freeParseInt(value.slice(2), isBinary ? 2 : 8) : reIsBadHex.test(value) ? NAN : +value;
+       }
 
-         var eventQueue = fillQueue(subject, clipping, sbbox, cbbox, operation); //console.timeEnd('fill queue');
+       /** Error message constants. */
 
-         trivial = compareBBoxes(subject, clipping, sbbox, cbbox, operation);
+       var FUNC_ERROR_TEXT$1 = 'Expected a function';
+       /* Built-in method references for those with the same name as other `lodash` methods. */
 
-         if (trivial) {
-           return trivial === EMPTY ? null : trivial;
-         } // console.time('subdivide edges');
+       var nativeMax = Math.max,
+           nativeMin = Math.min;
+       /**
+        * Creates a debounced function that delays invoking `func` until after `wait`
+        * milliseconds have elapsed since the last time the debounced function was
+        * invoked. The debounced function comes with a `cancel` method to cancel
+        * delayed `func` invocations and a `flush` method to immediately invoke them.
+        * Provide `options` to indicate whether `func` should be invoked on the
+        * leading and/or trailing edge of the `wait` timeout. The `func` is invoked
+        * with the last arguments provided to the debounced function. Subsequent
+        * calls to the debounced function return the result of the last `func`
+        * invocation.
+        *
+        * **Note:** If `leading` and `trailing` options are `true`, `func` is
+        * invoked on the trailing edge of the timeout only if the debounced function
+        * is invoked more than once during the `wait` timeout.
+        *
+        * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
+        * until to the next tick, similar to `setTimeout` with a timeout of `0`.
+        *
+        * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
+        * for details over the differences between `_.debounce` and `_.throttle`.
+        *
+        * @static
+        * @memberOf _
+        * @since 0.1.0
+        * @category Function
+        * @param {Function} func The function to debounce.
+        * @param {number} [wait=0] The number of milliseconds to delay.
+        * @param {Object} [options={}] The options object.
+        * @param {boolean} [options.leading=false]
+        *  Specify invoking on the leading edge of the timeout.
+        * @param {number} [options.maxWait]
+        *  The maximum time `func` is allowed to be delayed before it's invoked.
+        * @param {boolean} [options.trailing=true]
+        *  Specify invoking on the trailing edge of the timeout.
+        * @returns {Function} Returns the new debounced function.
+        * @example
+        *
+        * // Avoid costly calculations while the window size is in flux.
+        * jQuery(window).on('resize', _.debounce(calculateLayout, 150));
+        *
+        * // Invoke `sendMail` when clicked, debouncing subsequent calls.
+        * jQuery(element).on('click', _.debounce(sendMail, 300, {
+        *   'leading': true,
+        *   'trailing': false
+        * }));
+        *
+        * // Ensure `batchLog` is invoked once after 1 second of debounced calls.
+        * var debounced = _.debounce(batchLog, 250, { 'maxWait': 1000 });
+        * var source = new EventSource('/stream');
+        * jQuery(source).on('message', debounced);
+        *
+        * // Cancel the trailing debounced invocation.
+        * jQuery(window).on('popstate', debounced.cancel);
+        */
 
+       function debounce(func, wait, options) {
+         var lastArgs,
+             lastThis,
+             maxWait,
+             result,
+             timerId,
+             lastCallTime,
+             lastInvokeTime = 0,
+             leading = false,
+             maxing = false,
+             trailing = true;
 
-         var sortedEvents = subdivide(eventQueue, subject, clipping, sbbox, cbbox, operation); //console.timeEnd('subdivide edges');
-         // console.time('connect vertices');
+         if (typeof func != 'function') {
+           throw new TypeError(FUNC_ERROR_TEXT$1);
+         }
 
-         var contours = connectEdges(sortedEvents); //console.timeEnd('connect vertices');
-         // Convert contours to polygons
+         wait = toNumber(wait) || 0;
 
-         var polygons = [];
+         if (isObject$2(options)) {
+           leading = !!options.leading;
+           maxing = 'maxWait' in options;
+           maxWait = maxing ? nativeMax(toNumber(options.maxWait) || 0, wait) : maxWait;
+           trailing = 'trailing' in options ? !!options.trailing : trailing;
+         }
 
-         for (var i = 0; i < contours.length; i++) {
-           var contour = contours[i];
+         function invokeFunc(time) {
+           var args = lastArgs,
+               thisArg = lastThis;
+           lastArgs = lastThis = undefined;
+           lastInvokeTime = time;
+           result = func.apply(thisArg, args);
+           return result;
+         }
 
-           if (contour.isExterior()) {
-             // The exterior ring goes first
-             var rings = [contour.points]; // Followed by holes if any
+         function leadingEdge(time) {
+           // Reset any `maxWait` timer.
+           lastInvokeTime = time; // Start the timer for the trailing edge.
 
-             for (var j = 0; j < contour.holeIds.length; j++) {
-               var holeId = contour.holeIds[j];
-               rings.push(contours[holeId].points);
-             }
+           timerId = setTimeout(timerExpired, wait); // Invoke the leading edge.
 
-             polygons.push(rings);
-           }
+           return leading ? invokeFunc(time) : result;
          }
 
-         return polygons;
-       }
+         function remainingWait(time) {
+           var timeSinceLastCall = time - lastCallTime,
+               timeSinceLastInvoke = time - lastInvokeTime,
+               timeWaiting = wait - timeSinceLastCall;
+           return maxing ? nativeMin(timeWaiting, maxWait - timeSinceLastInvoke) : timeWaiting;
+         }
 
-       function union(subject, clipping) {
-         return _boolean(subject, clipping, UNION);
-       }
+         function shouldInvoke(time) {
+           var timeSinceLastCall = time - lastCallTime,
+               timeSinceLastInvoke = time - lastInvokeTime; // Either this is the first call, activity has stopped and we're at the
+           // trailing edge, the system time has gone backwards and we're treating
+           // it as the trailing edge, or we've hit the `maxWait` limit.
 
-       /*! ieee754. BSD-3-Clause License. Feross Aboukhadijeh <https://feross.org/opensource> */
-       var read$6 = function read(buffer, offset, isLE, mLen, nBytes) {
-         var e, m;
-         var eLen = nBytes * 8 - mLen - 1;
-         var eMax = (1 << eLen) - 1;
-         var eBias = eMax >> 1;
-         var nBits = -7;
-         var i = isLE ? nBytes - 1 : 0;
-         var d = isLE ? -1 : 1;
-         var s = buffer[offset + i];
-         i += d;
-         e = s & (1 << -nBits) - 1;
-         s >>= -nBits;
-         nBits += eLen;
+           return lastCallTime === undefined || timeSinceLastCall >= wait || timeSinceLastCall < 0 || maxing && timeSinceLastInvoke >= maxWait;
+         }
 
-         for (; nBits > 0; e = e * 256 + buffer[offset + i], i += d, nBits -= 8) {}
+         function timerExpired() {
+           var time = now();
 
-         m = e & (1 << -nBits) - 1;
-         e >>= -nBits;
-         nBits += mLen;
+           if (shouldInvoke(time)) {
+             return trailingEdge(time);
+           } // Restart the timer.
 
-         for (; nBits > 0; m = m * 256 + buffer[offset + i], i += d, nBits -= 8) {}
 
-         if (e === 0) {
-           e = 1 - eBias;
-         } else if (e === eMax) {
-           return m ? NaN : (s ? -1 : 1) * Infinity;
-         } else {
-           m = m + Math.pow(2, mLen);
-           e = e - eBias;
+           timerId = setTimeout(timerExpired, remainingWait(time));
          }
 
-         return (s ? -1 : 1) * m * Math.pow(2, e - mLen);
-       };
-
-       var write$6 = function write(buffer, value, offset, isLE, mLen, nBytes) {
-         var e, m, c;
-         var eLen = nBytes * 8 - mLen - 1;
-         var eMax = (1 << eLen) - 1;
-         var eBias = eMax >> 1;
-         var rt = mLen === 23 ? Math.pow(2, -24) - Math.pow(2, -77) : 0;
-         var i = isLE ? 0 : nBytes - 1;
-         var d = isLE ? 1 : -1;
-         var s = value < 0 || value === 0 && 1 / value < 0 ? 1 : 0;
-         value = Math.abs(value);
-
-         if (isNaN(value) || value === Infinity) {
-           m = isNaN(value) ? 1 : 0;
-           e = eMax;
-         } else {
-           e = Math.floor(Math.log(value) / Math.LN2);
+         function trailingEdge(time) {
+           timerId = undefined; // Only invoke if we have `lastArgs` which means `func` has been
+           // debounced at least once.
 
-           if (value * (c = Math.pow(2, -e)) < 1) {
-             e--;
-             c *= 2;
+           if (trailing && lastArgs) {
+             return invokeFunc(time);
            }
 
-           if (e + eBias >= 1) {
-             value += rt / c;
-           } else {
-             value += rt * Math.pow(2, 1 - eBias);
-           }
+           lastArgs = lastThis = undefined;
+           return result;
+         }
 
-           if (value * c >= 2) {
-             e++;
-             c /= 2;
+         function cancel() {
+           if (timerId !== undefined) {
+             clearTimeout(timerId);
            }
 
-           if (e + eBias >= eMax) {
-             m = 0;
-             e = eMax;
-           } else if (e + eBias >= 1) {
-             m = (value * c - 1) * Math.pow(2, mLen);
-             e = e + eBias;
-           } else {
-             m = value * Math.pow(2, eBias - 1) * Math.pow(2, mLen);
-             e = 0;
-           }
+           lastInvokeTime = 0;
+           lastArgs = lastCallTime = lastThis = timerId = undefined;
          }
 
-         for (; mLen >= 8; buffer[offset + i] = m & 0xff, i += d, m /= 256, mLen -= 8) {}
+         function flush() {
+           return timerId === undefined ? result : trailingEdge(now());
+         }
 
-         e = e << mLen | m;
-         eLen += mLen;
+         function debounced() {
+           var time = now(),
+               isInvoking = shouldInvoke(time);
+           lastArgs = arguments;
+           lastThis = this;
+           lastCallTime = time;
 
-         for (; eLen > 0; buffer[offset + i] = e & 0xff, i += d, e /= 256, eLen -= 8) {}
+           if (isInvoking) {
+             if (timerId === undefined) {
+               return leadingEdge(lastCallTime);
+             }
 
-         buffer[offset + i - d] |= s * 128;
-       };
+             if (maxing) {
+               // Handle invocations in a tight loop.
+               clearTimeout(timerId);
+               timerId = setTimeout(timerExpired, wait);
+               return invokeFunc(lastCallTime);
+             }
+           }
 
-       var ieee754$1 = {
-         read: read$6,
-         write: write$6
-       };
+           if (timerId === undefined) {
+             timerId = setTimeout(timerExpired, wait);
+           }
 
-       var pbf = Pbf;
+           return result;
+         }
 
-       function Pbf(buf) {
-         this.buf = ArrayBuffer.isView && ArrayBuffer.isView(buf) ? buf : new Uint8Array(buf || 0);
-         this.pos = 0;
-         this.type = 0;
-         this.length = this.buf.length;
+         debounced.cancel = cancel;
+         debounced.flush = flush;
+         return debounced;
        }
 
-       Pbf.Varint = 0; // varint: int32, int64, uint32, uint64, sint32, sint64, bool, enum
-
-       Pbf.Fixed64 = 1; // 64-bit: double, fixed64, sfixed64
-
-       Pbf.Bytes = 2; // length-delimited: string, bytes, embedded messages, packed repeated fields
-
-       Pbf.Fixed32 = 5; // 32-bit: float, fixed32, sfixed32
+       /*
+           iD.coreDifference represents the difference between two graphs.
+           It knows how to calculate the set of entities that were
+           created, modified, or deleted, and also contains the logic
+           for recursively extending a difference to the complete set
+           of entities that will require a redraw, taking into account
+           child and parent relationships.
+        */
 
-       var SHIFT_LEFT_32 = (1 << 16) * (1 << 16),
-           SHIFT_RIGHT_32 = 1 / SHIFT_LEFT_32; // Threshold chosen based on both benchmarking and knowledge about browser string
-       // data structures (which currently switch structure types at 12 bytes or more)
+       function coreDifference(base, head) {
+         var _changes = {};
+         var _didChange = {}; // 'addition', 'deletion', 'geometry', 'properties'
 
-       var TEXT_DECODER_MIN_LENGTH = 12;
-       var utf8TextDecoder = typeof TextDecoder === 'undefined' ? null : new TextDecoder('utf8');
-       Pbf.prototype = {
-         destroy: function destroy() {
-           this.buf = null;
-         },
-         // === READING =================================================================
-         readFields: function readFields(readField, result, end) {
-           end = end || this.length;
+         var _diff = {};
 
-           while (this.pos < end) {
-             var val = this.readVarint(),
-                 tag = val >> 3,
-                 startPos = this.pos;
-             this.type = val & 0x7;
-             readField(tag, result, this);
-             if (this.pos === startPos) this.skip(val);
-           }
+         function checkEntityID(id) {
+           var h = head.entities[id];
+           var b = base.entities[id];
+           if (h === b) return;
+           if (_changes[id]) return;
 
-           return result;
-         },
-         readMessage: function readMessage(readField, result) {
-           return this.readFields(readField, result, this.readVarint() + this.pos);
-         },
-         readFixed32: function readFixed32() {
-           var val = readUInt32(this.buf, this.pos);
-           this.pos += 4;
-           return val;
-         },
-         readSFixed32: function readSFixed32() {
-           var val = readInt32(this.buf, this.pos);
-           this.pos += 4;
-           return val;
-         },
-         // 64-bit int handling is based on github.com/dpw/node-buffer-more-ints (MIT-licensed)
-         readFixed64: function readFixed64() {
-           var val = readUInt32(this.buf, this.pos) + readUInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32;
-           this.pos += 8;
-           return val;
-         },
-         readSFixed64: function readSFixed64() {
-           var val = readUInt32(this.buf, this.pos) + readInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32;
-           this.pos += 8;
-           return val;
-         },
-         readFloat: function readFloat() {
-           var val = ieee754$1.read(this.buf, this.pos, true, 23, 4);
-           this.pos += 4;
-           return val;
-         },
-         readDouble: function readDouble() {
-           var val = ieee754$1.read(this.buf, this.pos, true, 52, 8);
-           this.pos += 8;
-           return val;
-         },
-         readVarint: function readVarint(isSigned) {
-           var buf = this.buf,
-               val,
-               b;
-           b = buf[this.pos++];
-           val = b & 0x7f;
-           if (b < 0x80) return val;
-           b = buf[this.pos++];
-           val |= (b & 0x7f) << 7;
-           if (b < 0x80) return val;
-           b = buf[this.pos++];
-           val |= (b & 0x7f) << 14;
-           if (b < 0x80) return val;
-           b = buf[this.pos++];
-           val |= (b & 0x7f) << 21;
-           if (b < 0x80) return val;
-           b = buf[this.pos];
-           val |= (b & 0x0f) << 28;
-           return readVarintRemainder(val, isSigned, this);
-         },
-         readVarint64: function readVarint64() {
-           // for compatibility with v2.0.1
-           return this.readVarint(true);
-         },
-         readSVarint: function readSVarint() {
-           var num = this.readVarint();
-           return num % 2 === 1 ? (num + 1) / -2 : num / 2; // zigzag encoding
-         },
-         readBoolean: function readBoolean() {
-           return Boolean(this.readVarint());
-         },
-         readString: function readString() {
-           var end = this.readVarint() + this.pos;
-           var pos = this.pos;
-           this.pos = end;
+           if (!h && b) {
+             _changes[id] = {
+               base: b,
+               head: h
+             };
+             _didChange.deletion = true;
+             return;
+           }
 
-           if (end - pos >= TEXT_DECODER_MIN_LENGTH && utf8TextDecoder) {
-             // longer strings are fast with the built-in browser TextDecoder API
-             return readUtf8TextDecoder(this.buf, pos, end);
-           } // short strings are fast with our custom implementation
+           if (h && !b) {
+             _changes[id] = {
+               base: b,
+               head: h
+             };
+             _didChange.addition = true;
+             return;
+           }
 
+           if (h && b) {
+             if (h.members && b.members && !fastDeepEqual(h.members, b.members)) {
+               _changes[id] = {
+                 base: b,
+                 head: h
+               };
+               _didChange.geometry = true;
+               _didChange.properties = true;
+               return;
+             }
 
-           return readUtf8(this.buf, pos, end);
-         },
-         readBytes: function readBytes() {
-           var end = this.readVarint() + this.pos,
-               buffer = this.buf.subarray(this.pos, end);
-           this.pos = end;
-           return buffer;
-         },
-         // verbose for performance reasons; doesn't affect gzipped size
-         readPackedVarint: function readPackedVarint(arr, isSigned) {
-           if (this.type !== Pbf.Bytes) return arr.push(this.readVarint(isSigned));
-           var end = readPackedEnd(this);
-           arr = arr || [];
+             if (h.loc && b.loc && !geoVecEqual(h.loc, b.loc)) {
+               _changes[id] = {
+                 base: b,
+                 head: h
+               };
+               _didChange.geometry = true;
+             }
 
-           while (this.pos < end) {
-             arr.push(this.readVarint(isSigned));
+             if (h.nodes && b.nodes && !fastDeepEqual(h.nodes, b.nodes)) {
+               _changes[id] = {
+                 base: b,
+                 head: h
+               };
+               _didChange.geometry = true;
+             }
+
+             if (h.tags && b.tags && !fastDeepEqual(h.tags, b.tags)) {
+               _changes[id] = {
+                 base: b,
+                 head: h
+               };
+               _didChange.properties = true;
+             }
            }
+         }
 
-           return arr;
-         },
-         readPackedSVarint: function readPackedSVarint(arr) {
-           if (this.type !== Pbf.Bytes) return arr.push(this.readSVarint());
-           var end = readPackedEnd(this);
-           arr = arr || [];
+         function load() {
+           // HOT CODE: there can be many thousands of downloaded entities, so looping
+           // through them all can become a performance bottleneck. Optimize by
+           // resolving duplicates and using a basic `for` loop
+           var ids = utilArrayUniq(Object.keys(head.entities).concat(Object.keys(base.entities)));
 
-           while (this.pos < end) {
-             arr.push(this.readSVarint());
+           for (var i = 0; i < ids.length; i++) {
+             checkEntityID(ids[i]);
            }
+         }
 
-           return arr;
-         },
-         readPackedBoolean: function readPackedBoolean(arr) {
-           if (this.type !== Pbf.Bytes) return arr.push(this.readBoolean());
-           var end = readPackedEnd(this);
-           arr = arr || [];
+         load();
 
-           while (this.pos < end) {
-             arr.push(this.readBoolean());
-           }
+         _diff.length = function length() {
+           return Object.keys(_changes).length;
+         };
 
-           return arr;
-         },
-         readPackedFloat: function readPackedFloat(arr) {
-           if (this.type !== Pbf.Bytes) return arr.push(this.readFloat());
-           var end = readPackedEnd(this);
-           arr = arr || [];
+         _diff.changes = function changes() {
+           return _changes;
+         };
 
-           while (this.pos < end) {
-             arr.push(this.readFloat());
-           }
+         _diff.didChange = _didChange; // pass true to include affected relation members
 
-           return arr;
-         },
-         readPackedDouble: function readPackedDouble(arr) {
-           if (this.type !== Pbf.Bytes) return arr.push(this.readDouble());
-           var end = readPackedEnd(this);
-           arr = arr || [];
+         _diff.extantIDs = function extantIDs(includeRelMembers) {
+           var result = new Set();
+           Object.keys(_changes).forEach(function (id) {
+             if (_changes[id].head) {
+               result.add(id);
+             }
 
-           while (this.pos < end) {
-             arr.push(this.readDouble());
-           }
+             var h = _changes[id].head;
+             var b = _changes[id].base;
+             var entity = h || b;
 
-           return arr;
-         },
-         readPackedFixed32: function readPackedFixed32(arr) {
-           if (this.type !== Pbf.Bytes) return arr.push(this.readFixed32());
-           var end = readPackedEnd(this);
-           arr = arr || [];
+             if (includeRelMembers && entity.type === 'relation') {
+               var mh = h ? h.members.map(function (m) {
+                 return m.id;
+               }) : [];
+               var mb = b ? b.members.map(function (m) {
+                 return m.id;
+               }) : [];
+               utilArrayUnion(mh, mb).forEach(function (memberID) {
+                 if (head.hasEntity(memberID)) {
+                   result.add(memberID);
+                 }
+               });
+             }
+           });
+           return Array.from(result);
+         };
 
-           while (this.pos < end) {
-             arr.push(this.readFixed32());
-           }
+         _diff.modified = function modified() {
+           var result = [];
+           Object.values(_changes).forEach(function (change) {
+             if (change.base && change.head) {
+               result.push(change.head);
+             }
+           });
+           return result;
+         };
 
-           return arr;
-         },
-         readPackedSFixed32: function readPackedSFixed32(arr) {
-           if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed32());
-           var end = readPackedEnd(this);
-           arr = arr || [];
+         _diff.created = function created() {
+           var result = [];
+           Object.values(_changes).forEach(function (change) {
+             if (!change.base && change.head) {
+               result.push(change.head);
+             }
+           });
+           return result;
+         };
 
-           while (this.pos < end) {
-             arr.push(this.readSFixed32());
-           }
+         _diff.deleted = function deleted() {
+           var result = [];
+           Object.values(_changes).forEach(function (change) {
+             if (change.base && !change.head) {
+               result.push(change.base);
+             }
+           });
+           return result;
+         };
 
-           return arr;
-         },
-         readPackedFixed64: function readPackedFixed64(arr) {
-           if (this.type !== Pbf.Bytes) return arr.push(this.readFixed64());
-           var end = readPackedEnd(this);
-           arr = arr || [];
+         _diff.summary = function summary() {
+           var relevant = {};
+           var keys = Object.keys(_changes);
 
-           while (this.pos < end) {
-             arr.push(this.readFixed64());
-           }
+           for (var i = 0; i < keys.length; i++) {
+             var change = _changes[keys[i]];
 
-           return arr;
-         },
-         readPackedSFixed64: function readPackedSFixed64(arr) {
-           if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed64());
-           var end = readPackedEnd(this);
-           arr = arr || [];
+             if (change.head && change.head.geometry(head) !== 'vertex') {
+               addEntity(change.head, head, change.base ? 'modified' : 'created');
+             } else if (change.base && change.base.geometry(base) !== 'vertex') {
+               addEntity(change.base, base, 'deleted');
+             } else if (change.base && change.head) {
+               // modified vertex
+               var moved = !fastDeepEqual(change.base.loc, change.head.loc);
+               var retagged = !fastDeepEqual(change.base.tags, change.head.tags);
 
-           while (this.pos < end) {
-             arr.push(this.readSFixed64());
+               if (moved) {
+                 addParents(change.head);
+               }
+
+               if (retagged || moved && change.head.hasInterestingTags()) {
+                 addEntity(change.head, head, 'modified');
+               }
+             } else if (change.head && change.head.hasInterestingTags()) {
+               // created vertex
+               addEntity(change.head, head, 'created');
+             } else if (change.base && change.base.hasInterestingTags()) {
+               // deleted vertex
+               addEntity(change.base, base, 'deleted');
+             }
            }
 
-           return arr;
-         },
-         skip: function skip(val) {
-           var type = val & 0x7;
-           if (type === Pbf.Varint) while (this.buf[this.pos++] > 0x7f) {} else if (type === Pbf.Bytes) this.pos = this.readVarint() + this.pos;else if (type === Pbf.Fixed32) this.pos += 4;else if (type === Pbf.Fixed64) this.pos += 8;else throw new Error('Unimplemented type: ' + type);
-         },
-         // === WRITING =================================================================
-         writeTag: function writeTag(tag, type) {
-           this.writeVarint(tag << 3 | type);
-         },
-         realloc: function realloc(min) {
-           var length = this.length || 16;
+           return Object.values(relevant);
 
-           while (length < this.pos + min) {
-             length *= 2;
+           function addEntity(entity, graph, changeType) {
+             relevant[entity.id] = {
+               entity: entity,
+               graph: graph,
+               changeType: changeType
+             };
            }
 
-           if (length !== this.length) {
-             var buf = new Uint8Array(length);
-             buf.set(this.buf);
-             this.buf = buf;
-             this.length = length;
-           }
-         },
-         finish: function finish() {
-           this.length = this.pos;
-           this.pos = 0;
-           return this.buf.subarray(0, this.length);
-         },
-         writeFixed32: function writeFixed32(val) {
-           this.realloc(4);
-           writeInt32(this.buf, val, this.pos);
-           this.pos += 4;
-         },
-         writeSFixed32: function writeSFixed32(val) {
-           this.realloc(4);
-           writeInt32(this.buf, val, this.pos);
-           this.pos += 4;
-         },
-         writeFixed64: function writeFixed64(val) {
-           this.realloc(8);
-           writeInt32(this.buf, val & -1, this.pos);
-           writeInt32(this.buf, Math.floor(val * SHIFT_RIGHT_32), this.pos + 4);
-           this.pos += 8;
-         },
-         writeSFixed64: function writeSFixed64(val) {
-           this.realloc(8);
-           writeInt32(this.buf, val & -1, this.pos);
-           writeInt32(this.buf, Math.floor(val * SHIFT_RIGHT_32), this.pos + 4);
-           this.pos += 8;
-         },
-         writeVarint: function writeVarint(val) {
-           val = +val || 0;
+           function addParents(entity) {
+             var parents = head.parentWays(entity);
 
-           if (val > 0xfffffff || val < 0) {
-             writeBigVarint(val, this);
-             return;
+             for (var j = parents.length - 1; j >= 0; j--) {
+               var parent = parents[j];
+
+               if (!(parent.id in relevant)) {
+                 addEntity(parent, head, 'modified');
+               }
+             }
            }
+         }; // returns complete set of entities that require a redraw
+         //  (optionally within given `extent`)
 
-           this.realloc(4);
-           this.buf[this.pos++] = val & 0x7f | (val > 0x7f ? 0x80 : 0);
-           if (val <= 0x7f) return;
-           this.buf[this.pos++] = (val >>>= 7) & 0x7f | (val > 0x7f ? 0x80 : 0);
-           if (val <= 0x7f) return;
-           this.buf[this.pos++] = (val >>>= 7) & 0x7f | (val > 0x7f ? 0x80 : 0);
-           if (val <= 0x7f) return;
-           this.buf[this.pos++] = val >>> 7 & 0x7f;
-         },
-         writeSVarint: function writeSVarint(val) {
-           this.writeVarint(val < 0 ? -val * 2 - 1 : val * 2);
-         },
-         writeBoolean: function writeBoolean(val) {
-           this.writeVarint(Boolean(val));
-         },
-         writeString: function writeString(str) {
-           str = String(str);
-           this.realloc(str.length * 4);
-           this.pos++; // reserve 1 byte for short string length
 
-           var startPos = this.pos; // write the string directly to the buffer and see how much was written
+         _diff.complete = function complete(extent) {
+           var result = {};
+           var id, change;
 
-           this.pos = writeUtf8(this.buf, str, this.pos);
-           var len = this.pos - startPos;
-           if (len >= 0x80) makeRoomForExtraLength(startPos, len, this); // finally, write the message length in the reserved place and restore the position
+           for (id in _changes) {
+             change = _changes[id];
+             var h = change.head;
+             var b = change.base;
+             var entity = h || b;
+             var i;
 
-           this.pos = startPos - 1;
-           this.writeVarint(len);
-           this.pos += len;
-         },
-         writeFloat: function writeFloat(val) {
-           this.realloc(4);
-           ieee754$1.write(this.buf, val, this.pos, true, 23, 4);
-           this.pos += 4;
-         },
-         writeDouble: function writeDouble(val) {
-           this.realloc(8);
-           ieee754$1.write(this.buf, val, this.pos, true, 52, 8);
-           this.pos += 8;
-         },
-         writeBytes: function writeBytes(buffer) {
-           var len = buffer.length;
-           this.writeVarint(len);
-           this.realloc(len);
+             if (extent && (!h || !h.intersects(extent, head)) && (!b || !b.intersects(extent, base))) {
+               continue;
+             }
 
-           for (var i = 0; i < len; i++) {
-             this.buf[this.pos++] = buffer[i];
-           }
-         },
-         writeRawMessage: function writeRawMessage(fn, obj) {
-           this.pos++; // reserve 1 byte for short message length
-           // write the message directly to the buffer and see how much was written
+             result[id] = h;
 
-           var startPos = this.pos;
-           fn(obj, this);
-           var len = this.pos - startPos;
-           if (len >= 0x80) makeRoomForExtraLength(startPos, len, this); // finally, write the message length in the reserved place and restore the position
+             if (entity.type === 'way') {
+               var nh = h ? h.nodes : [];
+               var nb = b ? b.nodes : [];
+               var diff;
+               diff = utilArrayDifference(nh, nb);
 
-           this.pos = startPos - 1;
-           this.writeVarint(len);
-           this.pos += len;
-         },
-         writeMessage: function writeMessage(tag, fn, obj) {
-           this.writeTag(tag, Pbf.Bytes);
-           this.writeRawMessage(fn, obj);
-         },
-         writePackedVarint: function writePackedVarint(tag, arr) {
-           if (arr.length) this.writeMessage(tag, _writePackedVarint, arr);
-         },
-         writePackedSVarint: function writePackedSVarint(tag, arr) {
-           if (arr.length) this.writeMessage(tag, _writePackedSVarint, arr);
-         },
-         writePackedBoolean: function writePackedBoolean(tag, arr) {
-           if (arr.length) this.writeMessage(tag, _writePackedBoolean, arr);
-         },
-         writePackedFloat: function writePackedFloat(tag, arr) {
-           if (arr.length) this.writeMessage(tag, _writePackedFloat, arr);
-         },
-         writePackedDouble: function writePackedDouble(tag, arr) {
-           if (arr.length) this.writeMessage(tag, _writePackedDouble, arr);
-         },
-         writePackedFixed32: function writePackedFixed32(tag, arr) {
-           if (arr.length) this.writeMessage(tag, _writePackedFixed, arr);
-         },
-         writePackedSFixed32: function writePackedSFixed32(tag, arr) {
-           if (arr.length) this.writeMessage(tag, _writePackedSFixed, arr);
-         },
-         writePackedFixed64: function writePackedFixed64(tag, arr) {
-           if (arr.length) this.writeMessage(tag, _writePackedFixed2, arr);
-         },
-         writePackedSFixed64: function writePackedSFixed64(tag, arr) {
-           if (arr.length) this.writeMessage(tag, _writePackedSFixed2, arr);
-         },
-         writeBytesField: function writeBytesField(tag, buffer) {
-           this.writeTag(tag, Pbf.Bytes);
-           this.writeBytes(buffer);
-         },
-         writeFixed32Field: function writeFixed32Field(tag, val) {
-           this.writeTag(tag, Pbf.Fixed32);
-           this.writeFixed32(val);
-         },
-         writeSFixed32Field: function writeSFixed32Field(tag, val) {
-           this.writeTag(tag, Pbf.Fixed32);
-           this.writeSFixed32(val);
-         },
-         writeFixed64Field: function writeFixed64Field(tag, val) {
-           this.writeTag(tag, Pbf.Fixed64);
-           this.writeFixed64(val);
-         },
-         writeSFixed64Field: function writeSFixed64Field(tag, val) {
-           this.writeTag(tag, Pbf.Fixed64);
-           this.writeSFixed64(val);
-         },
-         writeVarintField: function writeVarintField(tag, val) {
-           this.writeTag(tag, Pbf.Varint);
-           this.writeVarint(val);
-         },
-         writeSVarintField: function writeSVarintField(tag, val) {
-           this.writeTag(tag, Pbf.Varint);
-           this.writeSVarint(val);
-         },
-         writeStringField: function writeStringField(tag, str) {
-           this.writeTag(tag, Pbf.Bytes);
-           this.writeString(str);
-         },
-         writeFloatField: function writeFloatField(tag, val) {
-           this.writeTag(tag, Pbf.Fixed32);
-           this.writeFloat(val);
-         },
-         writeDoubleField: function writeDoubleField(tag, val) {
-           this.writeTag(tag, Pbf.Fixed64);
-           this.writeDouble(val);
-         },
-         writeBooleanField: function writeBooleanField(tag, val) {
-           this.writeVarintField(tag, Boolean(val));
-         }
-       };
+               for (i = 0; i < diff.length; i++) {
+                 result[diff[i]] = head.hasEntity(diff[i]);
+               }
 
-       function readVarintRemainder(l, s, p) {
-         var buf = p.buf,
-             h,
-             b;
-         b = buf[p.pos++];
-         h = (b & 0x70) >> 4;
-         if (b < 0x80) return toNum(l, h, s);
-         b = buf[p.pos++];
-         h |= (b & 0x7f) << 3;
-         if (b < 0x80) return toNum(l, h, s);
-         b = buf[p.pos++];
-         h |= (b & 0x7f) << 10;
-         if (b < 0x80) return toNum(l, h, s);
-         b = buf[p.pos++];
-         h |= (b & 0x7f) << 17;
-         if (b < 0x80) return toNum(l, h, s);
-         b = buf[p.pos++];
-         h |= (b & 0x7f) << 24;
-         if (b < 0x80) return toNum(l, h, s);
-         b = buf[p.pos++];
-         h |= (b & 0x01) << 31;
-         if (b < 0x80) return toNum(l, h, s);
-         throw new Error('Expected varint not more than 10 bytes');
-       }
+               diff = utilArrayDifference(nb, nh);
+
+               for (i = 0; i < diff.length; i++) {
+                 result[diff[i]] = head.hasEntity(diff[i]);
+               }
+             }
+
+             if (entity.type === 'relation' && entity.isMultipolygon()) {
+               var mh = h ? h.members.map(function (m) {
+                 return m.id;
+               }) : [];
+               var mb = b ? b.members.map(function (m) {
+                 return m.id;
+               }) : [];
+               var ids = utilArrayUnion(mh, mb);
+
+               for (i = 0; i < ids.length; i++) {
+                 var member = head.hasEntity(ids[i]);
+                 if (!member) continue; // not downloaded
+
+                 if (extent && !member.intersects(extent, head)) continue; // not visible
+
+                 result[ids[i]] = member;
+               }
+             }
+
+             addParents(head.parentWays(entity), result);
+             addParents(head.parentRelations(entity), result);
+           }
+
+           return result;
+
+           function addParents(parents, result) {
+             for (var i = 0; i < parents.length; i++) {
+               var parent = parents[i];
+               if (parent.id in result) continue;
+               result[parent.id] = parent;
+               addParents(head.parentRelations(parent), result);
+             }
+           }
+         };
 
-       function readPackedEnd(pbf) {
-         return pbf.type === Pbf.Bytes ? pbf.readVarint() + pbf.pos : pbf.pos + 1;
+         return _diff;
        }
 
-       function toNum(low, high, isSigned) {
-         if (isSigned) {
-           return high * 0x100000000 + (low >>> 0);
-         }
+       function coreTree(head) {
+         // tree for entities
+         var _rtree = new RBush();
 
-         return (high >>> 0) * 0x100000000 + (low >>> 0);
-       }
+         var _bboxes = {}; // maintain a separate tree for granular way segments
 
-       function writeBigVarint(val, pbf) {
-         var low, high;
+         var _segmentsRTree = new RBush();
 
-         if (val >= 0) {
-           low = val % 0x100000000 | 0;
-           high = val / 0x100000000 | 0;
-         } else {
-           low = ~(-val % 0x100000000);
-           high = ~(-val / 0x100000000);
+         var _segmentsBBoxes = {};
+         var _segmentsByWayId = {};
+         var tree = {};
 
-           if (low ^ 0xffffffff) {
-             low = low + 1 | 0;
-           } else {
-             low = 0;
-             high = high + 1 | 0;
-           }
+         function entityBBox(entity) {
+           var bbox = entity.extent(head).bbox();
+           bbox.id = entity.id;
+           _bboxes[entity.id] = bbox;
+           return bbox;
          }
 
-         if (val >= 0x10000000000000000 || val < -0x10000000000000000) {
-           throw new Error('Given varint doesn\'t fit into 10 bytes');
-         }
+         function segmentBBox(segment) {
+           var extent = segment.extent(head); // extent can be null if the node entities aren't in the graph for some reason
 
-         pbf.realloc(10);
-         writeBigVarintLow(low, high, pbf);
-         writeBigVarintHigh(high, pbf);
-       }
+           if (!extent) return null;
+           var bbox = extent.bbox();
+           bbox.segment = segment;
+           _segmentsBBoxes[segment.id] = bbox;
+           return bbox;
+         }
 
-       function writeBigVarintLow(low, high, pbf) {
-         pbf.buf[pbf.pos++] = low & 0x7f | 0x80;
-         low >>>= 7;
-         pbf.buf[pbf.pos++] = low & 0x7f | 0x80;
-         low >>>= 7;
-         pbf.buf[pbf.pos++] = low & 0x7f | 0x80;
-         low >>>= 7;
-         pbf.buf[pbf.pos++] = low & 0x7f | 0x80;
-         low >>>= 7;
-         pbf.buf[pbf.pos] = low & 0x7f;
-       }
+         function removeEntity(entity) {
+           _rtree.remove(_bboxes[entity.id]);
 
-       function writeBigVarintHigh(high, pbf) {
-         var lsb = (high & 0x07) << 4;
-         pbf.buf[pbf.pos++] |= lsb | ((high >>>= 3) ? 0x80 : 0);
-         if (!high) return;
-         pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0);
-         if (!high) return;
-         pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0);
-         if (!high) return;
-         pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0);
-         if (!high) return;
-         pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0);
-         if (!high) return;
-         pbf.buf[pbf.pos++] = high & 0x7f;
-       }
+           delete _bboxes[entity.id];
 
-       function makeRoomForExtraLength(startPos, len, pbf) {
-         var extraLen = len <= 0x3fff ? 1 : len <= 0x1fffff ? 2 : len <= 0xfffffff ? 3 : Math.floor(Math.log(len) / (Math.LN2 * 7)); // if 1 byte isn't enough for encoding message length, shift the data to the right
+           if (_segmentsByWayId[entity.id]) {
+             _segmentsByWayId[entity.id].forEach(function (segment) {
+               _segmentsRTree.remove(_segmentsBBoxes[segment.id]);
 
-         pbf.realloc(extraLen);
+               delete _segmentsBBoxes[segment.id];
+             });
 
-         for (var i = pbf.pos - 1; i >= startPos; i--) {
-           pbf.buf[i + extraLen] = pbf.buf[i];
+             delete _segmentsByWayId[entity.id];
+           }
          }
-       }
 
-       function _writePackedVarint(arr, pbf) {
-         for (var i = 0; i < arr.length; i++) {
-           pbf.writeVarint(arr[i]);
-         }
-       }
+         function loadEntities(entities) {
+           _rtree.load(entities.map(entityBBox));
 
-       function _writePackedSVarint(arr, pbf) {
-         for (var i = 0; i < arr.length; i++) {
-           pbf.writeSVarint(arr[i]);
+           var segments = [];
+           entities.forEach(function (entity) {
+             if (entity.segments) {
+               var entitySegments = entity.segments(head); // cache these to make them easy to remove later
+
+               _segmentsByWayId[entity.id] = entitySegments;
+               segments = segments.concat(entitySegments);
+             }
+           });
+           if (segments.length) _segmentsRTree.load(segments.map(segmentBBox).filter(Boolean));
          }
-       }
 
-       function _writePackedFloat(arr, pbf) {
-         for (var i = 0; i < arr.length; i++) {
-           pbf.writeFloat(arr[i]);
+         function updateParents(entity, insertions, memo) {
+           head.parentWays(entity).forEach(function (way) {
+             if (_bboxes[way.id]) {
+               removeEntity(way);
+               insertions[way.id] = way;
+             }
+
+             updateParents(way, insertions, memo);
+           });
+           head.parentRelations(entity).forEach(function (relation) {
+             if (memo[entity.id]) return;
+             memo[entity.id] = true;
+
+             if (_bboxes[relation.id]) {
+               removeEntity(relation);
+               insertions[relation.id] = relation;
+             }
+
+             updateParents(relation, insertions, memo);
+           });
          }
+
+         tree.rebase = function (entities, force) {
+           var insertions = {};
+
+           for (var i = 0; i < entities.length; i++) {
+             var entity = entities[i];
+             if (!entity.visible) continue;
+
+             if (head.entities.hasOwnProperty(entity.id) || _bboxes[entity.id]) {
+               if (!force) {
+                 continue;
+               } else if (_bboxes[entity.id]) {
+                 removeEntity(entity);
+               }
+             }
+
+             insertions[entity.id] = entity;
+             updateParents(entity, insertions, {});
+           }
+
+           loadEntities(Object.values(insertions));
+           return tree;
+         };
+
+         function updateToGraph(graph) {
+           if (graph === head) return;
+           var diff = coreDifference(head, graph);
+           head = graph;
+           var changed = diff.didChange;
+           if (!changed.addition && !changed.deletion && !changed.geometry) return;
+           var insertions = {};
+
+           if (changed.deletion) {
+             diff.deleted().forEach(function (entity) {
+               removeEntity(entity);
+             });
+           }
+
+           if (changed.geometry) {
+             diff.modified().forEach(function (entity) {
+               removeEntity(entity);
+               insertions[entity.id] = entity;
+               updateParents(entity, insertions, {});
+             });
+           }
+
+           if (changed.addition) {
+             diff.created().forEach(function (entity) {
+               insertions[entity.id] = entity;
+             });
+           }
+
+           loadEntities(Object.values(insertions));
+         } // returns an array of entities with bounding boxes overlapping `extent` for the given `graph`
+
+
+         tree.intersects = function (extent, graph) {
+           updateToGraph(graph);
+           return _rtree.search(extent.bbox()).map(function (bbox) {
+             return graph.entity(bbox.id);
+           });
+         }; // returns an array of segment objects with bounding boxes overlapping `extent` for the given `graph`
+
+
+         tree.waySegments = function (extent, graph) {
+           updateToGraph(graph);
+           return _segmentsRTree.search(extent.bbox()).map(function (bbox) {
+             return bbox.segment;
+           });
+         };
+
+         return tree;
        }
 
-       function _writePackedDouble(arr, pbf) {
-         for (var i = 0; i < arr.length; i++) {
-           pbf.writeDouble(arr[i]);
-         }
+       function svgIcon(name, svgklass, useklass) {
+         return function drawIcon(selection) {
+           selection.selectAll('svg.icon' + (svgklass ? '.' + svgklass.split(' ')[0] : '')).data([0]).enter().append('svg').attr('class', 'icon ' + (svgklass || '')).append('use').attr('xlink:href', name).attr('class', useklass);
+         };
        }
 
-       function _writePackedBoolean(arr, pbf) {
-         for (var i = 0; i < arr.length; i++) {
-           pbf.writeBoolean(arr[i]);
+       function uiModal(selection, blocking) {
+         var _this = this;
+
+         var keybinding = utilKeybinding('modal');
+         var previous = selection.select('div.modal');
+         var animate = previous.empty();
+         previous.transition().duration(200).style('opacity', 0).remove();
+         var shaded = selection.append('div').attr('class', 'shaded').style('opacity', 0);
+
+         shaded.close = function () {
+           shaded.transition().duration(200).style('opacity', 0).remove();
+           modal.transition().duration(200).style('top', '0px');
+           select(document).call(keybinding.unbind);
+         };
+
+         var modal = shaded.append('div').attr('class', 'modal fillL');
+         modal.append('input').attr('class', 'keytrap keytrap-first').on('focus.keytrap', moveFocusToLast);
+
+         if (!blocking) {
+           shaded.on('click.remove-modal', function (d3_event) {
+             if (d3_event.target === _this) {
+               shaded.close();
+             }
+           });
+           modal.append('button').attr('class', 'close').on('click', shaded.close).call(svgIcon('#iD-icon-close'));
+           keybinding.on('⌫', shaded.close).on('⎋', shaded.close);
+           select(document).call(keybinding);
          }
-       }
 
-       function _writePackedFixed(arr, pbf) {
-         for (var i = 0; i < arr.length; i++) {
-           pbf.writeFixed32(arr[i]);
+         modal.append('div').attr('class', 'content');
+         modal.append('input').attr('class', 'keytrap keytrap-last').on('focus.keytrap', moveFocusToFirst);
+
+         if (animate) {
+           shaded.transition().style('opacity', 1);
+         } else {
+           shaded.style('opacity', 1);
          }
-       }
 
-       function _writePackedSFixed(arr, pbf) {
-         for (var i = 0; i < arr.length; i++) {
-           pbf.writeSFixed32(arr[i]);
+         return shaded;
+
+         function moveFocusToFirst() {
+           var node = modal // there are additional rules about what's focusable, but this suits our purposes
+           .select('a, button, input:not(.keytrap), select, textarea').node();
+
+           if (node) {
+             node.focus();
+           } else {
+             select(this).node().blur();
+           }
          }
-       }
 
-       function _writePackedFixed2(arr, pbf) {
-         for (var i = 0; i < arr.length; i++) {
-           pbf.writeFixed64(arr[i]);
+         function moveFocusToLast() {
+           var nodes = modal.selectAll('a, button, input:not(.keytrap), select, textarea').nodes();
+
+           if (nodes.length) {
+             nodes[nodes.length - 1].focus();
+           } else {
+             select(this).node().blur();
+           }
          }
        }
 
-       function _writePackedSFixed2(arr, pbf) {
-         for (var i = 0; i < arr.length; i++) {
-           pbf.writeSFixed64(arr[i]);
-         }
-       } // Buffer code below from https://github.com/feross/buffer, MIT-licensed
+       function uiLoading(context) {
+         var _modalSelection = select(null);
 
+         var _message = '';
+         var _blocking = false;
 
-       function readUInt32(buf, pos) {
-         return (buf[pos] | buf[pos + 1] << 8 | buf[pos + 2] << 16) + buf[pos + 3] * 0x1000000;
-       }
+         var loading = function loading(selection) {
+           _modalSelection = uiModal(selection, _blocking);
 
-       function writeInt32(buf, val, pos) {
-         buf[pos] = val;
-         buf[pos + 1] = val >>> 8;
-         buf[pos + 2] = val >>> 16;
-         buf[pos + 3] = val >>> 24;
-       }
+           var loadertext = _modalSelection.select('.content').classed('loading-modal', true).append('div').attr('class', 'modal-section fillL');
 
-       function readInt32(buf, pos) {
-         return (buf[pos] | buf[pos + 1] << 8 | buf[pos + 2] << 16) + (buf[pos + 3] << 24);
+           loadertext.append('img').attr('class', 'loader').attr('src', context.imagePath('loader-white.gif'));
+           loadertext.append('h3').html(_message);
+
+           _modalSelection.select('button.close').attr('class', 'hide');
+
+           return loading;
+         };
+
+         loading.message = function (val) {
+           if (!arguments.length) return _message;
+           _message = val;
+           return loading;
+         };
+
+         loading.blocking = function (val) {
+           if (!arguments.length) return _blocking;
+           _blocking = val;
+           return loading;
+         };
+
+         loading.close = function () {
+           _modalSelection.remove();
+         };
+
+         loading.isShown = function () {
+           return _modalSelection && !_modalSelection.empty() && _modalSelection.node().parentNode;
+         };
+
+         return loading;
        }
 
-       function readUtf8(buf, pos, end) {
-         var str = '';
-         var i = pos;
+       function coreHistory(context) {
+         var dispatch = dispatch$8('reset', 'change', 'merge', 'restore', 'undone', 'redone');
 
-         while (i < end) {
-           var b0 = buf[i];
-           var c = null; // codepoint
+         var _lock = utilSessionMutex('lock'); // restorable if iD not open in another window/tab and a saved history exists in localStorage
 
-           var bytesPerSequence = b0 > 0xEF ? 4 : b0 > 0xDF ? 3 : b0 > 0xBF ? 2 : 1;
-           if (i + bytesPerSequence > end) break;
-           var b1, b2, b3;
 
-           if (bytesPerSequence === 1) {
-             if (b0 < 0x80) {
-               c = b0;
-             }
-           } else if (bytesPerSequence === 2) {
-             b1 = buf[i + 1];
+         var _hasUnresolvedRestorableChanges = _lock.lock() && !!corePreferences(getKey('saved_history'));
 
-             if ((b1 & 0xC0) === 0x80) {
-               c = (b0 & 0x1F) << 0x6 | b1 & 0x3F;
+         var duration = 150;
+         var _imageryUsed = [];
+         var _photoOverlaysUsed = [];
+         var _checkpoints = {};
 
-               if (c <= 0x7F) {
-                 c = null;
-               }
-             }
-           } else if (bytesPerSequence === 3) {
-             b1 = buf[i + 1];
-             b2 = buf[i + 2];
+         var _pausedGraph;
 
-             if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80) {
-               c = (b0 & 0xF) << 0xC | (b1 & 0x3F) << 0x6 | b2 & 0x3F;
+         var _stack;
 
-               if (c <= 0x7FF || c >= 0xD800 && c <= 0xDFFF) {
-                 c = null;
-               }
-             }
-           } else if (bytesPerSequence === 4) {
-             b1 = buf[i + 1];
-             b2 = buf[i + 2];
-             b3 = buf[i + 3];
+         var _index;
 
-             if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80 && (b3 & 0xC0) === 0x80) {
-               c = (b0 & 0xF) << 0x12 | (b1 & 0x3F) << 0xC | (b2 & 0x3F) << 0x6 | b3 & 0x3F;
+         var _tree; // internal _act, accepts list of actions and eased time
 
-               if (c <= 0xFFFF || c >= 0x110000) {
-                 c = null;
-               }
-             }
+
+         function _act(actions, t) {
+           actions = Array.prototype.slice.call(actions);
+           var annotation;
+
+           if (typeof actions[actions.length - 1] !== 'function') {
+             annotation = actions.pop();
            }
 
-           if (c === null) {
-             c = 0xFFFD;
-             bytesPerSequence = 1;
-           } else if (c > 0xFFFF) {
-             c -= 0x10000;
-             str += String.fromCharCode(c >>> 10 & 0x3FF | 0xD800);
-             c = 0xDC00 | c & 0x3FF;
+           var graph = _stack[_index].graph;
+
+           for (var i = 0; i < actions.length; i++) {
+             graph = actions[i](graph, t);
            }
 
-           str += String.fromCharCode(c);
-           i += bytesPerSequence;
-         }
+           return {
+             graph: graph,
+             annotation: annotation,
+             imageryUsed: _imageryUsed,
+             photoOverlaysUsed: _photoOverlaysUsed,
+             transform: context.projection.transform(),
+             selectedIDs: context.selectedIDs()
+           };
+         } // internal _perform with eased time
 
-         return str;
-       }
 
-       function readUtf8TextDecoder(buf, pos, end) {
-         return utf8TextDecoder.decode(buf.subarray(pos, end));
-       }
+         function _perform(args, t) {
+           var previous = _stack[_index].graph;
+           _stack = _stack.slice(0, _index + 1);
 
-       function writeUtf8(buf, str, pos) {
-         for (var i = 0, c, lead; i < str.length; i++) {
-           c = str.charCodeAt(i); // code point
+           var actionResult = _act(args, t);
 
-           if (c > 0xD7FF && c < 0xE000) {
-             if (lead) {
-               if (c < 0xDC00) {
-                 buf[pos++] = 0xEF;
-                 buf[pos++] = 0xBF;
-                 buf[pos++] = 0xBD;
-                 lead = c;
-                 continue;
-               } else {
-                 c = lead - 0xD800 << 10 | c - 0xDC00 | 0x10000;
-                 lead = null;
-               }
-             } else {
-               if (c > 0xDBFF || i + 1 === str.length) {
-                 buf[pos++] = 0xEF;
-                 buf[pos++] = 0xBF;
-                 buf[pos++] = 0xBD;
-               } else {
-                 lead = c;
-               }
+           _stack.push(actionResult);
+
+           _index++;
+           return change(previous);
+         } // internal _replace with eased time
+
+
+         function _replace(args, t) {
+           var previous = _stack[_index].graph; // assert(_index == _stack.length - 1)
+
+           var actionResult = _act(args, t);
+
+           _stack[_index] = actionResult;
+           return change(previous);
+         } // internal _overwrite with eased time
+
+
+         function _overwrite(args, t) {
+           var previous = _stack[_index].graph;
+
+           if (_index > 0) {
+             _index--;
+
+             _stack.pop();
+           }
+
+           _stack = _stack.slice(0, _index + 1);
+
+           var actionResult = _act(args, t);
+
+           _stack.push(actionResult);
+
+           _index++;
+           return change(previous);
+         } // determine difference and dispatch a change event
+
+
+         function change(previous) {
+           var difference = coreDifference(previous, history.graph());
+
+           if (!_pausedGraph) {
+             dispatch.call('change', this, difference);
+           }
+
+           return difference;
+         } // iD uses namespaced keys so multiple installations do not conflict
+
+
+         function getKey(n) {
+           return 'iD_' + window.location.origin + '_' + n;
+         }
+
+         var history = {
+           graph: function graph() {
+             return _stack[_index].graph;
+           },
+           tree: function tree() {
+             return _tree;
+           },
+           base: function base() {
+             return _stack[0].graph;
+           },
+           merge: function merge(entities
+           /*, extent*/
+           ) {
+             var stack = _stack.map(function (state) {
+               return state.graph;
+             });
 
-               continue;
-             }
-           } else if (lead) {
-             buf[pos++] = 0xEF;
-             buf[pos++] = 0xBF;
-             buf[pos++] = 0xBD;
-             lead = null;
-           }
+             _stack[0].graph.rebase(entities, stack, false);
 
-           if (c < 0x80) {
-             buf[pos++] = c;
-           } else {
-             if (c < 0x800) {
-               buf[pos++] = c >> 0x6 | 0xC0;
-             } else {
-               if (c < 0x10000) {
-                 buf[pos++] = c >> 0xC | 0xE0;
-               } else {
-                 buf[pos++] = c >> 0x12 | 0xF0;
-                 buf[pos++] = c >> 0xC & 0x3F | 0x80;
-               }
+             _tree.rebase(entities, false);
 
-               buf[pos++] = c >> 0x6 & 0x3F | 0x80;
+             dispatch.call('merge', this, entities);
+           },
+           perform: function perform() {
+             // complete any transition already in progress
+             select(document).interrupt('history.perform');
+             var transitionable = false;
+             var action0 = arguments[0];
+
+             if (arguments.length === 1 || arguments.length === 2 && typeof arguments[1] !== 'function') {
+               transitionable = !!action0.transitionable;
              }
 
-             buf[pos++] = c & 0x3F | 0x80;
-           }
-         }
+             if (transitionable) {
+               var origArguments = arguments;
+               select(document).transition('history.perform').duration(duration).ease(linear$1).tween('history.tween', function () {
+                 return function (t) {
+                   if (t < 1) _overwrite([action0], t);
+                 };
+               }).on('start', function () {
+                 _perform([action0], 0);
+               }).on('end interrupt', function () {
+                 _overwrite(origArguments, 1);
+               });
+             } else {
+               return _perform(arguments);
+             }
+           },
+           replace: function replace() {
+             select(document).interrupt('history.perform');
+             return _replace(arguments, 1);
+           },
+           // Same as calling pop and then perform
+           overwrite: function overwrite() {
+             select(document).interrupt('history.perform');
+             return _overwrite(arguments, 1);
+           },
+           pop: function pop(n) {
+             select(document).interrupt('history.perform');
+             var previous = _stack[_index].graph;
 
-         return pos;
-       }
+             if (isNaN(+n) || +n < 0) {
+               n = 1;
+             }
 
-       var pointGeometry = Point;
-       /**
-        * A standalone point geometry with useful accessor, comparison, and
-        * modification methods.
-        *
-        * @class Point
-        * @param {Number} x the x-coordinate. this could be longitude or screen
-        * pixels, or any other sort of unit.
-        * @param {Number} y the y-coordinate. this could be latitude or screen
-        * pixels, or any other sort of unit.
-        * @example
-        * var point = new Point(-77, 38);
-        */
+             while (n-- > 0 && _index > 0) {
+               _index--;
 
-       function Point(x, y) {
-         this.x = x;
-         this.y = y;
-       }
+               _stack.pop();
+             }
 
-       Point.prototype = {
-         /**
-          * Clone this point, returning a new point that can be modified
-          * without affecting the old one.
-          * @return {Point} the clone
-          */
-         clone: function clone() {
-           return new Point(this.x, this.y);
-         },
+             return change(previous);
+           },
+           // Back to the previous annotated state or _index = 0.
+           undo: function undo() {
+             select(document).interrupt('history.perform');
+             var previousStack = _stack[_index];
+             var previous = previousStack.graph;
 
-         /**
-          * Add this point's x & y coordinates to another point,
-          * yielding a new point.
-          * @param {Point} p the other point
-          * @return {Point} output point
-          */
-         add: function add(p) {
-           return this.clone()._add(p);
-         },
+             while (_index > 0) {
+               _index--;
+               if (_stack[_index].annotation) break;
+             }
 
-         /**
-          * Subtract this point's x & y coordinates to from point,
-          * yielding a new point.
-          * @param {Point} p the other point
-          * @return {Point} output point
-          */
-         sub: function sub(p) {
-           return this.clone()._sub(p);
-         },
+             dispatch.call('undone', this, _stack[_index], previousStack);
+             return change(previous);
+           },
+           // Forward to the next annotated state.
+           redo: function redo() {
+             select(document).interrupt('history.perform');
+             var previousStack = _stack[_index];
+             var previous = previousStack.graph;
+             var tryIndex = _index;
 
-         /**
-          * Multiply this point's x & y coordinates by point,
-          * yielding a new point.
-          * @param {Point} p the other point
-          * @return {Point} output point
-          */
-         multByPoint: function multByPoint(p) {
-           return this.clone()._multByPoint(p);
-         },
+             while (tryIndex < _stack.length - 1) {
+               tryIndex++;
 
-         /**
-          * Divide this point's x & y coordinates by point,
-          * yielding a new point.
-          * @param {Point} p the other point
-          * @return {Point} output point
-          */
-         divByPoint: function divByPoint(p) {
-           return this.clone()._divByPoint(p);
-         },
+               if (_stack[tryIndex].annotation) {
+                 _index = tryIndex;
+                 dispatch.call('redone', this, _stack[_index], previousStack);
+                 break;
+               }
+             }
 
-         /**
-          * Multiply this point's x & y coordinates by a factor,
-          * yielding a new point.
-          * @param {Point} k factor
-          * @return {Point} output point
-          */
-         mult: function mult(k) {
-           return this.clone()._mult(k);
-         },
+             return change(previous);
+           },
+           pauseChangeDispatch: function pauseChangeDispatch() {
+             if (!_pausedGraph) {
+               _pausedGraph = _stack[_index].graph;
+             }
+           },
+           resumeChangeDispatch: function resumeChangeDispatch() {
+             if (_pausedGraph) {
+               var previous = _pausedGraph;
+               _pausedGraph = null;
+               return change(previous);
+             }
+           },
+           undoAnnotation: function undoAnnotation() {
+             var i = _index;
 
-         /**
-          * Divide this point's x & y coordinates by a factor,
-          * yielding a new point.
-          * @param {Point} k factor
-          * @return {Point} output point
-          */
-         div: function div(k) {
-           return this.clone()._div(k);
-         },
+             while (i >= 0) {
+               if (_stack[i].annotation) return _stack[i].annotation;
+               i--;
+             }
+           },
+           redoAnnotation: function redoAnnotation() {
+             var i = _index + 1;
 
-         /**
-          * Rotate this point around the 0, 0 origin by an angle a,
-          * given in radians
-          * @param {Number} a angle to rotate around, in radians
-          * @return {Point} output point
-          */
-         rotate: function rotate(a) {
-           return this.clone()._rotate(a);
-         },
+             while (i <= _stack.length - 1) {
+               if (_stack[i].annotation) return _stack[i].annotation;
+               i++;
+             }
+           },
+           // Returns the entities from the active graph with bounding boxes
+           // overlapping the given `extent`.
+           intersects: function intersects(extent) {
+             return _tree.intersects(extent, _stack[_index].graph);
+           },
+           difference: function difference() {
+             var base = _stack[0].graph;
+             var head = _stack[_index].graph;
+             return coreDifference(base, head);
+           },
+           changes: function changes(action) {
+             var base = _stack[0].graph;
+             var head = _stack[_index].graph;
 
-         /**
-          * Rotate this point around p point by an angle a,
-          * given in radians
-          * @param {Number} a angle to rotate around, in radians
-          * @param {Point} p Point to rotate around
-          * @return {Point} output point
-          */
-         rotateAround: function rotateAround(a, p) {
-           return this.clone()._rotateAround(a, p);
-         },
+             if (action) {
+               head = action(head);
+             }
 
-         /**
-          * Multiply this point by a 4x1 transformation matrix
-          * @param {Array<Number>} m transformation matrix
-          * @return {Point} output point
-          */
-         matMult: function matMult(m) {
-           return this.clone()._matMult(m);
-         },
+             var difference = coreDifference(base, head);
+             return {
+               modified: difference.modified(),
+               created: difference.created(),
+               deleted: difference.deleted()
+             };
+           },
+           hasChanges: function hasChanges() {
+             return this.difference().length() > 0;
+           },
+           imageryUsed: function imageryUsed(sources) {
+             if (sources) {
+               _imageryUsed = sources;
+               return history;
+             } else {
+               var s = new Set();
 
-         /**
-          * Calculate this point but as a unit vector from 0, 0, meaning
-          * that the distance from the resulting point to the 0, 0
-          * coordinate will be equal to 1 and the angle from the resulting
-          * point to the 0, 0 coordinate will be the same as before.
-          * @return {Point} unit vector point
-          */
-         unit: function unit() {
-           return this.clone()._unit();
-         },
+               _stack.slice(1, _index + 1).forEach(function (state) {
+                 state.imageryUsed.forEach(function (source) {
+                   if (source !== 'Custom') {
+                     s.add(source);
+                   }
+                 });
+               });
 
-         /**
-          * Compute a perpendicular point, where the new y coordinate
-          * is the old x coordinate and the new x coordinate is the old y
-          * coordinate multiplied by -1
-          * @return {Point} perpendicular point
-          */
-         perp: function perp() {
-           return this.clone()._perp();
-         },
+               return Array.from(s);
+             }
+           },
+           photoOverlaysUsed: function photoOverlaysUsed(sources) {
+             if (sources) {
+               _photoOverlaysUsed = sources;
+               return history;
+             } else {
+               var s = new Set();
 
-         /**
-          * Return a version of this point with the x & y coordinates
-          * rounded to integers.
-          * @return {Point} rounded point
-          */
-         round: function round() {
-           return this.clone()._round();
-         },
+               _stack.slice(1, _index + 1).forEach(function (state) {
+                 if (state.photoOverlaysUsed && Array.isArray(state.photoOverlaysUsed)) {
+                   state.photoOverlaysUsed.forEach(function (photoOverlay) {
+                     s.add(photoOverlay);
+                   });
+                 }
+               });
 
-         /**
-          * Return the magitude of this point: this is the Euclidean
-          * distance from the 0, 0 coordinate to this point's x and y
-          * coordinates.
-          * @return {Number} magnitude
-          */
-         mag: function mag() {
-           return Math.sqrt(this.x * this.x + this.y * this.y);
-         },
+               return Array.from(s);
+             }
+           },
+           // save the current history state
+           checkpoint: function checkpoint(key) {
+             _checkpoints[key] = {
+               stack: _stack,
+               index: _index
+             };
+             return history;
+           },
+           // restore history state to a given checkpoint or reset completely
+           reset: function reset(key) {
+             if (key !== undefined && _checkpoints.hasOwnProperty(key)) {
+               _stack = _checkpoints[key].stack;
+               _index = _checkpoints[key].index;
+             } else {
+               _stack = [{
+                 graph: coreGraph()
+               }];
+               _index = 0;
+               _tree = coreTree(_stack[0].graph);
+               _checkpoints = {};
+             }
 
-         /**
-          * Judge whether this point is equal to another point, returning
-          * true or false.
-          * @param {Point} other the other point
-          * @return {boolean} whether the points are equal
-          */
-         equals: function equals(other) {
-           return this.x === other.x && this.y === other.y;
-         },
+             dispatch.call('reset');
+             dispatch.call('change');
+             return history;
+           },
+           // `toIntroGraph()` is used to export the intro graph used by the walkthrough.
+           //
+           // To use it:
+           //  1. Start the walkthrough.
+           //  2. Get to a "free editing" tutorial step
+           //  3. Make your edits to the walkthrough map
+           //  4. In your browser dev console run:
+           //        `id.history().toIntroGraph()`
+           //  5. This outputs stringified JSON to the browser console
+           //  6. Copy it to `data/intro_graph.json` and prettify it in your code editor
+           toIntroGraph: function toIntroGraph() {
+             var nextID = {
+               n: 0,
+               r: 0,
+               w: 0
+             };
+             var permIDs = {};
+             var graph = this.graph();
+             var baseEntities = {}; // clone base entities..
 
-         /**
-          * Calculate the distance from this point to another point
-          * @param {Point} p the other point
-          * @return {Number} distance
-          */
-         dist: function dist(p) {
-           return Math.sqrt(this.distSqr(p));
-         },
+             Object.values(graph.base().entities).forEach(function (entity) {
+               var copy = copyIntroEntity(entity);
+               baseEntities[copy.id] = copy;
+             }); // replace base entities with head entities..
 
-         /**
-          * Calculate the distance from this point to another point,
-          * without the square root step. Useful if you're comparing
-          * relative distances.
-          * @param {Point} p the other point
-          * @return {Number} distance
-          */
-         distSqr: function distSqr(p) {
-           var dx = p.x - this.x,
-               dy = p.y - this.y;
-           return dx * dx + dy * dy;
-         },
+             Object.keys(graph.entities).forEach(function (id) {
+               var entity = graph.entities[id];
 
-         /**
-          * Get the angle from the 0, 0 coordinate to this point, in radians
-          * coordinates.
-          * @return {Number} angle
-          */
-         angle: function angle() {
-           return Math.atan2(this.y, this.x);
-         },
+               if (entity) {
+                 var copy = copyIntroEntity(entity);
+                 baseEntities[copy.id] = copy;
+               } else {
+                 delete baseEntities[id];
+               }
+             }); // swap temporary for permanent ids..
 
-         /**
-          * Get the angle from this point to another point, in radians
-          * @param {Point} b the other point
-          * @return {Number} angle
-          */
-         angleTo: function angleTo(b) {
-           return Math.atan2(this.y - b.y, this.x - b.x);
-         },
+             Object.values(baseEntities).forEach(function (entity) {
+               if (Array.isArray(entity.nodes)) {
+                 entity.nodes = entity.nodes.map(function (node) {
+                   return permIDs[node] || node;
+                 });
+               }
 
-         /**
-          * Get the angle between this point and another point, in radians
-          * @param {Point} b the other point
-          * @return {Number} angle
-          */
-         angleWith: function angleWith(b) {
-           return this.angleWithSep(b.x, b.y);
-         },
+               if (Array.isArray(entity.members)) {
+                 entity.members = entity.members.map(function (member) {
+                   member.id = permIDs[member.id] || member.id;
+                   return member;
+                 });
+               }
+             });
+             return JSON.stringify({
+               dataIntroGraph: baseEntities
+             });
 
-         /*
-          * Find the angle of the two vectors, solving the formula for
-          * the cross product a x b = |a||b|sin(θ) for θ.
-          * @param {Number} x the x-coordinate
-          * @param {Number} y the y-coordinate
-          * @return {Number} the angle in radians
-          */
-         angleWithSep: function angleWithSep(x, y) {
-           return Math.atan2(this.x * y - this.y * x, this.x * x + this.y * y);
-         },
-         _matMult: function _matMult(m) {
-           var x = m[0] * this.x + m[1] * this.y,
-               y = m[2] * this.x + m[3] * this.y;
-           this.x = x;
-           this.y = y;
-           return this;
-         },
-         _add: function _add(p) {
-           this.x += p.x;
-           this.y += p.y;
-           return this;
-         },
-         _sub: function _sub(p) {
-           this.x -= p.x;
-           this.y -= p.y;
-           return this;
-         },
-         _mult: function _mult(k) {
-           this.x *= k;
-           this.y *= k;
-           return this;
-         },
-         _div: function _div(k) {
-           this.x /= k;
-           this.y /= k;
-           return this;
-         },
-         _multByPoint: function _multByPoint(p) {
-           this.x *= p.x;
-           this.y *= p.y;
-           return this;
-         },
-         _divByPoint: function _divByPoint(p) {
-           this.x /= p.x;
-           this.y /= p.y;
-           return this;
-         },
-         _unit: function _unit() {
-           this._div(this.mag());
+             function copyIntroEntity(source) {
+               var copy = utilObjectOmit(source, ['type', 'user', 'v', 'version', 'visible']); // Note: the copy is no longer an osmEntity, so it might not have `tags`
 
-           return this;
-         },
-         _perp: function _perp() {
-           var y = this.y;
-           this.y = this.x;
-           this.x = -y;
-           return this;
-         },
-         _rotate: function _rotate(angle) {
-           var cos = Math.cos(angle),
-               sin = Math.sin(angle),
-               x = cos * this.x - sin * this.y,
-               y = sin * this.x + cos * this.y;
-           this.x = x;
-           this.y = y;
-           return this;
-         },
-         _rotateAround: function _rotateAround(angle, p) {
-           var cos = Math.cos(angle),
-               sin = Math.sin(angle),
-               x = p.x + cos * (this.x - p.x) - sin * (this.y - p.y),
-               y = p.y + sin * (this.x - p.x) + cos * (this.y - p.y);
-           this.x = x;
-           this.y = y;
-           return this;
-         },
-         _round: function _round() {
-           this.x = Math.round(this.x);
-           this.y = Math.round(this.y);
-           return this;
-         }
-       };
-       /**
-        * Construct a point from an array if necessary, otherwise if the input
-        * is already a Point, or an unknown type, return it unchanged
-        * @param {Array<Number>|Point|*} a any kind of input value
-        * @return {Point} constructed point, or passed-through value.
-        * @example
-        * // this
-        * var point = Point.convert([0, 1]);
-        * // is equivalent to
-        * var point = new Point(0, 1);
-        */
+               if (copy.tags && !Object.keys(copy.tags)) {
+                 delete copy.tags;
+               }
 
-       Point.convert = function (a) {
-         if (a instanceof Point) {
-           return a;
-         }
+               if (Array.isArray(copy.loc)) {
+                 copy.loc[0] = +copy.loc[0].toFixed(6);
+                 copy.loc[1] = +copy.loc[1].toFixed(6);
+               }
 
-         if (Array.isArray(a)) {
-           return new Point(a[0], a[1]);
-         }
+               var match = source.id.match(/([nrw])-\d*/); // temporary id
 
-         return a;
-       };
+               if (match !== null) {
+                 var nrw = match[1];
+                 var permID;
 
-       var vectortilefeature = VectorTileFeature;
+                 do {
+                   permID = nrw + ++nextID[nrw];
+                 } while (baseEntities.hasOwnProperty(permID));
 
-       function VectorTileFeature(pbf, end, extent, keys, values) {
-         // Public
-         this.properties = {};
-         this.extent = extent;
-         this.type = 0; // Private
+                 copy.id = permIDs[source.id] = permID;
+               }
 
-         this._pbf = pbf;
-         this._geometry = -1;
-         this._keys = keys;
-         this._values = values;
-         pbf.readFields(readFeature, this, end);
-       }
+               return copy;
+             }
+           },
+           toJSON: function toJSON() {
+             if (!this.hasChanges()) return;
+             var allEntities = {};
+             var baseEntities = {};
+             var base = _stack[0];
 
-       function readFeature(tag, feature, pbf) {
-         if (tag == 1) feature.id = pbf.readVarint();else if (tag == 2) readTag(pbf, feature);else if (tag == 3) feature.type = pbf.readVarint();else if (tag == 4) feature._geometry = pbf.pos;
-       }
+             var s = _stack.map(function (i) {
+               var modified = [];
+               var deleted = [];
+               Object.keys(i.graph.entities).forEach(function (id) {
+                 var entity = i.graph.entities[id];
 
-       function readTag(pbf, feature) {
-         var end = pbf.readVarint() + pbf.pos;
+                 if (entity) {
+                   var key = osmEntity.key(entity);
+                   allEntities[key] = entity;
+                   modified.push(key);
+                 } else {
+                   deleted.push(id);
+                 } // make sure that the originals of changed or deleted entities get merged
+                 // into the base of the _stack after restoring the data from JSON.
 
-         while (pbf.pos < end) {
-           var key = feature._keys[pbf.readVarint()],
-               value = feature._values[pbf.readVarint()];
 
-           feature.properties[key] = value;
-         }
-       }
+                 if (id in base.graph.entities) {
+                   baseEntities[id] = base.graph.entities[id];
+                 }
 
-       VectorTileFeature.types = ['Unknown', 'Point', 'LineString', 'Polygon'];
+                 if (entity && entity.nodes) {
+                   // get originals of pre-existing child nodes
+                   entity.nodes.forEach(function (nodeID) {
+                     if (nodeID in base.graph.entities) {
+                       baseEntities[nodeID] = base.graph.entities[nodeID];
+                     }
+                   });
+                 } // get originals of parent entities too
 
-       VectorTileFeature.prototype.loadGeometry = function () {
-         var pbf = this._pbf;
-         pbf.pos = this._geometry;
-         var end = pbf.readVarint() + pbf.pos,
-             cmd = 1,
-             length = 0,
-             x = 0,
-             y = 0,
-             lines = [],
-             line;
 
-         while (pbf.pos < end) {
-           if (length <= 0) {
-             var cmdLen = pbf.readVarint();
-             cmd = cmdLen & 0x7;
-             length = cmdLen >> 3;
-           }
+                 var baseParents = base.graph._parentWays[id];
 
-           length--;
+                 if (baseParents) {
+                   baseParents.forEach(function (parentID) {
+                     if (parentID in base.graph.entities) {
+                       baseEntities[parentID] = base.graph.entities[parentID];
+                     }
+                   });
+                 }
+               });
+               var x = {};
+               if (modified.length) x.modified = modified;
+               if (deleted.length) x.deleted = deleted;
+               if (i.imageryUsed) x.imageryUsed = i.imageryUsed;
+               if (i.photoOverlaysUsed) x.photoOverlaysUsed = i.photoOverlaysUsed;
+               if (i.annotation) x.annotation = i.annotation;
+               if (i.transform) x.transform = i.transform;
+               if (i.selectedIDs) x.selectedIDs = i.selectedIDs;
+               return x;
+             });
 
-           if (cmd === 1 || cmd === 2) {
-             x += pbf.readSVarint();
-             y += pbf.readSVarint();
+             return JSON.stringify({
+               version: 3,
+               entities: Object.values(allEntities),
+               baseEntities: Object.values(baseEntities),
+               stack: s,
+               nextIDs: osmEntity.id.next,
+               index: _index,
+               // note the time the changes were saved
+               timestamp: new Date().getTime()
+             });
+           },
+           fromJSON: function fromJSON(json, loadChildNodes) {
+             var h = JSON.parse(json);
+             var loadComplete = true;
+             osmEntity.id.next = h.nextIDs;
+             _index = h.index;
 
-             if (cmd === 1) {
-               // moveTo
-               if (line) lines.push(line);
-               line = [];
-             }
+             if (h.version === 2 || h.version === 3) {
+               var allEntities = {};
+               h.entities.forEach(function (entity) {
+                 allEntities[osmEntity.key(entity)] = osmEntity(entity);
+               });
 
-             line.push(new pointGeometry(x, y));
-           } else if (cmd === 7) {
-             // Workaround for https://github.com/mapbox/mapnik-vector-tile/issues/90
-             if (line) {
-               line.push(line[0].clone()); // closePolygon
-             }
-           } else {
-             throw new Error('unknown command ' + cmd);
-           }
-         }
+               if (h.version === 3) {
+                 // This merges originals for changed entities into the base of
+                 // the _stack even if the current _stack doesn't have them (for
+                 // example when iD has been restarted in a different region)
+                 var baseEntities = h.baseEntities.map(function (d) {
+                   return osmEntity(d);
+                 });
 
-         if (line) lines.push(line);
-         return lines;
-       };
+                 var stack = _stack.map(function (state) {
+                   return state.graph;
+                 });
 
-       VectorTileFeature.prototype.bbox = function () {
-         var pbf = this._pbf;
-         pbf.pos = this._geometry;
-         var end = pbf.readVarint() + pbf.pos,
-             cmd = 1,
-             length = 0,
-             x = 0,
-             y = 0,
-             x1 = Infinity,
-             x2 = -Infinity,
-             y1 = Infinity,
-             y2 = -Infinity;
+                 _stack[0].graph.rebase(baseEntities, stack, true);
 
-         while (pbf.pos < end) {
-           if (length <= 0) {
-             var cmdLen = pbf.readVarint();
-             cmd = cmdLen & 0x7;
-             length = cmdLen >> 3;
-           }
+                 _tree.rebase(baseEntities, true); // When we restore a modified way, we also need to fetch any missing
+                 // childnodes that would normally have been downloaded with it.. #2142
 
-           length--;
 
-           if (cmd === 1 || cmd === 2) {
-             x += pbf.readSVarint();
-             y += pbf.readSVarint();
-             if (x < x1) x1 = x;
-             if (x > x2) x2 = x;
-             if (y < y1) y1 = y;
-             if (y > y2) y2 = y;
-           } else if (cmd !== 7) {
-             throw new Error('unknown command ' + cmd);
-           }
-         }
+                 if (loadChildNodes) {
+                   var osm = context.connection();
+                   var baseWays = baseEntities.filter(function (e) {
+                     return e.type === 'way';
+                   });
+                   var nodeIDs = baseWays.reduce(function (acc, way) {
+                     return utilArrayUnion(acc, way.nodes);
+                   }, []);
+                   var missing = nodeIDs.filter(function (n) {
+                     return !_stack[0].graph.hasEntity(n);
+                   });
 
-         return [x1, y1, x2, y2];
-       };
+                   if (missing.length && osm) {
+                     loadComplete = false;
+                     context.map().redrawEnable(false);
+                     var loading = uiLoading(context).blocking(true);
+                     context.container().call(loading);
 
-       VectorTileFeature.prototype.toGeoJSON = function (x, y, z) {
-         var size = this.extent * Math.pow(2, z),
-             x0 = this.extent * x,
-             y0 = this.extent * y,
-             coords = this.loadGeometry(),
-             type = VectorTileFeature.types[this.type],
-             i,
-             j;
+                     var childNodesLoaded = function childNodesLoaded(err, result) {
+                       if (!err) {
+                         var visibleGroups = utilArrayGroupBy(result.data, 'visible');
+                         var visibles = visibleGroups["true"] || []; // alive nodes
 
-         function project(line) {
-           for (var j = 0; j < line.length; j++) {
-             var p = line[j],
-                 y2 = 180 - (p.y + y0) * 360 / size;
-             line[j] = [(p.x + x0) * 360 / size - 180, 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90];
-           }
-         }
+                         var invisibles = visibleGroups["false"] || []; // deleted nodes
 
-         switch (this.type) {
-           case 1:
-             var points = [];
+                         if (visibles.length) {
+                           var visibleIDs = visibles.map(function (entity) {
+                             return entity.id;
+                           });
 
-             for (i = 0; i < coords.length; i++) {
-               points[i] = coords[i][0];
-             }
+                           var stack = _stack.map(function (state) {
+                             return state.graph;
+                           });
 
-             coords = points;
-             project(coords);
-             break;
+                           missing = utilArrayDifference(missing, visibleIDs);
 
-           case 2:
-             for (i = 0; i < coords.length; i++) {
-               project(coords[i]);
-             }
+                           _stack[0].graph.rebase(visibles, stack, true);
 
-             break;
+                           _tree.rebase(visibles, true);
+                         } // fetch older versions of nodes that were deleted..
 
-           case 3:
-             coords = classifyRings(coords);
 
-             for (i = 0; i < coords.length; i++) {
-               for (j = 0; j < coords[i].length; j++) {
-                 project(coords[i][j]);
-               }
-             }
+                         invisibles.forEach(function (entity) {
+                           osm.loadEntityVersion(entity.id, +entity.version - 1, childNodesLoaded);
+                         });
+                       }
 
-             break;
-         }
+                       if (err || !missing.length) {
+                         loading.close();
+                         context.map().redrawEnable(true);
+                         dispatch.call('change');
+                         dispatch.call('restore', this);
+                       }
+                     };
 
-         if (coords.length === 1) {
-           coords = coords[0];
-         } else {
-           type = 'Multi' + type;
-         }
+                     osm.loadMultiple(missing, childNodesLoaded);
+                   }
+                 }
+               }
 
-         var result = {
-           type: "Feature",
-           geometry: {
-             type: type,
-             coordinates: coords
-           },
-           properties: this.properties
-         };
+               _stack = h.stack.map(function (d) {
+                 var entities = {},
+                     entity;
 
-         if ('id' in this) {
-           result.id = this.id;
-         }
+                 if (d.modified) {
+                   d.modified.forEach(function (key) {
+                     entity = allEntities[key];
+                     entities[entity.id] = entity;
+                   });
+                 }
 
-         return result;
-       }; // classifies an array of rings into polygons with outer rings and holes
+                 if (d.deleted) {
+                   d.deleted.forEach(function (id) {
+                     entities[id] = undefined;
+                   });
+                 }
 
+                 return {
+                   graph: coreGraph(_stack[0].graph).load(entities),
+                   annotation: d.annotation,
+                   imageryUsed: d.imageryUsed,
+                   photoOverlaysUsed: d.photoOverlaysUsed,
+                   transform: d.transform,
+                   selectedIDs: d.selectedIDs
+                 };
+               });
+             } else {
+               // original version
+               _stack = h.stack.map(function (d) {
+                 var entities = {};
 
-       function classifyRings(rings) {
-         var len = rings.length;
-         if (len <= 1) return [rings];
-         var polygons = [],
-             polygon,
-             ccw;
+                 for (var i in d.entities) {
+                   var entity = d.entities[i];
+                   entities[i] = entity === 'undefined' ? undefined : osmEntity(entity);
+                 }
 
-         for (var i = 0; i < len; i++) {
-           var area = signedArea$1(rings[i]);
-           if (area === 0) continue;
-           if (ccw === undefined) ccw = area < 0;
+                 d.graph = coreGraph(_stack[0].graph).load(entities);
+                 return d;
+               });
+             }
 
-           if (ccw === area < 0) {
-             if (polygon) polygons.push(polygon);
-             polygon = [rings[i]];
-           } else {
-             polygon.push(rings[i]);
-           }
-         }
+             var transform = _stack[_index].transform;
 
-         if (polygon) polygons.push(polygon);
-         return polygons;
-       }
+             if (transform) {
+               context.map().transformEase(transform, 0); // 0 = immediate, no easing
+             }
 
-       function signedArea$1(ring) {
-         var sum = 0;
+             if (loadComplete) {
+               dispatch.call('change');
+               dispatch.call('restore', this);
+             }
 
-         for (var i = 0, len = ring.length, j = len - 1, p1, p2; i < len; j = i++) {
-           p1 = ring[i];
-           p2 = ring[j];
-           sum += (p2.x - p1.x) * (p1.y + p2.y);
-         }
+             return history;
+           },
+           lock: function lock() {
+             return _lock.lock();
+           },
+           unlock: function unlock() {
+             _lock.unlock();
+           },
+           save: function save() {
+             if (_lock.locked() && // don't overwrite existing, unresolved changes
+             !_hasUnresolvedRestorableChanges) {
+               corePreferences(getKey('saved_history'), history.toJSON() || null);
+             }
 
-         return sum;
-       }
+             return history;
+           },
+           // delete the history version saved in localStorage
+           clearSaved: function clearSaved() {
+             context.debouncedSave.cancel();
 
-       var vectortilelayer = VectorTileLayer;
+             if (_lock.locked()) {
+               _hasUnresolvedRestorableChanges = false;
+               corePreferences(getKey('saved_history'), null); // clear the changeset metadata associated with the saved history
 
-       function VectorTileLayer(pbf, end) {
-         // Public
-         this.version = 1;
-         this.name = null;
-         this.extent = 4096;
-         this.length = 0; // Private
+               corePreferences('comment', null);
+               corePreferences('hashtags', null);
+               corePreferences('source', null);
+             }
 
-         this._pbf = pbf;
-         this._keys = [];
-         this._values = [];
-         this._features = [];
-         pbf.readFields(readLayer, this, end);
-         this.length = this._features.length;
+             return history;
+           },
+           savedHistoryJSON: function savedHistoryJSON() {
+             return corePreferences(getKey('saved_history'));
+           },
+           hasRestorableChanges: function hasRestorableChanges() {
+             return _hasUnresolvedRestorableChanges;
+           },
+           // load history from a version stored in localStorage
+           restore: function restore() {
+             if (_lock.locked()) {
+               _hasUnresolvedRestorableChanges = false;
+               var json = this.savedHistoryJSON();
+               if (json) history.fromJSON(json, true);
+             }
+           },
+           _getKey: getKey
+         };
+         history.reset();
+         return utilRebind(history, dispatch, 'on');
        }
 
-       function readLayer(tag, layer, pbf) {
-         if (tag === 15) layer.version = pbf.readVarint();else if (tag === 1) layer.name = pbf.readString();else if (tag === 5) layer.extent = pbf.readVarint();else if (tag === 2) layer._features.push(pbf.pos);else if (tag === 3) layer._keys.push(pbf.readString());else if (tag === 4) layer._values.push(readValueMessage(pbf));
-       }
+       /**
+        * Look for roads that can be connected to other roads with a short extension
+        */
 
-       function readValueMessage(pbf) {
-         var value = null,
-             end = pbf.readVarint() + pbf.pos;
+       function validationAlmostJunction(context) {
+         var type = 'almost_junction';
+         var EXTEND_TH_METERS = 5;
+         var WELD_TH_METERS = 0.75; // Comes from considering bounding case of parallel ways
 
-         while (pbf.pos < end) {
-           var tag = pbf.readVarint() >> 3;
-           value = tag === 1 ? pbf.readString() : tag === 2 ? pbf.readFloat() : tag === 3 ? pbf.readDouble() : tag === 4 ? pbf.readVarint64() : tag === 5 ? pbf.readVarint() : tag === 6 ? pbf.readSVarint() : tag === 7 ? pbf.readBoolean() : null;
+         var CLOSE_NODE_TH = EXTEND_TH_METERS - WELD_TH_METERS; // Comes from considering bounding case of perpendicular ways
+
+         var SIG_ANGLE_TH = Math.atan(WELD_TH_METERS / EXTEND_TH_METERS);
+
+         function isHighway(entity) {
+           return entity.type === 'way' && osmRoutableHighwayTagValues[entity.tags.highway];
          }
 
-         return value;
-       } // return feature `i` from this layer as a `VectorTileFeature`
+         function isTaggedAsNotContinuing(node) {
+           return node.tags.noexit === 'yes' || node.tags.amenity === 'parking_entrance' || node.tags.entrance && node.tags.entrance !== 'no';
+         }
 
+         var validation = function checkAlmostJunction(entity, graph) {
+           if (!isHighway(entity)) return [];
+           if (entity.isDegenerate()) return [];
+           var tree = context.history().tree();
+           var extendableNodeInfos = findConnectableEndNodesByExtension(entity);
+           var issues = [];
+           extendableNodeInfos.forEach(function (extendableNodeInfo) {
+             issues.push(new validationIssue({
+               type: type,
+               subtype: 'highway-highway',
+               severity: 'warning',
+               message: function message(context) {
+                 var entity1 = context.hasEntity(this.entityIds[0]);
 
-       VectorTileLayer.prototype.feature = function (i) {
-         if (i < 0 || i >= this._features.length) throw new Error('feature index out of bounds');
-         this._pbf.pos = this._features[i];
+                 if (this.entityIds[0] === this.entityIds[2]) {
+                   return entity1 ? _t.html('issues.almost_junction.self.message', {
+                     feature: utilDisplayLabel(entity1, context.graph())
+                   }) : '';
+                 } else {
+                   var entity2 = context.hasEntity(this.entityIds[2]);
+                   return entity1 && entity2 ? _t.html('issues.almost_junction.message', {
+                     feature: utilDisplayLabel(entity1, context.graph()),
+                     feature2: utilDisplayLabel(entity2, context.graph())
+                   }) : '';
+                 }
+               },
+               reference: showReference,
+               entityIds: [entity.id, extendableNodeInfo.node.id, extendableNodeInfo.wid],
+               loc: extendableNodeInfo.node.loc,
+               hash: JSON.stringify(extendableNodeInfo.node.loc),
+               data: {
+                 midId: extendableNodeInfo.mid.id,
+                 edge: extendableNodeInfo.edge,
+                 cross_loc: extendableNodeInfo.cross_loc
+               },
+               dynamicFixes: makeFixes
+             }));
+           });
+           return issues;
 
-         var end = this._pbf.readVarint() + this._pbf.pos;
+           function makeFixes(context) {
+             var fixes = [new validationIssueFix({
+               icon: 'iD-icon-abutment',
+               title: _t.html('issues.fix.connect_features.title'),
+               onClick: function onClick(context) {
+                 var annotation = _t('issues.fix.connect_almost_junction.annotation');
 
-         return new vectortilefeature(this._pbf, end, this.extent, this._keys, this._values);
-       };
+                 var _this$issue$entityIds = _slicedToArray(this.issue.entityIds, 3),
+                     endNodeId = _this$issue$entityIds[1],
+                     crossWayId = _this$issue$entityIds[2];
 
-       var vectortile = VectorTile;
+                 var midNode = context.entity(this.issue.data.midId);
+                 var endNode = context.entity(endNodeId);
+                 var crossWay = context.entity(crossWayId); // When endpoints are close, just join if resulting small change in angle (#7201)
 
-       function VectorTile(pbf, end) {
-         this.layers = pbf.readFields(readTile, {}, end);
-       }
+                 var nearEndNodes = findNearbyEndNodes(endNode, crossWay);
 
-       function readTile(tag, layers, pbf) {
-         if (tag === 3) {
-           var layer = new vectortilelayer(pbf, pbf.readVarint() + pbf.pos);
-           if (layer.length) layers[layer.name] = layer;
-         }
-       }
+                 if (nearEndNodes.length > 0) {
+                   var collinear = findSmallJoinAngle(midNode, endNode, nearEndNodes);
 
-       var VectorTile$1 = vectortile;
-       var VectorTileFeature$1 = vectortilefeature;
-       var VectorTileLayer$1 = vectortilelayer;
-       var vectorTile = {
-         VectorTile: VectorTile$1,
-         VectorTileFeature: VectorTileFeature$1,
-         VectorTileLayer: VectorTileLayer$1
-       };
+                   if (collinear) {
+                     context.perform(actionMergeNodes([collinear.id, endNode.id], collinear.loc), annotation);
+                     return;
+                   }
+                 }
 
-       var tiler$7 = utilTiler().tileSize(512).margin(1);
-       var dispatch$8 = dispatch('loadedData');
+                 var targetEdge = this.issue.data.edge;
+                 var crossLoc = this.issue.data.cross_loc;
+                 var edgeNodes = [context.entity(targetEdge[0]), context.entity(targetEdge[1])];
+                 var closestNodeInfo = geoSphericalClosestNode(edgeNodes, crossLoc); // already a point nearby, just connect to that
 
-       var _vtCache;
+                 if (closestNodeInfo.distance < WELD_TH_METERS) {
+                   context.perform(actionMergeNodes([closestNodeInfo.node.id, endNode.id], closestNodeInfo.node.loc), annotation); // else add the end node to the edge way
+                 } else {
+                   context.perform(actionAddMidpoint({
+                     loc: crossLoc,
+                     edge: targetEdge
+                   }, endNode), annotation);
+                 }
+               }
+             })];
+             var node = context.hasEntity(this.entityIds[1]);
 
-       function abortRequest$7(controller) {
-         controller.abort();
-       }
+             if (node && !node.hasInterestingTags()) {
+               // node has no descriptive tags, suggest noexit fix
+               fixes.push(new validationIssueFix({
+                 icon: 'maki-barrier',
+                 title: _t.html('issues.fix.tag_as_disconnected.title'),
+                 onClick: function onClick(context) {
+                   var nodeID = this.issue.entityIds[1];
+                   var tags = Object.assign({}, context.entity(nodeID).tags);
+                   tags.noexit = 'yes';
+                   context.perform(actionChangeTags(nodeID, tags), _t('issues.fix.tag_as_disconnected.annotation'));
+                 }
+               }));
+             }
 
-       function vtToGeoJSON(data, tile, mergeCache) {
-         var vectorTile$1 = new vectorTile.VectorTile(new pbf(data));
-         var layers = Object.keys(vectorTile$1.layers);
+             return fixes;
+           }
 
-         if (!Array.isArray(layers)) {
-           layers = [layers];
-         }
+           function showReference(selection) {
+             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.almost_junction.highway-highway.reference'));
+           }
 
-         var features = [];
-         layers.forEach(function (layerID) {
-           var layer = vectorTile$1.layers[layerID];
+           function isExtendableCandidate(node, way) {
+             // can not accurately test vertices on tiles not downloaded from osm - #5938
+             var osm = services.osm;
 
-           if (layer) {
-             for (var i = 0; i < layer.length; i++) {
-               var feature = layer.feature(i).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
-               var geometry = feature.geometry; // Treat all Polygons as MultiPolygons
+             if (osm && !osm.isDataLoaded(node.loc)) {
+               return false;
+             }
 
-               if (geometry.type === 'Polygon') {
-                 geometry.type = 'MultiPolygon';
-                 geometry.coordinates = [geometry.coordinates];
-               }
+             if (isTaggedAsNotContinuing(node) || graph.parentWays(node).length !== 1) {
+               return false;
+             }
 
-               var isClipped = false; // Clip to tile bounds
+             var occurrences = 0;
 
-               if (geometry.type === 'MultiPolygon') {
-                 var featureClip = bboxClip(feature, tile.extent.rectangle());
+             for (var index in way.nodes) {
+               if (way.nodes[index] === node.id) {
+                 occurrences += 1;
 
-                 if (!fastDeepEqual(feature.geometry, featureClip.geometry)) {
-                   // feature = featureClip;
-                   isClipped = true;
+                 if (occurrences > 1) {
+                   return false;
                  }
+               }
+             }
 
-                 if (!feature.geometry.coordinates.length) continue; // not actually on this tile
-
-                 if (!feature.geometry.coordinates[0].length) continue; // not actually on this tile
-               } // Generate some unique IDs and add some metadata
-
+             return true;
+           }
 
-               var featurehash = utilHashcode(fastJsonStableStringify(feature));
-               var propertyhash = utilHashcode(fastJsonStableStringify(feature.properties || {}));
-               feature.__layerID__ = layerID.replace(/[^_a-zA-Z0-9\-]/g, '_');
-               feature.__featurehash__ = featurehash;
-               feature.__propertyhash__ = propertyhash;
-               features.push(feature); // Clipped Polygons at same zoom with identical properties can get merged
+           function findConnectableEndNodesByExtension(way) {
+             var results = [];
+             if (way.isClosed()) return results;
+             var testNodes;
+             var indices = [0, way.nodes.length - 1];
+             indices.forEach(function (nodeIndex) {
+               var nodeID = way.nodes[nodeIndex];
+               var node = graph.entity(nodeID);
+               if (!isExtendableCandidate(node, way)) return;
+               var connectionInfo = canConnectByExtend(way, nodeIndex);
+               if (!connectionInfo) return;
+               testNodes = graph.childNodes(way).slice(); // shallow copy
 
-               if (isClipped && geometry.type === 'MultiPolygon') {
-                 var merged = mergeCache[propertyhash];
+               testNodes[nodeIndex] = testNodes[nodeIndex].move(connectionInfo.cross_loc); // don't flag issue if connecting the ways would cause self-intersection
 
-                 if (merged && merged.length) {
-                   var other = merged[0];
-                   var coords = union(feature.geometry.coordinates, other.geometry.coordinates);
+               if (geoHasSelfIntersections(testNodes, nodeID)) return;
+               results.push(connectionInfo);
+             });
+             return results;
+           }
 
-                   if (!coords || !coords.length) {
-                     continue; // something failed in martinez union
-                   }
+           function findNearbyEndNodes(node, way) {
+             return [way.nodes[0], way.nodes[way.nodes.length - 1]].map(function (d) {
+               return graph.entity(d);
+             }).filter(function (d) {
+               // Node cannot be near to itself, but other endnode of same way could be
+               return d.id !== node.id && geoSphericalDistance(node.loc, d.loc) <= CLOSE_NODE_TH;
+             });
+           }
 
-                   merged.push(feature);
+           function findSmallJoinAngle(midNode, tipNode, endNodes) {
+             // Both nodes could be close, so want to join whichever is closest to collinear
+             var joinTo;
+             var minAngle = Infinity; // Checks midNode -> tipNode -> endNode for collinearity
 
-                   for (var j = 0; j < merged.length; j++) {
-                     // all these features get...
-                     merged[j].geometry.coordinates = coords; // same coords
+             endNodes.forEach(function (endNode) {
+               var a1 = geoAngle(midNode, tipNode, context.projection) + Math.PI;
+               var a2 = geoAngle(midNode, endNode, context.projection) + Math.PI;
+               var diff = Math.max(a1, a2) - Math.min(a1, a2);
 
-                     merged[j].__featurehash__ = featurehash; // same hash, so deduplication works
-                   }
-                 } else {
-                   mergeCache[propertyhash] = [feature];
-                 }
+               if (diff < minAngle) {
+                 joinTo = endNode;
+                 minAngle = diff;
                }
-             }
-           }
-         });
-         return features;
-       }
+             });
+             /* Threshold set by considering right angle triangle
+             based on node joining threshold and extension distance */
 
-       function loadTile(source, tile) {
-         if (source.loaded[tile.id] || source.inflight[tile.id]) return;
-         var url = source.template.replace('{x}', tile.xyz[0]).replace('{y}', tile.xyz[1]) // TMS-flipped y coordinate
-         .replace(/\{[t-]y\}/, Math.pow(2, tile.xyz[2]) - tile.xyz[1] - 1).replace(/\{z(oom)?\}/, tile.xyz[2]).replace(/\{switch:([^}]+)\}/, function (s, r) {
-           var subdomains = r.split(',');
-           return subdomains[(tile.xyz[0] + tile.xyz[1]) % subdomains.length];
-         });
-         var controller = new AbortController();
-         source.inflight[tile.id] = controller;
-         fetch(url, {
-           signal: controller.signal
-         }).then(function (response) {
-           if (!response.ok) {
-             throw new Error(response.status + ' ' + response.statusText);
+             if (minAngle <= SIG_ANGLE_TH) return joinTo;
+             return null;
            }
 
-           source.loaded[tile.id] = [];
-           delete source.inflight[tile.id];
-           return response.arrayBuffer();
-         }).then(function (data) {
-           if (!data) {
-             throw new Error('No Data');
+           function hasTag(tags, key) {
+             return tags[key] !== undefined && tags[key] !== 'no';
            }
 
-           var z = tile.xyz[2];
-
-           if (!source.canMerge[z]) {
-             source.canMerge[z] = {}; // initialize mergeCache
-           }
+           function canConnectWays(way, way2) {
+             // allow self-connections
+             if (way.id === way2.id) return true; // if one is bridge or tunnel, both must be bridge or tunnel
 
-           source.loaded[tile.id] = vtToGeoJSON(data, tile, source.canMerge[z]);
-           dispatch$8.call('loadedData');
-         })["catch"](function () {
-           source.loaded[tile.id] = [];
-           delete source.inflight[tile.id];
-         });
-       }
+             if ((hasTag(way.tags, 'bridge') || hasTag(way2.tags, 'bridge')) && !(hasTag(way.tags, 'bridge') && hasTag(way2.tags, 'bridge'))) return false;
+             if ((hasTag(way.tags, 'tunnel') || hasTag(way2.tags, 'tunnel')) && !(hasTag(way.tags, 'tunnel') && hasTag(way2.tags, 'tunnel'))) return false; // must have equivalent layers and levels
 
-       var serviceVectorTile = {
-         init: function init() {
-           if (!_vtCache) {
-             this.reset();
+             var layer1 = way.tags.layer || '0',
+                 layer2 = way2.tags.layer || '0';
+             if (layer1 !== layer2) return false;
+             var level1 = way.tags.level || '0',
+                 level2 = way2.tags.level || '0';
+             if (level1 !== level2) return false;
+             return true;
            }
 
-           this.event = utilRebind(this, dispatch$8, 'on');
-         },
-         reset: function reset() {
-           for (var sourceID in _vtCache) {
-             var source = _vtCache[sourceID];
+           function canConnectByExtend(way, endNodeIdx) {
+             var tipNid = way.nodes[endNodeIdx]; // the 'tip' node for extension point
 
-             if (source && source.inflight) {
-               Object.values(source.inflight).forEach(abortRequest$7);
-             }
-           }
+             var midNid = endNodeIdx === 0 ? way.nodes[1] : way.nodes[way.nodes.length - 2]; // the other node of the edge
 
-           _vtCache = {};
-         },
-         addSource: function addSource(sourceID, template) {
-           _vtCache[sourceID] = {
-             template: template,
-             inflight: {},
-             loaded: {},
-             canMerge: {}
-           };
-           return _vtCache[sourceID];
-         },
-         data: function data(sourceID, projection) {
-           var source = _vtCache[sourceID];
-           if (!source) return [];
-           var tiles = tiler$7.getTiles(projection);
-           var seen = {};
-           var results = [];
+             var tipNode = graph.entity(tipNid);
+             var midNode = graph.entity(midNid);
+             var lon = tipNode.loc[0];
+             var lat = tipNode.loc[1];
+             var lon_range = geoMetersToLon(EXTEND_TH_METERS, lat) / 2;
+             var lat_range = geoMetersToLat(EXTEND_TH_METERS) / 2;
+             var queryExtent = geoExtent([[lon - lon_range, lat - lat_range], [lon + lon_range, lat + lat_range]]); // first, extend the edge of [midNode -> tipNode] by EXTEND_TH_METERS and find the "extended tip" location
 
-           for (var i = 0; i < tiles.length; i++) {
-             var features = source.loaded[tiles[i].id];
-             if (!features || !features.length) continue;
+             var edgeLen = geoSphericalDistance(midNode.loc, tipNode.loc);
+             var t = EXTEND_TH_METERS / edgeLen + 1.0;
+             var extTipLoc = geoVecInterp(midNode.loc, tipNode.loc, t); // then, check if the extension part [tipNode.loc -> extTipLoc] intersects any other ways
 
-             for (var j = 0; j < features.length; j++) {
-               var feature = features[j];
-               var hash = feature.__featurehash__;
-               if (seen[hash]) continue;
-               seen[hash] = true; // return a shallow copy, because the hash may change
-               // later if this feature gets merged with another
+             var segmentInfos = tree.waySegments(queryExtent, graph);
 
-               results.push(Object.assign({}, feature)); // shallow copy
+             for (var i = 0; i < segmentInfos.length; i++) {
+               var segmentInfo = segmentInfos[i];
+               var way2 = graph.entity(segmentInfo.wayId);
+               if (!isHighway(way2)) continue;
+               if (!canConnectWays(way, way2)) continue;
+               var nAid = segmentInfo.nodes[0],
+                   nBid = segmentInfo.nodes[1];
+               if (nAid === tipNid || nBid === tipNid) continue;
+               var nA = graph.entity(nAid),
+                   nB = graph.entity(nBid);
+               var crossLoc = geoLineIntersection([tipNode.loc, extTipLoc], [nA.loc, nB.loc]);
+
+               if (crossLoc) {
+                 return {
+                   mid: midNode,
+                   node: tipNode,
+                   wid: way2.id,
+                   edge: [nA.id, nB.id],
+                   cross_loc: crossLoc
+                 };
+               }
              }
+
+             return null;
            }
+         };
 
-           return results;
-         },
-         loadTiles: function loadTiles(sourceID, template, projection) {
-           var source = _vtCache[sourceID];
+         validation.type = type;
+         return validation;
+       }
 
-           if (!source) {
-             source = this.addSource(sourceID, template);
+       function validationCloseNodes(context) {
+         var type = 'close_nodes';
+         var pointThresholdMeters = 0.2;
+
+         var validation = function validation(entity, graph) {
+           if (entity.type === 'node') {
+             return getIssuesForNode(entity);
+           } else if (entity.type === 'way') {
+             return getIssuesForWay(entity);
            }
 
-           var tiles = tiler$7.getTiles(projection); // abort inflight requests that are no longer needed
+           return [];
 
-           Object.keys(source.inflight).forEach(function (k) {
-             var wanted = tiles.find(function (tile) {
-               return k === tile.id;
-             });
+           function getIssuesForNode(node) {
+             var parentWays = graph.parentWays(node);
 
-             if (!wanted) {
-               abortRequest$7(source.inflight[k]);
-               delete source.inflight[k];
+             if (parentWays.length) {
+               return getIssuesForVertex(node, parentWays);
+             } else {
+               return getIssuesForDetachedPoint(node);
              }
-           });
-           tiles.forEach(function (tile) {
-             loadTile(source, tile);
-           });
-         },
-         cache: function cache() {
-           return _vtCache;
-         }
-       };
-
-       var apibase$3 = 'https://www.wikidata.org/w/api.php?';
-       var _wikidataCache = {};
-       var serviceWikidata = {
-         init: function init() {},
-         reset: function reset() {
-           _wikidataCache = {};
-         },
-         // Search for Wikidata items matching the query
-         itemsForSearchQuery: function itemsForSearchQuery(query, callback) {
-           if (!query) {
-             if (callback) callback('No query', {});
-             return;
            }
 
-           var lang = this.languagesToQuery()[0];
-           var url = apibase$3 + utilQsString({
-             action: 'wbsearchentities',
-             format: 'json',
-             formatversion: 2,
-             search: query,
-             type: 'item',
-             // the language to search
-             language: lang,
-             // the language for the label and description in the result
-             uselang: lang,
-             limit: 10,
-             origin: '*'
-           });
-           d3_json(url).then(function (result) {
-             if (result && result.error) {
-               throw new Error(result.error);
-             }
+           function wayTypeFor(way) {
+             if (way.tags.boundary && way.tags.boundary !== 'no') return 'boundary';
+             if (way.tags.indoor && way.tags.indoor !== 'no') return 'indoor';
+             if (way.tags.building && way.tags.building !== 'no' || way.tags['building:part'] && way.tags['building:part'] !== 'no') return 'building';
+             if (osmPathHighwayTagValues[way.tags.highway]) return 'path';
+             var parentRelations = graph.parentRelations(way);
 
-             if (callback) callback(null, result.search || {});
-           })["catch"](function (err) {
-             if (callback) callback(err.message, {});
-           });
-         },
-         // Given a Wikipedia language and article title,
-         // return an array of corresponding Wikidata entities.
-         itemsByTitle: function itemsByTitle(lang, title, callback) {
-           if (!title) {
-             if (callback) callback('No title', {});
-             return;
-           }
+             for (var i in parentRelations) {
+               var relation = parentRelations[i];
+               if (relation.tags.type === 'boundary') return 'boundary';
 
-           lang = lang || 'en';
-           var url = apibase$3 + utilQsString({
-             action: 'wbgetentities',
-             format: 'json',
-             formatversion: 2,
-             sites: lang.replace(/-/g, '_') + 'wiki',
-             titles: title,
-             languages: 'en',
-             // shrink response by filtering to one language
-             origin: '*'
-           });
-           d3_json(url).then(function (result) {
-             if (result && result.error) {
-               throw new Error(result.error);
+               if (relation.isMultipolygon()) {
+                 if (relation.tags.indoor && relation.tags.indoor !== 'no') return 'indoor';
+                 if (relation.tags.building && relation.tags.building !== 'no' || relation.tags['building:part'] && relation.tags['building:part'] !== 'no') return 'building';
+               }
              }
 
-             if (callback) callback(null, result.entities || {});
-           })["catch"](function (err) {
-             if (callback) callback(err.message, {});
-           });
-         },
-         languagesToQuery: function languagesToQuery() {
-           return _mainLocalizer.localeCodes().map(function (code) {
-             return code.toLowerCase();
-           }).filter(function (code) {
-             // HACK: en-us isn't a wikidata language. We should really be filtering by
-             // the languages known to be supported by wikidata.
-             return code !== 'en-us';
-           });
-         },
-         entityByQID: function entityByQID(qid, callback) {
-           if (!qid) {
-             callback('No qid', {});
-             return;
+             return 'other';
            }
 
-           if (_wikidataCache[qid]) {
-             if (callback) callback(null, _wikidataCache[qid]);
-             return;
+           function shouldCheckWay(way) {
+             // don't flag issues where merging would create degenerate ways
+             if (way.nodes.length <= 2 || way.isClosed() && way.nodes.length <= 4) return false;
+             var bbox = way.extent(graph).bbox();
+             var hypotenuseMeters = geoSphericalDistance([bbox.minX, bbox.minY], [bbox.maxX, bbox.maxY]); // don't flag close nodes in very small ways
+
+             if (hypotenuseMeters < 1.5) return false;
+             return true;
            }
 
-           var langs = this.languagesToQuery();
-           var url = apibase$3 + utilQsString({
-             action: 'wbgetentities',
-             format: 'json',
-             formatversion: 2,
-             ids: qid,
-             props: 'labels|descriptions|claims|sitelinks',
-             sitefilter: langs.map(function (d) {
-               return d + 'wiki';
-             }).join('|'),
-             languages: langs.join('|'),
-             languagefallback: 1,
-             origin: '*'
-           });
-           d3_json(url).then(function (result) {
-             if (result && result.error) {
-               throw new Error(result.error);
-             }
+           function getIssuesForWay(way) {
+             if (!shouldCheckWay(way)) return [];
+             var issues = [],
+                 nodes = graph.childNodes(way);
 
-             if (callback) callback(null, result.entities[qid] || {});
-           })["catch"](function (err) {
-             if (callback) callback(err.message, {});
-           });
-         },
-         // Pass `params` object of the form:
-         // {
-         //   qid: 'string'      // brand wikidata  (e.g. 'Q37158')
-         // }
-         //
-         // Get an result object used to display tag documentation
-         // {
-         //   title:        'string',
-         //   description:  'string',
-         //   editURL:      'string',
-         //   imageURL:     'string',
-         //   wiki:         { title: 'string', text: 'string', url: 'string' }
-         // }
-         //
-         getDocs: function getDocs(params, callback) {
-           var langs = this.languagesToQuery();
-           this.entityByQID(params.qid, function (err, entity) {
-             if (err || !entity) {
-               callback(err || 'No entity');
-               return;
+             for (var i = 0; i < nodes.length - 1; i++) {
+               var node1 = nodes[i];
+               var node2 = nodes[i + 1];
+               var issue = getWayIssueIfAny(node1, node2, way);
+               if (issue) issues.push(issue);
+             }
+
+             return issues;
+           }
+
+           function getIssuesForVertex(node, parentWays) {
+             var issues = [];
+
+             function checkForCloseness(node1, node2, way) {
+               var issue = getWayIssueIfAny(node1, node2, way);
+               if (issue) issues.push(issue);
              }
 
-             var i;
-             var description;
+             for (var i = 0; i < parentWays.length; i++) {
+               var parentWay = parentWays[i];
+               if (!shouldCheckWay(parentWay)) continue;
+               var lastIndex = parentWay.nodes.length - 1;
 
-             for (i in langs) {
-               var code = langs[i];
+               for (var j = 0; j < parentWay.nodes.length; j++) {
+                 if (j !== 0) {
+                   if (parentWay.nodes[j - 1] === node.id) {
+                     checkForCloseness(node, graph.entity(parentWay.nodes[j]), parentWay);
+                   }
+                 }
 
-               if (entity.descriptions[code] && entity.descriptions[code].language === code) {
-                 description = entity.descriptions[code];
-                 break;
+                 if (j !== lastIndex) {
+                   if (parentWay.nodes[j + 1] === node.id) {
+                     checkForCloseness(graph.entity(parentWay.nodes[j]), node, parentWay);
+                   }
+                 }
                }
              }
 
-             if (!description && Object.values(entity.descriptions).length) description = Object.values(entity.descriptions)[0]; // prepare result
+             return issues;
+           }
 
-             var result = {
-               title: entity.id,
-               description: description ? description.value : '',
-               descriptionLocaleCode: description ? description.language : '',
-               editURL: 'https://www.wikidata.org/wiki/' + entity.id
-             }; // add image
+           function thresholdMetersForWay(way) {
+             if (!shouldCheckWay(way)) return 0;
+             var wayType = wayTypeFor(way); // don't flag boundaries since they might be highly detailed and can't be easily verified
 
-             if (entity.claims) {
-               var imageroot = 'https://commons.wikimedia.org/w/index.php';
-               var props = ['P154', 'P18']; // logo image, image
+             if (wayType === 'boundary') return 0; // expect some features to be mapped with higher levels of detail
 
-               var prop, image;
+             if (wayType === 'indoor') return 0.01;
+             if (wayType === 'building') return 0.05;
+             if (wayType === 'path') return 0.1;
+             return 0.2;
+           }
 
-               for (i = 0; i < props.length; i++) {
-                 prop = entity.claims[props[i]];
+           function getIssuesForDetachedPoint(node) {
+             var issues = [];
+             var lon = node.loc[0];
+             var lat = node.loc[1];
+             var lon_range = geoMetersToLon(pointThresholdMeters, lat) / 2;
+             var lat_range = geoMetersToLat(pointThresholdMeters) / 2;
+             var queryExtent = geoExtent([[lon - lon_range, lat - lat_range], [lon + lon_range, lat + lat_range]]);
+             var intersected = context.history().tree().intersects(queryExtent, graph);
 
-                 if (prop && Object.keys(prop).length > 0) {
-                   image = prop[Object.keys(prop)[0]].mainsnak.datavalue.value;
+             for (var j = 0; j < intersected.length; j++) {
+               var nearby = intersected[j];
+               if (nearby.id === node.id) continue;
+               if (nearby.type !== 'node' || nearby.geometry(graph) !== 'point') continue;
 
-                   if (image) {
-                     result.imageURL = imageroot + '?' + utilQsString({
-                       title: 'Special:Redirect/file/' + image,
-                       width: 400
-                     });
+               if (nearby.loc === node.loc || geoSphericalDistance(node.loc, nearby.loc) < pointThresholdMeters) {
+                 // allow very close points if tags indicate the z-axis might vary
+                 var zAxisKeys = {
+                   layer: true,
+                   level: true,
+                   'addr:housenumber': true,
+                   'addr:unit': true
+                 };
+                 var zAxisDifferentiates = false;
+
+                 for (var key in zAxisKeys) {
+                   var nodeValue = node.tags[key] || '0';
+                   var nearbyValue = nearby.tags[key] || '0';
+
+                   if (nodeValue !== nearbyValue) {
+                     zAxisDifferentiates = true;
                      break;
                    }
                  }
+
+                 if (zAxisDifferentiates) continue;
+                 issues.push(new validationIssue({
+                   type: type,
+                   subtype: 'detached',
+                   severity: 'warning',
+                   message: function message(context) {
+                     var entity = context.hasEntity(this.entityIds[0]),
+                         entity2 = context.hasEntity(this.entityIds[1]);
+                     return entity && entity2 ? _t.html('issues.close_nodes.detached.message', {
+                       feature: utilDisplayLabel(entity, context.graph()),
+                       feature2: utilDisplayLabel(entity2, context.graph())
+                     }) : '';
+                   },
+                   reference: showReference,
+                   entityIds: [node.id, nearby.id],
+                   dynamicFixes: function dynamicFixes() {
+                     return [new validationIssueFix({
+                       icon: 'iD-operation-disconnect',
+                       title: _t.html('issues.fix.move_points_apart.title')
+                     }), new validationIssueFix({
+                       icon: 'iD-icon-layers',
+                       title: _t.html('issues.fix.use_different_layers_or_levels.title')
+                     })];
+                   }
+                 }));
                }
              }
 
-             if (entity.sitelinks) {
-               var englishLocale = _mainLocalizer.languageCode().toLowerCase() === 'en'; // must be one of these that we requested..
+             return issues;
 
-               for (i = 0; i < langs.length; i++) {
-                 // check each, in order of preference
-                 var w = langs[i] + 'wiki';
+             function showReference(selection) {
+               var referenceText = _t('issues.close_nodes.detached.reference');
+               selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(referenceText);
+             }
+           }
 
-                 if (entity.sitelinks[w]) {
-                   var title = entity.sitelinks[w].title;
-                   var tKey = 'inspector.wiki_reference';
+           function getWayIssueIfAny(node1, node2, way) {
+             if (node1.id === node2.id || node1.hasInterestingTags() && node2.hasInterestingTags()) {
+               return null;
+             }
 
-                   if (!englishLocale && langs[i] === 'en') {
-                     // user's locale isn't English but
-                     tKey = 'inspector.wiki_en_reference'; // we are sending them to enwiki anyway..
-                   }
+             if (node1.loc !== node2.loc) {
+               var parentWays1 = graph.parentWays(node1);
+               var parentWays2 = new Set(graph.parentWays(node2));
+               var sharedWays = parentWays1.filter(function (parentWay) {
+                 return parentWays2.has(parentWay);
+               });
+               var thresholds = sharedWays.map(function (parentWay) {
+                 return thresholdMetersForWay(parentWay);
+               });
+               var threshold = Math.min.apply(Math, _toConsumableArray(thresholds));
+               var distance = geoSphericalDistance(node1.loc, node2.loc);
+               if (distance > threshold) return null;
+             }
 
-                   result.wiki = {
-                     title: title,
-                     text: tKey,
-                     url: 'https://' + langs[i] + '.wikipedia.org/wiki/' + title.replace(/ /g, '_')
-                   };
-                   break;
-                 }
+             return new validationIssue({
+               type: type,
+               subtype: 'vertices',
+               severity: 'warning',
+               message: function message(context) {
+                 var entity = context.hasEntity(this.entityIds[0]);
+                 return entity ? _t.html('issues.close_nodes.message', {
+                   way: utilDisplayLabel(entity, context.graph())
+                 }) : '';
+               },
+               reference: showReference,
+               entityIds: [way.id, node1.id, node2.id],
+               loc: node1.loc,
+               dynamicFixes: function dynamicFixes() {
+                 return [new validationIssueFix({
+                   icon: 'iD-icon-plus',
+                   title: _t.html('issues.fix.merge_points.title'),
+                   onClick: function onClick(context) {
+                     var entityIds = this.issue.entityIds;
+                     var action = actionMergeNodes([entityIds[1], entityIds[2]]);
+                     context.perform(action, _t('issues.fix.merge_close_vertices.annotation'));
+                   }
+                 }), new validationIssueFix({
+                   icon: 'iD-operation-disconnect',
+                   title: _t.html('issues.fix.move_points_apart.title')
+                 })];
                }
+             });
+
+             function showReference(selection) {
+               var referenceText = _t('issues.close_nodes.reference');
+               selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(referenceText);
              }
+           }
+         };
 
-             callback(null, result);
-           });
-         }
-       };
+         validation.type = type;
+         return validation;
+       }
 
-       var endpoint = 'https://en.wikipedia.org/w/api.php?';
-       var serviceWikipedia = {
-         init: function init() {},
-         reset: function reset() {},
-         search: function search(lang, query, callback) {
-           if (!query) {
-             if (callback) callback('No Query', []);
-             return;
-           }
+       function validationCrossingWays(context) {
+         var type = 'crossing_ways'; // returns the way or its parent relation, whichever has a useful feature type
 
-           lang = lang || 'en';
-           var url = endpoint.replace('en', lang) + utilQsString({
-             action: 'query',
-             list: 'search',
-             srlimit: '10',
-             srinfo: 'suggestion',
-             format: 'json',
-             origin: '*',
-             srsearch: query
-           });
-           d3_json(url).then(function (result) {
-             if (result && result.error) {
-               throw new Error(result.error);
-             } else if (!result || !result.query || !result.query.search) {
-               throw new Error('No Results');
-             }
+         function getFeatureWithFeatureTypeTagsForWay(way, graph) {
+           if (getFeatureType(way, graph) === null) {
+             // if the way doesn't match a feature type, check its parent relations
+             var parentRels = graph.parentRelations(way);
 
-             if (callback) {
-               var titles = result.query.search.map(function (d) {
-                 return d.title;
-               });
-               callback(null, titles);
+             for (var i = 0; i < parentRels.length; i++) {
+               var rel = parentRels[i];
+
+               if (getFeatureType(rel, graph) !== null) {
+                 return rel;
+               }
              }
-           })["catch"](function (err) {
-             if (callback) callback(err, []);
-           });
-         },
-         suggestions: function suggestions(lang, query, callback) {
-           if (!query) {
-             if (callback) callback('', []);
-             return;
            }
 
-           lang = lang || 'en';
-           var url = endpoint.replace('en', lang) + utilQsString({
-             action: 'opensearch',
-             namespace: 0,
-             suggest: '',
-             format: 'json',
-             origin: '*',
-             search: query
-           });
-           d3_json(url).then(function (result) {
-             if (result && result.error) {
-               throw new Error(result.error);
-             } else if (!result || result.length < 2) {
-               throw new Error('No Results');
-             }
+           return way;
+         }
 
-             if (callback) callback(null, result[1] || []);
-           })["catch"](function (err) {
-             if (callback) callback(err.message, []);
-           });
-         },
-         translations: function translations(lang, title, callback) {
-           if (!title) {
-             if (callback) callback('No Title');
-             return;
-           }
+         function hasTag(tags, key) {
+           return tags[key] !== undefined && tags[key] !== 'no';
+         }
 
-           var url = endpoint.replace('en', lang) + utilQsString({
-             action: 'query',
-             prop: 'langlinks',
-             format: 'json',
-             origin: '*',
-             lllimit: 500,
-             titles: title
-           });
-           d3_json(url).then(function (result) {
-             if (result && result.error) {
-               throw new Error(result.error);
-             } else if (!result || !result.query || !result.query.pages) {
-               throw new Error('No Results');
-             }
+         function taggedAsIndoor(tags) {
+           return hasTag(tags, 'indoor') || hasTag(tags, 'level') || tags.highway === 'corridor';
+         }
 
-             if (callback) {
-               var list = result.query.pages[Object.keys(result.query.pages)[0]];
-               var translations = {};
+         function allowsBridge(featureType) {
+           return featureType === 'highway' || featureType === 'railway' || featureType === 'waterway';
+         }
 
-               if (list && list.langlinks) {
-                 list.langlinks.forEach(function (d) {
-                   translations[d.lang] = d['*'];
-                 });
-               }
+         function allowsTunnel(featureType) {
+           return featureType === 'highway' || featureType === 'railway' || featureType === 'waterway';
+         } // discard
 
-               callback(null, translations);
-             }
-           })["catch"](function (err) {
-             if (callback) callback(err.message);
-           });
+
+         var ignoredBuildings = {
+           demolished: true,
+           dismantled: true,
+           proposed: true,
+           razed: true
+         };
+
+         function getFeatureType(entity, graph) {
+           var geometry = entity.geometry(graph);
+           if (geometry !== 'line' && geometry !== 'area') return null;
+           var tags = entity.tags;
+           if (hasTag(tags, 'building') && !ignoredBuildings[tags.building]) return 'building';
+           if (hasTag(tags, 'highway') && osmRoutableHighwayTagValues[tags.highway]) return 'highway'; // don't check railway or waterway areas
+
+           if (geometry !== 'line') return null;
+           if (hasTag(tags, 'railway') && osmRailwayTrackTagValues[tags.railway]) return 'railway';
+           if (hasTag(tags, 'waterway') && osmFlowingWaterwayTagValues[tags.waterway]) return 'waterway';
+           return null;
          }
-       };
 
-       var services = {
-         geocoder: serviceNominatim,
-         keepRight: serviceKeepRight,
-         improveOSM: serviceImproveOSM,
-         osmose: serviceOsmose,
-         mapillary: serviceMapillary,
-         openstreetcam: serviceOpenstreetcam,
-         osm: serviceOsm,
-         osmWikibase: serviceOsmWikibase,
-         maprules: serviceMapRules,
-         streetside: serviceStreetside,
-         taginfo: serviceTaginfo,
-         vectorTile: serviceVectorTile,
-         wikidata: serviceWikidata,
-         wikipedia: serviceWikipedia
-       };
+         function isLegitCrossing(tags1, featureType1, tags2, featureType2) {
+           // assume 0 by default
+           var level1 = tags1.level || '0';
+           var level2 = tags2.level || '0';
 
-       function svgIcon(name, svgklass, useklass) {
-         return function drawIcon(selection) {
-           selection.selectAll('svg.icon' + (svgklass ? '.' + svgklass.split(' ')[0] : '')).data([0]).enter().append('svg').attr('class', 'icon ' + (svgklass || '')).append('use').attr('xlink:href', name).attr('class', useklass);
+           if (taggedAsIndoor(tags1) && taggedAsIndoor(tags2) && level1 !== level2) {
+             // assume features don't interact if they're indoor on different levels
+             return true;
+           } // assume 0 by default; don't use way.layer() since we account for structures here
+
+
+           var layer1 = tags1.layer || '0';
+           var layer2 = tags2.layer || '0';
+
+           if (allowsBridge(featureType1) && allowsBridge(featureType2)) {
+             if (hasTag(tags1, 'bridge') && !hasTag(tags2, 'bridge')) return true;
+             if (!hasTag(tags1, 'bridge') && hasTag(tags2, 'bridge')) return true; // crossing bridges must use different layers
+
+             if (hasTag(tags1, 'bridge') && hasTag(tags2, 'bridge') && layer1 !== layer2) return true;
+           } else if (allowsBridge(featureType1) && hasTag(tags1, 'bridge')) return true;else if (allowsBridge(featureType2) && hasTag(tags2, 'bridge')) return true;
+
+           if (allowsTunnel(featureType1) && allowsTunnel(featureType2)) {
+             if (hasTag(tags1, 'tunnel') && !hasTag(tags2, 'tunnel')) return true;
+             if (!hasTag(tags1, 'tunnel') && hasTag(tags2, 'tunnel')) return true; // crossing tunnels must use different layers
+
+             if (hasTag(tags1, 'tunnel') && hasTag(tags2, 'tunnel') && layer1 !== layer2) return true;
+           } else if (allowsTunnel(featureType1) && hasTag(tags1, 'tunnel')) return true;else if (allowsTunnel(featureType2) && hasTag(tags2, 'tunnel')) return true; // don't flag crossing waterways and pier/highways
+
+
+           if (featureType1 === 'waterway' && featureType2 === 'highway' && tags2.man_made === 'pier') return true;
+           if (featureType2 === 'waterway' && featureType1 === 'highway' && tags1.man_made === 'pier') return true;
+
+           if (featureType1 === 'building' || featureType2 === 'building') {
+             // for building crossings, different layers are enough
+             if (layer1 !== layer2) return true;
+           }
+
+           return false;
+         } // highway values for which we shouldn't recommend connecting to waterways
+
+
+         var highwaysDisallowingFords = {
+           motorway: true,
+           motorway_link: true,
+           trunk: true,
+           trunk_link: true,
+           primary: true,
+           primary_link: true,
+           secondary: true,
+           secondary_link: true
+         };
+         var nonCrossingHighways = {
+           track: true
          };
-       }
 
-       function uiNoteComments() {
-         var _note;
+         function tagsForConnectionNodeIfAllowed(entity1, entity2, graph) {
+           var featureType1 = getFeatureType(entity1, graph);
+           var featureType2 = getFeatureType(entity2, graph);
+           var geometry1 = entity1.geometry(graph);
+           var geometry2 = entity2.geometry(graph);
+           var bothLines = geometry1 === 'line' && geometry2 === 'line';
 
-         function noteComments(selection) {
-           if (_note.isNew()) return; // don't draw .comments-container
+           if (featureType1 === featureType2) {
+             if (featureType1 === 'highway') {
+               var entity1IsPath = osmPathHighwayTagValues[entity1.tags.highway];
+               var entity2IsPath = osmPathHighwayTagValues[entity2.tags.highway];
 
-           var comments = selection.selectAll('.comments-container').data([0]);
-           comments = comments.enter().append('div').attr('class', 'comments-container').merge(comments);
-           var commentEnter = comments.selectAll('.comment').data(_note.comments).enter().append('div').attr('class', 'comment');
-           commentEnter.append('div').attr('class', function (d) {
-             return 'comment-avatar user-' + d.uid;
-           }).call(svgIcon('#iD-icon-avatar', 'comment-avatar-icon'));
-           var mainEnter = commentEnter.append('div').attr('class', 'comment-main');
-           var metadataEnter = mainEnter.append('div').attr('class', 'comment-metadata');
-           metadataEnter.append('div').attr('class', 'comment-author').each(function (d) {
-             var selection = select(this);
-             var osm = services.osm;
+               if ((entity1IsPath || entity2IsPath) && entity1IsPath !== entity2IsPath) {
+                 // one feature is a path but not both
+                 var roadFeature = entity1IsPath ? entity2 : entity1;
 
-             if (osm && d.user) {
-               selection = selection.append('a').attr('class', 'comment-author-link').attr('href', osm.userURL(d.user)).attr('target', '_blank');
-             }
+                 if (nonCrossingHighways[roadFeature.tags.highway]) {
+                   // don't mark path connections with certain roads as crossings
+                   return {};
+                 }
 
-             selection.html(function (d) {
-               return d.user || _t.html('note.anonymous');
-             });
-           });
-           metadataEnter.append('div').attr('class', 'comment-date').html(function (d) {
-             return _t('note.status.' + d.action, {
-               when: localeDateString(d.date)
-             });
-           });
-           mainEnter.append('div').attr('class', 'comment-text').html(function (d) {
-             return d.html;
-           }).selectAll('a').attr('rel', 'noopener nofollow').attr('target', '_blank');
-           comments.call(replaceAvatars);
-         }
+                 var pathFeature = entity1IsPath ? entity1 : entity2;
 
-         function replaceAvatars(selection) {
-           var showThirdPartyIcons = corePreferences('preferences.privacy.thirdpartyicons') || 'true';
-           var osm = services.osm;
-           if (showThirdPartyIcons !== 'true' || !osm) return;
-           var uids = {}; // gather uids in the comment thread
+                 if (['marked', 'unmarked'].indexOf(pathFeature.tags.crossing) !== -1) {
+                   // if the path is a crossing, match the crossing type
+                   return bothLines ? {
+                     highway: 'crossing',
+                     crossing: pathFeature.tags.crossing
+                   } : {};
+                 } // don't add a `crossing` subtag to ambiguous crossings
 
-           _note.comments.forEach(function (d) {
-             if (d.uid) uids[d.uid] = true;
-           });
 
-           Object.keys(uids).forEach(function (uid) {
-             osm.loadUser(uid, function (err, user) {
-               if (!user || !user.image_url) return;
-               selection.selectAll('.comment-avatar.user-' + uid).html('').append('img').attr('class', 'icon comment-avatar-icon').attr('src', user.image_url).attr('alt', user.display_name);
-             });
-           });
-         }
+                 return bothLines ? {
+                   highway: 'crossing'
+                 } : {};
+               }
 
-         function localeDateString(s) {
-           if (!s) return null;
-           var options = {
-             day: 'numeric',
-             month: 'short',
-             year: 'numeric'
-           };
-           s = s.replace(/-/g, '/'); // fix browser-specific Date() issues
+               return {};
+             }
 
-           var d = new Date(s);
-           if (isNaN(d.getTime())) return null;
-           return d.toLocaleDateString(_mainLocalizer.localeCode(), options);
-         }
+             if (featureType1 === 'waterway') return {};
+             if (featureType1 === 'railway') return {};
+           } else {
+             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-tram connections use this tag
+                   if (isTram) return {
+                     railway: 'tram_crossing'
+                   }; // other path-rail connections use this tag
 
-         noteComments.note = function (val) {
-           if (!arguments.length) return _note;
-           _note = val;
-           return noteComments;
-         };
+                   return {
+                     railway: 'crossing'
+                   };
+                 } else {
+                   // path-tram connections use this tag
+                   if (isTram) return {
+                     railway: 'tram_level_crossing'
+                   }; // other road-rail connections use this tag
 
-         return noteComments;
-       }
+                   return {
+                     railway: 'level_crossing'
+                   };
+                 }
+               }
 
-       function uiNoteHeader() {
-         var _note;
+               if (featureTypes.indexOf('waterway') !== -1) {
+                 // do not allow fords on structures
+                 if (hasTag(entity1.tags, 'tunnel') && hasTag(entity2.tags, 'tunnel')) return null;
+                 if (hasTag(entity1.tags, 'bridge') && hasTag(entity2.tags, 'bridge')) return null;
 
-         function noteHeader(selection) {
-           var header = selection.selectAll('.note-header').data(_note ? [_note] : [], function (d) {
-             return d.status + d.id;
-           });
-           header.exit().remove();
-           var headerEnter = header.enter().append('div').attr('class', 'note-header');
-           var iconEnter = headerEnter.append('div').attr('class', function (d) {
-             return 'note-header-icon ' + d.status;
-           }).classed('new', function (d) {
-             return d.id < 0;
-           });
-           iconEnter.append('div').attr('class', 'preset-icon-28').call(svgIcon('#iD-icon-note', 'note-fill'));
-           iconEnter.each(function (d) {
-             var statusIcon = '#iD-icon-' + (d.id < 0 ? 'plus' : d.status === 'open' ? 'close' : 'apply');
-             iconEnter.append('div').attr('class', 'note-icon-annotation').call(svgIcon(statusIcon, 'icon-annotation'));
-           });
-           headerEnter.append('div').attr('class', 'note-header-label').html(function (d) {
-             if (_note.isNew()) {
-               return _t('note.new');
+                 if (highwaysDisallowingFords[entity1.tags.highway] || highwaysDisallowingFords[entity2.tags.highway]) {
+                   // do not allow fords on major highways
+                   return null;
+                 }
+
+                 return bothLines ? {
+                   ford: 'yes'
+                 } : {};
+               }
              }
+           }
 
-             return _t('note.note') + ' ' + d.id + ' ' + (d.status === 'closed' ? _t('note.closed') : '');
-           });
+           return null;
          }
 
-         noteHeader.note = function (val) {
-           if (!arguments.length) return _note;
-           _note = val;
-           return noteHeader;
-         };
+         function findCrossingsByWay(way1, graph, tree) {
+           var edgeCrossInfos = [];
+           if (way1.type !== 'way') return edgeCrossInfos;
+           var taggedFeature1 = getFeatureWithFeatureTypeTagsForWay(way1, graph);
+           var way1FeatureType = getFeatureType(taggedFeature1, graph);
+           if (way1FeatureType === null) return edgeCrossInfos;
+           var checkedSingleCrossingWays = {}; // declare vars ahead of time to reduce garbage collection
 
-         return noteHeader;
-       }
+           var i, j;
+           var extent;
+           var n1, n2, nA, nB, nAId, nBId;
+           var segment1, segment2;
+           var oneOnly;
+           var segmentInfos, segment2Info, way2, taggedFeature2, way2FeatureType;
+           var way1Nodes = graph.childNodes(way1);
+           var comparedWays = {};
 
-       function uiNoteReport() {
-         var _note;
+           for (i = 0; i < way1Nodes.length - 1; i++) {
+             n1 = way1Nodes[i];
+             n2 = way1Nodes[i + 1];
+             extent = geoExtent([[Math.min(n1.loc[0], n2.loc[0]), Math.min(n1.loc[1], n2.loc[1])], [Math.max(n1.loc[0], n2.loc[0]), Math.max(n1.loc[1], n2.loc[1])]]); // Optimize by only checking overlapping segments, not every segment
+             // of overlapping ways
 
-         function noteReport(selection) {
-           var url;
+             segmentInfos = tree.waySegments(extent, graph);
 
-           if (services.osm && _note instanceof osmNote && !_note.isNew()) {
-             url = services.osm.noteReportURL(_note);
-           }
+             for (j = 0; j < segmentInfos.length; j++) {
+               segment2Info = segmentInfos[j]; // don't check for self-intersection in this validation
 
-           var link = selection.selectAll('.note-report').data(url ? [url] : []); // exit
+               if (segment2Info.wayId === way1.id) continue; // skip if this way was already checked and only one issue is needed
 
-           link.exit().remove(); // enter
+               if (checkedSingleCrossingWays[segment2Info.wayId]) continue; // mark this way as checked even if there are no crossings
 
-           var linkEnter = link.enter().append('a').attr('class', 'note-report').attr('target', '_blank').attr('href', function (d) {
-             return d;
-           }).call(svgIcon('#iD-icon-out-link', 'inline'));
-           linkEnter.append('span').html(_t.html('note.report'));
-         }
+               comparedWays[segment2Info.wayId] = true;
+               way2 = graph.hasEntity(segment2Info.wayId);
+               if (!way2) continue;
+               taggedFeature2 = getFeatureWithFeatureTypeTagsForWay(way2, graph); // only check crossing highway, waterway, building, and railway
 
-         noteReport.note = function (val) {
-           if (!arguments.length) return _note;
-           _note = val;
-           return noteReport;
-         };
+               way2FeatureType = getFeatureType(taggedFeature2, graph);
 
-         return noteReport;
-       }
+               if (way2FeatureType === null || isLegitCrossing(taggedFeature1.tags, way1FeatureType, taggedFeature2.tags, way2FeatureType)) {
+                 continue;
+               } // create only one issue for building crossings
 
-       function uiViewOnOSM(context) {
-         var _what; // an osmEntity or osmNote
 
+               oneOnly = way1FeatureType === 'building' || way2FeatureType === 'building';
+               nAId = segment2Info.nodes[0];
+               nBId = segment2Info.nodes[1];
 
-         function viewOnOSM(selection) {
-           var url;
+               if (nAId === n1.id || nAId === n2.id || nBId === n1.id || nBId === n2.id) {
+                 // n1 or n2 is a connection node; skip
+                 continue;
+               }
 
-           if (_what instanceof osmEntity) {
-             url = context.connection().entityURL(_what);
-           } else if (_what instanceof osmNote) {
-             url = context.connection().noteURL(_what);
-           }
+               nA = graph.hasEntity(nAId);
+               if (!nA) continue;
+               nB = graph.hasEntity(nBId);
+               if (!nB) continue;
+               segment1 = [n1.loc, n2.loc];
+               segment2 = [nA.loc, nB.loc];
+               var point = geoLineIntersection(segment1, segment2);
 
-           var data = !_what || _what.isNew() ? [] : [_what];
-           var link = selection.selectAll('.view-on-osm').data(data, function (d) {
-             return d.id;
-           }); // exit
+               if (point) {
+                 edgeCrossInfos.push({
+                   wayInfos: [{
+                     way: way1,
+                     featureType: way1FeatureType,
+                     edge: [n1.id, n2.id]
+                   }, {
+                     way: way2,
+                     featureType: way2FeatureType,
+                     edge: [nA.id, nB.id]
+                   }],
+                   crossPoint: point
+                 });
 
-           link.exit().remove(); // enter
+                 if (oneOnly) {
+                   checkedSingleCrossingWays[way2.id] = true;
+                   break;
+                 }
+               }
+             }
+           }
 
-           var linkEnter = link.enter().append('a').attr('class', 'view-on-osm').attr('target', '_blank').attr('href', url).call(svgIcon('#iD-icon-out-link', 'inline'));
-           linkEnter.append('span').html(_t.html('inspector.view_on_osm'));
+           return edgeCrossInfos;
          }
 
-         viewOnOSM.what = function (_) {
-           if (!arguments.length) return _what;
-           _what = _;
-           return viewOnOSM;
-         };
+         function waysToCheck(entity, graph) {
+           var featureType = getFeatureType(entity, graph);
+           if (!featureType) return [];
 
-         return viewOnOSM;
-       }
+           if (entity.type === 'way') {
+             return [entity];
+           } else if (entity.type === 'relation') {
+             return entity.members.reduce(function (array, member) {
+               if (member.type === 'way' && ( // only look at geometry ways
+               !member.role || member.role === 'outer' || member.role === 'inner')) {
+                 var entity = graph.hasEntity(member.id); // don't add duplicates
 
-       function uiNoteEditor(context) {
-         var dispatch$1 = dispatch('change');
-         var noteComments = uiNoteComments();
-         var noteHeader = uiNoteHeader(); // var formFields = uiFormFields(context);
+                 if (entity && array.indexOf(entity) === -1) {
+                   array.push(entity);
+                 }
+               }
 
-         var _note;
+               return array;
+             }, []);
+           }
 
-         var _newNote; // var _fieldsArr;
+           return [];
+         }
 
+         var validation = function checkCrossingWays(entity, graph) {
+           var tree = context.history().tree();
+           var ways = waysToCheck(entity, graph);
+           var issues = []; // declare these here to reduce garbage collection
 
-         function noteEditor(selection) {
-           var header = selection.selectAll('.header').data([0]);
-           var headerEnter = header.enter().append('div').attr('class', 'header fillL');
-           headerEnter.append('button').attr('class', 'close').on('click', function () {
-             context.enter(modeBrowse(context));
-           }).call(svgIcon('#iD-icon-close'));
-           headerEnter.append('h3').html(_t.html('note.title'));
-           var body = selection.selectAll('.body').data([0]);
-           body = body.enter().append('div').attr('class', 'body').merge(body);
-           var editor = body.selectAll('.note-editor').data([0]);
-           editor.enter().append('div').attr('class', 'modal-section note-editor').merge(editor).call(noteHeader.note(_note)).call(noteComments.note(_note)).call(noteSaveSection);
-           var footer = selection.selectAll('.footer').data([0]);
-           footer.enter().append('div').attr('class', 'footer').merge(footer).call(uiViewOnOSM(context).what(_note)).call(uiNoteReport().note(_note)); // rerender the note editor on any auth change
+           var wayIndex, crossingIndex, crossings;
 
-           var osm = services.osm;
+           for (wayIndex in ways) {
+             crossings = findCrossingsByWay(ways[wayIndex], graph, tree);
 
-           if (osm) {
-             osm.on('change.note-save', function () {
-               selection.call(noteEditor);
-             });
+             for (crossingIndex in crossings) {
+               issues.push(createIssue(crossings[crossingIndex], graph));
+             }
            }
-         }
-
-         function noteSaveSection(selection) {
-           var isSelected = _note && _note.id === context.selectedNoteID();
 
-           var noteSave = selection.selectAll('.note-save').data(isSelected ? [_note] : [], function (d) {
-             return d.status + d.id;
-           }); // exit
+           return issues;
+         };
 
-           noteSave.exit().remove(); // enter
+         function createIssue(crossing, graph) {
+           // use the entities with the tags that define the feature type
+           crossing.wayInfos.sort(function (way1Info, way2Info) {
+             var type1 = way1Info.featureType;
+             var type2 = way2Info.featureType;
 
-           var noteSaveEnter = noteSave.enter().append('div').attr('class', 'note-save save-section cf'); // // if new note, show categories to pick from
-           // if (_note.isNew()) {
-           //     var presets = presetManager;
-           //     // NOTE: this key isn't a age and therefore there is no documentation (yet)
-           //     _fieldsArr = [
-           //         uiField(context, presets.field('category'), null, { show: true, revert: false }),
-           //     ];
-           //     _fieldsArr.forEach(function(field) {
-           //         field
-           //             .on('change', changeCategory);
-           //     });
-           //     noteSaveEnter
-           //         .append('div')
-           //         .attr('class', 'note-category')
-           //         .call(formFields.fieldsArr(_fieldsArr));
-           // }
-           // function changeCategory() {
-           //     // NOTE: perhaps there is a better way to get value
-           //     var val = context.container().select('input[name=\'category\']:checked').property('__data__') || undefined;
-           //     // store the unsaved category with the note itself
-           //     _note = _note.update({ newCategory: val });
-           //     var osm = services.osm;
-           //     if (osm) {
-           //         osm.replaceNote(_note);  // update note cache
-           //     }
-           //     noteSave
-           //         .call(noteSaveButtons);
-           // }
+             if (type1 === type2) {
+               return utilDisplayLabel(way1Info.way, graph) > utilDisplayLabel(way2Info.way, graph);
+             } else if (type1 === 'waterway') {
+               return true;
+             } else if (type2 === 'waterway') {
+               return false;
+             }
 
-           noteSaveEnter.append('h4').attr('class', '.note-save-header').html(function () {
-             return _note.isNew() ? _t('note.newDescription') : _t('note.newComment');
+             return type1 < type2;
            });
-           var commentTextarea = noteSaveEnter.append('textarea').attr('class', 'new-comment-input').attr('placeholder', _t('note.inputPlaceholder')).attr('maxlength', 1000).property('value', function (d) {
-             return d.newComment;
-           }).call(utilNoAuto).on('keydown.note-input', keydown).on('input.note-input', changeInput).on('blur.note-input', changeInput);
-
-           if (!commentTextarea.empty() && _newNote) {
-             // autofocus the comment field for new notes
-             commentTextarea.node().focus();
-           } // update
+           var entities = crossing.wayInfos.map(function (wayInfo) {
+             return getFeatureWithFeatureTypeTagsForWay(wayInfo.way, graph);
+           });
+           var edges = [crossing.wayInfos[0].edge, crossing.wayInfos[1].edge];
+           var featureTypes = [crossing.wayInfos[0].featureType, crossing.wayInfos[1].featureType];
+           var connectionTags = tagsForConnectionNodeIfAllowed(entities[0], entities[1], graph);
+           var featureType1 = crossing.wayInfos[0].featureType;
+           var featureType2 = crossing.wayInfos[1].featureType;
+           var isCrossingIndoors = taggedAsIndoor(entities[0].tags) && taggedAsIndoor(entities[1].tags);
+           var isCrossingTunnels = allowsTunnel(featureType1) && hasTag(entities[0].tags, 'tunnel') && allowsTunnel(featureType2) && hasTag(entities[1].tags, 'tunnel');
+           var isCrossingBridges = allowsBridge(featureType1) && hasTag(entities[0].tags, 'bridge') && allowsBridge(featureType2) && hasTag(entities[1].tags, 'bridge');
+           var subtype = [featureType1, featureType2].sort().join('-');
+           var crossingTypeID = subtype;
 
+           if (isCrossingIndoors) {
+             crossingTypeID = 'indoor-indoor';
+           } else if (isCrossingTunnels) {
+             crossingTypeID = 'tunnel-tunnel';
+           } else if (isCrossingBridges) {
+             crossingTypeID = 'bridge-bridge';
+           }
 
-           noteSave = noteSaveEnter.merge(noteSave).call(userDetails).call(noteSaveButtons); // fast submit if user presses cmd+enter
+           if (connectionTags && (isCrossingIndoors || isCrossingTunnels || isCrossingBridges)) {
+             crossingTypeID += '_connectable';
+           }
 
-           function keydown(d3_event) {
-             if (!(d3_event.keyCode === 13 && // ↩ Return
-             d3_event.metaKey)) return;
-             var osm = services.osm;
-             if (!osm) return;
-             var hasAuth = osm.authenticated();
-             if (!hasAuth) return;
-             if (!_note.newComment) return;
-             d3_event.preventDefault();
-             select(this).on('keydown.note-input', null); // focus on button and submit
+           return new validationIssue({
+             type: type,
+             subtype: subtype,
+             severity: 'warning',
+             message: function message(context) {
+               var graph = context.graph();
+               var entity1 = graph.hasEntity(this.entityIds[0]),
+                   entity2 = graph.hasEntity(this.entityIds[1]);
+               return entity1 && entity2 ? _t.html('issues.crossing_ways.message', {
+                 feature: utilDisplayLabel(entity1, graph),
+                 feature2: utilDisplayLabel(entity2, graph)
+               }) : '';
+             },
+             reference: showReference,
+             entityIds: entities.map(function (entity) {
+               return entity.id;
+             }),
+             data: {
+               edges: edges,
+               featureTypes: featureTypes,
+               connectionTags: connectionTags
+             },
+             // differentiate based on the loc since two ways can cross multiple times
+             hash: crossing.crossPoint.toString() + // if the edges change then so does the fix
+             edges.slice().sort(function (edge1, edge2) {
+               // order to assure hash is deterministic
+               return edge1[0] < edge2[0] ? -1 : 1;
+             }).toString() + // ensure the correct connection tags are added in the fix
+             JSON.stringify(connectionTags),
+             loc: crossing.crossPoint,
+             dynamicFixes: function dynamicFixes(context) {
+               var mode = context.mode();
+               if (!mode || mode.id !== 'select' || mode.selectedIDs().length !== 1) return [];
+               var selectedIndex = this.entityIds[0] === mode.selectedIDs()[0] ? 0 : 1;
+               var selectedFeatureType = this.data.featureTypes[selectedIndex];
+               var otherFeatureType = this.data.featureTypes[selectedIndex === 0 ? 1 : 0];
+               var fixes = [];
 
-             window.setTimeout(function () {
-               if (_note.isNew()) {
-                 noteSave.selectAll('.save-button').node().focus();
-                 clickSave();
-               } else {
-                 noteSave.selectAll('.comment-button').node().focus();
-                 clickComment();
+               if (connectionTags) {
+                 fixes.push(makeConnectWaysFix(this.data.connectionTags));
                }
-             }, 10);
-           }
-
-           function changeInput() {
-             var input = select(this);
-             var val = input.property('value').trim() || undefined; // store the unsaved comment with the note itself
 
-             _note = _note.update({
-               newComment: val
-             });
-             var osm = services.osm;
+               if (isCrossingIndoors) {
+                 fixes.push(new validationIssueFix({
+                   icon: 'iD-icon-layers',
+                   title: _t.html('issues.fix.use_different_levels.title')
+                 }));
+               } else if (isCrossingTunnels || isCrossingBridges || featureType1 === 'building' || featureType2 === 'building') {
+                 fixes.push(makeChangeLayerFix('higher'));
+                 fixes.push(makeChangeLayerFix('lower')); // can only add bridge/tunnel if both features are lines
+               } 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 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 uncommon
 
-             if (osm) {
-               osm.replaceNote(_note); // update note cache
-             }
 
-             noteSave.call(noteSaveButtons);
-           }
-         }
+                 var skipTunnelFix = otherFeatureType === 'waterway' && selectedFeatureType !== 'waterway';
 
-         function userDetails(selection) {
-           var detailSection = selection.selectAll('.detail-section').data([0]);
-           detailSection = detailSection.enter().append('div').attr('class', 'detail-section').merge(detailSection);
-           var osm = services.osm;
-           if (!osm) return; // Add warning if user is not logged in
+                 if (allowsTunnel(selectedFeatureType) && !skipTunnelFix) {
+                   fixes.push(makeAddBridgeOrTunnelFix('add_a_tunnel', 'temaki-tunnel', 'tunnel'));
+                 }
+               } // repositioning the features is always an option
 
-           var hasAuth = osm.authenticated();
-           var authWarning = detailSection.selectAll('.auth-warning').data(hasAuth ? [] : [0]);
-           authWarning.exit().transition().duration(200).style('opacity', 0).remove();
-           var authEnter = authWarning.enter().insert('div', '.tag-reference-body').attr('class', 'field-warning auth-warning').style('opacity', 0);
-           authEnter.call(svgIcon('#iD-icon-alert', 'inline'));
-           authEnter.append('span').html(_t.html('note.login'));
-           authEnter.append('a').attr('target', '_blank').call(svgIcon('#iD-icon-out-link', 'inline')).append('span').html(_t.html('login')).on('click.note-login', function (d3_event) {
-             d3_event.preventDefault();
-             osm.authenticate();
-           });
-           authEnter.transition().duration(200).style('opacity', 1);
-           var prose = detailSection.selectAll('.note-save-prose').data(hasAuth ? [0] : []);
-           prose.exit().remove();
-           prose = prose.enter().append('p').attr('class', 'note-save-prose').html(_t.html('note.upload_explanation')).merge(prose);
-           osm.userDetails(function (err, user) {
-             if (err) return;
-             var userLink = select(document.createElement('div'));
 
-             if (user.image_url) {
-               userLink.append('img').attr('src', user.image_url).attr('class', 'icon pre-text user-icon');
+               fixes.push(new validationIssueFix({
+                 icon: 'iD-operation-move',
+                 title: _t.html('issues.fix.reposition_features.title')
+               }));
+               return fixes;
              }
-
-             userLink.append('a').attr('class', 'user-info').html(user.display_name).attr('href', osm.userURL(user.display_name)).attr('target', '_blank');
-             prose.html(_t.html('note.upload_explanation_with_user', {
-               user: userLink.html()
-             }));
            });
-         }
 
-         function noteSaveButtons(selection) {
-           var osm = services.osm;
-           var hasAuth = osm && osm.authenticated();
+           function showReference(selection) {
+             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.crossing_ways.' + crossingTypeID + '.reference'));
+           }
+         }
 
-           var isSelected = _note && _note.id === context.selectedNoteID();
+         function makeAddBridgeOrTunnelFix(fixTitleID, iconName, bridgeOrTunnel) {
+           return new validationIssueFix({
+             icon: iconName,
+             title: _t.html('issues.fix.' + fixTitleID + '.title'),
+             onClick: function onClick(context) {
+               var mode = context.mode();
+               if (!mode || mode.id !== 'select') return;
+               var selectedIDs = mode.selectedIDs();
+               if (selectedIDs.length !== 1) return;
+               var selectedWayID = selectedIDs[0];
+               if (!context.hasEntity(selectedWayID)) return;
+               var resultWayIDs = [selectedWayID];
+               var edge, crossedEdge, crossedWayID;
 
-           var buttonSection = selection.selectAll('.buttons').data(isSelected ? [_note] : [], function (d) {
-             return d.status + d.id;
-           }); // exit
+               if (this.issue.entityIds[0] === selectedWayID) {
+                 edge = this.issue.data.edges[0];
+                 crossedEdge = this.issue.data.edges[1];
+                 crossedWayID = this.issue.entityIds[1];
+               } else {
+                 edge = this.issue.data.edges[1];
+                 crossedEdge = this.issue.data.edges[0];
+                 crossedWayID = this.issue.entityIds[0];
+               }
 
-           buttonSection.exit().remove(); // enter
+               var crossingLoc = this.issue.loc;
+               var projection = context.projection;
 
-           var buttonEnter = buttonSection.enter().append('div').attr('class', 'buttons');
+               var action = function actionAddStructure(graph) {
+                 var edgeNodes = [graph.entity(edge[0]), graph.entity(edge[1])];
+                 var crossedWay = graph.hasEntity(crossedWayID); // use the explicit width of the crossed feature as the structure length, if available
 
-           if (_note.isNew()) {
-             buttonEnter.append('button').attr('class', 'button cancel-button secondary-action').html(_t.html('confirm.cancel'));
-             buttonEnter.append('button').attr('class', 'button save-button action').html(_t.html('note.save'));
-           } else {
-             buttonEnter.append('button').attr('class', 'button status-button action');
-             buttonEnter.append('button').attr('class', 'button comment-button action').html(_t.html('note.comment'));
-           } // update
+                 var structLengthMeters = crossedWay && crossedWay.tags.width && parseFloat(crossedWay.tags.width);
 
+                 if (!structLengthMeters) {
+                   // if no explicit width is set, approximate the width based on the tags
+                   structLengthMeters = crossedWay && crossedWay.impliedLineWidthMeters();
+                 }
 
-           buttonSection = buttonSection.merge(buttonEnter);
-           buttonSection.select('.cancel-button') // select and propagate data
-           .on('click.cancel', clickCancel);
-           buttonSection.select('.save-button') // select and propagate data
-           .attr('disabled', isSaveDisabled).on('click.save', clickSave);
-           buttonSection.select('.status-button') // select and propagate data
-           .attr('disabled', hasAuth ? null : true).html(function (d) {
-             var action = d.status === 'open' ? 'close' : 'open';
-             var andComment = d.newComment ? '_comment' : '';
-             return _t('note.' + action + andComment);
-           }).on('click.status', clickStatus);
-           buttonSection.select('.comment-button') // select and propagate data
-           .attr('disabled', isSaveDisabled).on('click.comment', clickComment);
+                 if (structLengthMeters) {
+                   if (getFeatureType(crossedWay, graph) === 'railway') {
+                     // bridges over railways are generally much longer than the rail bed itself, compensate
+                     structLengthMeters *= 2;
+                   }
+                 } else {
+                   // should ideally never land here since all rail/water/road tags should have an implied width
+                   structLengthMeters = 8;
+                 }
 
-           function isSaveDisabled(d) {
-             return hasAuth && d.status === 'open' && d.newComment ? null : true;
-           }
-         }
+                 var a1 = geoAngle(edgeNodes[0], edgeNodes[1], projection) + Math.PI;
+                 var a2 = geoAngle(graph.entity(crossedEdge[0]), graph.entity(crossedEdge[1]), projection) + Math.PI;
+                 var crossingAngle = Math.max(a1, a2) - Math.min(a1, a2);
+                 if (crossingAngle > Math.PI) crossingAngle -= Math.PI; // lengthen the structure to account for the angle of the crossing
 
-         function clickCancel(d3_event, d) {
-           this.blur(); // avoid keeping focus on the button - #4641
+                 structLengthMeters = structLengthMeters / 2 / Math.sin(crossingAngle) * 2; // add padding since the structure must extend past the edges of the crossed feature
 
-           var osm = services.osm;
+                 structLengthMeters += 4; // clamp the length to a reasonable range
 
-           if (osm) {
-             osm.removeNote(d);
-           }
+                 structLengthMeters = Math.min(Math.max(structLengthMeters, 4), 50);
 
-           context.enter(modeBrowse(context));
-           dispatch$1.call('change');
-         }
+                 function geomToProj(geoPoint) {
+                   return [geoLonToMeters(geoPoint[0], geoPoint[1]), geoLatToMeters(geoPoint[1])];
+                 }
 
-         function clickSave(d3_event, d) {
-           this.blur(); // avoid keeping focus on the button - #4641
+                 function projToGeom(projPoint) {
+                   var lat = geoMetersToLat(projPoint[1]);
+                   return [geoMetersToLon(projPoint[0], lat), lat];
+                 }
 
-           var osm = services.osm;
+                 var projEdgeNode1 = geomToProj(edgeNodes[0].loc);
+                 var projEdgeNode2 = geomToProj(edgeNodes[1].loc);
+                 var projectedAngle = geoVecAngle(projEdgeNode1, projEdgeNode2);
+                 var projectedCrossingLoc = geomToProj(crossingLoc);
+                 var linearToSphericalMetersRatio = geoVecLength(projEdgeNode1, projEdgeNode2) / geoSphericalDistance(edgeNodes[0].loc, edgeNodes[1].loc);
 
-           if (osm) {
-             osm.postNoteCreate(d, function (err, note) {
-               dispatch$1.call('change', note);
-             });
-           }
-         }
+                 function locSphericalDistanceFromCrossingLoc(angle, distanceMeters) {
+                   var lengthSphericalMeters = distanceMeters * linearToSphericalMetersRatio;
+                   return projToGeom([projectedCrossingLoc[0] + Math.cos(angle) * lengthSphericalMeters, projectedCrossingLoc[1] + Math.sin(angle) * lengthSphericalMeters]);
+                 }
 
-         function clickStatus(d3_event, d) {
-           this.blur(); // avoid keeping focus on the button - #4641
+                 var endpointLocGetter1 = function endpointLocGetter1(lengthMeters) {
+                   return locSphericalDistanceFromCrossingLoc(projectedAngle, lengthMeters);
+                 };
 
-           var osm = services.osm;
+                 var endpointLocGetter2 = function endpointLocGetter2(lengthMeters) {
+                   return locSphericalDistanceFromCrossingLoc(projectedAngle + Math.PI, lengthMeters);
+                 }; // avoid creating very short edges from splitting too close to another node
 
-           if (osm) {
-             var setStatus = d.status === 'open' ? 'closed' : 'open';
-             osm.postNoteUpdate(d, setStatus, function (err, note) {
-               dispatch$1.call('change', note);
-             });
-           }
-         }
 
-         function clickComment(d3_event, d) {
-           this.blur(); // avoid keeping focus on the button - #4641
+                 var minEdgeLengthMeters = 0.55; // decide where to bound the structure along the way, splitting as necessary
 
-           var osm = services.osm;
+                 function determineEndpoint(edge, endNode, locGetter) {
+                   var newNode;
+                   var idealLengthMeters = structLengthMeters / 2; // distance between the crossing location and the end of the edge,
+                   // the maximum length of this side of the structure
 
-           if (osm) {
-             osm.postNoteUpdate(d, d.status, function (err, note) {
-               dispatch$1.call('change', note);
-             });
-           }
-         }
+                   var crossingToEdgeEndDistance = geoSphericalDistance(crossingLoc, endNode.loc);
 
-         noteEditor.note = function (val) {
-           if (!arguments.length) return _note;
-           _note = val;
-           return noteEditor;
-         };
+                   if (crossingToEdgeEndDistance - idealLengthMeters > minEdgeLengthMeters) {
+                     // the edge is long enough to insert a new node
+                     // the loc that would result in the full expected length
+                     var idealNodeLoc = locGetter(idealLengthMeters);
+                     newNode = osmNode();
+                     graph = actionAddMidpoint({
+                       loc: idealNodeLoc,
+                       edge: edge
+                     }, newNode)(graph);
+                   } else {
+                     var edgeCount = 0;
+                     endNode.parentIntersectionWays(graph).forEach(function (way) {
+                       way.nodes.forEach(function (nodeID) {
+                         if (nodeID === endNode.id) {
+                           if (endNode.id === way.first() && endNode.id !== way.last() || endNode.id === way.last() && endNode.id !== way.first()) {
+                             edgeCount += 1;
+                           } else {
+                             edgeCount += 2;
+                           }
+                         }
+                       });
+                     });
 
-         noteEditor.newNote = function (val) {
-           if (!arguments.length) return _newNote;
-           _newNote = val;
-           return noteEditor;
-         };
+                     if (edgeCount >= 3) {
+                       // the end node is a junction, try to leave a segment
+                       // between it and the structure - #7202
+                       var insetLength = crossingToEdgeEndDistance - minEdgeLengthMeters;
 
-         return utilRebind(noteEditor, dispatch$1, 'on');
-       }
+                       if (insetLength > minEdgeLengthMeters) {
+                         var insetNodeLoc = locGetter(insetLength);
+                         newNode = osmNode();
+                         graph = actionAddMidpoint({
+                           loc: insetNodeLoc,
+                           edge: edge
+                         }, newNode)(graph);
+                       }
+                     }
+                   } // if the edge is too short to subdivide as desired, then
+                   // just bound the structure at the existing end node
 
-       function modeSelectNote(context, selectedNoteID) {
-         var mode = {
-           id: 'select-note',
-           button: 'browse'
-         };
 
-         var _keybinding = utilKeybinding('select-note');
+                   if (!newNode) newNode = endNode;
+                   var splitAction = actionSplit([newNode.id]).limitWays(resultWayIDs); // only split selected or created ways
+                   // do the split
 
-         var _noteEditor = uiNoteEditor(context).on('change', function () {
-           context.map().pan([0, 0]); // trigger a redraw
+                   graph = splitAction(graph);
 
-           var note = checkSelectedID();
-           if (!note) return;
-           context.ui().sidebar.show(_noteEditor.note(note));
-         });
+                   if (splitAction.getCreatedWayIDs().length) {
+                     resultWayIDs.push(splitAction.getCreatedWayIDs()[0]);
+                   }
 
-         var _behaviors = [behaviorBreathe(), behaviorHover(context), behaviorSelect(context), behaviorLasso(context), modeDragNode(context).behavior, modeDragNote(context).behavior];
-         var _newFeature = false;
+                   return newNode;
+                 }
 
-         function checkSelectedID() {
-           if (!services.osm) return;
-           var note = services.osm.getNote(selectedNoteID);
+                 var structEndNode1 = determineEndpoint(edge, edgeNodes[1], endpointLocGetter1);
+                 var structEndNode2 = determineEndpoint([edgeNodes[0].id, structEndNode1.id], edgeNodes[0], endpointLocGetter2);
+                 var structureWay = resultWayIDs.map(function (id) {
+                   return graph.entity(id);
+                 }).find(function (way) {
+                   return way.nodes.indexOf(structEndNode1.id) !== -1 && way.nodes.indexOf(structEndNode2.id) !== -1;
+                 });
+                 var tags = Object.assign({}, structureWay.tags); // copy tags
 
-           if (!note) {
-             context.enter(modeBrowse(context));
-           }
+                 if (bridgeOrTunnel === 'bridge') {
+                   tags.bridge = 'yes';
+                   tags.layer = '1';
+                 } else {
+                   var tunnelValue = 'yes';
 
-           return note;
-         } // class the note as selected, or return to browse mode if the note is gone
+                   if (getFeatureType(structureWay, graph) === 'waterway') {
+                     // use `tunnel=culvert` for waterways by default
+                     tunnelValue = 'culvert';
+                   }
 
+                   tags.tunnel = tunnelValue;
+                   tags.layer = '-1';
+                 } // apply the structure tags to the way
 
-         function selectNote(d3_event, drawn) {
-           if (!checkSelectedID()) return;
-           var selection = context.surface().selectAll('.layer-notes .note-' + selectedNoteID);
 
-           if (selection.empty()) {
-             // Return to browse mode if selected DOM elements have
-             // disappeared because the user moved them out of view..
-             var source = d3_event && d3_event.type === 'zoom' && d3_event.sourceEvent;
+                 graph = actionChangeTags(structureWay.id, tags)(graph);
+                 return graph;
+               };
 
-             if (drawn && source && (source.type === 'pointermove' || source.type === 'mousemove' || source.type === 'touchmove')) {
-               context.enter(modeBrowse(context));
+               context.perform(action, _t('issues.fix.' + fixTitleID + '.annotation'));
+               context.enter(modeSelect(context, resultWayIDs));
              }
-           } else {
-             selection.classed('selected', true);
-             context.selectedNoteID(selectedNoteID);
-           }
-         }
-
-         function esc() {
-           if (context.container().select('.combobox').size()) return;
-           context.enter(modeBrowse(context));
+           });
          }
 
-         mode.zoomToSelected = function () {
-           if (!services.osm) return;
-           var note = services.osm.getNote(selectedNoteID);
+         function makeConnectWaysFix(connectionTags) {
+           var fixTitleID = 'connect_features';
 
-           if (note) {
-             context.map().centerZoomEase(note.loc, 20);
+           if (connectionTags.ford) {
+             fixTitleID = 'connect_using_ford';
            }
-         };
 
-         mode.newFeature = function (val) {
-           if (!arguments.length) return _newFeature;
-           _newFeature = val;
-           return mode;
-         };
+           return new validationIssueFix({
+             icon: 'iD-icon-crossing',
+             title: _t.html('issues.fix.' + fixTitleID + '.title'),
+             onClick: function onClick(context) {
+               var loc = this.issue.loc;
+               var connectionTags = this.issue.data.connectionTags;
+               var edges = this.issue.data.edges;
+               context.perform(function actionConnectCrossingWays(graph) {
+                 // create the new node for the points
+                 var node = osmNode({
+                   loc: loc,
+                   tags: connectionTags
+                 });
+                 graph = graph.replace(node);
+                 var nodesToMerge = [node.id];
+                 var mergeThresholdInMeters = 0.75;
+                 edges.forEach(function (edge) {
+                   var edgeNodes = [graph.entity(edge[0]), graph.entity(edge[1])];
+                   var nearby = geoSphericalClosestNode(edgeNodes, loc); // if there is already a suitable node nearby, use that
 
-         mode.enter = function () {
-           var note = checkSelectedID();
-           if (!note) return;
+                   if (!nearby.node.hasInterestingTags() && nearby.distance < mergeThresholdInMeters) {
+                     nodesToMerge.push(nearby.node.id); // else add the new node to the way
+                   } else {
+                     graph = actionAddMidpoint({
+                       loc: loc,
+                       edge: edge
+                     }, node)(graph);
+                   }
+                 });
 
-           _behaviors.forEach(context.install);
+                 if (nodesToMerge.length > 1) {
+                   // if we're using nearby nodes, merge them with the new node
+                   graph = actionMergeNodes(nodesToMerge, loc)(graph);
+                 }
 
-           _keybinding.on(_t('inspector.zoom_to.key'), mode.zoomToSelected).on('⎋', esc, true);
+                 return graph;
+               }, _t('issues.fix.connect_crossing_features.annotation'));
+             }
+           });
+         }
 
-           select(document).call(_keybinding);
-           selectNote();
-           var sidebar = context.ui().sidebar;
-           sidebar.show(_noteEditor.note(note).newNote(_newFeature)); // expand the sidebar, avoid obscuring the note if needed
+         function makeChangeLayerFix(higherOrLower) {
+           return new validationIssueFix({
+             icon: 'iD-icon-' + (higherOrLower === 'higher' ? 'up' : 'down'),
+             title: _t.html('issues.fix.tag_this_as_' + higherOrLower + '.title'),
+             onClick: function onClick(context) {
+               var mode = context.mode();
+               if (!mode || mode.id !== 'select') return;
+               var selectedIDs = mode.selectedIDs();
+               if (selectedIDs.length !== 1) return;
+               var selectedID = selectedIDs[0];
+               if (!this.issue.entityIds.some(function (entityId) {
+                 return entityId === selectedID;
+               })) return;
+               var entity = context.hasEntity(selectedID);
+               if (!entity) return;
+               var tags = Object.assign({}, entity.tags); // shallow copy
 
-           sidebar.expand(sidebar.intersects(note.extent()));
-           context.map().on('drawn.select', selectNote);
-         };
+               var layer = tags.layer && Number(tags.layer);
 
-         mode.exit = function () {
-           _behaviors.forEach(context.uninstall);
+               if (layer && !isNaN(layer)) {
+                 if (higherOrLower === 'higher') {
+                   layer += 1;
+                 } else {
+                   layer -= 1;
+                 }
+               } else {
+                 if (higherOrLower === 'higher') {
+                   layer = 1;
+                 } else {
+                   layer = -1;
+                 }
+               }
 
-           select(document).call(_keybinding.unbind);
-           context.surface().selectAll('.layer-notes .selected').classed('selected hover', false);
-           context.map().on('drawn.select', null);
-           context.ui().sidebar.hide();
-           context.selectedNoteID(null);
-         };
+               tags.layer = layer.toString();
+               context.perform(actionChangeTags(entity.id, tags), _t('operations.change_tags.annotation'));
+             }
+           });
+         }
 
-         return mode;
+         validation.type = type;
+         return validation;
        }
 
-       function modeDragNote(context) {
-         var mode = {
-           id: 'drag-note',
-           button: 'browse'
-         };
-         var edit = behaviorEdit(context);
-
-         var _nudgeInterval;
+       function behaviorDrawWay(context, wayID, mode, startGraph) {
+         var dispatch = dispatch$8('rejectedSelfIntersection');
+         var behavior = behaviorDraw(context); // Must be set by `drawWay.nodeIndex` before each install of this behavior.
 
-         var _lastLoc;
+         var _nodeIndex;
 
-         var _note; // most current note.. dragged note may have stale datum.
+         var _origWay;
 
+         var _wayGeometry;
 
-         function startNudge(d3_event, nudge) {
-           if (_nudgeInterval) window.clearInterval(_nudgeInterval);
-           _nudgeInterval = window.setInterval(function () {
-             context.map().pan(nudge);
-             doMove(d3_event, nudge);
-           }, 50);
-         }
+         var _headNodeID;
 
-         function stopNudge() {
-           if (_nudgeInterval) {
-             window.clearInterval(_nudgeInterval);
-             _nudgeInterval = null;
-           }
-         }
+         var _annotation;
 
-         function origin(note) {
-           return context.projection(note.loc);
-         }
+         var _pointerHasMoved = false; // The osmNode to be placed.
+         // This is temporary and just follows the mouse cursor until an "add" event occurs.
 
-         function start(d3_event, note) {
-           _note = note;
-           var osm = services.osm;
+         var _drawNode;
 
-           if (osm) {
-             // Get latest note from cache.. The marker may have a stale datum bound to it
-             // and dragging it around can sometimes delete the users note comment.
-             _note = osm.getNote(_note.id);
-           }
+         var _didResolveTempEdit = false;
 
-           context.surface().selectAll('.note-' + _note.id).classed('active', true);
-           context.perform(actionNoop());
-           context.enter(mode);
-           context.selectedNoteID(_note.id);
+         function createDrawNode(loc) {
+           // don't make the draw node until we actually need it
+           _drawNode = osmNode({
+             loc: loc
+           });
+           context.pauseChangeDispatch();
+           context.replace(function actionAddDrawNode(graph) {
+             // add the draw node to the graph and insert it into the way
+             var way = graph.entity(wayID);
+             return graph.replace(_drawNode).replace(way.addNode(_drawNode.id, _nodeIndex));
+           }, _annotation);
+           context.resumeChangeDispatch();
+           setActiveElements();
          }
 
-         function move(d3_event, entity, point) {
-           d3_event.stopPropagation();
-           _lastLoc = context.projection.invert(point);
-           doMove(d3_event);
-           var nudge = geoViewportEdge(point, context.map().dimensions());
-
-           if (nudge) {
-             startNudge(d3_event, nudge);
-           } else {
-             stopNudge();
-           }
+         function removeDrawNode() {
+           context.pauseChangeDispatch();
+           context.replace(function actionDeleteDrawNode(graph) {
+             var way = graph.entity(wayID);
+             return graph.replace(way.removeNode(_drawNode.id)).remove(_drawNode);
+           }, _annotation);
+           _drawNode = undefined;
+           context.resumeChangeDispatch();
          }
 
-         function doMove(d3_event, nudge) {
-           nudge = nudge || [0, 0];
-           var currPoint = d3_event && d3_event.point || context.projection(_lastLoc);
-           var currMouse = geoVecSubtract(currPoint, nudge);
-           var loc = context.projection.invert(currMouse);
-           _note = _note.move(loc);
-           var osm = services.osm;
+         function keydown(d3_event) {
+           if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
+             if (context.surface().classed('nope')) {
+               context.surface().classed('nope-suppressed', true);
+             }
 
-           if (osm) {
-             osm.replaceNote(_note); // update note cache
+             context.surface().classed('nope', false).classed('nope-disabled', true);
            }
-
-           context.replace(actionNoop()); // trigger redraw
          }
 
-         function end() {
-           context.replace(actionNoop()); // trigger redraw
-
-           context.selectedNoteID(_note.id).enter(modeSelectNote(context, _note.id));
-         }
-
-         var drag = behaviorDrag().selector('.layer-touch.markers .target.note.new').surface(context.container().select('.main-map').node()).origin(origin).on('start', start).on('move', move).on('end', end);
-
-         mode.enter = function () {
-           context.install(edit);
-         };
-
-         mode.exit = function () {
-           context.ui().sidebar.hover.cancel();
-           context.uninstall(edit);
-           context.surface().selectAll('.active').classed('active', false);
-           stopNudge();
-         };
-
-         mode.behavior = drag;
-         return mode;
-       }
-
-       function uiDataHeader() {
-         var _datum;
+         function keyup(d3_event) {
+           if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
+             if (context.surface().classed('nope-suppressed')) {
+               context.surface().classed('nope', true);
+             }
 
-         function dataHeader(selection) {
-           var header = selection.selectAll('.data-header').data(_datum ? [_datum] : [], function (d) {
-             return d.__featurehash__;
-           });
-           header.exit().remove();
-           var headerEnter = header.enter().append('div').attr('class', 'data-header');
-           var iconEnter = headerEnter.append('div').attr('class', 'data-header-icon');
-           iconEnter.append('div').attr('class', 'preset-icon-28').call(svgIcon('#iD-icon-data', 'note-fill'));
-           headerEnter.append('div').attr('class', 'data-header-label').html(_t.html('map_data.layers.custom.title'));
+             context.surface().classed('nope-suppressed', false).classed('nope-disabled', false);
+           }
          }
 
-         dataHeader.datum = function (val) {
-           if (!arguments.length) return _datum;
-           _datum = val;
-           return this;
-         };
-
-         return dataHeader;
-       }
-
-       // It is keyed on the `value` of the entry. Data should be an array of objects like:
-       //   [{
-       //       value:   'string value',  // required
-       //       display: 'label html'     // optional
-       //       title:   'hover text'     // optional
-       //       terms:   ['search terms'] // optional
-       //   }, ...]
-
-       var _comboHideTimerID;
+         function allowsVertex(d) {
+           return d.geometry(context.graph()) === 'vertex' || _mainPresetIndex.allowsVertex(d, context.graph());
+         } // related code
+         // - `mode/drag_node.js`     `doMove()`
+         // - `behavior/draw.js`      `click()`
+         // - `behavior/draw_way.js`  `move()`
 
-       function uiCombobox(context, klass) {
-         var dispatch$1 = dispatch('accept', 'cancel');
-         var container = context.container();
-         var _suggestions = [];
-         var _data = [];
-         var _fetched = {};
-         var _selected = null;
-         var _canAutocomplete = true;
-         var _caseSensitive = false;
-         var _cancelFetch = false;
-         var _minItems = 2;
-         var _tDown = 0;
 
-         var _mouseEnterHandler, _mouseLeaveHandler;
+         function move(d3_event, datum) {
+           var loc = context.map().mouseCoordinates();
+           if (!_drawNode) createDrawNode(loc);
+           context.surface().classed('nope-disabled', d3_event.altKey);
+           var targetLoc = datum && datum.properties && datum.properties.entity && allowsVertex(datum.properties.entity) && datum.properties.entity.loc;
+           var targetNodes = datum && datum.properties && datum.properties.nodes;
 
-         var _fetcher = function _fetcher(val, cb) {
-           cb(_data.filter(function (d) {
-             var terms = d.terms || [];
-             terms.push(d.value);
-             return terms.some(function (term) {
-               return term.toString().toLowerCase().indexOf(val.toLowerCase()) !== -1;
-             });
-           }));
-         };
+           if (targetLoc) {
+             // snap to node/vertex - a point target with `.loc`
+             loc = targetLoc;
+           } else if (targetNodes) {
+             // snap to way - a line target with `.nodes`
+             var choice = geoChooseEdge(targetNodes, context.map().mouse(), context.projection, _drawNode.id);
 
-         var combobox = function combobox(input, attachTo) {
-           if (!input || input.empty()) return;
-           input.classed('combobox-input', true).on('focus.combo-input', focus).on('blur.combo-input', blur).on('keydown.combo-input', keydown).on('keyup.combo-input', keyup).on('input.combo-input', change).on('mousedown.combo-input', mousedown).each(function () {
-             var parent = this.parentNode;
-             var sibling = this.nextSibling;
-             select(parent).selectAll('.combobox-caret').filter(function (d) {
-               return d === input.node();
-             }).data([input.node()]).enter().insert('div', function () {
-               return sibling;
-             }).attr('class', 'combobox-caret').on('mousedown.combo-caret', function (d3_event) {
-               d3_event.preventDefault(); // don't steal focus from input
+             if (choice) {
+               loc = choice.loc;
+             }
+           }
 
-               input.node().focus(); // focus the input as if it was clicked
+           context.replace(actionMoveNode(_drawNode.id, loc), _annotation);
+           _drawNode = context.entity(_drawNode.id);
+           checkGeometry(true
+           /* includeDrawNode */
+           );
+         } // Check whether this edit causes the geometry to break.
+         // If so, class the surface with a nope cursor.
+         // `includeDrawNode` - Only check the relevant line segments if finishing drawing
 
-               mousedown(d3_event);
-             }).on('mouseup.combo-caret', function (d3_event) {
-               d3_event.preventDefault(); // don't steal focus from input
 
-               mouseup(d3_event);
-             });
-           });
+         function checkGeometry(includeDrawNode) {
+           var nopeDisabled = context.surface().classed('nope-disabled');
+           var isInvalid = isInvalidGeometry(includeDrawNode);
 
-           function mousedown(d3_event) {
-             if (d3_event.button !== 0) return; // left click only
+           if (nopeDisabled) {
+             context.surface().classed('nope', false).classed('nope-suppressed', isInvalid);
+           } else {
+             context.surface().classed('nope', isInvalid).classed('nope-suppressed', false);
+           }
+         }
 
-             _tDown = +new Date(); // clear selection
+         function isInvalidGeometry(includeDrawNode) {
+           var testNode = _drawNode; // we only need to test the single way we're drawing
 
-             var start = input.property('selectionStart');
-             var end = input.property('selectionEnd');
+           var parentWay = context.graph().entity(wayID);
+           var nodes = context.graph().childNodes(parentWay).slice(); // shallow copy
 
-             if (start !== end) {
-               var val = utilGetSetValue(input);
-               input.node().setSelectionRange(val.length, val.length);
-               return;
+           if (includeDrawNode) {
+             if (parentWay.isClosed()) {
+               // don't test the last segment for closed ways - #4655
+               // (still test the first segment)
+               nodes.pop();
+             }
+           } else {
+             // discount the draw node
+             if (parentWay.isClosed()) {
+               if (nodes.length < 3) return false;
+               if (_drawNode) nodes.splice(-2, 1);
+               testNode = nodes[nodes.length - 2];
+             } else {
+               // there's nothing we need to test if we ignore the draw node on open ways
+               return false;
              }
-
-             input.on('mouseup.combo-input', mouseup);
            }
 
-           function mouseup(d3_event) {
-             input.on('mouseup.combo-input', null);
-             if (d3_event.button !== 0) return; // left click only
+           return testNode && geoHasSelfIntersections(nodes, testNode.id);
+         }
 
-             if (input.node() !== document.activeElement) return; // exit if this input is not focused
+         function undone() {
+           // undoing removed the temp edit
+           _didResolveTempEdit = true;
+           context.pauseChangeDispatch();
+           var nextMode;
 
-             var start = input.property('selectionStart');
-             var end = input.property('selectionEnd');
-             if (start !== end) return; // exit if user is selecting
-             // not showing or showing for a different field - try to show it.
+           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 {
+             // 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); // continue drawing
 
-             var combo = container.selectAll('.combobox');
+             nextMode = mode;
+           } // clear the redo stack by adding and removing a blank edit
 
-             if (combo.empty() || combo.datum() !== input.node()) {
-               var tOrig = _tDown;
-               window.setTimeout(function () {
-                 if (tOrig !== _tDown) return; // exit if user double clicked
 
-                 fetchComboData('', function () {
-                   show();
-                   render();
-                 });
-               }, 250);
-             } else {
-               hide();
-             }
-           }
+           context.perform(actionNoop());
+           context.pop(1);
+           context.resumeChangeDispatch();
+           context.enter(nextMode);
+         }
 
-           function focus() {
-             fetchComboData(''); // prefetch values (may warm taginfo cache)
-           }
+         function setActiveElements() {
+           if (!_drawNode) return;
+           context.surface().selectAll('.' + _drawNode.id).classed('active', true);
+         }
 
-           function blur() {
-             _comboHideTimerID = window.setTimeout(hide, 75);
+         function resetToStartGraph() {
+           while (context.graph() !== startGraph) {
+             context.pop();
            }
+         }
 
-           function show() {
-             hide(); // remove any existing
+         var drawWay = function drawWay(surface) {
+           _drawNode = undefined;
+           _didResolveTempEdit = false;
+           _origWay = context.entity(wayID);
 
-             container.insert('div', ':first-child').datum(input.node()).attr('class', 'combobox' + (klass ? ' combobox-' + klass : '')).style('position', 'absolute').style('display', 'block').style('left', '0px').on('mousedown.combo-container', function (d3_event) {
-               // prevent moving focus out of the input field
-               d3_event.preventDefault();
-             });
-             container.on('scroll.combo-scroll', render, true);
+           if (typeof _nodeIndex === 'number') {
+             _headNodeID = _origWay.nodes[_nodeIndex];
+           } else if (_origWay.isClosed()) {
+             _headNodeID = _origWay.nodes[_origWay.nodes.length - 2];
+           } else {
+             _headNodeID = _origWay.nodes[_origWay.nodes.length - 1];
            }
 
-           function hide() {
-             if (_comboHideTimerID) {
-               window.clearTimeout(_comboHideTimerID);
-               _comboHideTimerID = undefined;
-             }
+           _wayGeometry = _origWay.geometry(context.graph());
+           _annotation = _t((_origWay.nodes.length === (_origWay.isClosed() ? 2 : 1) ? 'operations.start.annotation.' : 'operations.continue.annotation.') + _wayGeometry);
+           _pointerHasMoved = false; // Push an annotated state for undo to return back to.
+           // We must make sure to replace or remove it later.
 
-             container.selectAll('.combobox').remove();
-             container.on('scroll.combo-scroll', null);
+           context.pauseChangeDispatch();
+           context.perform(actionNoop(), _annotation);
+           context.resumeChangeDispatch();
+           behavior.hover().initialNodeID(_headNodeID);
+           behavior.on('move', function () {
+             _pointerHasMoved = true;
+             move.apply(this, arguments);
+           }).on('down', function () {
+             move.apply(this, arguments);
+           }).on('downcancel', function () {
+             if (_drawNode) removeDrawNode();
+           }).on('click', drawWay.add).on('clickWay', drawWay.addWay).on('clickNode', drawWay.addNode).on('undo', context.undo).on('cancel', drawWay.cancel).on('finish', drawWay.finish);
+           select(window).on('keydown.drawWay', keydown).on('keyup.drawWay', keyup);
+           context.map().dblclickZoomEnable(false).on('drawn.draw', setActiveElements);
+           setActiveElements();
+           surface.call(behavior);
+           context.history().on('undone.draw', undone);
+         };
+
+         drawWay.off = function (surface) {
+           if (!_didResolveTempEdit) {
+             // Drawing was interrupted unexpectedly.
+             // This can happen if the user changes modes,
+             // clicks geolocate button, a hashchange event occurs, etc.
+             context.pauseChangeDispatch();
+             resetToStartGraph();
+             context.resumeChangeDispatch();
            }
 
-           function keydown(d3_event) {
-             var shown = !container.selectAll('.combobox').empty();
-             var tagName = input.node() ? input.node().tagName.toLowerCase() : '';
+           _drawNode = undefined;
+           _nodeIndex = undefined;
+           context.map().on('drawn.draw', null);
+           surface.call(behavior.off).selectAll('.active').classed('active', false);
+           surface.classed('nope', false).classed('nope-suppressed', false).classed('nope-disabled', false);
+           select(window).on('keydown.drawWay', null).on('keyup.drawWay', null);
+           context.history().on('undone.draw', null);
+         };
 
-             switch (d3_event.keyCode) {
-               case 8: // ⌫ Backspace
+         function attemptAdd(d, loc, doAdd) {
+           if (_drawNode) {
+             // move the node to the final loc in case move wasn't called
+             // consistently (e.g. on touch devices)
+             context.replace(actionMoveNode(_drawNode.id, loc), _annotation);
+             _drawNode = context.entity(_drawNode.id);
+           } else {
+             createDrawNode(loc);
+           }
 
-               case 46:
-                 // ⌦ Delete
-                 d3_event.stopPropagation();
-                 _selected = null;
-                 render();
-                 input.on('input.combo-input', function () {
-                   var start = input.property('selectionStart');
-                   input.node().setSelectionRange(start, start);
-                   input.on('input.combo-input', change);
-                 });
-                 break;
+           checkGeometry(true
+           /* includeDrawNode */
+           );
 
-               case 9:
-                 // ⇥ Tab
-                 accept();
-                 break;
+           if (d && d.properties && d.properties.nope || context.surface().classed('nope')) {
+             if (!_pointerHasMoved) {
+               // prevent the temporary draw node from appearing on touch devices
+               removeDrawNode();
+             }
 
-               case 13:
-                 // ↩ Return
-                 d3_event.preventDefault();
-                 d3_event.stopPropagation();
-                 break;
+             dispatch.call('rejectedSelfIntersection', this);
+             return; // can't click here
+           }
 
-               case 38:
-                 // ↑ Up arrow
-                 if (tagName === 'textarea' && !shown) return;
-                 d3_event.preventDefault();
+           context.pauseChangeDispatch();
+           doAdd(); // we just replaced the temporary edit with the real one
 
-                 if (tagName === 'input' && !shown) {
-                   show();
-                 }
+           _didResolveTempEdit = true;
+           context.resumeChangeDispatch();
+           context.enter(mode);
+         } // Accept the current position of the drawing node
 
-                 nav(-1);
-                 break;
 
-               case 40:
-                 // ↓ Down arrow
-                 if (tagName === 'textarea' && !shown) return;
-                 d3_event.preventDefault();
+         drawWay.add = function (loc, d) {
+           attemptAdd(d, loc, function () {// don't need to do anything extra
+           });
+         }; // Connect the way to an existing way
 
-                 if (tagName === 'input' && !shown) {
-                   show();
-                 }
 
-                 nav(+1);
-                 break;
-             }
-           }
+         drawWay.addWay = function (loc, edge, d) {
+           attemptAdd(d, loc, function () {
+             context.replace(actionAddMidpoint({
+               loc: loc,
+               edge: edge
+             }, _drawNode), _annotation);
+           });
+         }; // Connect the way to an existing node
 
-           function keyup(d3_event) {
-             switch (d3_event.keyCode) {
-               case 27:
-                 // ⎋ Escape
-                 cancel();
-                 break;
 
-               case 13:
-                 // ↩ Return
-                 accept();
-                 break;
-             }
-           } // Called whenever the input value is changed (e.g. on typing)
+         drawWay.addNode = function (node, d) {
+           // finish drawing if the mapper targets the prior node
+           if (node.id === _headNodeID || // or the first node when drawing an area
+           _origWay.isClosed() && node.id === _origWay.first()) {
+             drawWay.finish();
+             return;
+           }
 
+           attemptAdd(d, node.loc, function () {
+             context.replace(function actionReplaceDrawNode(graph) {
+               // remove the temporary draw node and insert the existing node
+               // at the same index
+               graph = graph.replace(graph.entity(wayID).removeNode(_drawNode.id)).remove(_drawNode);
+               return graph.replace(graph.entity(wayID).addNode(node.id, _nodeIndex));
+             }, _annotation);
+           });
+         }; // Finish the draw operation, removing the temporary edit.
+         // If the way has enough nodes to be valid, it's selected.
+         // Otherwise, delete everything and return to browse mode.
 
-           function change() {
-             fetchComboData(value(), function () {
-               _selected = null;
-               var val = input.property('value');
 
-               if (_suggestions.length) {
-                 if (input.property('selectionEnd') === val.length) {
-                   _selected = tryAutocomplete();
-                 }
+         drawWay.finish = function () {
+           checkGeometry(false
+           /* includeDrawNode */
+           );
 
-                 if (!_selected) {
-                   _selected = val;
-                 }
-               }
+           if (context.surface().classed('nope')) {
+             dispatch.call('rejectedSelfIntersection', this);
+             return; // can't click here
+           }
 
-               if (val.length) {
-                 var combo = container.selectAll('.combobox');
+           context.pauseChangeDispatch(); // remove the temporary edit
 
-                 if (combo.empty()) {
-                   show();
-                 }
-               } else {
-                 hide();
-               }
+           context.pop(1);
+           _didResolveTempEdit = true;
+           context.resumeChangeDispatch();
+           var way = context.hasEntity(wayID);
 
-               render();
-             });
-           } // Called when the user presses up/down arrows to navigate the list
+           if (!way || way.isDegenerate()) {
+             drawWay.cancel();
+             return;
+           }
 
+           window.setTimeout(function () {
+             context.map().dblclickZoomEnable(true);
+           }, 1000);
+           var isNewFeature = !mode.isContinuing;
+           context.enter(modeSelect(context, [wayID]).newFeature(isNewFeature));
+         }; // Cancel the draw operation, delete everything, and return to browse mode.
 
-           function nav(dir) {
-             if (_suggestions.length) {
-               // try to determine previously selected index..
-               var index = -1;
 
-               for (var i = 0; i < _suggestions.length; i++) {
-                 if (_selected && _suggestions[i].value === _selected) {
-                   index = i;
-                   break;
-                 }
-               } // pick new _selected
+         drawWay.cancel = function () {
+           context.pauseChangeDispatch();
+           resetToStartGraph();
+           context.resumeChangeDispatch();
+           window.setTimeout(function () {
+             context.map().dblclickZoomEnable(true);
+           }, 1000);
+           context.surface().classed('nope', false).classed('nope-disabled', false).classed('nope-suppressed', false);
+           context.enter(modeBrowse(context));
+         };
 
+         drawWay.nodeIndex = function (val) {
+           if (!arguments.length) return _nodeIndex;
+           _nodeIndex = val;
+           return drawWay;
+         };
 
-               index = Math.max(Math.min(index + dir, _suggestions.length - 1), 0);
-               _selected = _suggestions[index].value;
-               input.property('value', _selected);
-             }
+         drawWay.activeID = function () {
+           if (!arguments.length) return _drawNode && _drawNode.id; // no assign
 
-             render();
-             ensureVisible();
-           }
+           return drawWay;
+         };
 
-           function ensureVisible() {
-             var combo = container.selectAll('.combobox');
-             if (combo.empty()) return;
-             var containerRect = container.node().getBoundingClientRect();
-             var comboRect = combo.node().getBoundingClientRect();
+         return utilRebind(drawWay, dispatch, 'on');
+       }
 
-             if (comboRect.bottom > containerRect.bottom) {
-               var node = attachTo ? attachTo.node() : input.node();
-               node.scrollIntoView({
-                 behavior: 'instant',
-                 block: 'center'
-               });
-               render();
-             } // https://stackoverflow.com/questions/11039885/scrollintoview-causing-the-whole-page-to-move
+       function modeDrawLine(context, wayID, startGraph, button, affix, continuing) {
+         var mode = {
+           button: button,
+           id: 'draw-line'
+         };
+         var behavior = behaviorDrawWay(context, wayID, mode, startGraph).on('rejectedSelfIntersection.modeDrawLine', function () {
+           context.ui().flash.iconName('#iD-icon-no').label(_t('self_intersection.error.lines'))();
+         });
+         mode.wayID = wayID;
+         mode.isContinuing = continuing;
 
+         mode.enter = function () {
+           behavior.nodeIndex(affix === 'prefix' ? 0 : undefined);
+           context.install(behavior);
+         };
 
-             var selected = combo.selectAll('.combobox-option.selected').node();
+         mode.exit = function () {
+           context.uninstall(behavior);
+         };
 
-             if (selected) {
-               selected.scrollIntoView({
-                 behavior: 'smooth',
-                 block: 'nearest'
-               });
-             }
-           }
+         mode.selectedIDs = function () {
+           return [wayID];
+         };
 
-           function value() {
-             var value = input.property('value');
-             var start = input.property('selectionStart');
-             var end = input.property('selectionEnd');
+         mode.activeID = function () {
+           return behavior && behavior.activeID() || [];
+         };
 
-             if (start && end) {
-               value = value.substring(0, start);
-             }
+         return mode;
+       }
 
-             return value;
-           }
+       function validationDisconnectedWay() {
+         var type = 'disconnected_way';
 
-           function fetchComboData(v, cb) {
-             _cancelFetch = false;
+         function isTaggedAsHighway(entity) {
+           return osmRoutableHighwayTagValues[entity.tags.highway];
+         }
 
-             _fetcher.call(input, v, function (results) {
-               // already chose a value, don't overwrite or autocomplete it
-               if (_cancelFetch) return;
-               _suggestions = results;
-               results.forEach(function (d) {
-                 _fetched[d.value] = d;
+         var validation = function checkDisconnectedWay(entity, graph) {
+           var routingIslandWays = routingIslandForEntity(entity);
+           if (!routingIslandWays) return [];
+           return [new validationIssue({
+             type: type,
+             subtype: 'highway',
+             severity: 'warning',
+             message: function message(context) {
+               var entity = this.entityIds.length && context.hasEntity(this.entityIds[0]);
+               var label = entity && utilDisplayLabel(entity, context.graph());
+               return _t.html('issues.disconnected_way.routable.message', {
+                 count: this.entityIds.length,
+                 highway: label
                });
+             },
+             reference: showReference,
+             entityIds: Array.from(routingIslandWays).map(function (way) {
+               return way.id;
+             }),
+             dynamicFixes: makeFixes
+           })];
 
-               if (cb) {
-                 cb();
-               }
-             });
-           }
-
-           function tryAutocomplete() {
-             if (!_canAutocomplete) return;
-             var val = _caseSensitive ? value() : value().toLowerCase();
-             if (!val) return; // Don't autocomplete if user is typing a number - #4935
-
-             if (!isNaN(parseFloat(val)) && isFinite(val)) return;
-             var bestIndex = -1;
+           function makeFixes(context) {
+             var fixes = [];
+             var singleEntity = this.entityIds.length === 1 && context.hasEntity(this.entityIds[0]);
 
-             for (var i = 0; i < _suggestions.length; i++) {
-               var suggestion = _suggestions[i].value;
-               var compare = _caseSensitive ? suggestion : suggestion.toLowerCase(); // if search string matches suggestion exactly, pick it..
+             if (singleEntity) {
+               if (singleEntity.type === 'way' && !singleEntity.isClosed()) {
+                 var textDirection = _mainLocalizer.textDirection();
+                 var startFix = makeContinueDrawingFixIfAllowed(textDirection, singleEntity.first(), 'start');
+                 if (startFix) fixes.push(startFix);
+                 var endFix = makeContinueDrawingFixIfAllowed(textDirection, singleEntity.last(), 'end');
+                 if (endFix) fixes.push(endFix);
+               }
 
-               if (compare === val) {
-                 bestIndex = i;
-                 break; // otherwise lock in the first result that starts with the search string..
-               } else if (bestIndex === -1 && compare.indexOf(val) === 0) {
-                 bestIndex = i;
+               if (!fixes.length) {
+                 fixes.push(new validationIssueFix({
+                   title: _t.html('issues.fix.connect_feature.title')
+                 }));
                }
-             }
 
-             if (bestIndex !== -1) {
-               var bestVal = _suggestions[bestIndex].value;
-               input.property('value', bestVal);
-               input.node().setSelectionRange(val.length, bestVal.length);
-               return bestVal;
+               fixes.push(new validationIssueFix({
+                 icon: 'iD-operation-delete',
+                 title: _t.html('issues.fix.delete_feature.title'),
+                 entityIds: [singleEntity.id],
+                 onClick: function onClick(context) {
+                   var id = this.issue.entityIds[0];
+                   var operation = operationDelete(context, [id]);
+
+                   if (!operation.disabled()) {
+                     operation();
+                   }
+                 }
+               }));
+             } else {
+               fixes.push(new validationIssueFix({
+                 title: _t.html('issues.fix.connect_features.title')
+               }));
              }
+
+             return fixes;
            }
 
-           function render() {
-             if (_suggestions.length < _minItems || document.activeElement !== input.node()) {
-               hide();
-               return;
-             }
+           function showReference(selection) {
+             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.disconnected_way.routable.reference'));
+           }
 
-             var shown = !container.selectAll('.combobox').empty();
-             if (!shown) return;
-             var combo = container.selectAll('.combobox');
-             var options = combo.selectAll('.combobox-option').data(_suggestions, function (d) {
-               return d.value;
-             });
-             options.exit().remove(); // enter/update
+           function routingIslandForEntity(entity) {
+             var routingIsland = new Set(); // the interconnected routable features
 
-             options.enter().append('a').attr('class', 'combobox-option').attr('title', function (d) {
-               return d.title;
-             }).html(function (d) {
-               return d.display || d.value;
-             }).on('mouseenter', _mouseEnterHandler).on('mouseleave', _mouseLeaveHandler).merge(options).classed('selected', function (d) {
-               return d.value === _selected;
-             }).on('click.combo-option', accept).order();
-             var node = attachTo ? attachTo.node() : input.node();
-             var containerRect = container.node().getBoundingClientRect();
-             var rect = node.getBoundingClientRect();
-             combo.style('left', rect.left + 5 - containerRect.left + 'px').style('width', rect.width - 10 + 'px').style('top', rect.height + rect.top - containerRect.top + 'px');
-           } // Dispatches an 'accept' event
-           // Then hides the combobox.
+             var waysToCheck = []; // the queue of remaining routable ways to traverse
 
+             function queueParentWays(node) {
+               graph.parentWays(node).forEach(function (parentWay) {
+                 if (!routingIsland.has(parentWay) && // only check each feature once
+                 isRoutableWay(parentWay, false)) {
+                   // only check routable features
+                   routingIsland.add(parentWay);
+                   waysToCheck.push(parentWay);
+                 }
+               });
+             }
 
-           function accept(d3_event, d) {
-             _cancelFetch = true;
-             var thiz = input.node();
+             if (entity.type === 'way' && isRoutableWay(entity, true)) {
+               routingIsland.add(entity);
+               waysToCheck.push(entity);
+             } else if (entity.type === 'node' && isRoutableNode(entity)) {
+               routingIsland.add(entity);
+               queueParentWays(entity);
+             } else {
+               // this feature isn't routable, cannot be a routing island
+               return null;
+             }
 
-             if (d) {
-               // user clicked on a suggestion
-               utilGetSetValue(input, d.value); // replace field contents
+             while (waysToCheck.length) {
+               var wayToCheck = waysToCheck.pop();
+               var childNodes = graph.childNodes(wayToCheck);
 
-               utilTriggerEvent(input, 'change');
-             } // clear (and keep) selection
+               for (var i in childNodes) {
+                 var vertex = childNodes[i];
 
+                 if (isConnectedVertex(vertex)) {
+                   // found a link to the wider network, not a routing island
+                   return null;
+                 }
 
-             var val = utilGetSetValue(input);
-             thiz.setSelectionRange(val.length, val.length);
-             d = _fetched[val];
-             dispatch$1.call('accept', thiz, d, val);
-             hide();
-           } // Dispatches an 'cancel' event
-           // Then hides the combobox.
+                 if (isRoutableNode(vertex)) {
+                   routingIsland.add(vertex);
+                 }
 
+                 queueParentWays(vertex);
+               }
+             } // no network link found, this is a routing island, return its members
 
-           function cancel() {
-             _cancelFetch = true;
-             var thiz = input.node(); // clear (and remove) selection, and replace field contents
 
-             var val = utilGetSetValue(input);
-             var start = input.property('selectionStart');
-             var end = input.property('selectionEnd');
-             val = val.slice(0, start) + val.slice(end);
-             utilGetSetValue(input, val);
-             thiz.setSelectionRange(val.length, val.length);
-             dispatch$1.call('cancel', thiz);
-             hide();
+             return routingIsland;
            }
-         };
-
-         combobox.canAutocomplete = function (val) {
-           if (!arguments.length) return _canAutocomplete;
-           _canAutocomplete = val;
-           return combobox;
-         };
-
-         combobox.caseSensitive = function (val) {
-           if (!arguments.length) return _caseSensitive;
-           _caseSensitive = val;
-           return combobox;
-         };
-
-         combobox.data = function (val) {
-           if (!arguments.length) return _data;
-           _data = val;
-           return combobox;
-         };
 
-         combobox.fetcher = function (val) {
-           if (!arguments.length) return _fetcher;
-           _fetcher = val;
-           return combobox;
-         };
+           function isConnectedVertex(vertex) {
+             // assume ways overlapping unloaded tiles are connected to the wider road network  - #5938
+             var osm = services.osm;
+             if (osm && !osm.isDataLoaded(vertex.loc)) return true; // entrances are considered connected
 
-         combobox.minItems = function (val) {
-           if (!arguments.length) return _minItems;
-           _minItems = val;
-           return combobox;
-         };
+             if (vertex.tags.entrance && vertex.tags.entrance !== 'no') return true;
+             if (vertex.tags.amenity === 'parking_entrance') return true;
+             return false;
+           }
 
-         combobox.itemsMouseEnter = function (val) {
-           if (!arguments.length) return _mouseEnterHandler;
-           _mouseEnterHandler = val;
-           return combobox;
-         };
+           function isRoutableNode(node) {
+             // treat elevators as distinct features in the highway network
+             if (node.tags.highway === 'elevator') return true;
+             return false;
+           }
 
-         combobox.itemsMouseLeave = function (val) {
-           if (!arguments.length) return _mouseLeaveHandler;
-           _mouseLeaveHandler = val;
-           return combobox;
-         };
+           function isRoutableWay(way, ignoreInnerWays) {
+             if (isTaggedAsHighway(way) || way.tags.route === 'ferry') return true;
+             return graph.parentRelations(way).some(function (parentRelation) {
+               if (parentRelation.tags.type === 'route' && parentRelation.tags.route === 'ferry') return true;
+               if (parentRelation.isMultipolygon() && isTaggedAsHighway(parentRelation) && (!ignoreInnerWays || parentRelation.memberById(way.id).role !== 'inner')) return true;
+               return false;
+             });
+           }
 
-         return utilRebind(combobox, dispatch$1, 'on');
-       }
+           function makeContinueDrawingFixIfAllowed(textDirection, vertexID, whichEnd) {
+             var vertex = graph.hasEntity(vertexID);
+             if (!vertex || vertex.tags.noexit === 'yes') return null;
+             var useLeftContinue = whichEnd === 'start' && textDirection === 'ltr' || whichEnd === 'end' && textDirection === 'rtl';
+             return new validationIssueFix({
+               icon: 'iD-operation-continue' + (useLeftContinue ? '-left' : ''),
+               title: _t.html('issues.fix.continue_from_' + whichEnd + '.title'),
+               entityIds: [vertexID],
+               onClick: function onClick(context) {
+                 var wayId = this.issue.entityIds[0];
+                 var way = context.hasEntity(wayId);
+                 var vertexId = this.entityIds[0];
+                 var vertex = context.hasEntity(vertexId);
+                 if (!way || !vertex) return; // make sure the vertex is actually visible and editable
 
-       uiCombobox.off = function (input, context) {
-         input.on('focus.combo-input', null).on('blur.combo-input', null).on('keydown.combo-input', null).on('keyup.combo-input', null).on('input.combo-input', null).on('mousedown.combo-input', null).on('mouseup.combo-input', null);
-         context.container().on('scroll.combo-scroll', null);
-       };
+                 var map = context.map();
 
-       // hide class, which sets display=none, and a d3 transition for opacity.
-       // this will cause blinking when called repeatedly, so check that the
-       // value actually changes between calls.
+                 if (!context.editable() || !map.trimmedExtent().contains(vertex.loc)) {
+                   map.zoomToEase(vertex);
+                 }
 
-       function uiToggle(show, callback) {
-         return function (selection) {
-           selection.style('opacity', show ? 0 : 1).classed('hide', false).transition().style('opacity', show ? 1 : 0).on('end', function () {
-             select(this).classed('hide', !show).style('opacity', null);
-             if (callback) callback.apply(this);
-           });
+                 context.enter(modeDrawLine(context, wayId, context.graph(), 'line', way.affix(vertexId), true));
+               }
+             });
+           }
          };
-       }
-
-       function uiDisclosure(context, key, expandedDefault) {
-         var dispatch$1 = dispatch('toggled');
 
-         var _expanded;
+         validation.type = type;
+         return validation;
+       }
 
-         var _label = utilFunctor('');
+       function validationFormatting() {
+         var type = 'invalid_format';
 
-         var _updatePreference = true;
+         var validation = function validation(entity) {
+           var issues = [];
 
-         var _content = function _content() {};
+           function isValidEmail(email) {
+             // Emails in OSM are going to be official so they should be pretty simple
+             // Using negated lists to better support all possible unicode characters (#6494)
+             var valid_email = /^[^\(\)\\,":;<>@\[\]]+@[^\(\)\\,":;<>@\[\]\.]+(?:\.[a-z0-9-]+)*$/i; // An empty value is also acceptable
 
-         var disclosure = function disclosure(selection) {
-           if (_expanded === undefined || _expanded === null) {
-             // loading _expanded here allows it to be reset by calling `disclosure.expanded(null)`
-             var preference = corePreferences('disclosure.' + key + '.expanded');
-             _expanded = preference === null ? !!expandedDefault : preference === 'true';
+             return !email || valid_email.test(email);
+           }
+           /*
+           function isSchemePresent(url) {
+               var valid_scheme = /^https?:\/\//i;
+               return (!url || valid_scheme.test(url));
            }
+           */
 
-           var hideToggle = selection.selectAll('.hide-toggle-' + key).data([0]); // enter
 
-           var hideToggleEnter = hideToggle.enter().append('a').attr('href', '#').attr('class', 'hide-toggle hide-toggle-' + key).call(svgIcon('', 'pre-text', 'hide-toggle-icon'));
-           hideToggleEnter.append('span').attr('class', 'hide-toggle-text'); // update
+           function showReferenceEmail(selection) {
+             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.invalid_format.email.reference'));
+           }
+           /*
+           function showReferenceWebsite(selection) {
+               selection.selectAll('.issue-reference')
+                   .data([0])
+                   .enter()
+                   .append('div')
+                   .attr('class', 'issue-reference')
+                   .html(t.html('issues.invalid_format.website.reference'));
+           }
+            if (entity.tags.website) {
+               // Multiple websites are possible
+               // If ever we support ES6, arrow functions make this nicer
+               var websites = entity.tags.website
+                   .split(';')
+                   .map(function(s) { return s.trim(); })
+                   .filter(function(x) { return !isSchemePresent(x); });
+                if (websites.length) {
+                   issues.push(new validationIssue({
+                       type: type,
+                       subtype: 'website',
+                       severity: 'warning',
+                       message: function(context) {
+                           var entity = context.hasEntity(this.entityIds[0]);
+                           return entity ? t.html('issues.invalid_format.website.message' + this.data,
+                               { feature: utilDisplayLabel(entity, context.graph()), site: websites.join(', ') }) : '';
+                       },
+                       reference: showReferenceWebsite,
+                       entityIds: [entity.id],
+                       hash: websites.join(),
+                       data: (websites.length > 1) ? '_multi' : ''
+                   }));
+               }
+           }
+           */
 
-           hideToggle = hideToggleEnter.merge(hideToggle);
-           hideToggle.on('click', toggle).classed('expanded', _expanded);
-           hideToggle.selectAll('.hide-toggle-text').html(_label());
-           hideToggle.selectAll('.hide-toggle-icon').attr('xlink:href', _expanded ? '#iD-icon-down' : _mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-backward' : '#iD-icon-forward');
-           var wrap = selection.selectAll('.disclosure-wrap').data([0]); // enter/update
 
-           wrap = wrap.enter().append('div').attr('class', 'disclosure-wrap disclosure-wrap-' + key).merge(wrap).classed('hide', !_expanded);
+           if (entity.tags.email) {
+             // Multiple emails are possible
+             var emails = entity.tags.email.split(';').map(function (s) {
+               return s.trim();
+             }).filter(function (x) {
+               return !isValidEmail(x);
+             });
 
-           if (_expanded) {
-             wrap.call(_content);
+             if (emails.length) {
+               issues.push(new validationIssue({
+                 type: type,
+                 subtype: 'email',
+                 severity: 'warning',
+                 message: function message(context) {
+                   var entity = context.hasEntity(this.entityIds[0]);
+                   return entity ? _t.html('issues.invalid_format.email.message' + this.data, {
+                     feature: utilDisplayLabel(entity, context.graph()),
+                     email: emails.join(', ')
+                   }) : '';
+                 },
+                 reference: showReferenceEmail,
+                 entityIds: [entity.id],
+                 hash: emails.join(),
+                 data: emails.length > 1 ? '_multi' : ''
+               }));
+             }
            }
 
-           function toggle(d3_event) {
-             d3_event.preventDefault();
-             _expanded = !_expanded;
+           return issues;
+         };
 
-             if (_updatePreference) {
-               corePreferences('disclosure.' + key + '.expanded', _expanded);
-             }
+         validation.type = type;
+         return validation;
+       }
 
-             hideToggle.classed('expanded', _expanded);
-             hideToggle.selectAll('.hide-toggle-icon').attr('xlink:href', _expanded ? '#iD-icon-down' : _mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-backward' : '#iD-icon-forward');
-             wrap.call(uiToggle(_expanded));
+       function validationHelpRequest(context) {
+         var type = 'help_request';
 
-             if (_expanded) {
-               wrap.call(_content);
-             }
+         var validation = function checkFixmeTag(entity) {
+           if (!entity.tags.fixme) return []; // don't flag fixmes on features added by the user
 
-             dispatch$1.call('toggled', this, _expanded);
-           }
-         };
+           if (entity.version === undefined) return [];
 
-         disclosure.label = function (val) {
-           if (!arguments.length) return _label;
-           _label = utilFunctor(val);
-           return disclosure;
-         };
+           if (entity.v !== undefined) {
+             var baseEntity = context.history().base().hasEntity(entity.id); // don't flag fixmes added by the user on existing features
 
-         disclosure.expanded = function (val) {
-           if (!arguments.length) return _expanded;
-           _expanded = val;
-           return disclosure;
-         };
+             if (!baseEntity || !baseEntity.tags.fixme) return [];
+           }
 
-         disclosure.updatePreference = function (val) {
-           if (!arguments.length) return _updatePreference;
-           _updatePreference = val;
-           return disclosure;
-         };
+           return [new validationIssue({
+             type: type,
+             subtype: 'fixme_tag',
+             severity: 'warning',
+             message: function message(context) {
+               var entity = context.hasEntity(this.entityIds[0]);
+               return entity ? _t.html('issues.fixme_tag.message', {
+                 feature: utilDisplayLabel(entity, context.graph(), true
+                 /* verbose */
+                 )
+               }) : '';
+             },
+             dynamicFixes: function dynamicFixes() {
+               return [new validationIssueFix({
+                 title: _t.html('issues.fix.address_the_concern.title')
+               })];
+             },
+             reference: showReference,
+             entityIds: [entity.id]
+           })];
 
-         disclosure.content = function (val) {
-           if (!arguments.length) return _content;
-           _content = val;
-           return disclosure;
+           function showReference(selection) {
+             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.fixme_tag.reference'));
+           }
          };
 
-         return utilRebind(disclosure, dispatch$1, 'on');
+         validation.type = type;
+         return validation;
        }
 
-       // Can be labeled and collapsible.
-
-       function uiSection(id, context) {
-         var _classes = utilFunctor('');
+       function validationImpossibleOneway() {
+         var type = 'impossible_oneway';
 
-         var _shouldDisplay;
+         var validation = function checkImpossibleOneway(entity, graph) {
+           if (entity.type !== 'way' || entity.geometry(graph) !== 'line') return [];
+           if (entity.isClosed()) return [];
+           if (!typeForWay(entity)) return [];
+           if (!isOneway(entity)) return [];
+           var firstIssues = issuesForNode(entity, entity.first());
+           var lastIssues = issuesForNode(entity, entity.last());
+           return firstIssues.concat(lastIssues);
 
-         var _content;
+           function typeForWay(way) {
+             if (way.geometry(graph) !== 'line') return null;
+             if (osmRoutableHighwayTagValues[way.tags.highway]) return 'highway';
+             if (osmFlowingWaterwayTagValues[way.tags.waterway]) return 'waterway';
+             return null;
+           }
 
-         var _disclosure;
+           function isOneway(way) {
+             if (way.tags.oneway === 'yes') return true;
+             if (way.tags.oneway) return false;
 
-         var _label;
+             for (var key in way.tags) {
+               if (osmOneWayTags[key] && osmOneWayTags[key][way.tags[key]]) {
+                 return true;
+               }
+             }
 
-         var _expandedByDefault = utilFunctor(true);
+             return false;
+           }
 
-         var _disclosureContent;
+           function nodeOccursMoreThanOnce(way, nodeID) {
+             var occurrences = 0;
 
-         var _disclosureExpanded;
+             for (var index in way.nodes) {
+               if (way.nodes[index] === nodeID) {
+                 occurrences += 1;
+                 if (occurrences > 1) return true;
+               }
+             }
 
-         var _containerSelection = select(null);
+             return false;
+           }
 
-         var section = {
-           id: id
-         };
+           function isConnectedViaOtherTypes(way, node) {
+             var wayType = typeForWay(way);
 
-         section.classes = function (val) {
-           if (!arguments.length) return _classes;
-           _classes = utilFunctor(val);
-           return section;
-         };
+             if (wayType === 'highway') {
+               // entrances are considered connected
+               if (node.tags.entrance && node.tags.entrance !== 'no') return true;
+               if (node.tags.amenity === 'parking_entrance') return true;
+             } else if (wayType === 'waterway') {
+               if (node.id === way.first()) {
+                 // multiple waterways may start at the same spring
+                 if (node.tags.natural === 'spring') return true;
+               } else {
+                 // multiple waterways may end at the same drain
+                 if (node.tags.manhole === 'drain') return true;
+               }
+             }
 
-         section.label = function (val) {
-           if (!arguments.length) return _label;
-           _label = utilFunctor(val);
-           return section;
-         };
+             return graph.parentWays(node).some(function (parentWay) {
+               if (parentWay.id === way.id) return false;
 
-         section.expandedByDefault = function (val) {
-           if (!arguments.length) return _expandedByDefault;
-           _expandedByDefault = utilFunctor(val);
-           return section;
-         };
+               if (wayType === 'highway') {
+                 // allow connections to highway areas
+                 if (parentWay.geometry(graph) === 'area' && osmRoutableHighwayTagValues[parentWay.tags.highway]) return true; // count connections to ferry routes as connected
 
-         section.shouldDisplay = function (val) {
-           if (!arguments.length) return _shouldDisplay;
-           _shouldDisplay = utilFunctor(val);
-           return section;
-         };
+                 if (parentWay.tags.route === 'ferry') return true;
+                 return graph.parentRelations(parentWay).some(function (parentRelation) {
+                   if (parentRelation.tags.type === 'route' && parentRelation.tags.route === 'ferry') return true; // allow connections to highway multipolygons
 
-         section.content = function (val) {
-           if (!arguments.length) return _content;
-           _content = val;
-           return section;
-         };
+                   return parentRelation.isMultipolygon() && osmRoutableHighwayTagValues[parentRelation.tags.highway];
+                 });
+               } else if (wayType === 'waterway') {
+                 // multiple waterways may start or end at a water body at the same node
+                 if (parentWay.tags.natural === 'water' || parentWay.tags.natural === 'coastline') return true;
+               }
 
-         section.disclosureContent = function (val) {
-           if (!arguments.length) return _disclosureContent;
-           _disclosureContent = val;
-           return section;
-         };
+               return false;
+             });
+           }
 
-         section.disclosureExpanded = function (val) {
-           if (!arguments.length) return _disclosureExpanded;
-           _disclosureExpanded = val;
-           return section;
-         }; // may be called multiple times
+           function issuesForNode(way, nodeID) {
+             var isFirst = nodeID === way.first();
+             var wayType = typeForWay(way); // ignore if this way is self-connected at this node
 
+             if (nodeOccursMoreThanOnce(way, nodeID)) return [];
+             var osm = services.osm;
+             if (!osm) return [];
+             var node = graph.hasEntity(nodeID); // ignore if this node or its tile are unloaded
 
-         section.render = function (selection) {
-           _containerSelection = selection.selectAll('.section-' + id).data([0]);
+             if (!node || !osm.isDataLoaded(node.loc)) return [];
+             if (isConnectedViaOtherTypes(way, node)) return [];
+             var attachedWaysOfSameType = graph.parentWays(node).filter(function (parentWay) {
+               if (parentWay.id === way.id) return false;
+               return typeForWay(parentWay) === wayType;
+             }); // assume it's okay for waterways to start or end disconnected for now
 
-           var sectionEnter = _containerSelection.enter().append('div').attr('class', 'section section-' + id + ' ' + (_classes && _classes() || ''));
+             if (wayType === 'waterway' && attachedWaysOfSameType.length === 0) return [];
+             var attachedOneways = attachedWaysOfSameType.filter(function (attachedWay) {
+               return isOneway(attachedWay);
+             }); // ignore if the way is connected to some non-oneway features
 
-           _containerSelection = sectionEnter.merge(_containerSelection);
+             if (attachedOneways.length < attachedWaysOfSameType.length) return [];
 
-           _containerSelection.call(renderContent);
-         };
+             if (attachedOneways.length) {
+               var connectedEndpointsOkay = attachedOneways.some(function (attachedOneway) {
+                 if ((isFirst ? attachedOneway.first() : attachedOneway.last()) !== nodeID) return true;
+                 if (nodeOccursMoreThanOnce(attachedOneway, nodeID)) return true;
+                 return false;
+               });
+               if (connectedEndpointsOkay) return [];
+             }
 
-         section.reRender = function () {
-           _containerSelection.call(renderContent);
-         };
+             var placement = isFirst ? 'start' : 'end',
+                 messageID = wayType + '.',
+                 referenceID = wayType + '.';
 
-         section.selection = function () {
-           return _containerSelection;
-         };
+             if (wayType === 'waterway') {
+               messageID += 'connected.' + placement;
+               referenceID += 'connected';
+             } else {
+               messageID += placement;
+               referenceID += placement;
+             }
 
-         section.disclosure = function () {
-           return _disclosure;
-         }; // may be called multiple times
+             return [new validationIssue({
+               type: type,
+               subtype: wayType,
+               severity: 'warning',
+               message: function message(context) {
+                 var entity = context.hasEntity(this.entityIds[0]);
+                 return entity ? _t.html('issues.impossible_oneway.' + messageID + '.message', {
+                   feature: utilDisplayLabel(entity, context.graph())
+                 }) : '';
+               },
+               reference: getReference(referenceID),
+               entityIds: [way.id, node.id],
+               dynamicFixes: function dynamicFixes() {
+                 var fixes = [];
 
+                 if (attachedOneways.length) {
+                   fixes.push(new validationIssueFix({
+                     icon: 'iD-operation-reverse',
+                     title: _t.html('issues.fix.reverse_feature.title'),
+                     entityIds: [way.id],
+                     onClick: function onClick(context) {
+                       var id = this.issue.entityIds[0];
+                       context.perform(actionReverse(id), _t('operations.reverse.annotation.line', {
+                         n: 1
+                       }));
+                     }
+                   }));
+                 }
 
-         function renderContent(selection) {
-           if (_shouldDisplay) {
-             var shouldDisplay = _shouldDisplay();
+                 if (node.tags.noexit !== 'yes') {
+                   var textDirection = _mainLocalizer.textDirection();
+                   var useLeftContinue = isFirst && textDirection === 'ltr' || !isFirst && textDirection === 'rtl';
+                   fixes.push(new validationIssueFix({
+                     icon: 'iD-operation-continue' + (useLeftContinue ? '-left' : ''),
+                     title: _t.html('issues.fix.continue_from_' + (isFirst ? 'start' : 'end') + '.title'),
+                     onClick: function onClick(context) {
+                       var entityID = this.issue.entityIds[0];
+                       var vertexID = this.issue.entityIds[1];
+                       var way = context.entity(entityID);
+                       var vertex = context.entity(vertexID);
+                       continueDrawing(way, vertex, context);
+                     }
+                   }));
+                 }
 
-             selection.classed('hide', !shouldDisplay);
+                 return fixes;
+               },
+               loc: node.loc
+             })];
 
-             if (!shouldDisplay) {
-               selection.html('');
-               return;
+             function getReference(referenceID) {
+               return function showReference(selection) {
+                 selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.impossible_oneway.' + referenceID + '.reference'));
+               };
              }
            }
+         };
 
-           if (_disclosureContent) {
-             if (!_disclosure) {
-               _disclosure = uiDisclosure(context, id.replace(/-/g, '_'), _expandedByDefault()).label(_label || '')
-               /*.on('toggled', function(expanded) {
-                   if (expanded) { selection.node().parentNode.scrollTop += 200; }
-               })*/
-               .content(_disclosureContent);
-             }
-
-             if (_disclosureExpanded !== undefined) {
-               _disclosure.expanded(_disclosureExpanded);
-
-               _disclosureExpanded = undefined;
-             }
+         function continueDrawing(way, vertex, context) {
+           // make sure the vertex is actually visible and editable
+           var map = context.map();
 
-             selection.call(_disclosure);
-             return;
+           if (!context.editable() || !map.trimmedExtent().contains(vertex.loc)) {
+             map.zoomToEase(vertex);
            }
 
-           if (_content) {
-             selection.call(_content);
-           }
+           context.enter(modeDrawLine(context, way.id, context.graph(), 'line', way.affix(vertex.id), true));
          }
 
-         return section;
+         validation.type = type;
+         return validation;
        }
 
-       // {
-       //   key: 'string',     // required
-       //   value: 'string'    // optional
-       // }
-       //   -or-
-       // {
-       //   qid: 'string'      // brand wikidata  (e.g. 'Q37158')
-       // }
-       //
-
-       function uiTagReference(what) {
-         var wikibase = what.qid ? services.wikidata : services.osmWikibase;
-         var tagReference = {};
+       function validationIncompatibleSource() {
+         var type = 'incompatible_source';
+         var invalidSources = [{
+           id: 'google',
+           regex: 'google',
+           exceptRegex: 'books.google|Google Books|drive.google|googledrive|Google Drive'
+         }];
 
-         var _button = select(null);
+         var validation = function checkIncompatibleSource(entity) {
+           var entitySources = entity.tags && entity.tags.source && entity.tags.source.split(';');
+           if (!entitySources) return [];
+           var issues = [];
+           invalidSources.forEach(function (invalidSource) {
+             var hasInvalidSource = entitySources.some(function (source) {
+               if (!source.match(new RegExp(invalidSource.regex, 'i'))) return false;
+               if (invalidSource.exceptRegex && source.match(new RegExp(invalidSource.exceptRegex, 'i'))) return false;
+               return true;
+             });
+             if (!hasInvalidSource) return;
+             issues.push(new validationIssue({
+               type: type,
+               severity: 'warning',
+               message: function message(context) {
+                 var entity = context.hasEntity(this.entityIds[0]);
+                 return entity ? _t.html('issues.incompatible_source.' + invalidSource.id + '.feature.message', {
+                   feature: utilDisplayLabel(entity, context.graph(), true
+                   /* verbose */
+                   )
+                 }) : '';
+               },
+               reference: getReference(invalidSource.id),
+               entityIds: [entity.id],
+               dynamicFixes: function dynamicFixes() {
+                 return [new validationIssueFix({
+                   title: _t.html('issues.fix.remove_proprietary_data.title')
+                 })];
+               }
+             }));
+           });
+           return issues;
 
-         var _body = select(null);
+           function getReference(id) {
+             return function showReference(selection) {
+               selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.incompatible_source.' + id + '.reference'));
+             };
+           }
+         };
 
-         var _loaded;
+         validation.type = type;
+         return validation;
+       }
 
-         var _showing;
+       function validationMaprules() {
+         var type = 'maprules';
 
-         function load() {
-           if (!wikibase) return;
+         var validation = function checkMaprules(entity, graph) {
+           if (!services.maprules) return [];
+           var rules = services.maprules.validationRules();
+           var issues = [];
 
-           _button.classed('tag-reference-loading', true);
+           for (var i = 0; i < rules.length; i++) {
+             var rule = rules[i];
+             rule.findIssues(entity, graph, issues);
+           }
 
-           wikibase.getDocs(what, gotDocs);
-         }
+           return issues;
+         };
 
-         function gotDocs(err, docs) {
-           _body.html('');
+         validation.type = type;
+         return validation;
+       }
 
-           if (!docs || !docs.title) {
-             _body.append('p').attr('class', 'tag-reference-description').html(_t.html('inspector.no_documentation_key'));
+       function validationMismatchedGeometry() {
+         var type = 'mismatched_geometry';
 
-             done();
-             return;
-           }
+         function tagSuggestingLineIsArea(entity) {
+           if (entity.type !== 'way' || entity.isClosed()) return null;
+           var tagSuggestingArea = entity.tagSuggestingArea();
 
-           if (docs.imageURL) {
-             _body.append('img').attr('class', 'tag-reference-wiki-image').attr('src', docs.imageURL).on('load', function () {
-               done();
-             }).on('error', function () {
-               select(this).remove();
-               done();
-             });
-           } else {
-             done();
+           if (!tagSuggestingArea) {
+             return null;
            }
 
-           _body.append('p').attr('class', 'tag-reference-description').html(docs.description ? _mainLocalizer.htmlForLocalizedText(docs.description, docs.descriptionLocaleCode) : _t.html('inspector.no_documentation_key')).append('a').attr('class', 'tag-reference-edit').attr('target', '_blank').attr('title', _t('inspector.edit_reference')).attr('href', docs.editURL).call(svgIcon('#iD-icon-edit', 'inline'));
-
-           if (docs.wiki) {
-             _body.append('a').attr('class', 'tag-reference-link').attr('target', '_blank').attr('href', docs.wiki.url).call(svgIcon('#iD-icon-out-link', 'inline')).append('span').html(_t.html(docs.wiki.text));
-           } // Add link to info about "good changeset comments" - #2923
-
+           var asLine = _mainPresetIndex.matchTags(tagSuggestingArea, 'line');
+           var asArea = _mainPresetIndex.matchTags(tagSuggestingArea, 'area');
 
-           if (what.key === 'comment') {
-             _body.append('a').attr('class', 'tag-reference-comment-link').attr('target', '_blank').call(svgIcon('#iD-icon-out-link', 'inline')).attr('href', _t('commit.about_changeset_comments_link')).append('span').html(_t.html('commit.about_changeset_comments'));
+           if (asLine && asArea && asLine === asArea) {
+             // these tags also allow lines and making this an area wouldn't matter
+             return null;
            }
-         }
 
-         function done() {
-           _loaded = true;
-
-           _button.classed('tag-reference-loading', false);
+           return tagSuggestingArea;
+         }
 
-           _body.classed('expanded', true).transition().duration(200).style('max-height', '200px').style('opacity', '1');
+         function makeConnectEndpointsFixOnClick(way, graph) {
+           // must have at least three nodes to close this automatically
+           if (way.nodes.length < 3) return null;
+           var nodes = graph.childNodes(way),
+               testNodes;
+           var firstToLastDistanceMeters = geoSphericalDistance(nodes[0].loc, nodes[nodes.length - 1].loc); // if the distance is very small, attempt to merge the endpoints
 
-           _showing = true;
+           if (firstToLastDistanceMeters < 0.75) {
+             testNodes = nodes.slice(); // shallow copy
 
-           _button.selectAll('svg.icon use').each(function () {
-             var iconUse = select(this);
+             testNodes.pop();
+             testNodes.push(testNodes[0]); // make sure this will not create a self-intersection
 
-             if (iconUse.attr('href') === '#iD-icon-info') {
-               iconUse.attr('href', '#iD-icon-info-filled');
+             if (!geoHasSelfIntersections(testNodes, testNodes[0].id)) {
+               return function (context) {
+                 var way = context.entity(this.issue.entityIds[0]);
+                 context.perform(actionMergeNodes([way.nodes[0], way.nodes[way.nodes.length - 1]], nodes[0].loc), _t('issues.fix.connect_endpoints.annotation'));
+               };
              }
-           });
-         }
+           } // if the points were not merged, attempt to close the way
 
-         function hide() {
-           _body.transition().duration(200).style('max-height', '0px').style('opacity', '0').on('end', function () {
-             _body.classed('expanded', false);
-           });
 
-           _showing = false;
+           testNodes = nodes.slice(); // shallow copy
 
-           _button.selectAll('svg.icon use').each(function () {
-             var iconUse = select(this);
+           testNodes.push(testNodes[0]); // make sure this will not create a self-intersection
 
-             if (iconUse.attr('href') === '#iD-icon-info-filled') {
-               iconUse.attr('href', '#iD-icon-info');
-             }
-           });
+           if (!geoHasSelfIntersections(testNodes, testNodes[0].id)) {
+             return function (context) {
+               var wayId = this.issue.entityIds[0];
+               var way = context.entity(wayId);
+               var nodeId = way.nodes[0];
+               var index = way.nodes.length;
+               context.perform(actionAddVertex(wayId, nodeId, index), _t('issues.fix.connect_endpoints.annotation'));
+             };
+           }
          }
 
-         tagReference.button = function (selection, klass, iconName) {
-           _button = selection.selectAll('.tag-reference-button').data([0]);
-           _button = _button.enter().append('button').attr('class', 'tag-reference-button ' + (klass || '')).attr('title', _t('icons.information')).call(svgIcon('#iD-icon-' + (iconName || 'inspect'))).merge(_button);
+         function lineTaggedAsAreaIssue(entity) {
+           var tagSuggestingArea = tagSuggestingLineIsArea(entity);
+           if (!tagSuggestingArea) return null;
+           return new validationIssue({
+             type: type,
+             subtype: 'area_as_line',
+             severity: 'warning',
+             message: function message(context) {
+               var entity = context.hasEntity(this.entityIds[0]);
+               return entity ? _t.html('issues.tag_suggests_area.message', {
+                 feature: utilDisplayLabel(entity, 'area', true
+                 /* verbose */
+                 ),
+                 tag: utilTagText({
+                   tags: tagSuggestingArea
+                 })
+               }) : '';
+             },
+             reference: showReference,
+             entityIds: [entity.id],
+             hash: JSON.stringify(tagSuggestingArea),
+             dynamicFixes: function dynamicFixes(context) {
+               var fixes = [];
+               var entity = context.entity(this.entityIds[0]);
+               var connectEndsOnClick = makeConnectEndpointsFixOnClick(entity, context.graph());
+               fixes.push(new validationIssueFix({
+                 title: _t.html('issues.fix.connect_endpoints.title'),
+                 onClick: connectEndsOnClick
+               }));
+               fixes.push(new validationIssueFix({
+                 icon: 'iD-operation-delete',
+                 title: _t.html('issues.fix.remove_tag.title'),
+                 onClick: function onClick(context) {
+                   var entityId = this.issue.entityIds[0];
+                   var entity = context.entity(entityId);
+                   var tags = Object.assign({}, entity.tags); // shallow copy
 
-           _button.on('click', function (d3_event) {
-             d3_event.stopPropagation();
-             d3_event.preventDefault();
-             this.blur(); // avoid keeping focus on the button - #4641
+                   for (var key in tagSuggestingArea) {
+                     delete tags[key];
+                   }
 
-             if (_showing) {
-               hide();
-             } else if (_loaded) {
-               done();
-             } else {
-               load();
+                   context.perform(actionChangeTags(entityId, tags), _t('issues.fix.remove_tag.annotation'));
+                 }
+               }));
+               return fixes;
              }
            });
-         };
-
-         tagReference.body = function (selection) {
-           var itemID = what.qid || what.key + '-' + (what.value || '');
-           _body = selection.selectAll('.tag-reference-body').data([itemID], function (d) {
-             return d;
-           });
 
-           _body.exit().remove();
-
-           _body = _body.enter().append('div').attr('class', 'tag-reference-body').style('max-height', '0').style('opacity', '0').merge(_body);
-
-           if (_showing === false) {
-             hide();
+           function showReference(selection) {
+             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.tag_suggests_area.reference'));
            }
-         };
+         }
 
-         tagReference.showing = function (val) {
-           if (!arguments.length) return _showing;
-           _showing = val;
-           return tagReference;
-         };
+         function vertexPointIssue(entity, graph) {
+           // we only care about nodes
+           if (entity.type !== 'node') return null; // ignore tagless points
 
-         return tagReference;
-       }
+           if (Object.keys(entity.tags).length === 0) return null; // address lines are special so just ignore them
 
-       function uiSectionRawTagEditor(id, context) {
-         var section = uiSection(id, context).classes('raw-tag-editor').label(function () {
-           var count = Object.keys(_tags).filter(function (d) {
-             return d;
-           }).length;
-           return _t('inspector.title_count', {
-             title: _t.html('inspector.tags'),
-             count: count
-           });
-         }).expandedByDefault(false).disclosureContent(renderDisclosureContent);
-         var taginfo = services.taginfo;
-         var dispatch$1 = dispatch('change');
-         var availableViews = [{
-           id: 'list',
-           icon: '#fas-th-list'
-         }, {
-           id: 'text',
-           icon: '#fas-i-cursor'
-         }];
+           if (entity.isOnAddressLine(graph)) return null;
+           var geometry = entity.geometry(graph);
+           var allowedGeometries = osmNodeGeometriesForTags(entity.tags);
 
-         var _tagView = corePreferences('raw-tag-editor-view') || 'list'; // 'list, 'text'
+           if (geometry === 'point' && !allowedGeometries.point && allowedGeometries.vertex) {
+             return new validationIssue({
+               type: type,
+               subtype: 'vertex_as_point',
+               severity: 'warning',
+               message: function message(context) {
+                 var entity = context.hasEntity(this.entityIds[0]);
+                 return entity ? _t.html('issues.vertex_as_point.message', {
+                   feature: utilDisplayLabel(entity, 'vertex', true
+                   /* verbose */
+                   )
+                 }) : '';
+               },
+               reference: function showReference(selection) {
+                 selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.vertex_as_point.reference'));
+               },
+               entityIds: [entity.id]
+             });
+           } else if (geometry === 'vertex' && !allowedGeometries.vertex && allowedGeometries.point) {
+             return new validationIssue({
+               type: type,
+               subtype: 'point_as_vertex',
+               severity: 'warning',
+               message: function message(context) {
+                 var entity = context.hasEntity(this.entityIds[0]);
+                 return entity ? _t.html('issues.point_as_vertex.message', {
+                   feature: utilDisplayLabel(entity, 'point', true
+                   /* verbose */
+                   )
+                 }) : '';
+               },
+               reference: function showReference(selection) {
+                 selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.point_as_vertex.reference'));
+               },
+               entityIds: [entity.id],
+               dynamicFixes: extractPointDynamicFixes
+             });
+           }
 
+           return null;
+         }
 
-         var _readOnlyTags = []; // the keys in the order we want them to display
+         function otherMismatchIssue(entity, graph) {
+           // ignore boring features
+           if (!entity.hasInterestingTags()) return null;
+           if (entity.type !== 'node' && entity.type !== 'way') return null; // address lines are special so just ignore them
+
+           if (entity.type === 'node' && entity.isOnAddressLine(graph)) return null;
+           var sourceGeom = entity.geometry(graph);
+           var targetGeoms = entity.type === 'way' ? ['point', 'vertex'] : ['line', 'area'];
+           if (sourceGeom === 'area') targetGeoms.unshift('line');
+           var targetGeom = targetGeoms.find(function (nodeGeom) {
+             var asSource = _mainPresetIndex.matchTags(entity.tags, sourceGeom);
+             var asTarget = _mainPresetIndex.matchTags(entity.tags, nodeGeom);
+             if (!asSource || !asTarget || asSource === asTarget || // sometimes there are two presets with the same tags for different geometries
+             fastDeepEqual(asSource.tags, asTarget.tags)) return false;
+             if (asTarget.isFallback()) return false;
+             var primaryKey = Object.keys(asTarget.tags)[0]; // special case: buildings-as-points are discouraged by iD, but common in OSM, so ignore them
+
+             if (primaryKey === 'building') return false;
+             if (asTarget.tags[primaryKey] === '*') return false;
+             return asSource.isFallback() || asSource.tags[primaryKey] === '*';
+           });
+           if (!targetGeom) return null;
+           var subtype = targetGeom + '_as_' + sourceGeom;
+           if (targetGeom === 'vertex') targetGeom = 'point';
+           if (sourceGeom === 'vertex') sourceGeom = 'point';
+           var referenceId = targetGeom + '_as_' + sourceGeom;
+           var dynamicFixes;
+
+           if (targetGeom === 'point') {
+             dynamicFixes = extractPointDynamicFixes;
+           } else if (sourceGeom === 'area' && targetGeom === 'line') {
+             dynamicFixes = lineToAreaDynamicFixes;
+           }
 
-         var _orderedKeys = [];
-         var _showBlank = false;
-         var _pendingChange = null;
+           return new validationIssue({
+             type: type,
+             subtype: subtype,
+             severity: 'warning',
+             message: function message(context) {
+               var entity = context.hasEntity(this.entityIds[0]);
+               return entity ? _t.html('issues.' + referenceId + '.message', {
+                 feature: utilDisplayLabel(entity, targetGeom, true
+                 /* verbose */
+                 )
+               }) : '';
+             },
+             reference: function showReference(selection) {
+               selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.mismatched_geometry.reference'));
+             },
+             entityIds: [entity.id],
+             dynamicFixes: dynamicFixes
+           });
+         }
 
-         var _state;
+         function lineToAreaDynamicFixes(context) {
+           var convertOnClick;
+           var entityId = this.entityIds[0];
+           var entity = context.entity(entityId);
+           var tags = Object.assign({}, entity.tags); // shallow copy
 
-         var _presets;
+           delete tags.area;
 
-         var _tags;
+           if (!osmTagSuggestingArea(tags)) {
+             // if removing the area tag would make this a line, offer that as a quick fix
+             convertOnClick = function convertOnClick(context) {
+               var entityId = this.issue.entityIds[0];
+               var entity = context.entity(entityId);
+               var tags = Object.assign({}, entity.tags); // shallow copy
 
-         var _entityIDs;
+               if (tags.area) {
+                 delete tags.area;
+               }
 
-         var _didInteract = false;
+               context.perform(actionChangeTags(entityId, tags), _t('issues.fix.convert_to_line.annotation'));
+             };
+           }
 
-         function interacted() {
-           _didInteract = true;
+           return [new validationIssueFix({
+             icon: 'iD-icon-line',
+             title: _t.html('issues.fix.convert_to_line.title'),
+             onClick: convertOnClick
+           })];
          }
 
-         function renderDisclosureContent(wrap) {
-           // remove deleted keys
-           _orderedKeys = _orderedKeys.filter(function (key) {
-             return _tags[key] !== undefined;
-           }); // When switching to a different entity or changing the state (hover/select)
-           // reorder the keys alphabetically.
-           // We trigger this by emptying the `_orderedKeys` array, then it will be rebuilt here.
-           // Otherwise leave their order alone - #5857, #5927
-
-           var all = Object.keys(_tags).sort();
-           var missingKeys = utilArrayDifference(all, _orderedKeys);
-
-           for (var i in missingKeys) {
-             _orderedKeys.push(missingKeys[i]);
-           } // assemble row data
+         function extractPointDynamicFixes(context) {
+           var entityId = this.entityIds[0];
+           var extractOnClick = null;
 
+           if (!context.hasHiddenConnections(entityId)) {
+             extractOnClick = function extractOnClick(context) {
+               var entityId = this.issue.entityIds[0];
+               var action = actionExtract(entityId, context.projection);
+               context.perform(action, _t('operations.extract.annotation', {
+                 n: 1
+               })); // re-enter mode to trigger updates
 
-           var rowData = _orderedKeys.map(function (key, i) {
-             return {
-               index: i,
-               key: key,
-               value: _tags[key]
+               context.enter(modeSelect(context, [action.getExtractedNodeID()]));
              };
-           }); // append blank row last, if necessary
+           }
 
+           return [new validationIssueFix({
+             icon: 'iD-operation-extract',
+             title: _t.html('issues.fix.extract_point.title'),
+             onClick: extractOnClick
+           })];
+         }
 
-           if (!rowData.length || _showBlank) {
-             _showBlank = false;
-             rowData.push({
-               index: rowData.length,
-               key: '',
-               value: ''
-             });
-           } // View Options
+         function unclosedMultipolygonPartIssues(entity, graph) {
+           if (entity.type !== 'relation' || !entity.isMultipolygon() || entity.isDegenerate() || // cannot determine issues for incompletely-downloaded relations
+           !entity.isComplete(graph)) return [];
+           var sequences = osmJoinWays(entity.members, graph);
+           var issues = [];
 
+           for (var i in sequences) {
+             var sequence = sequences[i];
+             if (!sequence.nodes) continue;
+             var firstNode = sequence.nodes[0];
+             var lastNode = sequence.nodes[sequence.nodes.length - 1]; // part is closed if the first and last nodes are the same
 
-           var options = wrap.selectAll('.raw-tag-options').data([0]);
-           options.exit().remove();
-           var optionsEnter = options.enter().insert('div', ':first-child').attr('class', 'raw-tag-options');
-           var optionEnter = optionsEnter.selectAll('.raw-tag-option').data(availableViews, function (d) {
-             return d.id;
-           }).enter();
-           optionEnter.append('button').attr('class', function (d) {
-             return 'raw-tag-option raw-tag-option-' + d.id + (_tagView === d.id ? ' selected' : '');
-           }).attr('title', function (d) {
-             return _t('icons.' + d.id);
-           }).on('click', function (d3_event, d) {
-             _tagView = d.id;
-             corePreferences('raw-tag-editor-view', d.id);
-             wrap.selectAll('.raw-tag-option').classed('selected', function (datum) {
-               return datum === d;
+             if (firstNode === lastNode) continue;
+             var issue = new validationIssue({
+               type: type,
+               subtype: 'unclosed_multipolygon_part',
+               severity: 'warning',
+               message: function message(context) {
+                 var entity = context.hasEntity(this.entityIds[0]);
+                 return entity ? _t.html('issues.unclosed_multipolygon_part.message', {
+                   feature: utilDisplayLabel(entity, context.graph(), true
+                   /* verbose */
+                   )
+                 }) : '';
+               },
+               reference: showReference,
+               loc: sequence.nodes[0].loc,
+               entityIds: [entity.id],
+               hash: sequence.map(function (way) {
+                 return way.id;
+               }).join()
              });
-             wrap.selectAll('.tag-text').classed('hide', d.id !== 'text').each(setTextareaHeight);
-             wrap.selectAll('.tag-list, .add-row').classed('hide', d.id !== 'list');
-           }).each(function (d) {
-             select(this).call(svgIcon(d.icon));
-           }); // View as Text
+             issues.push(issue);
+           }
 
-           var textData = rowsToText(rowData);
-           var textarea = wrap.selectAll('.tag-text').data([0]);
-           textarea = textarea.enter().append('textarea').attr('class', 'tag-text' + (_tagView !== 'text' ? ' hide' : '')).call(utilNoAuto).attr('placeholder', _t('inspector.key_value')).attr('spellcheck', 'false').merge(textarea);
-           textarea.call(utilGetSetValue, textData).each(setTextareaHeight).on('input', setTextareaHeight).on('focus', interacted).on('blur', textChanged).on('change', textChanged); // View as List
+           return issues;
 
-           var list = wrap.selectAll('.tag-list').data([0]);
-           list = list.enter().append('ul').attr('class', 'tag-list' + (_tagView !== 'list' ? ' hide' : '')).merge(list); // Container for the Add button
+           function showReference(selection) {
+             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.unclosed_multipolygon_part.reference'));
+           }
+         }
 
-           var addRowEnter = wrap.selectAll('.add-row').data([0]).enter().append('div').attr('class', 'add-row' + (_tagView !== 'list' ? ' hide' : ''));
-           addRowEnter.append('button').attr('class', 'add-tag').call(svgIcon('#iD-icon-plus', 'light')).on('click', addTag);
-           addRowEnter.append('div').attr('class', 'space-value'); // preserve space
+         var validation = function checkMismatchedGeometry(entity, graph) {
+           var vertexPoint = vertexPointIssue(entity, graph);
+           if (vertexPoint) return [vertexPoint];
+           var lineAsArea = lineTaggedAsAreaIssue(entity);
+           if (lineAsArea) return [lineAsArea];
+           var mismatch = otherMismatchIssue(entity, graph);
+           if (mismatch) return [mismatch];
+           return unclosedMultipolygonPartIssues(entity, graph);
+         };
 
-           addRowEnter.append('div').attr('class', 'space-buttons'); // preserve space
-           // Tag list items
+         validation.type = type;
+         return validation;
+       }
 
-           var items = list.selectAll('.tag-row').data(rowData, function (d) {
-             return d.key;
-           });
-           items.exit().each(unbind).remove(); // Enter
+       function validationMissingRole() {
+         var type = 'missing_role';
 
-           var itemsEnter = items.enter().append('li').attr('class', 'tag-row').classed('readonly', isReadOnly);
-           var innerWrap = itemsEnter.append('div').attr('class', 'inner-wrap');
-           innerWrap.append('div').attr('class', 'key-wrap').append('input').property('type', 'text').attr('class', 'key').call(utilNoAuto).on('focus', interacted).on('blur', keyChange).on('change', keyChange);
-           innerWrap.append('div').attr('class', 'value-wrap').append('input').property('type', 'text').attr('class', 'value').call(utilNoAuto).on('focus', interacted).on('blur', valueChange).on('change', valueChange).on('keydown.push-more', pushMore);
-           innerWrap.append('button').attr('class', 'form-field-button remove').attr('title', _t('icons.remove')).call(svgIcon('#iD-operation-delete')); // Update
+         var validation = function checkMissingRole(entity, graph) {
+           var issues = [];
 
-           items = items.merge(itemsEnter).sort(function (a, b) {
-             return a.index - b.index;
-           });
-           items.each(function (d) {
-             var row = select(this);
-             var key = row.select('input.key'); // propagate bound data
+           if (entity.type === 'way') {
+             graph.parentRelations(entity).forEach(function (relation) {
+               if (!relation.isMultipolygon()) return;
+               var member = relation.memberById(entity.id);
 
-             var value = row.select('input.value'); // propagate bound data
+               if (member && isMissingRole(member)) {
+                 issues.push(makeIssue(entity, relation, member));
+               }
+             });
+           } else if (entity.type === 'relation' && entity.isMultipolygon()) {
+             entity.indexedMembers().forEach(function (member) {
+               var way = graph.hasEntity(member.id);
 
-             if (_entityIDs && taginfo && _state !== 'hover') {
-               bindTypeahead(key, value);
-             }
+               if (way && isMissingRole(member)) {
+                 issues.push(makeIssue(way, entity, member));
+               }
+             });
+           }
 
-             var referenceOptions = {
-               key: d.key
-             };
+           return issues;
+         };
 
-             if (typeof d.value === 'string') {
-               referenceOptions.value = d.value;
+         function isMissingRole(member) {
+           return !member.role || !member.role.trim().length;
+         }
+
+         function makeIssue(way, relation, member) {
+           return new validationIssue({
+             type: type,
+             severity: 'warning',
+             message: function message(context) {
+               var member = context.hasEntity(this.entityIds[1]),
+                   relation = context.hasEntity(this.entityIds[0]);
+               return member && relation ? _t.html('issues.missing_role.message', {
+                 member: utilDisplayLabel(member, context.graph()),
+                 relation: utilDisplayLabel(relation, context.graph())
+               }) : '';
+             },
+             reference: showReference,
+             entityIds: [relation.id, way.id],
+             data: {
+               member: member
+             },
+             hash: member.index.toString(),
+             dynamicFixes: function dynamicFixes() {
+               return [makeAddRoleFix('inner'), makeAddRoleFix('outer'), new validationIssueFix({
+                 icon: 'iD-operation-delete',
+                 title: _t.html('issues.fix.remove_from_relation.title'),
+                 onClick: function onClick(context) {
+                   context.perform(actionDeleteMember(this.issue.entityIds[0], this.issue.data.member.index), _t('operations.delete_member.annotation', {
+                     n: 1
+                   }));
+                 }
+               })];
              }
+           });
 
-             var reference = uiTagReference(referenceOptions);
+           function showReference(selection) {
+             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.missing_role.multipolygon.reference'));
+           }
+         }
 
-             if (_state === 'hover') {
-               reference.showing(false);
+         function makeAddRoleFix(role) {
+           return new validationIssueFix({
+             title: _t.html('issues.fix.set_as_' + role + '.title'),
+             onClick: function onClick(context) {
+               var oldMember = this.issue.data.member;
+               var member = {
+                 id: this.issue.entityIds[1],
+                 type: oldMember.type,
+                 role: role
+               };
+               context.perform(actionChangeMember(this.issue.entityIds[0], member, oldMember.index), _t('operations.change_role.annotation', {
+                 n: 1
+               }));
              }
-
-             row.select('.inner-wrap') // propagate bound data
-             .call(reference.button);
-             row.call(reference.body);
-             row.select('button.remove'); // propagate bound data
-           });
-           items.selectAll('input.key').attr('title', function (d) {
-             return d.key;
-           }).call(utilGetSetValue, function (d) {
-             return d.key;
-           }).attr('readonly', function (d) {
-             return isReadOnly(d) || typeof d.value !== 'string' || null;
-           });
-           items.selectAll('input.value').attr('title', function (d) {
-             return Array.isArray(d.value) ? d.value.filter(Boolean).join('\n') : d.value;
-           }).classed('mixed', function (d) {
-             return Array.isArray(d.value);
-           }).attr('placeholder', function (d) {
-             return typeof d.value === 'string' ? null : _t('inspector.multiple_values');
-           }).call(utilGetSetValue, function (d) {
-             return typeof d.value === 'string' ? d.value : '';
-           }).attr('readonly', function (d) {
-             return isReadOnly(d) || null;
            });
-           items.selectAll('button.remove').on(('PointerEvent' in window ? 'pointer' : 'mouse') + 'down', removeTag); // 'click' fires too late - #5878
          }
 
-         function isReadOnly(d) {
-           for (var i = 0; i < _readOnlyTags.length; i++) {
-             if (d.key.match(_readOnlyTags[i]) !== null) {
-               return true;
-             }
+         validation.type = type;
+         return validation;
+       }
+
+       function validationMissingTag(context) {
+         var type = 'missing_tag';
+
+         function hasDescriptiveTags(entity, graph) {
+           var onlyAttributeKeys = ['description', 'name', 'note', 'start_date'];
+           var entityDescriptiveKeys = Object.keys(entity.tags).filter(function (k) {
+             if (k === 'area' || !osmIsInterestingTag(k)) return false;
+             return !onlyAttributeKeys.some(function (attributeKey) {
+               return k === attributeKey || k.indexOf(attributeKey + ':') === 0;
+             });
+           });
+
+           if (entity.type === 'relation' && entityDescriptiveKeys.length === 1 && entity.tags.type === 'multipolygon') {
+             // this relation's only interesting tag just says its a multipolygon,
+             // which is not descriptive enough
+             // It's okay for a simple multipolygon to have no descriptive tags
+             // if its outer way has them (old model, see `outdated_tags.js`)
+             return osmOldMultipolygonOuterMemberOfRelation(entity, graph);
            }
 
-           return false;
+           return entityDescriptiveKeys.length > 0;
          }
 
-         function setTextareaHeight() {
-           if (_tagView !== 'text') return;
-           var selection = select(this);
-           var matches = selection.node().value.match(/\n/g);
-           var lineCount = 2 + Number(matches && matches.length);
-           var lineHeight = 20;
-           selection.style('height', lineCount * lineHeight + 'px');
+         function isUnknownRoad(entity) {
+           return entity.type === 'way' && entity.tags.highway === 'road';
          }
 
-         function stringify(s) {
-           return JSON.stringify(s).slice(1, -1); // without leading/trailing "
+         function isUntypedRelation(entity) {
+           return entity.type === 'relation' && !entity.tags.type;
          }
 
-         function unstringify(s) {
-           var leading = '';
-           var trailing = '';
+         var validation = function checkMissingTag(entity, graph) {
+           var subtype;
+           var osm = context.connection();
+           var isUnloadedNode = entity.type === 'node' && osm && !osm.isDataLoaded(entity.loc); // we can't know if the node is a vertex if the tile is undownloaded
 
-           if (s.length < 1 || s.charAt(0) !== '"') {
-             leading = '"';
-           }
+           if (!isUnloadedNode && // allow untagged nodes that are part of ways
+           entity.geometry(graph) !== 'vertex' && // allow untagged entities that are part of relations
+           !entity.hasParentRelations(graph)) {
+             if (Object.keys(entity.tags).length === 0) {
+               subtype = 'any';
+             } else if (!hasDescriptiveTags(entity, graph)) {
+               subtype = 'descriptive';
+             } else if (isUntypedRelation(entity)) {
+               subtype = 'relation_type';
+             }
+           } // flag an unknown road even if it's a member of a relation
 
-           if (s.length < 2 || s.charAt(s.length - 1) !== '"' || s.charAt(s.length - 1) === '"' && s.charAt(s.length - 2) === '\\') {
-             trailing = '"';
+
+           if (!subtype && isUnknownRoad(entity)) {
+             subtype = 'highway_classification';
            }
 
-           return JSON.parse(leading + s + trailing);
-         }
+           if (!subtype) return [];
+           var messageID = subtype === 'highway_classification' ? 'unknown_road' : 'missing_tag.' + subtype;
+           var referenceID = subtype === 'highway_classification' ? 'unknown_road' : 'missing_tag'; // can always delete if the user created it in the first place..
 
-         function rowsToText(rows) {
-           var str = rows.filter(function (row) {
-             return row.key && row.key.trim() !== '';
-           }).map(function (row) {
-             var rawVal = row.value;
-             if (typeof rawVal !== 'string') rawVal = '*';
-             var val = rawVal ? stringify(rawVal) : '';
-             return stringify(row.key) + '=' + val;
-           }).join('\n');
+           var canDelete = entity.version === undefined || entity.v !== undefined;
+           var severity = canDelete && subtype !== 'highway_classification' ? 'error' : 'warning';
+           return [new validationIssue({
+             type: type,
+             subtype: subtype,
+             severity: severity,
+             message: function message(context) {
+               var entity = context.hasEntity(this.entityIds[0]);
+               return entity ? _t.html('issues.' + messageID + '.message', {
+                 feature: utilDisplayLabel(entity, context.graph())
+               }) : '';
+             },
+             reference: showReference,
+             entityIds: [entity.id],
+             dynamicFixes: function dynamicFixes(context) {
+               var fixes = [];
+               var selectFixType = subtype === 'highway_classification' ? 'select_road_type' : 'select_preset';
+               fixes.push(new validationIssueFix({
+                 icon: 'iD-icon-search',
+                 title: _t.html('issues.fix.' + selectFixType + '.title'),
+                 onClick: function onClick(context) {
+                   context.ui().sidebar.showPresetList();
+                 }
+               }));
+               var deleteOnClick;
+               var id = this.entityIds[0];
+               var operation = operationDelete(context, [id]);
+               var disabledReasonID = operation.disabled();
 
-           if (_state !== 'hover' && str.length) {
-             return str + '\n';
+               if (!disabledReasonID) {
+                 deleteOnClick = function deleteOnClick(context) {
+                   var id = this.issue.entityIds[0];
+                   var operation = operationDelete(context, [id]);
+
+                   if (!operation.disabled()) {
+                     operation();
+                   }
+                 };
+               }
+
+               fixes.push(new validationIssueFix({
+                 icon: 'iD-operation-delete',
+                 title: _t.html('issues.fix.delete_feature.title'),
+                 disabledReason: disabledReasonID ? _t('operations.delete.' + disabledReasonID + '.single') : undefined,
+                 onClick: deleteOnClick
+               }));
+               return fixes;
+             }
+           })];
+
+           function showReference(selection) {
+             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.' + referenceID + '.reference'));
            }
+         };
 
-           return str;
-         }
+         validation.type = type;
+         return validation;
+       }
 
-         function textChanged() {
-           var newText = this.value.trim();
-           var newTags = {};
-           newText.split('\n').forEach(function (row) {
-             var m = row.match(/^\s*([^=]+)=(.*)$/);
+       function validationOutdatedTags() {
+         var type = 'outdated_tags';
+         var _waitingForDeprecated = true;
 
-             if (m !== null) {
-               var k = context.cleanTagKey(unstringify(m[1].trim()));
-               var v = context.cleanTagValue(unstringify(m[2].trim()));
-               newTags[k] = v;
-             }
-           });
-           var tagDiff = utilTagDiff(_tags, newTags);
-           if (!tagDiff.length) return;
-           _pendingChange = _pendingChange || {};
-           tagDiff.forEach(function (change) {
-             if (isReadOnly({
-               key: change.key
-             })) return; // skip unchanged multiselection placeholders
+         var _dataDeprecated; // fetch deprecated tags
 
-             if (change.newVal === '*' && typeof change.oldVal !== 'string') return;
 
-             if (change.type === '-') {
-               _pendingChange[change.key] = undefined;
-             } else if (change.type === '+') {
-               _pendingChange[change.key] = change.newVal || '';
-             }
-           });
+         _mainFileFetcher.get('deprecated').then(function (d) {
+           return _dataDeprecated = d;
+         })["catch"](function () {
+           /* ignore */
+         })["finally"](function () {
+           return _waitingForDeprecated = false;
+         });
 
-           if (Object.keys(_pendingChange).length === 0) {
-             _pendingChange = null;
-             return;
-           }
+         function oldTagIssues(entity, graph) {
+           var oldTags = Object.assign({}, entity.tags); // shallow copy
 
-           scheduleChange();
-         }
+           var preset = _mainPresetIndex.match(entity, graph);
+           var subtype = 'deprecated_tags';
+           if (!preset) return [];
+           if (!entity.hasInterestingTags()) return []; // Upgrade preset, if a replacement is available..
 
-         function pushMore(d3_event) {
-           // if pressing Tab on the last value field with content, add a blank row
-           if (d3_event.keyCode === 9 && !d3_event.shiftKey && section.selection().selectAll('.tag-list li:last-child input.value').node() === this && utilGetSetValue(select(this))) {
-             addTag();
-           }
-         }
+           if (preset.replacement) {
+             var newPreset = _mainPresetIndex.item(preset.replacement);
+             graph = actionChangePreset(entity.id, preset, newPreset, true
+             /* skip field defaults */
+             )(graph);
+             entity = graph.entity(entity.id);
+             preset = newPreset;
+           } // Upgrade deprecated tags..
 
-         function bindTypeahead(key, value) {
-           if (isReadOnly(key.datum())) return;
 
-           if (Array.isArray(value.datum().value)) {
-             value.call(uiCombobox(context, 'tag-value').minItems(1).fetcher(function (value, callback) {
-               var keyString = utilGetSetValue(key);
-               if (!_tags[keyString]) return;
+           if (_dataDeprecated) {
+             var deprecatedTags = entity.deprecatedTags(_dataDeprecated);
 
-               var data = _tags[keyString].filter(Boolean).map(function (tagValue) {
-                 return {
-                   value: tagValue,
-                   title: tagValue
-                 };
+             if (deprecatedTags.length) {
+               deprecatedTags.forEach(function (tag) {
+                 graph = actionUpgradeTags(entity.id, tag.old, tag.replace)(graph);
                });
+               entity = graph.entity(entity.id);
+             }
+           } // Add missing addTags from the detected preset
 
-               callback(data);
-             }));
-             return;
-           }
 
-           var geometry = context.graph().geometry(_entityIDs[0]);
-           key.call(uiCombobox(context, 'tag-key').fetcher(function (value, callback) {
-             taginfo.keys({
-               debounce: true,
-               geometry: geometry,
-               query: value
-             }, function (err, data) {
-               if (!err) {
-                 var filtered = data.filter(function (d) {
-                   return _tags[d.value] === undefined;
-                 });
-                 callback(sort(value, filtered));
+           var newTags = Object.assign({}, entity.tags); // shallow copy
+
+           if (preset.tags !== preset.addTags) {
+             Object.keys(preset.addTags).forEach(function (k) {
+               if (!newTags[k]) {
+                 if (preset.addTags[k] === '*') {
+                   newTags[k] = 'yes';
+                 } else {
+                   newTags[k] = preset.addTags[k];
+                 }
                }
              });
+           } // Attempt to match a canonical record in the name-suggestion-index.
+
+
+           var nsi = services.nsi;
+           var waitingForNsi = false;
+
+           if (nsi) {
+             waitingForNsi = nsi.status() === 'loading';
+
+             if (!waitingForNsi) {
+               var loc = entity.extent(graph).center();
+               var result = nsi.upgradeTags(newTags, loc);
+
+               if (result) {
+                 newTags = result;
+                 subtype = 'noncanonical_brand';
+               }
+             }
+           }
+
+           var issues = [];
+           issues.provisional = _waitingForDeprecated || waitingForNsi; // determine diff
+
+           var tagDiff = utilTagDiff(oldTags, newTags);
+           if (!tagDiff.length) return issues;
+           var isOnlyAddingTags = tagDiff.every(function (d) {
+             return d.type === '+';
+           });
+           var prefix = '';
+
+           if (subtype === 'noncanonical_brand') {
+             prefix = 'noncanonical_brand.';
+           } else if (subtype === 'deprecated_tags' && isOnlyAddingTags) {
+             subtype = 'incomplete_tags';
+             prefix = 'incomplete.';
+           } // don't allow autofixing brand tags
+
+
+           var autoArgs = subtype !== 'noncanonical_brand' ? [doUpgrade, _t('issues.fix.upgrade_tags.annotation')] : null;
+           issues.push(new validationIssue({
+             type: type,
+             subtype: subtype,
+             severity: 'warning',
+             message: showMessage,
+             reference: showReference,
+             entityIds: [entity.id],
+             hash: utilHashcode(JSON.stringify(tagDiff)),
+             dynamicFixes: function dynamicFixes() {
+               return [new validationIssueFix({
+                 autoArgs: autoArgs,
+                 title: _t.html('issues.fix.upgrade_tags.title'),
+                 onClick: function onClick(context) {
+                   context.perform(doUpgrade, _t('issues.fix.upgrade_tags.annotation'));
+                 }
+               })];
+             }
            }));
-           value.call(uiCombobox(context, 'tag-value').fetcher(function (value, callback) {
-             taginfo.values({
-               debounce: true,
-               key: utilGetSetValue(key),
-               geometry: geometry,
-               query: value
-             }, function (err, data) {
-               if (!err) callback(sort(value, data));
-             });
-           }));
+           return issues;
 
-           function sort(value, data) {
-             var sameletter = [];
-             var other = [];
+           function doUpgrade(graph) {
+             var currEntity = graph.hasEntity(entity.id);
+             if (!currEntity) return graph;
+             var newTags = Object.assign({}, currEntity.tags); // shallow copy
 
-             for (var i = 0; i < data.length; i++) {
-               if (data[i].value.substring(0, value.length) === value) {
-                 sameletter.push(data[i]);
-               } else {
-                 other.push(data[i]);
+             tagDiff.forEach(function (diff) {
+               if (diff.type === '-') {
+                 delete newTags[diff.key];
+               } else if (diff.type === '+') {
+                 newTags[diff.key] = diff.newVal;
                }
+             });
+             return actionChangeTags(currEntity.id, newTags)(graph);
+           }
+
+           function showMessage(context) {
+             var currEntity = context.hasEntity(entity.id);
+             if (!currEntity) return '';
+             var messageID = "issues.outdated_tags.".concat(prefix, "message");
+
+             if (subtype === 'noncanonical_brand' && isOnlyAddingTags) {
+               messageID += '_incomplete';
              }
 
-             return sameletter.concat(other);
+             return _t.html(messageID, {
+               feature: utilDisplayLabel(currEntity, context.graph(), true
+               /* verbose */
+               )
+             });
            }
-         }
 
-         function unbind() {
-           var row = select(this);
-           row.selectAll('input.key').call(uiCombobox.off, context);
-           row.selectAll('input.value').call(uiCombobox.off, context);
+           function showReference(selection) {
+             var enter = selection.selectAll('.issue-reference').data([0]).enter();
+             enter.append('div').attr('class', 'issue-reference').html(_t.html("issues.outdated_tags.".concat(prefix, "reference")));
+             enter.append('strong').html(_t.html('issues.suggested'));
+             enter.append('table').attr('class', 'tagDiff-table').selectAll('.tagDiff-row').data(tagDiff).enter().append('tr').attr('class', 'tagDiff-row').append('td').attr('class', function (d) {
+               var klass = d.type === '+' ? 'add' : 'remove';
+               return "tagDiff-cell tagDiff-cell-".concat(klass);
+             }).html(function (d) {
+               return d.display;
+             });
+           }
          }
 
-         function keyChange(d3_event, d) {
-           if (select(this).attr('readonly')) return;
-           var kOld = d.key; // exit if we are currently about to delete this row anyway - #6366
+         function oldMultipolygonIssues(entity, graph) {
+           var multipolygon, outerWay;
 
-           if (_pendingChange && _pendingChange.hasOwnProperty(kOld) && _pendingChange[kOld] === undefined) return;
-           var kNew = context.cleanTagKey(this.value.trim()); // allow no change if the key should be readonly
+           if (entity.type === 'relation') {
+             outerWay = osmOldMultipolygonOuterMemberOfRelation(entity, graph);
+             multipolygon = entity;
+           } else if (entity.type === 'way') {
+             multipolygon = osmIsOldMultipolygonOuterMember(entity, graph);
+             outerWay = entity;
+           } else {
+             return [];
+           }
+
+           if (!multipolygon || !outerWay) return [];
+           return [new validationIssue({
+             type: type,
+             subtype: 'old_multipolygon',
+             severity: 'warning',
+             message: showMessage,
+             reference: showReference,
+             entityIds: [outerWay.id, multipolygon.id],
+             dynamicFixes: function dynamicFixes() {
+               return [new validationIssueFix({
+                 autoArgs: [doUpgrade, _t('issues.fix.move_tags.annotation')],
+                 title: _t.html('issues.fix.move_tags.title'),
+                 onClick: function onClick(context) {
+                   context.perform(doUpgrade, _t('issues.fix.move_tags.annotation'));
+                 }
+               })];
+             }
+           })];
 
-           if (isReadOnly({
-             key: kNew
-           })) {
-             this.value = kOld;
-             return;
+           function doUpgrade(graph) {
+             var currMultipolygon = graph.hasEntity(multipolygon.id);
+             var currOuterWay = graph.hasEntity(outerWay.id);
+             if (!currMultipolygon || !currOuterWay) return graph;
+             currMultipolygon = currMultipolygon.mergeTags(currOuterWay.tags);
+             graph = graph.replace(currMultipolygon);
+             return actionChangeTags(currOuterWay.id, {})(graph);
            }
 
-           if (kNew && kNew !== kOld && _tags[kNew] !== undefined) {
-             // new key is already in use, switch focus to the existing row
-             this.value = kOld; // reset the key
-
-             section.selection().selectAll('.tag-list input.value').each(function (d) {
-               if (d.key === kNew) {
-                 // send focus to that other value combo instead
-                 var input = select(this).node();
-                 input.focus();
-                 input.select();
-               }
+           function showMessage(context) {
+             var currMultipolygon = context.hasEntity(multipolygon.id);
+             if (!currMultipolygon) return '';
+             return _t.html('issues.old_multipolygon.message', {
+               multipolygon: utilDisplayLabel(currMultipolygon, context.graph(), true
+               /* verbose */
+               )
              });
-             return;
            }
 
-           var row = this.parentNode.parentNode;
-           var inputVal = select(row).selectAll('input.value');
-           var vNew = context.cleanTagValue(utilGetSetValue(inputVal));
-           _pendingChange = _pendingChange || {};
-
-           if (kOld) {
-             _pendingChange[kOld] = undefined;
+           function showReference(selection) {
+             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.old_multipolygon.reference'));
            }
+         }
 
-           _pendingChange[kNew] = vNew; // update the ordered key index so this row doesn't change position
+         var validation = function checkOutdatedTags(entity, graph) {
+           var issues = oldMultipolygonIssues(entity, graph);
+           if (!issues.length) issues = oldTagIssues(entity, graph);
+           return issues;
+         };
 
-           var existingKeyIndex = _orderedKeys.indexOf(kOld);
+         validation.type = type;
+         return validation;
+       }
 
-           if (existingKeyIndex !== -1) _orderedKeys[existingKeyIndex] = kNew;
-           d.key = kNew; // update datum to avoid exit/enter on tag update
+       function validationPrivateData() {
+         var type = 'private_data'; // assume that some buildings are private
 
-           d.value = vNew;
-           this.value = kNew;
-           utilGetSetValue(inputVal, vNew);
-           scheduleChange();
-         }
+         var privateBuildingValues = {
+           detached: true,
+           farm: true,
+           house: true,
+           houseboat: true,
+           residential: true,
+           semidetached_house: true,
+           static_caravan: true
+         }; // but they might be public if they have one of these other tags
 
-         function valueChange(d3_event, d) {
-           if (isReadOnly(d)) return; // exit if this is a multiselection and no value was entered
+         var publicKeys = {
+           amenity: true,
+           craft: true,
+           historic: true,
+           leisure: true,
+           office: true,
+           shop: true,
+           tourism: true
+         }; // these tags may contain personally identifying info
 
-           if (typeof d.value !== 'string' && !this.value) return; // exit if we are currently about to delete this row anyway - #6366
+         var personalTags = {
+           'contact:email': true,
+           'contact:fax': true,
+           'contact:phone': true,
+           email: true,
+           fax: true,
+           phone: true
+         };
 
-           if (_pendingChange && _pendingChange.hasOwnProperty(d.key) && _pendingChange[d.key] === undefined) return;
-           _pendingChange = _pendingChange || {};
-           _pendingChange[d.key] = context.cleanTagValue(this.value);
-           scheduleChange();
-         }
+         var validation = function checkPrivateData(entity) {
+           var tags = entity.tags;
+           if (!tags.building || !privateBuildingValues[tags.building]) return [];
+           var keepTags = {};
 
-         function removeTag(d3_event, d) {
-           if (isReadOnly(d)) return;
+           for (var k in tags) {
+             if (publicKeys[k]) return []; // probably a public feature
 
-           if (d.key === '') {
-             // removing the blank row
-             _showBlank = false;
-             section.reRender();
-           } else {
-             // remove the key from the ordered key index
-             _orderedKeys = _orderedKeys.filter(function (key) {
-               return key !== d.key;
-             });
-             _pendingChange = _pendingChange || {};
-             _pendingChange[d.key] = undefined;
-             scheduleChange();
+             if (!personalTags[k]) {
+               keepTags[k] = tags[k];
+             }
            }
-         }
-
-         function addTag() {
-           // Delay render in case this click is blurring an edited combo.
-           // Without the setTimeout, the `content` render would wipe out the pending tag change.
-           window.setTimeout(function () {
-             _showBlank = true;
-             section.reRender();
-             section.selection().selectAll('.tag-list li:last-child input.key').node().focus();
-           }, 20);
-         }
 
-         function scheduleChange() {
-           // Cache IDs in case the editor is reloaded before the change event is called. - #6028
-           var entityIDs = _entityIDs; // Delay change in case this change is blurring an edited combo. - #5878
+           var tagDiff = utilTagDiff(tags, keepTags);
+           if (!tagDiff.length) return [];
+           var fixID = tagDiff.length === 1 ? 'remove_tag' : 'remove_tags';
+           return [new validationIssue({
+             type: type,
+             severity: 'warning',
+             message: showMessage,
+             reference: showReference,
+             entityIds: [entity.id],
+             dynamicFixes: function dynamicFixes() {
+               return [new validationIssueFix({
+                 icon: 'iD-operation-delete',
+                 title: _t.html('issues.fix.' + fixID + '.title'),
+                 onClick: function onClick(context) {
+                   context.perform(doUpgrade, _t('issues.fix.upgrade_tags.annotation'));
+                 }
+               })];
+             }
+           })];
 
-           window.setTimeout(function () {
-             if (!_pendingChange) return;
-             dispatch$1.call('change', this, entityIDs, _pendingChange);
-             _pendingChange = null;
-           }, 10);
-         }
+           function doUpgrade(graph) {
+             var currEntity = graph.hasEntity(entity.id);
+             if (!currEntity) return graph;
+             var newTags = Object.assign({}, currEntity.tags); // shallow copy
 
-         section.state = function (val) {
-           if (!arguments.length) return _state;
+             tagDiff.forEach(function (diff) {
+               if (diff.type === '-') {
+                 delete newTags[diff.key];
+               } else if (diff.type === '+') {
+                 newTags[diff.key] = diff.newVal;
+               }
+             });
+             return actionChangeTags(currEntity.id, newTags)(graph);
+           }
 
-           if (_state !== val) {
-             _orderedKeys = [];
-             _state = val;
+           function showMessage(context) {
+             var currEntity = context.hasEntity(this.entityIds[0]);
+             if (!currEntity) return '';
+             return _t.html('issues.private_data.contact.message', {
+               feature: utilDisplayLabel(currEntity, context.graph())
+             });
            }
 
-           return section;
+           function showReference(selection) {
+             var enter = selection.selectAll('.issue-reference').data([0]).enter();
+             enter.append('div').attr('class', 'issue-reference').html(_t.html('issues.private_data.reference'));
+             enter.append('strong').html(_t.html('issues.suggested'));
+             enter.append('table').attr('class', 'tagDiff-table').selectAll('.tagDiff-row').data(tagDiff).enter().append('tr').attr('class', 'tagDiff-row').append('td').attr('class', function (d) {
+               var klass = d.type === '+' ? 'add' : 'remove';
+               return 'tagDiff-cell tagDiff-cell-' + klass;
+             }).html(function (d) {
+               return d.display;
+             });
+           }
          };
 
-         section.presets = function (val) {
-           if (!arguments.length) return _presets;
-           _presets = val;
+         validation.type = type;
+         return validation;
+       }
 
-           if (_presets && _presets.length && _presets[0].isFallback()) {
-             section.disclosureExpanded(true); // don't collapse the disclosure if the mapper used the raw tag editor - #1881
-           } else if (!_didInteract) {
-             section.disclosureExpanded(null);
+       function validationSuspiciousName() {
+         var type = 'suspicious_name';
+         var keysToTestForGenericValues = ['aerialway', 'aeroway', 'amenity', 'building', 'craft', 'highway', 'leisure', 'railway', 'man_made', 'office', 'shop', 'tourism', 'waterway'];
+         var _waitingForNsi = false; // Attempt to match a generic record in the name-suggestion-index.
+
+         function isGenericMatchInNsi(tags) {
+           var nsi = services.nsi;
+
+           if (nsi) {
+             _waitingForNsi = nsi.status() === 'loading';
+
+             if (!_waitingForNsi) {
+               return nsi.isGenericName(tags);
+             }
            }
 
-           return section;
-         };
+           return false;
+         } // Test if the name is just the key or tag value (e.g. "park")
 
-         section.tags = function (val) {
-           if (!arguments.length) return _tags;
-           _tags = val;
-           return section;
-         };
 
-         section.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs;
+         function nameMatchesRawTag(lowercaseName, tags) {
+           for (var i = 0; i < keysToTestForGenericValues.length; i++) {
+             var key = keysToTestForGenericValues[i];
+             var val = tags[key];
 
-           if (!_entityIDs || !val || !utilArrayIdentical(_entityIDs, val)) {
-             _entityIDs = val;
-             _orderedKeys = [];
-           }
+             if (val) {
+               val = val.toLowerCase();
 
-           return section;
-         }; // pass an array of regular expressions to test against the tag key
+               if (key === lowercaseName || val === lowercaseName || key.replace(/\_/g, ' ') === lowercaseName || val.replace(/\_/g, ' ') === lowercaseName) {
+                 return true;
+               }
+             }
+           }
 
+           return false;
+         }
 
-         section.readOnlyTags = function (val) {
-           if (!arguments.length) return _readOnlyTags;
-           _readOnlyTags = val;
-           return section;
-         };
+         function isGenericName(name, tags) {
+           name = name.toLowerCase();
+           return nameMatchesRawTag(name, tags) || isGenericMatchInNsi(tags);
+         }
 
-         return utilRebind(section, dispatch$1, 'on');
-       }
+         function makeGenericNameIssue(entityId, nameKey, genericName, langCode) {
+           return new validationIssue({
+             type: type,
+             subtype: 'generic_name',
+             severity: 'warning',
+             message: function message(context) {
+               var entity = context.hasEntity(this.entityIds[0]);
+               if (!entity) return '';
+               var preset = _mainPresetIndex.match(entity, context.graph());
+               var langName = langCode && _mainLocalizer.languageName(langCode);
+               return _t.html('issues.generic_name.message' + (langName ? '_language' : ''), {
+                 feature: preset.name(),
+                 name: genericName,
+                 language: langName
+               });
+             },
+             reference: showReference,
+             entityIds: [entityId],
+             hash: "".concat(nameKey, "=").concat(genericName),
+             dynamicFixes: function dynamicFixes() {
+               return [new validationIssueFix({
+                 icon: 'iD-operation-delete',
+                 title: _t.html('issues.fix.remove_the_name.title'),
+                 onClick: function onClick(context) {
+                   var entityId = this.issue.entityIds[0];
+                   var entity = context.entity(entityId);
+                   var tags = Object.assign({}, entity.tags); // shallow copy
 
-       function uiDataEditor(context) {
-         var dataHeader = uiDataHeader();
-         var rawTagEditor = uiSectionRawTagEditor('custom-data-tag-editor', context).expandedByDefault(true).readOnlyTags([/./]);
+                   delete tags[nameKey];
+                   context.perform(actionChangeTags(entityId, tags), _t('issues.fix.remove_generic_name.annotation'));
+                 }
+               })];
+             }
+           });
 
-         var _datum;
+           function showReference(selection) {
+             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.generic_name.reference'));
+           }
+         }
 
-         function dataEditor(selection) {
-           var header = selection.selectAll('.header').data([0]);
-           var headerEnter = header.enter().append('div').attr('class', 'header fillL');
-           headerEnter.append('button').attr('class', 'close').on('click', function () {
-             context.enter(modeBrowse(context));
-           }).call(svgIcon('#iD-icon-close'));
-           headerEnter.append('h3').html(_t.html('map_data.title'));
-           var body = selection.selectAll('.body').data([0]);
-           body = body.enter().append('div').attr('class', 'body').merge(body);
-           var editor = body.selectAll('.data-editor').data([0]); // enter/update
+         function makeIncorrectNameIssue(entityId, nameKey, incorrectName, langCode) {
+           return new validationIssue({
+             type: type,
+             subtype: 'not_name',
+             severity: 'warning',
+             message: function message(context) {
+               var entity = context.hasEntity(this.entityIds[0]);
+               if (!entity) return '';
+               var preset = _mainPresetIndex.match(entity, context.graph());
+               var langName = langCode && _mainLocalizer.languageName(langCode);
+               return _t.html('issues.incorrect_name.message' + (langName ? '_language' : ''), {
+                 feature: preset.name(),
+                 name: incorrectName,
+                 language: langName
+               });
+             },
+             reference: showReference,
+             entityIds: [entityId],
+             hash: "".concat(nameKey, "=").concat(incorrectName),
+             dynamicFixes: function dynamicFixes() {
+               return [new validationIssueFix({
+                 icon: 'iD-operation-delete',
+                 title: _t.html('issues.fix.remove_the_name.title'),
+                 onClick: function onClick(context) {
+                   var entityId = this.issue.entityIds[0];
+                   var entity = context.entity(entityId);
+                   var tags = Object.assign({}, entity.tags); // shallow copy
 
-           editor.enter().append('div').attr('class', 'modal-section data-editor').merge(editor).call(dataHeader.datum(_datum));
-           var rte = body.selectAll('.raw-tag-editor').data([0]); // enter/update
+                   delete tags[nameKey];
+                   context.perform(actionChangeTags(entityId, tags), _t('issues.fix.remove_mistaken_name.annotation'));
+                 }
+               })];
+             }
+           });
 
-           rte.enter().append('div').attr('class', 'raw-tag-editor data-editor').merge(rte).call(rawTagEditor.tags(_datum && _datum.properties || {}).state('hover').render).selectAll('textarea.tag-text').attr('readonly', true).classed('readonly', true);
+           function showReference(selection) {
+             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.generic_name.reference'));
+           }
          }
 
-         dataEditor.datum = function (val) {
-           if (!arguments.length) return _datum;
-           _datum = val;
-           return this;
-         };
+         var validation = function checkGenericName(entity) {
+           var tags = entity.tags; // a generic name is allowed if it's a known brand or entity
 
-         return dataEditor;
-       }
+           var hasWikidata = !!tags.wikidata || !!tags['brand:wikidata'] || !!tags['operator:wikidata'];
+           if (hasWikidata) return [];
+           var issues = [];
+           var notNames = (tags['not:name'] || '').split(';');
 
-       function modeSelectData(context, selectedDatum) {
-         var mode = {
-           id: 'select-data',
-           button: 'browse'
-         };
-         var keybinding = utilKeybinding('select-data');
-         var dataEditor = uiDataEditor(context);
-         var behaviors = [behaviorBreathe(), behaviorHover(context), behaviorSelect(context), behaviorLasso(context), modeDragNode(context).behavior, modeDragNote(context).behavior]; // class the data as selected, or return to browse mode if the data is gone
+           for (var key in tags) {
+             var m = key.match(/^name(?:(?::)([a-zA-Z_-]+))?$/);
+             if (!m) continue;
+             var langCode = m.length >= 2 ? m[1] : null;
+             var value = tags[key];
 
-         function selectData(d3_event, drawn) {
-           var selection = context.surface().selectAll('.layer-mapdata .data' + selectedDatum.__featurehash__);
+             if (notNames.length) {
+               for (var i in notNames) {
+                 var notName = notNames[i];
 
-           if (selection.empty()) {
-             // Return to browse mode if selected DOM elements have
-             // disappeared because the user moved them out of view..
-             var source = d3_event && d3_event.type === 'zoom' && d3_event.sourceEvent;
+                 if (notName && value === notName) {
+                   issues.push(makeIncorrectNameIssue(entity.id, key, value, langCode));
+                   continue;
+                 }
+               }
+             }
 
-             if (drawn && source && (source.type === 'pointermove' || source.type === 'mousemove' || source.type === 'touchmove')) {
-               context.enter(modeBrowse(context));
+             if (isGenericName(value, tags)) {
+               issues.provisional = _waitingForNsi; // retry later if we are waiting on NSI to finish loading
+
+               issues.push(makeGenericNameIssue(entity.id, key, value, langCode));
              }
-           } else {
-             selection.classed('selected', true);
            }
-         }
-
-         function esc() {
-           if (context.container().select('.combobox').size()) return;
-           context.enter(modeBrowse(context));
-         }
 
-         mode.zoomToSelected = function () {
-           var extent = geoExtent(d3_geoBounds(selectedDatum));
-           context.map().centerZoomEase(extent.center(), context.map().trimmedExtentZoom(extent));
+           return issues;
          };
 
-         mode.enter = function () {
-           behaviors.forEach(context.install);
-           keybinding.on(_t('inspector.zoom_to.key'), mode.zoomToSelected).on('⎋', esc, true);
-           select(document).call(keybinding);
-           selectData();
-           var sidebar = context.ui().sidebar;
-           sidebar.show(dataEditor.datum(selectedDatum)); // expand the sidebar, avoid obscuring the data if needed
+         validation.type = type;
+         return validation;
+       }
 
-           var extent = geoExtent(d3_geoBounds(selectedDatum));
-           sidebar.expand(sidebar.intersects(extent));
-           context.map().on('drawn.select-data', selectData);
-         };
+       function validationUnsquareWay(context) {
+         var type = 'unsquare_way';
+         var DEFAULT_DEG_THRESHOLD = 5; // see also issues.js
+         // use looser epsilon for detection to reduce warnings of buildings that are essentially square already
 
-         mode.exit = function () {
-           behaviors.forEach(context.uninstall);
-           select(document).call(keybinding.unbind);
-           context.surface().selectAll('.layer-mapdata .selected').classed('selected hover', false);
-           context.map().on('drawn.select-data', null);
-           context.ui().sidebar.hide();
-         };
+         var epsilon = 0.05;
+         var nodeThreshold = 10;
 
-         return mode;
-       }
+         function isBuilding(entity, graph) {
+           if (entity.type !== 'way' || entity.geometry(graph) !== 'area') return false;
+           return entity.tags.building && entity.tags.building !== 'no';
+         }
 
-       function uiImproveOsmComments() {
-         var _qaItem;
+         var validation = function checkUnsquareWay(entity, graph) {
+           if (!isBuilding(entity, graph)) return []; // don't flag ways marked as physically unsquare
 
-         function issueComments(selection) {
-           // make the div immediately so it appears above the buttons
-           var comments = selection.selectAll('.comments-container').data([0]);
-           comments = comments.enter().append('div').attr('class', 'comments-container').merge(comments); // must retrieve comments from API before they can be displayed
+           if (entity.tags.nonsquare === 'yes') return [];
+           var isClosed = entity.isClosed();
+           if (!isClosed) return []; // this building has bigger problems
+           // don't flag ways with lots of nodes since they are likely detail-mapped
 
-           services.improveOSM.getComments(_qaItem).then(function (d) {
-             if (!d.comments) return; // nothing to do here
+           var nodes = graph.childNodes(entity).slice(); // shallow copy
 
-             var commentEnter = comments.selectAll('.comment').data(d.comments).enter().append('div').attr('class', 'comment');
-             commentEnter.append('div').attr('class', 'comment-avatar').call(svgIcon('#iD-icon-avatar', 'comment-avatar-icon'));
-             var mainEnter = commentEnter.append('div').attr('class', 'comment-main');
-             var metadataEnter = mainEnter.append('div').attr('class', 'comment-metadata');
-             metadataEnter.append('div').attr('class', 'comment-author').each(function (d) {
-               var osm = services.osm;
-               var selection = select(this);
+           if (nodes.length > nodeThreshold + 1) return []; // +1 because closing node appears twice
+           // ignore if not all nodes are fully downloaded
 
-               if (osm && d.username) {
-                 selection = selection.append('a').attr('class', 'comment-author-link').attr('href', osm.userURL(d.username)).attr('target', '_blank');
-               }
+           var osm = services.osm;
+           if (!osm || nodes.some(function (node) {
+             return !osm.isDataLoaded(node.loc);
+           })) return []; // don't flag connected ways to avoid unresolvable unsquare loops
 
-               selection.html(function (d) {
-                 return d.username;
-               });
-             });
-             metadataEnter.append('div').attr('class', 'comment-date').html(function (d) {
-               return _t.html('note.status.commented', {
-                 when: localeDateString(d.timestamp)
+           var hasConnectedSquarableWays = nodes.some(function (node) {
+             return graph.parentWays(node).some(function (way) {
+               if (way.id === entity.id) return false;
+               if (isBuilding(way, graph)) return true;
+               return graph.parentRelations(way).some(function (parentRelation) {
+                 return parentRelation.isMultipolygon() && parentRelation.tags.building && parentRelation.tags.building !== 'no';
                });
              });
-             mainEnter.append('div').attr('class', 'comment-text').append('p').html(function (d) {
-               return d.text;
-             });
-           })["catch"](function (err) {
-             console.log(err); // eslint-disable-line no-console
            });
-         }
+           if (hasConnectedSquarableWays) return []; // user-configurable square threshold
+
+           var storedDegreeThreshold = corePreferences('validate-square-degrees');
+           var degreeThreshold = isNaN(storedDegreeThreshold) ? DEFAULT_DEG_THRESHOLD : parseFloat(storedDegreeThreshold);
+           var points = nodes.map(function (node) {
+             return context.projection(node.loc);
+           });
+           if (!geoOrthoCanOrthogonalize(points, isClosed, epsilon, degreeThreshold, true)) return [];
+           var autoArgs; // don't allow autosquaring features linked to wikidata
+
+           if (!entity.tags.wikidata) {
+             // use same degree threshold as for detection
+             var autoAction = actionOrthogonalize(entity.id, context.projection, undefined, degreeThreshold);
+             autoAction.transitionable = false; // when autofixing, do it instantly
+
+             autoArgs = [autoAction, _t('operations.orthogonalize.annotation.feature', {
+               n: 1
+             })];
+           }
+
+           return [new validationIssue({
+             type: type,
+             subtype: 'building',
+             severity: 'warning',
+             message: function message(context) {
+               var entity = context.hasEntity(this.entityIds[0]);
+               return entity ? _t.html('issues.unsquare_way.message', {
+                 feature: utilDisplayLabel(entity, context.graph())
+               }) : '';
+             },
+             reference: showReference,
+             entityIds: [entity.id],
+             hash: degreeThreshold,
+             dynamicFixes: function dynamicFixes() {
+               return [new validationIssueFix({
+                 icon: 'iD-operation-orthogonalize',
+                 title: _t.html('issues.fix.square_feature.title'),
+                 autoArgs: autoArgs,
+                 onClick: function onClick(context, completionHandler) {
+                   var entityId = this.issue.entityIds[0]; // use same degree threshold as for detection
 
-         function localeDateString(s) {
-           if (!s) return null;
-           var options = {
-             day: 'numeric',
-             month: 'short',
-             year: 'numeric'
-           };
-           var d = new Date(s * 1000); // timestamp is served in seconds, date takes ms
+                   context.perform(actionOrthogonalize(entityId, context.projection, undefined, degreeThreshold), _t('operations.orthogonalize.annotation.feature', {
+                     n: 1
+                   })); // run after the squaring transition (currently 150ms)
 
-           if (isNaN(d.getTime())) return null;
-           return d.toLocaleDateString(_mainLocalizer.localeCode(), options);
-         }
+                   window.setTimeout(function () {
+                     completionHandler();
+                   }, 175);
+                 }
+               })
+               /*
+               new validationIssueFix({
+                   title: t.html('issues.fix.tag_as_unsquare.title'),
+                   onClick: function(context) {
+                       var entityId = this.issue.entityIds[0];
+                       var entity = context.entity(entityId);
+                       var tags = Object.assign({}, entity.tags);  // shallow copy
+                       tags.nonsquare = 'yes';
+                       context.perform(
+                           actionChangeTags(entityId, tags),
+                           t('issues.fix.tag_as_unsquare.annotation')
+                       );
+                   }
+               })
+               */
+               ];
+             }
+           })];
 
-         issueComments.issue = function (val) {
-           if (!arguments.length) return _qaItem;
-           _qaItem = val;
-           return issueComments;
+           function showReference(selection) {
+             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.unsquare_way.buildings.reference'));
+           }
          };
 
-         return issueComments;
+         validation.type = type;
+         return validation;
        }
 
-       function uiImproveOsmDetails(context) {
-         var _qaItem;
+       var Validations = /*#__PURE__*/Object.freeze({
+               __proto__: null,
+               validationAlmostJunction: validationAlmostJunction,
+               validationCloseNodes: validationCloseNodes,
+               validationCrossingWays: validationCrossingWays,
+               validationDisconnectedWay: validationDisconnectedWay,
+               validationFormatting: validationFormatting,
+               validationHelpRequest: validationHelpRequest,
+               validationImpossibleOneway: validationImpossibleOneway,
+               validationIncompatibleSource: validationIncompatibleSource,
+               validationMaprules: validationMaprules,
+               validationMismatchedGeometry: validationMismatchedGeometry,
+               validationMissingRole: validationMissingRole,
+               validationMissingTag: validationMissingTag,
+               validationOutdatedTags: validationOutdatedTags,
+               validationPrivateData: validationPrivateData,
+               validationSuspiciousName: validationSuspiciousName,
+               validationUnsquareWay: validationUnsquareWay
+       });
 
-         function issueDetail(d) {
-           if (d.desc) return d.desc;
-           var issueKey = d.issueKey;
-           d.replacements = d.replacements || {};
-           d.replacements["default"] = _t.html('inspector.unknown'); // special key `default` works as a fallback string
+       function coreValidator(context) {
+         var _this = this;
 
-           return _t.html("QA.improveOSM.error_types.".concat(issueKey, ".description"), d.replacements);
-         }
+         var dispatch = dispatch$8('validated', 'focusedIssue');
+         var validator = utilRebind({}, dispatch, 'on');
+         var _rules = {};
+         var _disabledRules = {};
 
-         function improveOsmDetails(selection) {
-           var details = selection.selectAll('.error-details').data(_qaItem ? [_qaItem] : [], function (d) {
-             return "".concat(d.id, "-").concat(d.status || 0);
-           });
-           details.exit().remove();
-           var detailsEnter = details.enter().append('div').attr('class', 'error-details qa-details-container'); // description
+         var _ignoredIssueIDs = new Set();
 
-           var descriptionEnter = detailsEnter.append('div').attr('class', 'qa-details-subsection');
-           descriptionEnter.append('h4').html(_t.html('QA.keepRight.detail_description'));
-           descriptionEnter.append('div').attr('class', 'qa-details-description-text').html(issueDetail); // If there are entity links in the error message..
+         var _resolvedIssueIDs = new Set();
 
-           var relatedEntities = [];
-           descriptionEnter.selectAll('.error_entity_link, .error_object_link').attr('href', '#').each(function () {
-             var link = select(this);
-             var isObjectLink = link.classed('error_object_link');
-             var entityID = isObjectLink ? utilEntityRoot(_qaItem.objectType) + _qaItem.objectId : this.textContent;
-             var entity = context.hasEntity(entityID);
-             relatedEntities.push(entityID); // Add click handler
+         var _baseCache = validationCache(); // issues before any user edits
 
-             link.on('mouseenter', function () {
-               utilHighlightEntities([entityID], true, context);
-             }).on('mouseleave', function () {
-               utilHighlightEntities([entityID], false, context);
-             }).on('click', function (d3_event) {
-               d3_event.preventDefault();
-               utilHighlightEntities([entityID], false, context);
-               var osmlayer = context.layers().layer('osm');
 
-               if (!osmlayer.enabled()) {
-                 osmlayer.enabled(true);
-               }
+         var _headCache = validationCache(); // issues after all user edits
 
-               context.map().centerZoom(_qaItem.loc, 20);
 
-               if (entity) {
-                 context.enter(modeSelect(context, [entityID]));
-               } else {
-                 context.loadEntity(entityID, function () {
-                   context.enter(modeSelect(context, [entityID]));
-                 });
-               }
-             }); // Replace with friendly name if possible
-             // (The entity may not yet be loaded into the graph)
+         var _headGraph = null;
+         var _headIsCurrent = false;
 
-             if (entity) {
-               var name = utilDisplayName(entity); // try to use common name
+         var _deferredRIC = new Set(); // Set( RequestIdleCallback handles )
 
-               if (!name && !isObjectLink) {
-                 var preset = _mainPresetIndex.match(entity, context.graph());
-                 name = preset && !preset.isFallback() && preset.name(); // fallback to preset name
-               }
 
-               if (name) {
-                 this.innerText = name;
-               }
-             }
-           }); // Don't hide entities related to this error - #5880
+         var _deferredST = new Set(); // Set( SetTimeout handles )
 
-           context.features().forceVisible(relatedEntities);
-           context.map().pan([0, 0]); // trigger a redraw
-         }
 
-         improveOsmDetails.issue = function (val) {
-           if (!arguments.length) return _qaItem;
-           _qaItem = val;
-           return improveOsmDetails;
-         };
+         var _headPromise; // Promise fulfilled when validation is performed up to headGraph snapshot
 
-         return improveOsmDetails;
-       }
 
-       function uiImproveOsmHeader() {
-         var _qaItem;
+         var RETRY = 5000; // wait 5sec before revalidating provisional entities
+         // Allow validation severity to be overridden by url queryparams...
+         // Each param should contain a urlencoded comma separated list of
+         // `type/subtype` rules.  `*` may be used as a wildcard..
+         // Examples:
+         //  `validationError=disconnected_way/*`
+         //  `validationError=disconnected_way/highway`
+         //  `validationError=crossing_ways/bridge*`
+         //  `validationError=crossing_ways/bridge*,crossing_ways/tunnel*`
 
-         function issueTitle(d) {
-           var issueKey = d.issueKey;
-           d.replacements = d.replacements || {};
-           d.replacements["default"] = _t.html('inspector.unknown'); // special key `default` works as a fallback string
+         var _errorOverrides = parseHashParam(context.initialHashParams.validationError);
 
-           return _t.html("QA.improveOSM.error_types.".concat(issueKey, ".title"), d.replacements);
-         }
+         var _warningOverrides = parseHashParam(context.initialHashParams.validationWarning);
 
-         function improveOsmHeader(selection) {
-           var header = selection.selectAll('.qa-header').data(_qaItem ? [_qaItem] : [], function (d) {
-             return "".concat(d.id, "-").concat(d.status || 0);
-           });
-           header.exit().remove();
-           var headerEnter = header.enter().append('div').attr('class', 'qa-header');
-           var svgEnter = headerEnter.append('div').attr('class', 'qa-header-icon').classed('new', function (d) {
-             return d.id < 0;
-           }).append('svg').attr('width', '20px').attr('height', '30px').attr('viewbox', '0 0 20 30').attr('class', function (d) {
-             return "preset-icon-28 qaItem ".concat(d.service, " itemId-").concat(d.id, " itemType-").concat(d.itemType);
-           });
-           svgEnter.append('polygon').attr('fill', 'currentColor').attr('class', 'qaItem-fill').attr('points', '16,3 4,3 1,6 1,17 4,20 7,20 10,27 13,20 16,20 19,17.033 19,6');
-           svgEnter.append('use').attr('class', 'icon-annotation').attr('width', '13px').attr('height', '13px').attr('transform', 'translate(3.5, 5)').attr('xlink:href', function (d) {
-             var picon = d.icon;
+         var _disableOverrides = parseHashParam(context.initialHashParams.validationDisable);
 
-             if (!picon) {
-               return '';
-             } else {
-               var isMaki = /^maki-/.test(picon);
-               return "#".concat(picon).concat(isMaki ? '-11' : '');
-             }
+         function parseHashParam(param) {
+           var result = [];
+           var rules = (param || '').split(',');
+           rules.forEach(function (rule) {
+             rule = rule.trim();
+             var parts = rule.split('/', 2); // "type/subtype"
+
+             var type = parts[0];
+             var subtype = parts[1] || '*';
+             if (!type || !subtype) return;
+             result.push({
+               type: makeRegExp(type),
+               subtype: makeRegExp(subtype)
+             });
            });
-           headerEnter.append('div').attr('class', 'qa-header-label').html(issueTitle);
+           return result;
          }
 
-         improveOsmHeader.issue = function (val) {
-           if (!arguments.length) return _qaItem;
-           _qaItem = val;
-           return improveOsmHeader;
-         };
+         function makeRegExp(str) {
+           var escaped = str.replace(/[-\/\\^$+?.()|[\]{}]/g, '\\$&') // escape all reserved chars except for the '*'
+           .replace(/\*/g, '.*'); // treat a '*' like '.*'
 
-         return improveOsmHeader;
-       }
+           return new RegExp('^' + escaped + '$');
+         } // `init()`
+         // Initialize the validator, called once on iD startup
+         //
 
-       function uiImproveOsmEditor(context) {
-         var dispatch$1 = dispatch('change');
-         var qaDetails = uiImproveOsmDetails(context);
-         var qaComments = uiImproveOsmComments();
-         var qaHeader = uiImproveOsmHeader();
 
-         var _qaItem;
+         validator.init = function () {
+           Object.values(Validations).forEach(function (validation) {
+             if (typeof validation !== 'function') return;
+             var fn = validation(context);
+             var key = fn.type;
+             _rules[key] = fn;
+           });
+           var disabledRules = corePreferences('validate-disabledRules');
 
-         function improveOsmEditor(selection) {
-           var headerEnter = selection.selectAll('.header').data([0]).enter().append('div').attr('class', 'header fillL');
-           headerEnter.append('button').attr('class', 'close').on('click', function () {
-             return context.enter(modeBrowse(context));
-           }).call(svgIcon('#iD-icon-close'));
-           headerEnter.append('h3').html(_t.html('QA.improveOSM.title'));
-           var body = selection.selectAll('.body').data([0]);
-           body = body.enter().append('div').attr('class', 'body').merge(body);
-           var editor = body.selectAll('.qa-editor').data([0]);
-           editor.enter().append('div').attr('class', 'modal-section qa-editor').merge(editor).call(qaHeader.issue(_qaItem)).call(qaDetails.issue(_qaItem)).call(qaComments.issue(_qaItem)).call(improveOsmSaveSection);
-         }
+           if (disabledRules) {
+             disabledRules.split(',').forEach(function (k) {
+               return _disabledRules[k] = true;
+             });
+           }
+         }; // `reset()`   (private)
+         // Cancels deferred work and resets all caches
+         //
+         // Arguments
+         //   `resetIgnored` - `true` to clear the list of user-ignored issues
+         //
 
-         function improveOsmSaveSection(selection) {
-           var isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
 
-           var isShown = _qaItem && (isSelected || _qaItem.newComment || _qaItem.comment);
-           var saveSection = selection.selectAll('.qa-save').data(isShown ? [_qaItem] : [], function (d) {
-             return "".concat(d.id, "-").concat(d.status || 0);
-           }); // exit
+         function reset(resetIgnored) {
+           // cancel deferred work
+           _deferredRIC.forEach(window.cancelIdleCallback);
 
-           saveSection.exit().remove(); // enter
+           _deferredRIC.clear();
 
-           var saveSectionEnter = saveSection.enter().append('div').attr('class', 'qa-save save-section cf');
-           saveSectionEnter.append('h4').attr('class', '.qa-save-header').html(_t.html('note.newComment'));
-           saveSectionEnter.append('textarea').attr('class', 'new-comment-input').attr('placeholder', _t('QA.keepRight.comment_placeholder')).attr('maxlength', 1000).property('value', function (d) {
-             return d.newComment;
-           }).call(utilNoAuto).on('input', changeInput).on('blur', changeInput); // update
+           _deferredST.forEach(window.clearTimeout);
 
-           saveSection = saveSectionEnter.merge(saveSection).call(qaSaveButtons);
+           _deferredST.clear(); // empty queues and resolve any pending promise
 
-           function changeInput() {
-             var input = select(this);
-             var val = input.property('value').trim();
 
-             if (val === '') {
-               val = undefined;
-             } // store the unsaved comment with the issue itself
+           _baseCache.queue = [];
+           _headCache.queue = [];
+           processQueue(_headCache);
+           processQueue(_baseCache); // clear caches
 
+           if (resetIgnored) _ignoredIssueIDs.clear();
 
-             _qaItem = _qaItem.update({
-               newComment: val
-             });
-             var qaService = services.improveOSM;
+           _resolvedIssueIDs.clear();
 
-             if (qaService) {
-               qaService.replaceItem(_qaItem);
-             }
+           _baseCache = validationCache();
+           _headCache = validationCache();
+           _headGraph = null;
+           _headIsCurrent = false;
+         } // `reset()`
+         // clear caches, called whenever iD resets after a save or switches sources
+         // (clears out the _ignoredIssueIDs set also)
+         //
 
-             saveSection.call(qaSaveButtons);
-           }
-         }
 
-         function qaSaveButtons(selection) {
-           var isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
+         validator.reset = function () {
+           reset(true);
+         }; // `resetIgnoredIssues()`
+         // clears out the _ignoredIssueIDs Set
+         //
 
-           var buttonSection = selection.selectAll('.buttons').data(isSelected ? [_qaItem] : [], function (d) {
-             return d.status + d.id;
-           }); // exit
 
-           buttonSection.exit().remove(); // enter
+         validator.resetIgnoredIssues = function () {
+           _ignoredIssueIDs.clear();
 
-           var buttonEnter = buttonSection.enter().append('div').attr('class', 'buttons');
-           buttonEnter.append('button').attr('class', 'button comment-button action').html(_t.html('QA.keepRight.save_comment'));
-           buttonEnter.append('button').attr('class', 'button close-button action');
-           buttonEnter.append('button').attr('class', 'button ignore-button action'); // update
+           dispatch.call('validated'); // redraw UI
+         }; // `revalidateUnsquare()`
+         // Called whenever the user changes the unsquare threshold
+         // It reruns just the "unsquare_way" validation on all buildings.
+         //
 
-           buttonSection = buttonSection.merge(buttonEnter);
-           buttonSection.select('.comment-button').attr('disabled', function (d) {
-             return d.newComment ? null : true;
-           }).on('click.comment', function (d3_event, d) {
-             this.blur(); // avoid keeping focus on the button - #4641
 
-             var qaService = services.improveOSM;
+         validator.revalidateUnsquare = function () {
+           revalidateUnsquare(_headCache, _headGraph);
+           revalidateUnsquare(_baseCache, context.history().base());
+           dispatch.call('validated');
+         };
 
-             if (qaService) {
-               qaService.postUpdate(d, function (err, item) {
-                 return dispatch$1.call('change', item);
-               });
-             }
-           });
-           buttonSection.select('.close-button').html(function (d) {
-             var andComment = d.newComment ? '_comment' : '';
-             return _t.html("QA.keepRight.close".concat(andComment));
-           }).on('click.close', function (d3_event, d) {
-             this.blur(); // avoid keeping focus on the button - #4641
+         function revalidateUnsquare(cache, graph) {
+           var checkUnsquareWay = _rules.unsquare_way;
+           if (!graph || typeof checkUnsquareWay !== 'function') return; // uncache existing
 
-             var qaService = services.improveOSM;
+           cache.uncacheIssuesOfType('unsquare_way');
+           var buildings = context.history().tree().intersects(geoExtent([-180, -90], [180, 90]), graph) // everywhere
+           .filter(function (entity) {
+             return entity.type === 'way' && entity.tags.building && entity.tags.building !== 'no';
+           }); // rerun for all buildings
 
-             if (qaService) {
-               d.newStatus = 'SOLVED';
-               qaService.postUpdate(d, function (err, item) {
-                 return dispatch$1.call('change', item);
-               });
-             }
+           buildings.forEach(function (entity) {
+             var detected = checkUnsquareWay(entity, graph);
+             if (!detected.length) return;
+             cache.cacheIssues(detected);
            });
-           buttonSection.select('.ignore-button').html(function (d) {
-             var andComment = d.newComment ? '_comment' : '';
-             return _t.html("QA.keepRight.ignore".concat(andComment));
-           }).on('click.ignore', function (d3_event, d) {
-             this.blur(); // avoid keeping focus on the button - #4641
+         } // `getIssues()`
+         // Gets all issues that match the given options
+         // This is called by many other places
+         //
+         // Arguments
+         //   `options` Object like:
+         //   {
+         //     what: 'all',                  // 'all' or 'edited'
+         //     where: 'all',                 // 'all' or 'visible'
+         //     includeIgnored: false,        // true, false, or 'only'
+         //     includeDisabledRules: false   // true, false, or 'only'
+         //   }
+         //
+         // Returns
+         //   An Array containing the issues
+         //
 
-             var qaService = services.improveOSM;
 
-             if (qaService) {
-               d.newStatus = 'INVALID';
-               qaService.postUpdate(d, function (err, item) {
-                 return dispatch$1.call('change', item);
-               });
-             }
-           });
-         } // NOTE: Don't change method name until UI v3 is merged
+         validator.getIssues = function (options) {
+           var opts = Object.assign({
+             what: 'all',
+             where: 'all',
+             includeIgnored: false,
+             includeDisabledRules: false
+           }, options);
+           var view = context.map().extent();
+           var issues = [];
+           var seen = new Set(); // collect head issues - caused by user edits
 
+           var cache = _headCache;
 
-         improveOsmEditor.error = function (val) {
-           if (!arguments.length) return _qaItem;
-           _qaItem = val;
-           return improveOsmEditor;
-         };
+           if (_headGraph) {
+             Object.values(cache.issuesByIssueID).forEach(function (issue) {
+               if (!filter(issue, _headGraph, cache)) return;
+               seen.add(issue.id);
+               issues.push(issue);
+             });
+           } // collect base issues - not caused by user edits
 
-         return utilRebind(improveOsmEditor, dispatch$1, 'on');
-       }
 
-       function uiKeepRightDetails(context) {
-         var _qaItem;
+           if (opts.what === 'all') {
+             cache = _baseCache;
+             Object.values(cache.issuesByIssueID).forEach(function (issue) {
+               if (!filter(issue, context.history().base(), cache)) return;
+               seen.add(issue.id);
+               issues.push(issue);
+             });
+           }
 
-         function issueDetail(d) {
-           var itemType = d.itemType,
-               parentIssueType = d.parentIssueType;
-           var unknown = _t.html('inspector.unknown');
-           var replacements = d.replacements || {};
-           replacements["default"] = unknown; // special key `default` works as a fallback string
+           return issues;
 
-           var detail = _t.html("QA.keepRight.errorTypes.".concat(itemType, ".description"), replacements);
+           function filter(issue, resolver, cache) {
+             if (!issue) return false;
+             if (seen.has(issue.id)) return false;
+             if (_resolvedIssueIDs.has(issue.id)) return false;
+             if (opts.includeDisabledRules === 'only' && !_disabledRules[issue.type]) return false;
+             if (!opts.includeDisabledRules && _disabledRules[issue.type]) return false;
+             if (opts.includeIgnored === 'only' && !_ignoredIssueIDs.has(issue.id)) return false;
+             if (!opts.includeIgnored && _ignoredIssueIDs.has(issue.id)) return false; // Sanity check:  This issue may be for an entity that not longer exists.
+             // If we detect this, uncache and return false so it is not included..
 
-           if (detail === unknown) {
-             detail = _t.html("QA.keepRight.errorTypes.".concat(parentIssueType, ".description"), replacements);
+             var entityIDs = issue.entityIds || [];
+
+             for (var i = 0; i < entityIDs.length; i++) {
+               var entityID = entityIDs[i];
+
+               if (!resolver.hasEntity(entityID)) {
+                 cache.uncacheEntityID(entityID);
+                 return false;
+               }
+             }
+
+             if (opts.where === 'visible') {
+               var extent = issue.extent(resolver);
+               if (!view.intersects(extent)) return false;
+             }
+
+             return true;
            }
+         }; // `getResolvedIssues()`
+         // Gets the issues that have been fixed by the user.
+         // Resolved issues are tracked in the `_resolvedIssueIDs` Set
+         //
+         // Returns
+         //   An Array containing the issues
+         //
 
-           return detail;
-         }
 
-         function keepRightDetails(selection) {
-           var details = selection.selectAll('.error-details').data(_qaItem ? [_qaItem] : [], function (d) {
-             return "".concat(d.id, "-").concat(d.status || 0);
+         validator.getResolvedIssues = function () {
+           var collected = new Set();
+           Object.values(_baseCache.issuesByIssueID).forEach(function (issue) {
+             if (_resolvedIssueIDs.has(issue.id)) collected.add(issue);
            });
-           details.exit().remove();
-           var detailsEnter = details.enter().append('div').attr('class', 'error-details qa-details-container'); // description
+           Object.values(_headCache.issuesByIssueID).forEach(function (issue) {
+             if (_resolvedIssueIDs.has(issue.id)) collected.add(issue);
+           });
+           return Array.from(collected);
+         }; // `focusIssue()`
+         // Adjusts the map to focus on the given issue.
+         // (requires the issue to have a reasonable extent defined)
+         //
+         // Arguments
+         //   `issue` - the issue to focus on
+         //
 
-           var descriptionEnter = detailsEnter.append('div').attr('class', 'qa-details-subsection');
-           descriptionEnter.append('h4').html(_t.html('QA.keepRight.detail_description'));
-           descriptionEnter.append('div').attr('class', 'qa-details-description-text').html(issueDetail); // If there are entity links in the error message..
 
-           var relatedEntities = [];
-           descriptionEnter.selectAll('.error_entity_link, .error_object_link').attr('href', '#').each(function () {
-             var link = select(this);
-             var isObjectLink = link.classed('error_object_link');
-             var entityID = isObjectLink ? utilEntityRoot(_qaItem.objectType) + _qaItem.objectId : this.textContent;
-             var entity = context.hasEntity(entityID);
-             relatedEntities.push(entityID); // Add click handler
+         validator.focusIssue = function (issue) {
+           var extent = issue.extent(context.graph());
+           if (!extent) return;
+           var setZoom = Math.max(context.map().zoom(), 19);
+           context.map().unobscuredCenterZoomEase(extent.center(), setZoom); // select the first entity
 
-             link.on('mouseenter', function () {
-               utilHighlightEntities([entityID], true, context);
-             }).on('mouseleave', function () {
-               utilHighlightEntities([entityID], false, context);
-             }).on('click', function (d3_event) {
-               d3_event.preventDefault();
-               utilHighlightEntities([entityID], false, context);
-               var osmlayer = context.layers().layer('osm');
+           if (issue.entityIds && issue.entityIds.length) {
+             window.setTimeout(function () {
+               var ids = issue.entityIds;
+               context.enter(modeSelect(context, [ids[0]]));
+               dispatch.call('focusedIssue', _this, issue);
+             }, 250); // after ease
+           }
+         }; // `getIssuesBySeverity()`
+         // Gets the issues then groups them by error/warning
+         // (This just calls getIssues, then puts issues in groups)
+         //
+         // Arguments
+         //   `options` - (see `getIssues`)
+         // Returns
+         //   Object result like:
+         //   {
+         //     error:    Array of errors,
+         //     warning:  Array of warnings
+         //   }
+         //
 
-               if (!osmlayer.enabled()) {
-                 osmlayer.enabled(true);
-               }
 
-               context.map().centerZoomEase(_qaItem.loc, 20);
+         validator.getIssuesBySeverity = function (options) {
+           var groups = utilArrayGroupBy(validator.getIssues(options), 'severity');
+           groups.error = groups.error || [];
+           groups.warning = groups.warning || [];
+           return groups;
+         }; // `getEntityIssues()`
+         // Gets the issues that the given entity IDs have in common, matching the given options
+         // (This just calls getIssues, then filters for the given entity IDs)
+         // The issues are sorted for relevance
+         //
+         // Arguments
+         //   `entityIDs` - Array or Set of entityIDs to get issues for
+         //   `options` - (see `getIssues`)
+         // Returns
+         //   An Array containing the issues
+         //
 
-               if (entity) {
-                 context.enter(modeSelect(context, [entityID]));
-               } else {
-                 context.loadEntity(entityID, function () {
-                   context.enter(modeSelect(context, [entityID]));
-                 });
-               }
-             }); // Replace with friendly name if possible
-             // (The entity may not yet be loaded into the graph)
 
-             if (entity) {
-               var name = utilDisplayName(entity); // try to use common name
+         validator.getSharedEntityIssues = function (entityIDs, options) {
+           // show some issue types in a particular order
+           var orderedIssueTypes = [// flag missing data first
+           'missing_tag', 'missing_role', // then flag identity issues
+           'outdated_tags', 'mismatched_geometry', // flag geometry issues where fixing them might solve connectivity issues
+           'crossing_ways', 'almost_junction', // then flag connectivity issues
+           'disconnected_way', 'impossible_oneway'];
+           var allIssues = validator.getIssues(options);
+           var forEntityIDs = new Set(entityIDs);
+           return allIssues.filter(function (issue) {
+             return (issue.entityIds || []).some(function (entityID) {
+               return forEntityIDs.has(entityID);
+             });
+           }).sort(function (issue1, issue2) {
+             if (issue1.type === issue2.type) {
+               // issues of the same type, sort deterministically
+               return issue1.id < issue2.id ? -1 : 1;
+             }
 
-               if (!name && !isObjectLink) {
-                 var preset = _mainPresetIndex.match(entity, context.graph());
-                 name = preset && !preset.isFallback() && preset.name(); // fallback to preset name
-               }
+             var index1 = orderedIssueTypes.indexOf(issue1.type);
+             var index2 = orderedIssueTypes.indexOf(issue2.type);
 
-               if (name) {
-                 this.innerText = name;
-               }
+             if (index1 !== -1 && index2 !== -1) {
+               // both issue types have explicit sort orders
+               return index1 - index2;
+             } else if (index1 === -1 && index2 === -1) {
+               // neither issue type has an explicit sort order, sort by type
+               return issue1.type < issue2.type ? -1 : 1;
+             } else {
+               // order explicit types before everything else
+               return index1 !== -1 ? -1 : 1;
              }
-           }); // Don't hide entities related to this issue - #5880
+           });
+         }; // `getEntityIssues()`
+         // Get an array of detected issues for the given entityID.
+         // (This just calls getSharedEntityIssues for a single entity)
+         //
+         // Arguments
+         //   `entityID` - the entity ID to get the issues for
+         //   `options` - (see `getIssues`)
+         // Returns
+         //   An Array containing the issues
+         //
 
-           context.features().forceVisible(relatedEntities);
-           context.map().pan([0, 0]); // trigger a redraw
-         }
 
-         keepRightDetails.issue = function (val) {
-           if (!arguments.length) return _qaItem;
-           _qaItem = val;
-           return keepRightDetails;
-         };
+         validator.getEntityIssues = function (entityID, options) {
+           return validator.getSharedEntityIssues([entityID], options);
+         }; // `getRuleKeys()`
+         //
+         // Returns
+         //   An Array containing the rule keys
+         //
 
-         return keepRightDetails;
-       }
 
-       function uiKeepRightHeader() {
-         var _qaItem;
+         validator.getRuleKeys = function () {
+           return Object.keys(_rules);
+         }; // `isRuleEnabled()`
+         //
+         // Arguments
+         //   `key` - the rule to check (e.g. 'crossing_ways')
+         // Returns
+         //   `true`/`false`
+         //
 
-         function issueTitle(d) {
-           var itemType = d.itemType,
-               parentIssueType = d.parentIssueType;
-           var unknown = _t.html('inspector.unknown');
-           var replacements = d.replacements || {};
-           replacements["default"] = unknown; // special key `default` works as a fallback string
 
-           var title = _t.html("QA.keepRight.errorTypes.".concat(itemType, ".title"), replacements);
+         validator.isRuleEnabled = function (key) {
+           return !_disabledRules[key];
+         }; // `toggleRule()`
+         // Toggles a single validation rule,
+         // then reruns the validation so that the user sees something happen in the UI
+         //
+         // Arguments
+         //   `key` - the rule to toggle (e.g. 'crossing_ways')
+         //
 
-           if (title === unknown) {
-             title = _t.html("QA.keepRight.errorTypes.".concat(parentIssueType, ".title"), replacements);
+
+         validator.toggleRule = function (key) {
+           if (_disabledRules[key]) {
+             delete _disabledRules[key];
+           } else {
+             _disabledRules[key] = true;
            }
 
-           return title;
-         }
+           corePreferences('validate-disabledRules', Object.keys(_disabledRules).join(','));
+           validator.validate();
+         }; // `disableRules()`
+         // Disables given validation rules,
+         // then reruns the validation so that the user sees something happen in the UI
+         //
+         // Arguments
+         //   `keys` - Array or Set containing rule keys to disable
+         //
 
-         function keepRightHeader(selection) {
-           var header = selection.selectAll('.qa-header').data(_qaItem ? [_qaItem] : [], function (d) {
-             return "".concat(d.id, "-").concat(d.status || 0);
-           });
-           header.exit().remove();
-           var headerEnter = header.enter().append('div').attr('class', 'qa-header');
-           var iconEnter = headerEnter.append('div').attr('class', 'qa-header-icon').classed('new', function (d) {
-             return d.id < 0;
+
+         validator.disableRules = function (keys) {
+           _disabledRules = {};
+           keys.forEach(function (k) {
+             return _disabledRules[k] = true;
            });
-           iconEnter.append('div').attr('class', function (d) {
-             return "preset-icon-28 qaItem ".concat(d.service, " itemId-").concat(d.id, " itemType-").concat(d.parentIssueType);
-           }).call(svgIcon('#iD-icon-bolt', 'qaItem-fill'));
-           headerEnter.append('div').attr('class', 'qa-header-label').html(issueTitle);
-         }
+           corePreferences('validate-disabledRules', Object.keys(_disabledRules).join(','));
+           validator.validate();
+         }; // `ignoreIssue()`
+         // Don't show the given issue in lists
+         //
+         // Arguments
+         //   `issueID` - the issueID
+         //
 
-         keepRightHeader.issue = function (val) {
-           if (!arguments.length) return _qaItem;
-           _qaItem = val;
-           return keepRightHeader;
-         };
 
-         return keepRightHeader;
-       }
+         validator.ignoreIssue = function (issueID) {
+           _ignoredIssueIDs.add(issueID);
+         }; // `validate()`
+         // Validates anything that has changed in the head graph since the last time it was run.
+         // (head graph contains user's edits)
+         //
+         // Returns
+         //   A Promise fulfilled when the validation has completed and then dispatches a `validated` event.
+         //   This may take time but happen in the background during browser idle time.
+         //
+
 
-       function uiViewOnKeepRight() {
-         var _qaItem;
+         validator.validate = function () {
+           var currGraph = context.graph();
 
-         function viewOnKeepRight(selection) {
-           var url;
+           var prevGraph = _headGraph || context.history().base();
 
-           if (services.keepRight && _qaItem instanceof QAItem) {
-             url = services.keepRight.issueURL(_qaItem);
+           if (currGraph === prevGraph) {
+             // _headGraph is current - we are caught up
+             _headIsCurrent = true;
+             dispatch.call('validated');
+             return Promise.resolve();
            }
 
-           var link = selection.selectAll('.view-on-keepRight').data(url ? [url] : []); // exit
+           if (_headPromise) {
+             // Validation already in process, but we aren't caught up to current
+             _headIsCurrent = false; // We will need to catch up after the validation promise fulfills
 
-           link.exit().remove(); // enter
+             return _headPromise;
+           }
 
-           var linkEnter = link.enter().append('a').attr('class', 'view-on-keepRight').attr('target', '_blank').attr('rel', 'noopener') // security measure
-           .attr('href', function (d) {
-             return d;
-           }).call(svgIcon('#iD-icon-out-link', 'inline'));
-           linkEnter.append('span').html(_t.html('inspector.view_on_keepRight'));
-         }
+           _headGraph = currGraph; // take snapshot
 
-         viewOnKeepRight.what = function (val) {
-           if (!arguments.length) return _qaItem;
-           _qaItem = val;
-           return viewOnKeepRight;
-         };
+           var difference = coreDifference(prevGraph, _headGraph); // Gather all entities related to this difference..
+           // For created/modified, use the head graph
 
-         return viewOnKeepRight;
-       }
+           var entityIDs = difference.extantIDs(true); // created/modified (true = w/relation members)
 
-       function uiKeepRightEditor(context) {
-         var dispatch$1 = dispatch('change');
-         var qaDetails = uiKeepRightDetails(context);
-         var qaHeader = uiKeepRightHeader();
+           entityIDs = entityIDsToValidate(entityIDs, _headGraph); // For modified/deleted, use the previous graph
+           // (e.g. deleting the only highway connected to a road should create a disconnected highway issue)
 
-         var _qaItem;
+           var previousEntityIDs = difference.deleted().concat(difference.modified()).map(function (entity) {
+             return entity.id;
+           });
+           previousEntityIDs = entityIDsToValidate(previousEntityIDs, prevGraph);
+           previousEntityIDs.forEach(entityIDs.add, entityIDs); // concat the sets
 
-         function keepRightEditor(selection) {
-           var headerEnter = selection.selectAll('.header').data([0]).enter().append('div').attr('class', 'header fillL');
-           headerEnter.append('button').attr('class', 'close').on('click', function () {
-             return context.enter(modeBrowse(context));
-           }).call(svgIcon('#iD-icon-close'));
-           headerEnter.append('h3').html(_t.html('QA.keepRight.title'));
-           var body = selection.selectAll('.body').data([0]);
-           body = body.enter().append('div').attr('class', 'body').merge(body);
-           var editor = body.selectAll('.qa-editor').data([0]);
-           editor.enter().append('div').attr('class', 'modal-section qa-editor').merge(editor).call(qaHeader.issue(_qaItem)).call(qaDetails.issue(_qaItem)).call(keepRightSaveSection);
-           var footer = selection.selectAll('.footer').data([0]);
-           footer.enter().append('div').attr('class', 'footer').merge(footer).call(uiViewOnKeepRight().what(_qaItem));
-         }
+           if (!entityIDs.size) {
+             dispatch.call('validated');
+             return Promise.resolve();
+           }
 
-         function keepRightSaveSection(selection) {
-           var isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
+           _headPromise = validateEntitiesAsync(entityIDs, _headGraph, _headCache).then(function () {
+             return updateResolvedIssues(entityIDs);
+           }).then(function () {
+             return dispatch.call('validated');
+           })["catch"](function () {
+             /* ignore */
+           }).then(function () {
+             _headPromise = null;
 
-           var isShown = _qaItem && (isSelected || _qaItem.newComment || _qaItem.comment);
-           var saveSection = selection.selectAll('.qa-save').data(isShown ? [_qaItem] : [], function (d) {
-             return "".concat(d.id, "-").concat(d.status || 0);
-           }); // exit
+             if (!_headIsCurrent) {
+               validator.validate(); // run it again to catch up to current graph
+             }
+           });
+           return _headPromise;
+         }; // register event handlers:
+         // WHEN TO RUN VALIDATION:
+         // When history changes:
 
-           saveSection.exit().remove(); // enter
 
-           var saveSectionEnter = saveSection.enter().append('div').attr('class', 'qa-save save-section cf');
-           saveSectionEnter.append('h4').attr('class', '.qa-save-header').html(_t.html('QA.keepRight.comment'));
-           saveSectionEnter.append('textarea').attr('class', 'new-comment-input').attr('placeholder', _t('QA.keepRight.comment_placeholder')).attr('maxlength', 1000).property('value', function (d) {
-             return d.newComment || d.comment;
-           }).call(utilNoAuto).on('input', changeInput).on('blur', changeInput); // update
+         context.history().on('restore.validator', validator.validate) // on restore saved history
+         .on('undone.validator', validator.validate) // on undo
+         .on('redone.validator', validator.validate) // on redo
+         .on('reset.validator', function () {
+           // on history reset - happens after save, or enter/exit walkthrough
+           reset(false); // cached issues aren't valid any longer if the history has been reset
 
-           saveSection = saveSectionEnter.merge(saveSection).call(qaSaveButtons);
+           validator.validate();
+         }); // but not on 'change' (e.g. while drawing)
+         // When user changes editing modes (to catch recent changes e.g. drawing)
 
-           function changeInput() {
-             var input = select(this);
-             var val = input.property('value').trim();
+         context.on('exit.validator', validator.validate); // When merging fetched data, validate base graph:
 
-             if (val === _qaItem.comment) {
-               val = undefined;
-             } // store the unsaved comment with the issue itself
+         context.history().on('merge.validator', function (entities) {
+           if (!entities) return;
+           var baseGraph = context.history().base();
+           var entityIDs = entities.map(function (entity) {
+             return entity.id;
+           });
+           entityIDs = entityIDsToValidate(entityIDs, baseGraph);
+           validateEntitiesAsync(entityIDs, baseGraph, _baseCache);
+         }); // `validateEntity()`   (private)
+         // Runs all validation rules on a single entity.
+         // Some things to note:
+         //  - Graph is passed in from whenever the validation was started.  Validators shouldn't use
+         //   `context.graph()` because this all happens async, and the graph might have changed
+         //   (for example, nodes getting deleted before the validation can run)
+         //  - Validator functions may still be waiting on something and return a "provisional" result.
+         //    In this situation, we will schedule to revalidate the entity sometime later.
+         //
+         // Arguments
+         //   `entity` - The entity
+         //   `graph` - graph containing the entity
+         //
+         // Returns
+         //   Object result like:
+         //   {
+         //     issues:       Array of detected issues
+         //     provisional:  `true` if provisional result, `false` if final result
+         //   }
+         //
 
+         function validateEntity(entity, graph) {
+           var result = {
+             issues: [],
+             provisional: false
+           }; // runs validation and appends resulting issues
 
-             _qaItem = _qaItem.update({
-               newComment: val
-             });
-             var qaService = services.keepRight;
+           function runValidation(key) {
+             var fn = _rules[key];
 
-             if (qaService) {
-               qaService.replaceItem(_qaItem); // update keepright cache
+             if (typeof fn !== 'function') {
+               console.error('no such validation rule = ' + key); // eslint-disable-line no-console
+
+               return;
              }
 
-             saveSection.call(qaSaveButtons);
-           }
-         }
+             var detected = fn(entity, graph).filter(applySeverityOverrides);
 
-         function qaSaveButtons(selection) {
-           var isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
+             if (detected.provisional) {
+               // this validation should be run again later
+               result.provisional = true;
+             }
 
-           var buttonSection = selection.selectAll('.buttons').data(isSelected ? [_qaItem] : [], function (d) {
-             return d.status + d.id;
-           }); // exit
+             result.issues = result.issues.concat(detected);
+           } // run all rules
 
-           buttonSection.exit().remove(); // enter
 
-           var buttonEnter = buttonSection.enter().append('div').attr('class', 'buttons');
-           buttonEnter.append('button').attr('class', 'button comment-button action').html(_t.html('QA.keepRight.save_comment'));
-           buttonEnter.append('button').attr('class', 'button close-button action');
-           buttonEnter.append('button').attr('class', 'button ignore-button action'); // update
+           Object.keys(_rules).forEach(runValidation);
+           return result;
+         } // If there are any override rules that match the issue type/subtype,
+         // adjust severity (or disable it) and keep/discard as quickly as possible.
 
-           buttonSection = buttonSection.merge(buttonEnter);
-           buttonSection.select('.comment-button') // select and propagate data
-           .attr('disabled', function (d) {
-             return d.newComment ? null : true;
-           }).on('click.comment', function (d3_event, d) {
-             this.blur(); // avoid keeping focus on the button - #4641
 
-             var qaService = services.keepRight;
+         function applySeverityOverrides(issue) {
+           var type = issue.type;
+           var subtype = issue.subtype || '';
+           var i;
 
-             if (qaService) {
-               qaService.postUpdate(d, function (err, item) {
-                 return dispatch$1.call('change', item);
-               });
+           for (i = 0; i < _errorOverrides.length; i++) {
+             if (_errorOverrides[i].type.test(type) && _errorOverrides[i].subtype.test(subtype)) {
+               issue.severity = 'error';
+               return true;
              }
-           });
-           buttonSection.select('.close-button') // select and propagate data
-           .html(function (d) {
-             var andComment = d.newComment ? '_comment' : '';
-             return _t.html("QA.keepRight.close".concat(andComment));
-           }).on('click.close', function (d3_event, d) {
-             this.blur(); // avoid keeping focus on the button - #4641
-
-             var qaService = services.keepRight;
-
-             if (qaService) {
-               d.newStatus = 'ignore_t'; // ignore temporarily (item fixed)
+           }
 
-               qaService.postUpdate(d, function (err, item) {
-                 return dispatch$1.call('change', item);
-               });
+           for (i = 0; i < _warningOverrides.length; i++) {
+             if (_warningOverrides[i].type.test(type) && _warningOverrides[i].subtype.test(subtype)) {
+               issue.severity = 'warning';
+               return true;
              }
-           });
-           buttonSection.select('.ignore-button') // select and propagate data
-           .html(function (d) {
-             var andComment = d.newComment ? '_comment' : '';
-             return _t.html("QA.keepRight.ignore".concat(andComment));
-           }).on('click.ignore', function (d3_event, d) {
-             this.blur(); // avoid keeping focus on the button - #4641
+           }
 
-             var qaService = services.keepRight;
+           for (i = 0; i < _disableOverrides.length; i++) {
+             if (_disableOverrides[i].type.test(type) && _disableOverrides[i].subtype.test(subtype)) {
+               return false;
+             }
+           }
 
-             if (qaService) {
-               d.newStatus = 'ignore'; // ignore permanently (false positive)
+           return true;
+         } // `entityIDsToValidate()`   (private)
+         // Collects the complete list of entityIDs related to the input entityIDs.
+         //
+         // Arguments
+         //   `entityIDs` - Set or Array containing entityIDs.
+         //   `graph` - graph containing the entities
+         //
+         // Returns
+         //   Set containing entityIDs
+         //
 
-               qaService.postUpdate(d, function (err, item) {
-                 return dispatch$1.call('change', item);
-               });
-             }
-           });
-         } // NOTE: Don't change method name until UI v3 is merged
 
+         function entityIDsToValidate(entityIDs, graph) {
+           var seen = new Set();
+           var collected = new Set();
+           entityIDs.forEach(function (entityID) {
+             // keep `seen` separate from `collected` because an `entityID`
+             // could have been added to `collected` as a related entity through an earlier pass
+             if (seen.has(entityID)) return;
+             seen.add(entityID);
+             var entity = graph.hasEntity(entityID);
+             if (!entity) return;
+             collected.add(entityID); // collect self
 
-         keepRightEditor.error = function (val) {
-           if (!arguments.length) return _qaItem;
-           _qaItem = val;
-           return keepRightEditor;
-         };
+             var checkParentRels = [entity];
 
-         return utilRebind(keepRightEditor, dispatch$1, 'on');
-       }
+             if (entity.type === 'node') {
+               graph.parentWays(entity).forEach(function (parentWay) {
+                 collected.add(parentWay.id); // collect parent ways
 
-       function uiOsmoseDetails(context) {
-         var _qaItem;
+                 checkParentRels.push(parentWay);
+               });
+             } else if (entity.type === 'relation') {
+               entity.members.forEach(function (member) {
+                 return collected.add(member.id);
+               }); // collect members
+             } else if (entity.type === 'way') {
+               entity.nodes.forEach(function (nodeID) {
+                 collected.add(nodeID); // collect child nodes
 
-         function issueString(d, type) {
-           if (!d) return ''; // Issue strings are cached from Osmose API
+                 graph._parentWays[nodeID].forEach(function (wayID) {
+                   return collected.add(wayID);
+                 }); // collect connected ways
 
-           var s = services.osmose.getStrings(d.itemType);
-           return type in s ? s[type] : '';
-         }
+               });
+             }
 
-         function osmoseDetails(selection) {
-           var details = selection.selectAll('.error-details').data(_qaItem ? [_qaItem] : [], function (d) {
-             return "".concat(d.id, "-").concat(d.status || 0);
+             checkParentRels.forEach(function (entity) {
+               // collect parent relations
+               if (entity.type !== 'relation') {
+                 // but not super-relations
+                 graph.parentRelations(entity).forEach(function (parentRelation) {
+                   return collected.add(parentRelation.id);
+                 });
+               }
+             });
            });
-           details.exit().remove();
-           var detailsEnter = details.enter().append('div').attr('class', 'error-details qa-details-container'); // Description
+           return collected;
+         } // `updateResolvedIssues()`   (private)
+         // Determine if any issues were resolved for the given entities.
+         // This is called by `validate()` after validation of the head graph
+         //
+         // Arguments
+         //   `entityIDs` - Set containing entity IDs.
+         //
 
-           if (issueString(_qaItem, 'detail')) {
-             var div = detailsEnter.append('div').attr('class', 'qa-details-subsection');
-             div.append('h4').html(_t.html('QA.keepRight.detail_description'));
-             div.append('p').attr('class', 'qa-details-description-text').html(function (d) {
-               return issueString(d, 'detail');
-             }).selectAll('a').attr('rel', 'noopener').attr('target', '_blank');
-           } // Elements (populated later as data is requested)
 
+         function updateResolvedIssues(entityIDs) {
+           entityIDs.forEach(function (entityID) {
+             var headIssues = _headCache.issuesByEntityID[entityID];
+             var baseIssues = _baseCache.issuesByEntityID[entityID];
+             if (!baseIssues) return;
+             baseIssues.forEach(function (issueID) {
+               if (headIssues && headIssues.has(issueID)) {
+                 // issue still not resolved
+                 _resolvedIssueIDs["delete"](issueID); // (did undo, or possibly fixed and then re-caused the issue)
 
-           var detailsDiv = detailsEnter.append('div').attr('class', 'qa-details-subsection');
-           var elemsDiv = detailsEnter.append('div').attr('class', 'qa-details-subsection'); // Suggested Fix (mustn't exist for every issue type)
+               } else {
+                 _resolvedIssueIDs.add(issueID);
+               }
+             });
+           });
+         } // `validateEntitiesAsync()`   (private)
+         // Schedule validation for many entities.
+         //
+         // Arguments
+         //   `entityIDs` - Set containing entity IDs.
+         //   `graph` - the graph to validate that contains those entities
+         //   `cache` - the cache to store results in (_headCache or _baseCache)
+         //
+         // Returns
+         //   A Promise fulfilled when the validation has completed.
+         //   This may take time but happen in the background during browser idle time.
+         //
 
-           if (issueString(_qaItem, 'fix')) {
-             var _div = detailsEnter.append('div').attr('class', 'qa-details-subsection');
 
-             _div.append('h4').html(_t.html('QA.osmose.fix_title'));
+         function validateEntitiesAsync(entityIDs, graph, cache) {
+           // Enqueue the work
+           var jobs = Array.from(entityIDs).map(function (entityID) {
+             if (cache.queuedEntityIDs.has(entityID)) return null; // queued already
 
-             _div.append('p').html(function (d) {
-               return issueString(d, 'fix');
-             }).selectAll('a').attr('rel', 'noopener').attr('target', '_blank');
-           } // Common Pitfalls (mustn't exist for every issue type)
+             cache.queuedEntityIDs.add(entityID);
+             return function () {
+               // clear caches for existing issues related to this entity
+               cache.uncacheEntityID(entityID);
+               cache.queuedEntityIDs["delete"](entityID); // detect new issues and update caches
 
+               var entity = graph.hasEntity(entityID); // Sanity check: don't validate deleted entities
 
-           if (issueString(_qaItem, 'trap')) {
-             var _div2 = detailsEnter.append('div').attr('class', 'qa-details-subsection');
+               if (entity) {
+                 var result = validateEntity(entity, graph);
 
-             _div2.append('h4').html(_t.html('QA.osmose.trap_title'));
+                 if (result.provisional) {
+                   // provisional result
+                   cache.provisionalEntityIDs.add(entityID); // we'll need to revalidate this entity again later
+                 }
 
-             _div2.append('p').html(function (d) {
-               return issueString(d, 'trap');
-             }).selectAll('a').attr('rel', 'noopener').attr('target', '_blank');
-           } // Save current item to check if UI changed by time request resolves
+                 cache.cacheIssues(result.issues); // update cache
+               }
+             };
+           }).filter(Boolean); // Perform the work in chunks.
+           // Because this will happen during idle callbacks, we want to choose a chunk size
+           // that won't make the browser stutter too badly.
 
+           cache.queue = cache.queue.concat(utilArrayChunk(jobs, 100)); // Perform the work
 
-           var thisItem = _qaItem;
-           services.osmose.loadIssueDetail(_qaItem).then(function (d) {
-             // No details to add if there are no associated issue elements
-             if (!d.elems || d.elems.length === 0) return; // Do nothing if UI has moved on by the time this resolves
+           if (cache.queuePromise) return cache.queuePromise;
+           cache.queuePromise = processQueue(cache).then(function () {
+             return revalidateProvisionalEntities(cache);
+           })["catch"](function () {
+             /* ignore */
+           })["finally"](function () {
+             return cache.queuePromise = null;
+           });
+           return cache.queuePromise;
+         } // `revalidateProvisionalEntities()`   (private)
+         // Sometimes a validator will return a "provisional" result.
+         // In this situation, we'll need to revalidate the entity later.
+         // This function waits a delay, then places them back into the validation queue.
+         //
+         // Arguments
+         //   `cache` - The cache (_headCache or _baseCache)
+         //
 
-             if (context.selectedErrorID() !== thisItem.id && context.container().selectAll(".qaItem.osmose.hover.itemId-".concat(thisItem.id)).empty()) return; // Things like keys and values are dynamically added to a subtitle string
 
-             if (d.detail) {
-               detailsDiv.append('h4').html(_t.html('QA.osmose.detail_title'));
-               detailsDiv.append('p').html(function (d) {
-                 return d.detail;
-               }).selectAll('a').attr('rel', 'noopener').attr('target', '_blank');
-             } // Create list of linked issue elements
+         function revalidateProvisionalEntities(cache) {
+           if (!cache.provisionalEntityIDs.size) return; // nothing to do
 
+           var handle = window.setTimeout(function () {
+             _deferredST["delete"](handle);
 
-             elemsDiv.append('h4').html(_t.html('QA.osmose.elems_title'));
-             elemsDiv.append('ul').selectAll('li').data(d.elems).enter().append('li').append('a').attr('href', '#').attr('class', 'error_entity_link').html(function (d) {
-               return d;
-             }).each(function () {
-               var link = select(this);
-               var entityID = this.textContent;
-               var entity = context.hasEntity(entityID); // Add click handler
+             if (!cache.provisionalEntityIDs.size) return; // nothing to do
 
-               link.on('mouseenter', function () {
-                 utilHighlightEntities([entityID], true, context);
-               }).on('mouseleave', function () {
-                 utilHighlightEntities([entityID], false, context);
-               }).on('click', function (d3_event) {
-                 d3_event.preventDefault();
-                 utilHighlightEntities([entityID], false, context);
-                 var osmlayer = context.layers().layer('osm');
+             var graph = cache === _headCache ? _headGraph : context.history().base();
+             validateEntitiesAsync(cache.provisionalEntityIDs, graph, cache);
+           }, RETRY);
 
-                 if (!osmlayer.enabled()) {
-                   osmlayer.enabled(true);
-                 }
+           _deferredST.add(handle);
+         } // `processQueue(queue)`   (private)
+         // Process the next chunk of deferred validation work
+         //
+         // Arguments
+         //   `cache` - The cache (_headCache or _baseCache)
+         //
+         // Returns
+         //   A Promise fulfilled when the validation has completed.
+         //   This may take time but happen in the background during browser idle time.
+         //
 
-                 context.map().centerZoom(d.loc, 20);
 
-                 if (entity) {
-                   context.enter(modeSelect(context, [entityID]));
-                 } else {
-                   context.loadEntity(entityID, function () {
-                     context.enter(modeSelect(context, [entityID]));
-                   });
-                 }
-               }); // Replace with friendly name if possible
-               // (The entity may not yet be loaded into the graph)
+         function processQueue(cache) {
+           // const which = (cache === _headCache) ? 'head' : 'base';
+           // console.log(`${which} queue length ${cache.queue.length}`);
+           if (!cache.queue.length) return Promise.resolve(); // we're done
 
-               if (entity) {
-                 var name = utilDisplayName(entity); // try to use common name
+           var chunk = cache.queue.pop();
+           return new Promise(function (resolvePromise) {
+             var handle = window.requestIdleCallback(function () {
+               _deferredRIC["delete"](handle); // const t0 = performance.now();
 
-                 if (!name) {
-                   var preset = _mainPresetIndex.match(entity, context.graph());
-                   name = preset && !preset.isFallback() && preset.name(); // fallback to preset name
-                 }
 
-                 if (name) {
-                   this.innerText = name;
-                 }
-               }
-             }); // Don't hide entities related to this issue - #5880
+               chunk.forEach(function (job) {
+                 return job();
+               }); // const t1 = performance.now();
+               // console.log('chunk processed in ' + (t1 - t0) + ' ms');
 
-             context.features().forceVisible(d.elems);
-             context.map().pan([0, 0]); // trigger a redraw
-           })["catch"](function (err) {
-             console.log(err); // eslint-disable-line no-console
+               resolvePromise();
+             });
+
+             _deferredRIC.add(handle);
+           }).then(function () {
+             // dispatch an event sometimes to redraw various UI things
+             if (cache.queue.length % 25 === 0) dispatch.call('validated');
+           }).then(function () {
+             return processQueue(cache);
            });
          }
 
-         osmoseDetails.issue = function (val) {
-           if (!arguments.length) return _qaItem;
-           _qaItem = val;
-           return osmoseDetails;
-         };
-
-         return osmoseDetails;
-       }
+         return validator;
+       } // `validationCache()`   (private)
+       // Creates a cache to store validation state
+       // We create 2 of these:
+       //   `_baseCache` for validation on the base graph (unedited)
+       //   `_headCache` for validation on the head graph (user edits applied)
+       //
 
-       function uiOsmoseHeader() {
-         var _qaItem;
+       function validationCache() {
+         var cache = {
+           queue: [],
+           queuePromise: null,
+           queuedEntityIDs: new Set(),
+           provisionalEntityIDs: new Set(),
+           issuesByIssueID: {},
+           // issue.id -> issue
+           issuesByEntityID: {} // entity.id -> set(issue.id)
 
-         function issueTitle(d) {
-           var unknown = _t('inspector.unknown');
-           if (!d) return unknown; // Issue titles supplied by Osmose
+         };
 
-           var s = services.osmose.getStrings(d.itemType);
-           return 'title' in s ? s.title : unknown;
-         }
+         cache.cacheIssues = function (issues) {
+           issues.forEach(function (issue) {
+             var entityIDs = issue.entityIds || [];
+             entityIDs.forEach(function (entityID) {
+               if (!cache.issuesByEntityID[entityID]) {
+                 cache.issuesByEntityID[entityID] = new Set();
+               }
 
-         function osmoseHeader(selection) {
-           var header = selection.selectAll('.qa-header').data(_qaItem ? [_qaItem] : [], function (d) {
-             return "".concat(d.id, "-").concat(d.status || 0);
-           });
-           header.exit().remove();
-           var headerEnter = header.enter().append('div').attr('class', 'qa-header');
-           var svgEnter = headerEnter.append('div').attr('class', 'qa-header-icon').classed('new', function (d) {
-             return d.id < 0;
-           }).append('svg').attr('width', '20px').attr('height', '30px').attr('viewbox', '0 0 20 30').attr('class', function (d) {
-             return "preset-icon-28 qaItem ".concat(d.service, " itemId-").concat(d.id, " itemType-").concat(d.itemType);
+               cache.issuesByEntityID[entityID].add(issue.id);
+             });
+             cache.issuesByIssueID[issue.id] = issue;
            });
-           svgEnter.append('polygon').attr('fill', function (d) {
-             return services.osmose.getColor(d.item);
-           }).attr('class', 'qaItem-fill').attr('points', '16,3 4,3 1,6 1,17 4,20 7,20 10,27 13,20 16,20 19,17.033 19,6');
-           svgEnter.append('use').attr('class', 'icon-annotation').attr('width', '13px').attr('height', '13px').attr('transform', 'translate(3.5, 5)').attr('xlink:href', function (d) {
-             var picon = d.icon;
+         };
 
-             if (!picon) {
-               return '';
-             } else {
-               var isMaki = /^maki-/.test(picon);
-               return "#".concat(picon).concat(isMaki ? '-11' : '');
+         cache.uncacheIssue = function (issue) {
+           // When multiple entities are involved (e.g. crossing_ways),
+           // remove this issue from the other entity caches too..
+           var entityIDs = issue.entityIds || [];
+           entityIDs.forEach(function (entityID) {
+             if (cache.issuesByEntityID[entityID]) {
+               cache.issuesByEntityID[entityID]["delete"](issue.id);
              }
            });
-           headerEnter.append('div').attr('class', 'qa-header-label').html(issueTitle);
-         }
-
-         osmoseHeader.issue = function (val) {
-           if (!arguments.length) return _qaItem;
-           _qaItem = val;
-           return osmoseHeader;
+           delete cache.issuesByIssueID[issue.id];
          };
 
-         return osmoseHeader;
-       }
-
-       function uiViewOnOsmose() {
-         var _qaItem;
+         cache.uncacheIssues = function (issues) {
+           issues.forEach(cache.uncacheIssue);
+         };
 
-         function viewOnOsmose(selection) {
-           var url;
+         cache.uncacheIssuesOfType = function (type) {
+           var issuesOfType = Object.values(cache.issuesByIssueID).filter(function (issue) {
+             return issue.type === type;
+           });
+           cache.uncacheIssues(issuesOfType);
+         }; // Remove a single entity and all its related issues from the caches
 
-           if (services.osmose && _qaItem instanceof QAItem) {
-             url = services.osmose.itemURL(_qaItem);
-           }
 
-           var link = selection.selectAll('.view-on-osmose').data(url ? [url] : []); // exit
+         cache.uncacheEntityID = function (entityID) {
+           var issueIDs = cache.issuesByEntityID[entityID];
 
-           link.exit().remove(); // enter
+           if (issueIDs) {
+             issueIDs.forEach(function (issueID) {
+               var issue = cache.issuesByIssueID[issueID];
 
-           var linkEnter = link.enter().append('a').attr('class', 'view-on-osmose').attr('target', '_blank').attr('rel', 'noopener') // security measure
-           .attr('href', function (d) {
-             return d;
-           }).call(svgIcon('#iD-icon-out-link', 'inline'));
-           linkEnter.append('span').html(_t.html('inspector.view_on_osmose'));
-         }
+               if (issue) {
+                 cache.uncacheIssue(issue);
+               } else {
+                 delete cache.issuesByIssueID[issueID];
+               }
+             });
+           }
 
-         viewOnOsmose.what = function (val) {
-           if (!arguments.length) return _qaItem;
-           _qaItem = val;
-           return viewOnOsmose;
+           delete cache.issuesByEntityID[entityID];
+           cache.provisionalEntityIDs["delete"](entityID);
          };
 
-         return viewOnOsmose;
+         return cache;
        }
 
-       function uiOsmoseEditor(context) {
-         var dispatch$1 = dispatch('change');
-         var qaDetails = uiOsmoseDetails(context);
-         var qaHeader = uiOsmoseHeader();
-
-         var _qaItem;
-
-         function osmoseEditor(selection) {
-           var header = selection.selectAll('.header').data([0]);
-           var headerEnter = header.enter().append('div').attr('class', 'header fillL');
-           headerEnter.append('button').attr('class', 'close').on('click', function () {
-             return context.enter(modeBrowse(context));
-           }).call(svgIcon('#iD-icon-close'));
-           headerEnter.append('h3').html(_t.html('QA.osmose.title'));
-           var body = selection.selectAll('.body').data([0]);
-           body = body.enter().append('div').attr('class', 'body').merge(body);
-           var editor = body.selectAll('.qa-editor').data([0]);
-           editor.enter().append('div').attr('class', 'modal-section qa-editor').merge(editor).call(qaHeader.issue(_qaItem)).call(qaDetails.issue(_qaItem)).call(osmoseSaveSection);
-           var footer = selection.selectAll('.footer').data([0]);
-           footer.enter().append('div').attr('class', 'footer').merge(footer).call(uiViewOnOsmose().what(_qaItem));
-         }
-
-         function osmoseSaveSection(selection) {
-           var isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
-
-           var isShown = _qaItem && isSelected;
-           var saveSection = selection.selectAll('.qa-save').data(isShown ? [_qaItem] : [], function (d) {
-             return "".concat(d.id, "-").concat(d.status || 0);
-           }); // exit
-
-           saveSection.exit().remove(); // enter
-
-           var saveSectionEnter = saveSection.enter().append('div').attr('class', 'qa-save save-section cf'); // update
+       function coreUploader(context) {
+         var dispatch = dispatch$8( // Start and end events are dispatched exactly once each per legitimate outside call to `save`
+         'saveStarted', // dispatched as soon as a call to `save` has been deemed legitimate
+         'saveEnded', // dispatched after the result event has been dispatched
+         'willAttemptUpload', // dispatched before the actual upload call occurs, if it will
+         'progressChanged', // Each save results in one of these outcomes:
+         'resultNoChanges', // upload wasn't attempted since there were no edits
+         'resultErrors', // upload failed due to errors
+         'resultConflicts', // upload failed due to data conflicts
+         'resultSuccess' // upload completed without errors
+         );
+         var _isSaving = false;
+         var _conflicts = [];
+         var _errors = [];
 
-           saveSection = saveSectionEnter.merge(saveSection).call(qaSaveButtons);
-         }
+         var _origChanges;
 
-         function qaSaveButtons(selection) {
-           var isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
+         var _discardTags = {};
+         _mainFileFetcher.get('discarded').then(function (d) {
+           _discardTags = d;
+         })["catch"](function () {
+           /* ignore */
+         });
+         var uploader = utilRebind({}, dispatch, 'on');
 
-           var buttonSection = selection.selectAll('.buttons').data(isSelected ? [_qaItem] : [], function (d) {
-             return d.status + d.id;
-           }); // exit
+         uploader.isSaving = function () {
+           return _isSaving;
+         };
 
-           buttonSection.exit().remove(); // enter
+         uploader.save = function (changeset, tryAgain, checkConflicts) {
+           // Guard against accidentally entering save code twice - #4641
+           if (_isSaving && !tryAgain) {
+             return;
+           }
 
-           var buttonEnter = buttonSection.enter().append('div').attr('class', 'buttons');
-           buttonEnter.append('button').attr('class', 'button close-button action');
-           buttonEnter.append('button').attr('class', 'button ignore-button action'); // update
+           var osm = context.connection();
+           if (!osm) return; // If user somehow got logged out mid-save, try to reauthenticate..
+           // This can happen if they were logged in from before, but the tokens are no longer valid.
 
-           buttonSection = buttonSection.merge(buttonEnter);
-           buttonSection.select('.close-button').html(_t.html('QA.keepRight.close')).on('click.close', function (d3_event, d) {
-             this.blur(); // avoid keeping focus on the button - #4641
+           if (!osm.authenticated()) {
+             osm.authenticate(function (err) {
+               if (!err) {
+                 uploader.save(changeset, tryAgain, checkConflicts); // continue where we left off..
+               }
+             });
+             return;
+           }
 
-             var qaService = services.osmose;
+           if (!_isSaving) {
+             _isSaving = true;
+             dispatch.call('saveStarted', this);
+           }
 
-             if (qaService) {
-               d.newStatus = 'done';
-               qaService.postUpdate(d, function (err, item) {
-                 return dispatch$1.call('change', item);
-               });
-             }
-           });
-           buttonSection.select('.ignore-button').html(_t.html('QA.keepRight.ignore')).on('click.ignore', function (d3_event, d) {
-             this.blur(); // avoid keeping focus on the button - #4641
+           var history = context.history();
+           _conflicts = [];
+           _errors = []; // Store original changes, in case user wants to download them as an .osc file
 
-             var qaService = services.osmose;
+           _origChanges = history.changes(actionDiscardTags(history.difference(), _discardTags)); // First time, `history.perform` a no-op action.
+           // Any conflict resolutions will be done as `history.replace`
+           // Remember to pop this later if needed
 
-             if (qaService) {
-               d.newStatus = 'false';
-               qaService.postUpdate(d, function (err, item) {
-                 return dispatch$1.call('change', item);
-               });
-             }
-           });
-         } // NOTE: Don't change method name until UI v3 is merged
+           if (!tryAgain) {
+             history.perform(actionNoop());
+           } // Attempt a fast upload.. If there are conflicts, re-enter with `checkConflicts = true`
 
 
-         osmoseEditor.error = function (val) {
-           if (!arguments.length) return _qaItem;
-           _qaItem = val;
-           return osmoseEditor;
+           if (!checkConflicts) {
+             upload(changeset); // Do the full (slow) conflict check..
+           } else {
+             performFullConflictCheck(changeset);
+           }
          };
 
-         return utilRebind(osmoseEditor, dispatch$1, 'on');
-       }
+         function performFullConflictCheck(changeset) {
+           var osm = context.connection();
+           if (!osm) return;
+           var history = context.history();
+           var localGraph = context.graph();
+           var remoteGraph = coreGraph(history.base(), true);
+           var summary = history.difference().summary();
+           var _toCheck = [];
 
-       function modeSelectError(context, selectedErrorID, selectedErrorService) {
-         var mode = {
-           id: 'select-error',
-           button: 'browse'
-         };
-         var keybinding = utilKeybinding('select-error');
-         var errorService = services[selectedErrorService];
-         var errorEditor;
+           for (var i = 0; i < summary.length; i++) {
+             var item = summary[i];
 
-         switch (selectedErrorService) {
-           case 'improveOSM':
-             errorEditor = uiImproveOsmEditor(context).on('change', function () {
-               context.map().pan([0, 0]); // trigger a redraw
+             if (item.changeType === 'modified') {
+               _toCheck.push(item.entity.id);
+             }
+           }
 
-               var error = checkSelectedID();
-               if (!error) return;
-               context.ui().sidebar.show(errorEditor.error(error));
-             });
-             break;
+           var _toLoad = withChildNodes(_toCheck, localGraph);
 
-           case 'keepRight':
-             errorEditor = uiKeepRightEditor(context).on('change', function () {
-               context.map().pan([0, 0]); // trigger a redraw
+           var _loaded = {};
+           var _toLoadCount = 0;
+           var _toLoadTotal = _toLoad.length;
 
-               var error = checkSelectedID();
-               if (!error) return;
-               context.ui().sidebar.show(errorEditor.error(error));
+           if (_toCheck.length) {
+             dispatch.call('progressChanged', this, _toLoadCount, _toLoadTotal);
+
+             _toLoad.forEach(function (id) {
+               _loaded[id] = false;
              });
-             break;
 
-           case 'osmose':
-             errorEditor = uiOsmoseEditor(context).on('change', function () {
-               context.map().pan([0, 0]); // trigger a redraw
+             osm.loadMultiple(_toLoad, loaded);
+           } else {
+             upload(changeset);
+           }
 
-               var error = checkSelectedID();
-               if (!error) return;
-               context.ui().sidebar.show(errorEditor.error(error));
+           return;
+
+           function withChildNodes(ids, graph) {
+             var s = new Set(ids);
+             ids.forEach(function (id) {
+               var entity = graph.entity(id);
+               if (entity.type !== 'way') return;
+               graph.childNodes(entity).forEach(function (child) {
+                 if (child.version !== undefined) {
+                   s.add(child.id);
+                 }
+               });
              });
-             break;
-         }
+             return Array.from(s);
+           } // Reload modified entities into an alternate graph and check for conflicts..
 
-         var behaviors = [behaviorBreathe(), behaviorHover(context), behaviorSelect(context), behaviorLasso(context), modeDragNode(context).behavior, modeDragNote(context).behavior];
 
-         function checkSelectedID() {
-           if (!errorService) return;
-           var error = errorService.getError(selectedErrorID);
+           function loaded(err, result) {
+             if (_errors.length) return;
 
-           if (!error) {
-             context.enter(modeBrowse(context));
-           }
+             if (err) {
+               _errors.push({
+                 msg: err.message || err.responseText,
+                 details: [_t('save.status_code', {
+                   code: err.status
+                 })]
+               });
 
-           return error;
-         }
+               didResultInErrors();
+             } else {
+               var loadMore = [];
+               result.data.forEach(function (entity) {
+                 remoteGraph.replace(entity);
+                 _loaded[entity.id] = true;
+                 _toLoad = _toLoad.filter(function (val) {
+                   return val !== entity.id;
+                 });
+                 if (!entity.visible) return; // Because loadMultiple doesn't download /full like loadEntity,
+                 // need to also load children that aren't already being checked..
 
-         mode.zoomToSelected = function () {
-           if (!errorService) return;
-           var error = errorService.getError(selectedErrorID);
+                 var i, id;
 
-           if (error) {
-             context.map().centerZoomEase(error.loc, 20);
-           }
-         };
+                 if (entity.type === 'way') {
+                   for (i = 0; i < entity.nodes.length; i++) {
+                     id = entity.nodes[i];
 
-         mode.enter = function () {
-           var error = checkSelectedID();
-           if (!error) return;
-           behaviors.forEach(context.install);
-           keybinding.on(_t('inspector.zoom_to.key'), mode.zoomToSelected).on('⎋', esc, true);
-           select(document).call(keybinding);
-           selectError();
-           var sidebar = context.ui().sidebar;
-           sidebar.show(errorEditor.error(error));
-           context.map().on('drawn.select-error', selectError); // class the error as selected, or return to browse mode if the error is gone
+                     if (_loaded[id] === undefined) {
+                       _loaded[id] = false;
+                       loadMore.push(id);
+                     }
+                   }
+                 } else if (entity.type === 'relation' && entity.isMultipolygon()) {
+                   for (i = 0; i < entity.members.length; i++) {
+                     id = entity.members[i].id;
 
-           function selectError(d3_event, drawn) {
-             if (!checkSelectedID()) return;
-             var selection = context.surface().selectAll('.itemId-' + selectedErrorID + '.' + selectedErrorService);
+                     if (_loaded[id] === undefined) {
+                       _loaded[id] = false;
+                       loadMore.push(id);
+                     }
+                   }
+                 }
+               });
+               _toLoadCount += result.data.length;
+               _toLoadTotal += loadMore.length;
+               dispatch.call('progressChanged', this, _toLoadCount, _toLoadTotal);
 
-             if (selection.empty()) {
-               // Return to browse mode if selected DOM elements have
-               // disappeared because the user moved them out of view..
-               var source = d3_event && d3_event.type === 'zoom' && d3_event.sourceEvent;
+               if (loadMore.length) {
+                 _toLoad.push.apply(_toLoad, loadMore);
 
-               if (drawn && source && (source.type === 'pointermove' || source.type === 'mousemove' || source.type === 'touchmove')) {
-                 context.enter(modeBrowse(context));
+                 osm.loadMultiple(loadMore, loaded);
+               }
+
+               if (!_toLoad.length) {
+                 detectConflicts();
+                 upload(changeset);
                }
-             } else {
-               selection.classed('selected', true);
-               context.selectedErrorID(selectedErrorID);
              }
            }
 
-           function esc() {
-             if (context.container().select('.combobox').size()) return;
-             context.enter(modeBrowse(context));
-           }
-         };
+           function detectConflicts() {
+             function choice(id, text, _action) {
+               return {
+                 id: id,
+                 text: text,
+                 action: function action() {
+                   history.replace(_action);
+                 }
+               };
+             }
 
-         mode.exit = function () {
-           behaviors.forEach(context.uninstall);
-           select(document).call(keybinding.unbind);
-           context.surface().selectAll('.qaItem.selected').classed('selected hover', false);
-           context.map().on('drawn.select-error', null);
-           context.ui().sidebar.hide();
-           context.selectedErrorID(null);
-           context.features().forceVisible([]);
-         };
+             function formatUser(d) {
+               return '<a href="' + osm.userURL(d) + '" target="_blank">' + d + '</a>';
+             }
 
-         return mode;
-       }
+             function entityName(entity) {
+               return utilDisplayName(entity) || utilDisplayType(entity.id) + ' ' + entity.id;
+             }
 
-       function behaviorSelect(context) {
-         var _tolerancePx = 4; // see also behaviorDrag
+             function sameVersions(local, remote) {
+               if (local.version !== remote.version) return false;
 
-         var _lastMouseEvent = null;
-         var _showMenu = false;
-         var _downPointers = {};
-         var _longPressTimeout = null;
-         var _lastInteractionType = null; // the id of the down pointer that's enabling multiselection while down
+               if (local.type === 'way') {
+                 var children = utilArrayUnion(local.nodes, remote.nodes);
 
-         var _multiselectionPointerId = null; // use pointer events on supported platforms; fallback to mouse events
+                 for (var i = 0; i < children.length; i++) {
+                   var a = localGraph.hasEntity(children[i]);
+                   var b = remoteGraph.hasEntity(children[i]);
+                   if (a && b && a.version !== b.version) return false;
+                 }
+               }
 
-         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
+               return true;
+             }
 
-         function keydown(d3_event) {
-           if (d3_event.keyCode === 32) {
-             // don't react to spacebar events during text input
-             var activeNode = document.activeElement;
-             if (activeNode && new Set(['INPUT', 'TEXTAREA']).has(activeNode.nodeName)) return;
-           }
+             _toCheck.forEach(function (id) {
+               var local = localGraph.entity(id);
+               var remote = remoteGraph.entity(id);
+               if (sameVersions(local, remote)) return;
+               var merge = actionMergeRemoteChanges(id, localGraph, remoteGraph, _discardTags, formatUser);
+               history.replace(merge);
+               var mergeConflicts = merge.conflicts();
+               if (!mergeConflicts.length) return; // merged safely
 
-           if (d3_event.keyCode === 93 || // context menu key
-           d3_event.keyCode === 32) {
-             // spacebar
-             d3_event.preventDefault();
-           }
+               var forceLocal = actionMergeRemoteChanges(id, localGraph, remoteGraph, _discardTags).withOption('force_local');
+               var forceRemote = actionMergeRemoteChanges(id, localGraph, remoteGraph, _discardTags).withOption('force_remote');
+               var keepMine = _t('save.conflict.' + (remote.visible ? 'keep_local' : 'restore'));
+               var keepTheirs = _t('save.conflict.' + (remote.visible ? 'keep_remote' : 'delete'));
 
-           if (d3_event.repeat) return; // ignore repeated events for held keys
-           // if any key is pressed the user is probably doing something other than long-pressing
+               _conflicts.push({
+                 id: id,
+                 name: entityName(local),
+                 details: mergeConflicts,
+                 chosen: 1,
+                 choices: [choice(id, keepMine, forceLocal), choice(id, keepTheirs, forceRemote)]
+               });
+             });
+           }
+         }
 
-           cancelLongPress();
+         function upload(changeset) {
+           var osm = context.connection();
 
-           if (d3_event.shiftKey) {
-             context.surface().classed('behavior-multiselect', true);
+           if (!osm) {
+             _errors.push({
+               msg: 'No OSM Service'
+             });
            }
 
-           if (d3_event.keyCode === 32) {
-             // spacebar
-             if (!_downPointers.spacebar && _lastMouseEvent) {
-               cancelLongPress();
-               _longPressTimeout = window.setTimeout(didLongPress, 500, 'spacebar', 'spacebar');
-               _downPointers.spacebar = {
-                 firstEvent: _lastMouseEvent,
-                 lastEvent: _lastMouseEvent
-               };
+           if (_conflicts.length) {
+             didResultInConflicts(changeset);
+           } else if (_errors.length) {
+             didResultInErrors();
+           } else {
+             var history = context.history();
+             var changes = history.changes(actionDiscardTags(history.difference(), _discardTags));
+
+             if (changes.modified.length || changes.created.length || changes.deleted.length) {
+               dispatch.call('willAttemptUpload', this);
+               osm.putChangeset(changeset, changes, uploadCallback);
+             } else {
+               // changes were insignificant or reverted by user
+               didResultInNoChanges();
              }
            }
          }
 
-         function keyup(d3_event) {
-           cancelLongPress();
-
-           if (!d3_event.shiftKey) {
-             context.surface().classed('behavior-multiselect', false);
-           }
-
-           if (d3_event.keyCode === 93) {
-             // context menu key
-             d3_event.preventDefault();
-             _lastInteractionType = 'menukey';
-             contextmenu(d3_event);
-           } else if (d3_event.keyCode === 32) {
-             // spacebar
-             var pointer = _downPointers.spacebar;
+         function uploadCallback(err, changeset) {
+           if (err) {
+             if (err.status === 409) {
+               // 409 Conflict
+               uploader.save(changeset, true, true); // tryAgain = true, checkConflicts = true
+             } else {
+               _errors.push({
+                 msg: err.message || err.responseText,
+                 details: [_t('save.status_code', {
+                   code: err.status
+                 })]
+               });
 
-             if (pointer) {
-               delete _downPointers.spacebar;
-               if (pointer.done) return;
-               d3_event.preventDefault();
-               _lastInteractionType = 'spacebar';
-               click(pointer.firstEvent, pointer.lastEvent, 'spacebar');
+               didResultInErrors();
              }
+           } else {
+             didResultInSuccess(changeset);
            }
          }
 
-         function pointerdown(d3_event) {
-           var id = (d3_event.pointerId || 'mouse').toString();
-           cancelLongPress();
-           if (d3_event.buttons && d3_event.buttons !== 1) return;
-           context.ui().closeEditMenu();
-           _longPressTimeout = window.setTimeout(didLongPress, 500, id, 'longdown-' + (d3_event.pointerType || 'mouse'));
-           _downPointers[id] = {
-             firstEvent: d3_event,
-             lastEvent: d3_event
-           };
+         function didResultInNoChanges() {
+           dispatch.call('resultNoChanges', this);
+           endSave();
+           context.flush(); // reset iD
          }
 
-         function didLongPress(id, interactionType) {
-           var pointer = _downPointers[id];
-           if (!pointer) return;
+         function didResultInErrors() {
+           context.history().pop();
+           dispatch.call('resultErrors', this, _errors);
+           endSave();
+         }
 
-           for (var i in _downPointers) {
-             // don't allow this or any currently down pointer to trigger another click
-             _downPointers[i].done = true;
-           } // treat long presses like right-clicks
+         function didResultInConflicts(changeset) {
+           _conflicts.sort(function (a, b) {
+             return b.id.localeCompare(a.id);
+           });
 
+           dispatch.call('resultConflicts', this, changeset, _conflicts, _origChanges);
+           endSave();
+         }
 
-           _longPressTimeout = null;
-           _lastInteractionType = interactionType;
-           _showMenu = true;
-           click(pointer.firstEvent, pointer.lastEvent, id);
+         function didResultInSuccess(changeset) {
+           // delete the edit stack cached to local storage
+           context.history().clearSaved();
+           dispatch.call('resultSuccess', this, changeset); // Add delay to allow for postgres replication #1646 #2678
+
+           window.setTimeout(function () {
+             endSave();
+             context.flush(); // reset iD
+           }, 2500);
          }
 
-         function pointermove(d3_event) {
-           var id = (d3_event.pointerId || 'mouse').toString();
+         function endSave() {
+           _isSaving = false;
+           dispatch.call('saveEnded', this);
+         }
 
-           if (_downPointers[id]) {
-             _downPointers[id].lastEvent = d3_event;
-           }
+         uploader.cancelConflictResolution = function () {
+           context.history().pop();
+         };
 
-           if (!d3_event.pointerType || d3_event.pointerType === 'mouse') {
-             _lastMouseEvent = d3_event;
+         uploader.processResolvedConflicts = function (changeset) {
+           var history = context.history();
 
-             if (_downPointers.spacebar) {
-               _downPointers.spacebar.lastEvent = d3_event;
+           for (var i = 0; i < _conflicts.length; i++) {
+             if (_conflicts[i].chosen === 1) {
+               // user chose "use theirs"
+               var entity = context.hasEntity(_conflicts[i].id);
+
+               if (entity && entity.type === 'way') {
+                 var children = utilArrayUniq(entity.nodes);
+
+                 for (var j = 0; j < children.length; j++) {
+                   history.replace(actionRevert(children[j]));
+                 }
+               }
+
+               history.replace(actionRevert(_conflicts[i].id));
              }
            }
-         }
 
-         function pointerup(d3_event) {
-           var id = (d3_event.pointerId || 'mouse').toString();
-           var pointer = _downPointers[id];
-           if (!pointer) return;
-           delete _downPointers[id];
+           uploader.save(changeset, true, false); // tryAgain = true, checkConflicts = false
+         };
 
-           if (_multiselectionPointerId === id) {
-             _multiselectionPointerId = null;
-           }
+         uploader.reset = function () {};
 
-           if (pointer.done) return;
-           click(pointer.firstEvent, d3_event, id);
-         }
+         return uploader;
+       }
 
-         function pointercancel(d3_event) {
-           var id = (d3_event.pointerId || 'mouse').toString();
-           if (!_downPointers[id]) return;
-           delete _downPointers[id];
+       var abs = Math.abs;
+       var exp = Math.exp;
+       var E = Math.E;
 
-           if (_multiselectionPointerId === id) {
-             _multiselectionPointerId = null;
-           }
+       var FORCED = fails(function () {
+         // eslint-disable-next-line es/no-math-sinh -- required for testing
+         return Math.sinh(-2e-17) != -2e-17;
+       });
+
+       // `Math.sinh` method
+       // https://tc39.es/ecma262/#sec-math.sinh
+       // V8 near Chromium 38 has a problem with very small numbers
+       _export({ target: 'Math', stat: true, forced: FORCED }, {
+         sinh: function sinh(x) {
+           return abs(x = +x) < 1 ? (mathExpm1(x) - mathExpm1(-x)) / 2 : (exp(x - 1) - exp(-x - 1)) * (E / 2);
          }
+       });
 
-         function contextmenu(d3_event) {
-           d3_event.preventDefault();
+       var isRetina = window.devicePixelRatio && window.devicePixelRatio >= 2; // listen for DPI change, e.g. when dragging a browser window from a retina to non-retina screen
 
-           if (!+d3_event.clientX && !+d3_event.clientY) {
-             if (_lastMouseEvent) {
-               d3_event.sourceEvent = _lastMouseEvent;
-             } else {
-               return;
-             }
-           } else {
-             _lastMouseEvent = d3_event;
-             _lastInteractionType = 'rightclick';
-           }
+       window.matchMedia("\n        (-webkit-min-device-pixel-ratio: 2), /* Safari */\n        (min-resolution: 2dppx),             /* standard */\n        (min-resolution: 192dpi)             /* fallback */\n    ").addListener(function () {
+         isRetina = window.devicePixelRatio && window.devicePixelRatio >= 2;
+       });
 
-           _showMenu = true;
-           click(d3_event, d3_event);
-         }
+       function localeDateString(s) {
+         if (!s) return null;
+         var options = {
+           day: 'numeric',
+           month: 'short',
+           year: 'numeric'
+         };
+         var d = new Date(s);
+         if (isNaN(d.getTime())) return null;
+         return d.toLocaleDateString(_mainLocalizer.localeCode(), options);
+       }
 
-         function click(firstEvent, lastEvent, pointerId) {
-           cancelLongPress();
-           var mapNode = context.container().select('.main-map').node(); // Use the `main-map` coordinate system since the surface and supersurface
-           // are transformed when drag-panning.
+       function vintageRange(vintage) {
+         var s;
 
-           var pointGetter = utilFastMouse(mapNode);
-           var p1 = pointGetter(firstEvent);
-           var p2 = pointGetter(lastEvent);
-           var dist = geoVecLength(p1, p2);
+         if (vintage.start || vintage.end) {
+           s = vintage.start || '?';
 
-           if (dist > _tolerancePx || !mapContains(lastEvent)) {
-             resetProperties();
-             return;
+           if (vintage.start !== vintage.end) {
+             s += ' - ' + (vintage.end || '?');
            }
+         }
 
-           var targetDatum = lastEvent.target.__data__;
-           var multiselectEntityId;
+         return s;
+       }
 
-           if (!_multiselectionPointerId) {
-             // If a different pointer than the one triggering this click is down on a
-             // feature, treat this and all future clicks as multiselection until that
-             // pointer is raised.
-             var selectPointerInfo = pointerDownOnSelection(pointerId);
+       function rendererBackgroundSource(data) {
+         var source = Object.assign({}, data); // shallow copy
 
-             if (selectPointerInfo) {
-               _multiselectionPointerId = selectPointerInfo.pointerId; // if the other feature isn't selected yet, make sure we select it
+         var _offset = [0, 0];
+         var _name = source.name;
+         var _description = source.description;
+
+         var _best = !!source.best;
+
+         var _template = source.encrypted ? utilAesDecrypt(source.template) : source.template;
 
-               multiselectEntityId = !selectPointerInfo.selected && selectPointerInfo.entityId;
-               _downPointers[selectPointerInfo.pointerId].done = true;
-             }
-           } // support multiselect if data is already selected
+         source.tileSize = data.tileSize || 256;
+         source.zoomExtent = data.zoomExtent || [0, 22];
+         source.overzoom = data.overzoom !== false;
 
+         source.offset = function (val) {
+           if (!arguments.length) return _offset;
+           _offset = val;
+           return source;
+         };
 
-           var isMultiselect = context.mode().id === 'select' && ( // and shift key is down
-           lastEvent && lastEvent.shiftKey || // or we're lasso-selecting
-           context.surface().select('.lasso').node() || // or a pointer is down over a selected feature
-           _multiselectionPointerId && !multiselectEntityId);
+         source.nudge = function (val, zoomlevel) {
+           _offset[0] += val[0] / Math.pow(2, zoomlevel);
+           _offset[1] += val[1] / Math.pow(2, zoomlevel);
+           return source;
+         };
 
-           processClick(targetDatum, isMultiselect, p2, multiselectEntityId);
+         source.name = function () {
+           var id_safe = source.id.replace(/\./g, '<TX_DOT>');
+           return _t('imagery.' + id_safe + '.name', {
+             "default": _name
+           });
+         };
 
-           function mapContains(event) {
-             var rect = mapNode.getBoundingClientRect();
-             return event.clientX >= rect.left && event.clientX <= rect.right && event.clientY >= rect.top && event.clientY <= rect.bottom;
-           }
+         source.label = function () {
+           var id_safe = source.id.replace(/\./g, '<TX_DOT>');
+           return _t.html('imagery.' + id_safe + '.name', {
+             "default": _name
+           });
+         };
 
-           function pointerDownOnSelection(skipPointerId) {
-             var mode = context.mode();
-             var selectedIDs = mode.id === 'select' ? mode.selectedIDs() : [];
+         source.description = function () {
+           var id_safe = source.id.replace(/\./g, '<TX_DOT>');
+           return _t.html('imagery.' + id_safe + '.description', {
+             "default": _description
+           });
+         };
 
-             for (var pointerId in _downPointers) {
-               if (pointerId === 'spacebar' || pointerId === skipPointerId) continue;
-               var pointerInfo = _downPointers[pointerId];
-               var p1 = pointGetter(pointerInfo.firstEvent);
-               var p2 = pointGetter(pointerInfo.lastEvent);
-               if (geoVecLength(p1, p2) > _tolerancePx) continue;
-               var datum = pointerInfo.firstEvent.target.__data__;
-               var entity = datum && datum.properties && datum.properties.entity || datum;
-               if (context.graph().hasEntity(entity.id)) return {
-                 pointerId: pointerId,
-                 entityId: entity.id,
-                 selected: selectedIDs.indexOf(entity.id) !== -1
-               };
-             }
+         source.best = function () {
+           return _best;
+         };
 
-             return null;
-           }
-         }
+         source.area = function () {
+           if (!data.polygon) return Number.MAX_VALUE; // worldwide
 
-         function processClick(datum, isMultiselect, point, alsoSelectId) {
-           var mode = context.mode();
-           var showMenu = _showMenu;
-           var interactionType = _lastInteractionType;
-           var entity = datum && datum.properties && datum.properties.entity;
-           if (entity) datum = entity;
+           var area = d3_geoArea({
+             type: 'MultiPolygon',
+             coordinates: [data.polygon]
+           });
+           return isNaN(area) ? 0 : area;
+         };
 
-           if (datum && datum.type === 'midpoint') {
-             // treat targeting midpoints as if targeting the parent way
-             datum = datum.parents[0];
-           }
+         source.imageryUsed = function () {
+           return _name || source.id;
+         };
 
-           var newMode;
+         source.template = function (val) {
+           if (!arguments.length) return _template;
 
-           if (datum instanceof osmEntity) {
-             // targeting an entity
-             var selectedIDs = context.selectedIDs();
-             context.selectedNoteID(null);
-             context.selectedErrorID(null);
+           if (source.id === 'custom' || source.id === 'Bing') {
+             _template = val;
+           }
 
-             if (!isMultiselect) {
-               // don't change the selection if we're toggling the menu atop a multiselection
-               if (!showMenu || selectedIDs.length <= 1 || selectedIDs.indexOf(datum.id) === -1) {
-                 if (alsoSelectId === datum.id) alsoSelectId = null;
-                 selectedIDs = (alsoSelectId ? [alsoSelectId] : []).concat([datum.id]); // always enter modeSelect even if the entity is already
-                 // selected since listeners may expect `context.enter` events,
-                 // e.g. in the walkthrough
+           return source;
+         };
 
-                 newMode = mode.id === 'select' ? mode.selectedIDs(selectedIDs) : modeSelect(context, selectedIDs).selectBehavior(behavior);
-                 context.enter(newMode);
-               }
-             } else {
-               if (selectedIDs.indexOf(datum.id) !== -1) {
-                 // clicked entity is already in the selectedIDs list..
-                 if (!showMenu) {
-                   // deselect clicked entity, then reenter select mode or return to browse mode..
-                   selectedIDs = selectedIDs.filter(function (id) {
-                     return id !== datum.id;
-                   });
-                   newMode = selectedIDs.length ? mode.selectedIDs(selectedIDs) : modeBrowse(context).selectBehavior(behavior);
-                   context.enter(newMode);
-                 }
-               } else {
-                 // clicked entity is not in the selected list, add it..
-                 selectedIDs = selectedIDs.concat([datum.id]);
-                 newMode = mode.selectedIDs(selectedIDs);
-                 context.enter(newMode);
-               }
-             }
-           } else if (datum && datum.__featurehash__ && !isMultiselect) {
-             // targeting custom data
-             context.selectedNoteID(null).enter(modeSelectData(context, datum));
-           } else if (datum instanceof osmNote && !isMultiselect) {
-             // targeting a note
-             context.selectedNoteID(datum.id).enter(modeSelectNote(context, datum.id));
-           } else if (datum instanceof QAItem & !isMultiselect) {
-             // targeting an external QA issue
-             context.selectedErrorID(datum.id).enter(modeSelectError(context, datum.id, datum.service));
-           } else {
-             // targeting nothing
-             context.selectedNoteID(null);
-             context.selectedErrorID(null);
+         source.url = function (coord) {
+           var result = _template;
+           if (result === '') return result; // source 'none'
+           // Guess a type based on the tokens present in the template
+           // (This is for 'custom' source, where we don't know)
 
-             if (!isMultiselect && mode.id !== 'browse') {
-               context.enter(modeBrowse(context));
+           if (!source.type) {
+             if (/SERVICE=WMS|\{(proj|wkid|bbox)\}/.test(_template)) {
+               source.type = 'wms';
+               source.projection = 'EPSG:3857'; // guess
+             } else if (/\{(x|y)\}/.test(_template)) {
+               source.type = 'tms';
+             } else if (/\{u\}/.test(_template)) {
+               source.type = 'bing';
              }
            }
 
-           context.ui().closeEditMenu(); // always request to show the edit menu in case the mode needs it
-
-           if (showMenu) context.ui().showEditMenu(point, interactionType);
-           resetProperties();
-         }
+           if (source.type === 'wms') {
+             var tileToProjectedCoords = function tileToProjectedCoords(x, y, z) {
+               //polyfill for IE11, PhantomJS
+               var sinh = Math.sinh || function (x) {
+                 var y = Math.exp(x);
+                 return (y - 1 / y) / 2;
+               };
 
-         function cancelLongPress() {
-           if (_longPressTimeout) window.clearTimeout(_longPressTimeout);
-           _longPressTimeout = null;
-         }
+               var zoomSize = Math.pow(2, z);
+               var lon = x / zoomSize * Math.PI * 2 - Math.PI;
+               var lat = Math.atan(sinh(Math.PI * (1 - 2 * y / zoomSize)));
 
-         function resetProperties() {
-           cancelLongPress();
-           _showMenu = false;
-           _lastInteractionType = null; // don't reset _lastMouseEvent since it might still be useful
-         }
+               switch (source.projection) {
+                 case 'EPSG:4326':
+                   return {
+                     x: lon * 180 / Math.PI,
+                     y: lat * 180 / Math.PI
+                   };
 
-         function behavior(selection) {
-           resetProperties();
-           _lastMouseEvent = context.map().lastPointerEvent();
-           select(window).on('keydown.select', keydown).on('keyup.select', keyup).on(_pointerPrefix + 'move.select', pointermove, true).on(_pointerPrefix + 'up.select', pointerup, true).on('pointercancel.select', pointercancel, true).on('contextmenu.select-window', function (d3_event) {
-             // Edge and IE really like to show the contextmenu on the
-             // menubar when user presses a keyboard menu button
-             // even after we've already preventdefaulted the key event.
-             var e = d3_event;
+                 default:
+                   // EPSG:3857 and synonyms
+                   var mercCoords = mercatorRaw(lon, lat);
+                   return {
+                     x: 20037508.34 / Math.PI * mercCoords[0],
+                     y: 20037508.34 / Math.PI * mercCoords[1]
+                   };
+               }
+             };
 
-             if (+e.clientX === 0 && +e.clientY === 0) {
-               d3_event.preventDefault();
-             }
-           });
-           selection.on(_pointerPrefix + 'down.select', pointerdown).on('contextmenu.select', contextmenu);
-           /*if (d3_event && d3_event.shiftKey) {
-               context.surface()
-                   .classed('behavior-multiselect', true);
-           }*/
-         }
+             var tileSize = source.tileSize;
+             var projection = source.projection;
+             var minXmaxY = tileToProjectedCoords(coord[0], coord[1], coord[2]);
+             var maxXminY = tileToProjectedCoords(coord[0] + 1, coord[1] + 1, coord[2]);
+             result = result.replace(/\{(\w+)\}/g, function (token, key) {
+               switch (key) {
+                 case 'width':
+                 case 'height':
+                   return tileSize;
 
-         behavior.off = function (selection) {
-           cancelLongPress();
-           select(window).on('keydown.select', null).on('keyup.select', null).on('contextmenu.select-window', null).on(_pointerPrefix + 'move.select', null, true).on(_pointerPrefix + 'up.select', null, true).on('pointercancel.select', null, true);
-           selection.on(_pointerPrefix + 'down.select', null).on('contextmenu.select', null);
-           context.surface().classed('behavior-multiselect', false);
-         };
+                 case 'proj':
+                   return projection;
 
-         return behavior;
-       }
+                 case 'wkid':
+                   return projection.replace(/^EPSG:/, '');
 
-       function behaviorDrawWay(context, wayID, mode, startGraph) {
-         var dispatch$1 = dispatch('rejectedSelfIntersection');
-         var behavior = behaviorDraw(context); // Must be set by `drawWay.nodeIndex` before each install of this behavior.
+                 case 'bbox':
+                   // WMS 1.3 flips x/y for some coordinate systems including EPSG:4326 - #7557
+                   if (projection === 'EPSG:4326' && // The CRS parameter implies version 1.3 (prior versions use SRS)
+                   /VERSION=1.3|CRS={proj}/.test(source.template().toUpperCase())) {
+                     return maxXminY.y + ',' + minXmaxY.x + ',' + minXmaxY.y + ',' + maxXminY.x;
+                   } else {
+                     return minXmaxY.x + ',' + maxXminY.y + ',' + maxXminY.x + ',' + minXmaxY.y;
+                   }
 
-         var _nodeIndex;
+                 case 'w':
+                   return minXmaxY.x;
 
-         var _origWay;
+                 case 's':
+                   return maxXminY.y;
 
-         var _wayGeometry;
+                 case 'n':
+                   return maxXminY.x;
 
-         var _headNodeID;
+                 case 'e':
+                   return minXmaxY.y;
 
-         var _annotation;
+                 default:
+                   return token;
+               }
+             });
+           } else if (source.type === 'tms') {
+             result = result.replace('{x}', coord[0]).replace('{y}', coord[1]) // TMS-flipped y coordinate
+             .replace(/\{[t-]y\}/, Math.pow(2, coord[2]) - coord[1] - 1).replace(/\{z(oom)?\}/, coord[2]) // only fetch retina tiles for retina screens
+             .replace(/\{@2x\}|\{r\}/, isRetina ? '@2x' : '');
+           } else if (source.type === 'bing') {
+             result = result.replace('{u}', function () {
+               var u = '';
 
-         var _pointerHasMoved = false; // The osmNode to be placed.
-         // This is temporary and just follows the mouse cursor until an "add" event occurs.
+               for (var zoom = coord[2]; zoom > 0; zoom--) {
+                 var b = 0;
+                 var mask = 1 << zoom - 1;
+                 if ((coord[0] & mask) !== 0) b++;
+                 if ((coord[1] & mask) !== 0) b += 2;
+                 u += b.toString();
+               }
 
-         var _drawNode;
+               return u;
+             });
+           } // these apply to any type..
 
-         var _didResolveTempEdit = false;
 
-         function createDrawNode(loc) {
-           // don't make the draw node until we actually need it
-           _drawNode = osmNode({
-             loc: loc
+           result = result.replace(/\{switch:([^}]+)\}/, function (s, r) {
+             var subdomains = r.split(',');
+             return subdomains[(coord[0] + coord[1]) % subdomains.length];
            });
-           context.pauseChangeDispatch();
-           context.replace(function actionAddDrawNode(graph) {
-             // add the draw node to the graph and insert it into the way
-             var way = graph.entity(wayID);
-             return graph.replace(_drawNode).replace(way.addNode(_drawNode.id, _nodeIndex));
-           }, _annotation);
-           context.resumeChangeDispatch();
-           setActiveElements();
-         }
+           return result;
+         };
 
-         function removeDrawNode() {
-           context.pauseChangeDispatch();
-           context.replace(function actionDeleteDrawNode(graph) {
-             var way = graph.entity(wayID);
-             return graph.replace(way.removeNode(_drawNode.id)).remove(_drawNode);
-           }, _annotation);
-           _drawNode = undefined;
-           context.resumeChangeDispatch();
-         }
+         source.validZoom = function (z) {
+           return source.zoomExtent[0] <= z && (source.overzoom || source.zoomExtent[1] > z);
+         };
 
-         function keydown(d3_event) {
-           if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
-             if (context.surface().classed('nope')) {
-               context.surface().classed('nope-suppressed', true);
-             }
+         source.isLocatorOverlay = function () {
+           return source.id === 'mapbox_locator_overlay';
+         };
+         /* hides a source from the list, but leaves it available for use */
 
-             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);
-             }
+         source.isHidden = function () {
+           return source.id === 'DigitalGlobe-Premium-vintage' || source.id === 'DigitalGlobe-Standard-vintage';
+         };
 
-             context.surface().classed('nope-suppressed', false).classed('nope-disabled', false);
-           }
-         }
+         source.copyrightNotices = function () {};
 
-         function allowsVertex(d) {
-           return d.geometry(context.graph()) === 'vertex' || _mainPresetIndex.allowsVertex(d, context.graph());
-         } // related code
-         // - `mode/drag_node.js`     `doMove()`
-         // - `behavior/draw.js`      `click()`
-         // - `behavior/draw_way.js`  `move()`
+         source.getMetadata = function (center, tileCoord, callback) {
+           var vintage = {
+             start: localeDateString(source.startDate),
+             end: localeDateString(source.endDate)
+           };
+           vintage.range = vintageRange(vintage);
+           var metadata = {
+             vintage: vintage
+           };
+           callback(null, metadata);
+         };
 
+         return source;
+       }
 
-         function move(d3_event, datum) {
-           var loc = context.map().mouseCoordinates();
-           if (!_drawNode) createDrawNode(loc);
-           context.surface().classed('nope-disabled', d3_event.altKey);
-           var targetLoc = datum && datum.properties && datum.properties.entity && allowsVertex(datum.properties.entity) && datum.properties.entity.loc;
-           var targetNodes = datum && datum.properties && datum.properties.nodes;
+       rendererBackgroundSource.Bing = function (data, dispatch) {
+         // https://docs.microsoft.com/en-us/bingmaps/rest-services/imagery/get-imagery-metadata
+         // https://docs.microsoft.com/en-us/bingmaps/rest-services/directly-accessing-the-bing-maps-tiles
+         //fallback url template
+         data.template = 'https://ecn.t{switch:0,1,2,3}.tiles.virtualearth.net/tiles/a{u}.jpeg?g=10555&n=z';
+         var bing = rendererBackgroundSource(data); //var key = 'Arzdiw4nlOJzRwOz__qailc8NiR31Tt51dN2D7cm57NrnceZnCpgOkmJhNpGoppU'; // P2, JOSM, etc
 
-           if (targetLoc) {
-             // snap to node/vertex - a point target with `.loc`
-             loc = targetLoc;
-           } else if (targetNodes) {
-             // snap to way - a line target with `.nodes`
-             var choice = geoChooseEdge(targetNodes, context.map().mouse(), context.projection, _drawNode.id);
+         var key = 'Ak5oTE46TUbjRp08OFVcGpkARErDobfpuyNKa-W2mQ8wbt1K1KL8p1bIRwWwcF-Q'; // iD
 
-             if (choice) {
-               loc = choice.loc;
-             }
-           }
+         /*
+         missing tile image strictness param (n=)
+         •   n=f -> (Fail) returns a 404
+         •   n=z -> (Empty) returns a 200 with 0 bytes (no content)
+         •   n=t -> (Transparent) returns a 200 with a transparent (png) tile
+         */
 
-           context.replace(actionMoveNode(_drawNode.id, loc), _annotation);
-           _drawNode = context.entity(_drawNode.id);
-           checkGeometry(true
-           /* includeDrawNode */
-           );
-         } // Check whether this edit causes the geometry to break.
-         // If so, class the surface with a nope cursor.
-         // `includeDrawNode` - Only check the relevant line segments if finishing drawing
+         var strictParam = 'n';
+         var url = 'https://dev.virtualearth.net/REST/v1/Imagery/Metadata/Aerial?include=ImageryProviders&uriScheme=https&key=' + key;
+         var cache = {};
+         var inflight = {};
+         var providers = [];
+         d3_json(url).then(function (json) {
+           var imageryResource = json.resourceSets[0].resources[0]; //retrieve and prepare up to date imagery template
 
+           var template = imageryResource.imageUrl; //https://ecn.{subdomain}.tiles.virtualearth.net/tiles/a{quadkey}.jpeg?g=10339
 
-         function checkGeometry(includeDrawNode) {
-           var nopeDisabled = context.surface().classed('nope-disabled');
-           var isInvalid = isInvalidGeometry(includeDrawNode);
+           var subDomains = imageryResource.imageUrlSubdomains; //["t0, t1, t2, t3"]
 
-           if (nopeDisabled) {
-             context.surface().classed('nope', false).classed('nope-suppressed', isInvalid);
-           } else {
-             context.surface().classed('nope', isInvalid).classed('nope-suppressed', false);
+           var subDomainNumbers = subDomains.map(function (subDomain) {
+             return subDomain.substring(1);
+           }).join(',');
+           template = template.replace('{subdomain}', "t{switch:".concat(subDomainNumbers, "}")).replace('{quadkey}', '{u}');
+
+           if (!new URLSearchParams(template).has(strictParam)) {
+             template += "&".concat(strictParam, "=z");
            }
-         }
 
-         function isInvalidGeometry(includeDrawNode) {
-           var testNode = _drawNode; // we only need to test the single way we're drawing
+           bing.template(template);
+           providers = imageryResource.imageryProviders.map(function (provider) {
+             return {
+               attribution: provider.attribution,
+               areas: provider.coverageAreas.map(function (area) {
+                 return {
+                   zoom: [area.zoomMin, area.zoomMax],
+                   extent: geoExtent([area.bbox[1], area.bbox[0]], [area.bbox[3], area.bbox[2]])
+                 };
+               })
+             };
+           });
+           dispatch.call('change');
+         })["catch"](function () {
+           /* ignore */
+         });
 
-           var parentWay = context.graph().entity(wayID);
-           var nodes = context.graph().childNodes(parentWay).slice(); // shallow copy
+         bing.copyrightNotices = function (zoom, extent) {
+           zoom = Math.min(zoom, 21);
+           return providers.filter(function (provider) {
+             return provider.areas.some(function (area) {
+               return extent.intersects(area.extent) && area.zoom[0] <= zoom && area.zoom[1] >= zoom;
+             });
+           }).map(function (provider) {
+             return provider.attribution;
+           }).join(', ');
+         };
 
-           if (includeDrawNode) {
-             if (parentWay.isClosed()) {
-               // don't test the last segment for closed ways - #4655
-               // (still test the first segment)
-               nodes.pop();
-             }
-           } else {
-             // discount the draw node
-             if (parentWay.isClosed()) {
-               if (nodes.length < 3) return false;
-               if (_drawNode) nodes.splice(-2, 1);
-               testNode = nodes[nodes.length - 2];
-             } else {
-               // there's nothing we need to test if we ignore the draw node on open ways
-               return false;
-             }
-           }
+         bing.getMetadata = function (center, tileCoord, callback) {
+           var tileID = tileCoord.slice(0, 3).join('/');
+           var zoom = Math.min(tileCoord[2], 21);
+           var centerPoint = center[1] + ',' + center[0]; // lat,lng
 
-           return testNode && geoHasSelfIntersections(nodes, testNode.id);
-         }
+           var url = 'https://dev.virtualearth.net/REST/v1/Imagery/Metadata/Aerial/' + centerPoint + '?zl=' + zoom + '&key=' + key;
+           if (inflight[tileID]) return;
 
-         function undone() {
-           // undoing removed the temp edit
-           _didResolveTempEdit = true;
-           context.pauseChangeDispatch();
-           var nextMode;
+           if (!cache[tileID]) {
+             cache[tileID] = {};
+           }
 
-           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 {
-             // 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); // continue drawing
+           if (cache[tileID] && cache[tileID].metadata) {
+             return callback(null, cache[tileID].metadata);
+           }
 
-             nextMode = mode;
-           } // clear the redo stack by adding and removing a blank edit
+           inflight[tileID] = true;
+           d3_json(url).then(function (result) {
+             delete inflight[tileID];
 
+             if (!result) {
+               throw new Error('Unknown Error');
+             }
 
-           context.perform(actionNoop());
-           context.pop(1);
-           context.resumeChangeDispatch();
-           context.enter(nextMode);
-         }
+             var vintage = {
+               start: localeDateString(result.resourceSets[0].resources[0].vintageStart),
+               end: localeDateString(result.resourceSets[0].resources[0].vintageEnd)
+             };
+             vintage.range = vintageRange(vintage);
+             var metadata = {
+               vintage: vintage
+             };
+             cache[tileID].metadata = metadata;
+             if (callback) callback(null, metadata);
+           })["catch"](function (err) {
+             delete inflight[tileID];
+             if (callback) callback(err.message);
+           });
+         };
 
-         function setActiveElements() {
-           if (!_drawNode) return;
-           context.surface().selectAll('.' + _drawNode.id).classed('active', true);
-         }
+         bing.terms_url = 'https://blog.openstreetmap.org/2010/11/30/microsoft-imagery-details';
+         return bing;
+       };
 
-         function resetToStartGraph() {
-           while (context.graph() !== startGraph) {
-             context.pop();
-           }
+       rendererBackgroundSource.Esri = function (data) {
+         // in addition to using the tilemap at zoom level 20, overzoom real tiles - #4327 (deprecated technique, but it works)
+         if (data.template.match(/blankTile/) === null) {
+           data.template = data.template + '?blankTile=false';
          }
 
-         var drawWay = function drawWay(surface) {
-           _drawNode = undefined;
-           _didResolveTempEdit = false;
-           _origWay = context.entity(wayID);
-           _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.nodes.length === (_origWay.isClosed() ? 2 : 1) ? 'operations.start.annotation.' : 'operations.continue.annotation.') + _wayGeometry);
-           _pointerHasMoved = false; // Push an annotated state for undo to return back to.
-           // We must make sure to replace or remove it later.
+         var esri = rendererBackgroundSource(data);
+         var cache = {};
+         var inflight = {};
+
+         var _prevCenter; // use a tilemap service to set maximum zoom for esri tiles dynamically
+         // https://developers.arcgis.com/documentation/tiled-elevation-service/
 
-           context.pauseChangeDispatch();
-           context.perform(actionNoop(), _annotation);
-           context.resumeChangeDispatch();
-           behavior.hover().initialNodeID(_headNodeID);
-           behavior.on('move', function () {
-             _pointerHasMoved = true;
-             move.apply(this, arguments);
-           }).on('down', function () {
-             move.apply(this, arguments);
-           }).on('downcancel', function () {
-             if (_drawNode) removeDrawNode();
-           }).on('click', drawWay.add).on('clickWay', drawWay.addWay).on('clickNode', drawWay.addNode).on('undo', context.undo).on('cancel', drawWay.cancel).on('finish', drawWay.finish);
-           select(window).on('keydown.drawWay', keydown).on('keyup.drawWay', keyup);
-           context.map().dblclickZoomEnable(false).on('drawn.draw', setActiveElements);
-           setActiveElements();
-           surface.call(behavior);
-           context.history().on('undone.draw', undone);
-         };
 
-         drawWay.off = function (surface) {
-           if (!_didResolveTempEdit) {
-             // Drawing was interrupted unexpectedly.
-             // This can happen if the user changes modes,
-             // clicks geolocate button, a hashchange event occurs, etc.
-             context.pauseChangeDispatch();
-             resetToStartGraph();
-             context.resumeChangeDispatch();
-           }
+         esri.fetchTilemap = function (center) {
+           // skip if we have already fetched a tilemap within 5km
+           if (_prevCenter && geoSphericalDistance(center, _prevCenter) < 5000) return;
+           _prevCenter = center; // tiles are available globally to zoom level 19, afterward they may or may not be present
 
-           _drawNode = undefined;
-           _nodeIndex = undefined;
-           context.map().on('drawn.draw', null);
-           surface.call(behavior.off).selectAll('.active').classed('active', false);
-           surface.classed('nope', false).classed('nope-suppressed', false).classed('nope-disabled', false);
-           select(window).on('keydown.drawWay', null).on('keyup.drawWay', null);
-           context.history().on('undone.draw', null);
-         };
+           var z = 20; // first generate a random url using the template
 
-         function attemptAdd(d, loc, doAdd) {
-           if (_drawNode) {
-             // move the node to the final loc in case move wasn't called
-             // consistently (e.g. on touch devices)
-             context.replace(actionMoveNode(_drawNode.id, loc), _annotation);
-             _drawNode = context.entity(_drawNode.id);
-           } else {
-             createDrawNode(loc);
-           }
+           var dummyUrl = esri.url([1, 2, 3]); // calculate url z/y/x from the lat/long of the center of the map
 
-           checkGeometry(true
-           /* includeDrawNode */
-           );
+           var x = Math.floor((center[0] + 180) / 360 * Math.pow(2, z));
+           var y = Math.floor((1 - Math.log(Math.tan(center[1] * Math.PI / 180) + 1 / Math.cos(center[1] * Math.PI / 180)) / Math.PI) / 2 * Math.pow(2, z)); // fetch an 8x8 grid to leverage cache
 
-           if (d && d.properties && d.properties.nope || context.surface().classed('nope')) {
-             if (!_pointerHasMoved) {
-               // prevent the temporary draw node from appearing on touch devices
-               removeDrawNode();
-             }
+           var tilemapUrl = dummyUrl.replace(/tile\/[0-9]+\/[0-9]+\/[0-9]+\?blankTile=false/, 'tilemap') + '/' + z + '/' + y + '/' + x + '/8/8'; // make the request and introspect the response from the tilemap server
 
-             dispatch$1.call('rejectedSelfIntersection', this);
-             return; // can't click here
-           }
+           d3_json(tilemapUrl).then(function (tilemap) {
+             if (!tilemap) {
+               throw new Error('Unknown Error');
+             }
 
-           context.pauseChangeDispatch();
-           doAdd(); // we just replaced the temporary edit with the real one
+             var hasTiles = true;
 
-           _didResolveTempEdit = true;
-           context.resumeChangeDispatch();
-           context.enter(mode);
-         } // Accept the current position of the drawing node
+             for (var i = 0; i < tilemap.data.length; i++) {
+               // 0 means an individual tile in the grid doesn't exist
+               if (!tilemap.data[i]) {
+                 hasTiles = false;
+                 break;
+               }
+             } // if any tiles are missing at level 20 we restrict maxZoom to 19
 
 
-         drawWay.add = function (loc, d) {
-           attemptAdd(d, loc, function () {// don't need to do anything extra
+             esri.zoomExtent[1] = hasTiles ? 22 : 19;
+           })["catch"](function () {
+             /* ignore */
            });
-         }; // Connect the way to an existing way
+         };
 
+         esri.getMetadata = function (center, tileCoord, callback) {
+           var tileID = tileCoord.slice(0, 3).join('/');
+           var zoom = Math.min(tileCoord[2], esri.zoomExtent[1]);
+           var centerPoint = center[0] + ',' + center[1]; // long, lat (as it should be)
 
-         drawWay.addWay = function (loc, edge, d) {
-           attemptAdd(d, loc, function () {
-             context.replace(actionAddMidpoint({
-               loc: loc,
-               edge: edge
-             }, _drawNode), _annotation);
-           });
-         }; // Connect the way to an existing node
+           var unknown = _t('info_panels.background.unknown');
+           var metadataLayer;
+           var vintage = {};
+           var metadata = {};
+           if (inflight[tileID]) return;
 
+           switch (true) {
+             case zoom >= 20 && esri.id === 'EsriWorldImageryClarity':
+               metadataLayer = 4;
+               break;
 
-         drawWay.addNode = function (node, d) {
-           // finish drawing if the mapper targets the prior node
-           if (node.id === _headNodeID || // or the first node when drawing an area
-           _origWay.isClosed() && node.id === _origWay.first()) {
-             drawWay.finish();
-             return;
+             case zoom >= 19:
+               metadataLayer = 3;
+               break;
+
+             case zoom >= 17:
+               metadataLayer = 2;
+               break;
+
+             case zoom >= 13:
+               metadataLayer = 0;
+               break;
+
+             default:
+               metadataLayer = 99;
            }
 
-           attemptAdd(d, node.loc, function () {
-             context.replace(function actionReplaceDrawNode(graph) {
-               // remove the temporary draw node and insert the existing node
-               // at the same index
-               graph = graph.replace(graph.entity(wayID).removeNode(_drawNode.id)).remove(_drawNode);
-               return graph.replace(graph.entity(wayID).addNode(node.id, _nodeIndex));
-             }, _annotation);
-           });
-         }; // Finish the draw operation, removing the temporary edit.
-         // If the way has enough nodes to be valid, it's selected.
-         // Otherwise, delete everything and return to browse mode.
+           var url; // build up query using the layer appropriate to the current zoom
 
+           if (esri.id === 'EsriWorldImagery') {
+             url = 'https://services.arcgisonline.com/arcgis/rest/services/World_Imagery/MapServer/';
+           } else if (esri.id === 'EsriWorldImageryClarity') {
+             url = 'https://serviceslab.arcgisonline.com/arcgis/rest/services/Clarity_World_Imagery/MapServer/';
+           }
 
-         drawWay.finish = function () {
-           checkGeometry(false
-           /* includeDrawNode */
-           );
+           url += metadataLayer + '/query?returnGeometry=false&geometry=' + centerPoint + '&inSR=4326&geometryType=esriGeometryPoint&outFields=*&f=json';
 
-           if (context.surface().classed('nope')) {
-             dispatch$1.call('rejectedSelfIntersection', this);
-             return; // can't click here
+           if (!cache[tileID]) {
+             cache[tileID] = {};
            }
 
-           context.pauseChangeDispatch(); // remove the temporary edit
+           if (cache[tileID] && cache[tileID].metadata) {
+             return callback(null, cache[tileID].metadata);
+           } // accurate metadata is only available >= 13
 
-           context.pop(1);
-           _didResolveTempEdit = true;
-           context.resumeChangeDispatch();
-           var way = context.hasEntity(wayID);
 
-           if (!way || way.isDegenerate()) {
-             drawWay.cancel();
-             return;
-           }
+           if (metadataLayer === 99) {
+             vintage = {
+               start: null,
+               end: null,
+               range: null
+             };
+             metadata = {
+               vintage: null,
+               source: unknown,
+               description: unknown,
+               resolution: unknown,
+               accuracy: unknown
+             };
+             callback(null, metadata);
+           } else {
+             inflight[tileID] = true;
+             d3_json(url).then(function (result) {
+               delete inflight[tileID];
 
-           window.setTimeout(function () {
-             context.map().dblclickZoomEnable(true);
-           }, 1000);
-           var isNewFeature = !mode.isContinuing;
-           context.enter(modeSelect(context, [wayID]).newFeature(isNewFeature));
-         }; // Cancel the draw operation, delete everything, and return to browse mode.
+               if (!result) {
+                 throw new Error('Unknown Error');
+               } else if (result.features && result.features.length < 1) {
+                 throw new Error('No Results');
+               } else if (result.error && result.error.message) {
+                 throw new Error(result.error.message);
+               } // pass through the discrete capture date from metadata
 
 
-         drawWay.cancel = function () {
-           context.pauseChangeDispatch();
-           resetToStartGraph();
-           context.resumeChangeDispatch();
-           window.setTimeout(function () {
-             context.map().dblclickZoomEnable(true);
-           }, 1000);
-           context.surface().classed('nope', false).classed('nope-disabled', false).classed('nope-suppressed', false);
-           context.enter(modeBrowse(context));
-         };
+               var captureDate = localeDateString(result.features[0].attributes.SRC_DATE2);
+               vintage = {
+                 start: captureDate,
+                 end: captureDate,
+                 range: captureDate
+               };
+               metadata = {
+                 vintage: vintage,
+                 source: clean(result.features[0].attributes.NICE_NAME),
+                 description: clean(result.features[0].attributes.NICE_DESC),
+                 resolution: clean(+parseFloat(result.features[0].attributes.SRC_RES).toFixed(4)),
+                 accuracy: clean(+parseFloat(result.features[0].attributes.SRC_ACC).toFixed(4))
+               }; // append units - meters
 
-         drawWay.nodeIndex = function (val) {
-           if (!arguments.length) return _nodeIndex;
-           _nodeIndex = val;
-           return drawWay;
-         };
+               if (isFinite(metadata.resolution)) {
+                 metadata.resolution += ' m';
+               }
 
-         drawWay.activeID = function () {
-           if (!arguments.length) return _drawNode && _drawNode.id; // no assign
+               if (isFinite(metadata.accuracy)) {
+                 metadata.accuracy += ' m';
+               }
 
-           return drawWay;
+               cache[tileID].metadata = metadata;
+               if (callback) callback(null, metadata);
+             })["catch"](function (err) {
+               delete inflight[tileID];
+               if (callback) callback(err.message);
+             });
+           }
+
+           function clean(val) {
+             return String(val).trim() || unknown;
+           }
          };
 
-         return utilRebind(drawWay, dispatch$1, 'on');
-       }
+         return esri;
+       };
 
-       function modeDrawLine(context, wayID, startGraph, button, affix, continuing) {
-         var mode = {
-           button: button,
-           id: 'draw-line'
-         };
-         var behavior = behaviorDrawWay(context, wayID, mode, startGraph).on('rejectedSelfIntersection.modeDrawLine', function () {
-           context.ui().flash.iconName('#iD-icon-no').label(_t('self_intersection.error.lines'))();
+       rendererBackgroundSource.None = function () {
+         var source = rendererBackgroundSource({
+           id: 'none',
+           template: ''
          });
-         mode.wayID = wayID;
-         mode.isContinuing = continuing;
 
-         mode.enter = function () {
-           behavior.nodeIndex(affix === 'prefix' ? 0 : undefined);
-           context.install(behavior);
+         source.name = function () {
+           return _t('background.none');
          };
 
-         mode.exit = function () {
-           context.uninstall(behavior);
+         source.label = function () {
+           return _t.html('background.none');
          };
 
-         mode.selectedIDs = function () {
-           return [wayID];
+         source.imageryUsed = function () {
+           return null;
          };
 
-         mode.activeID = function () {
-           return behavior && behavior.activeID() || [];
+         source.area = function () {
+           return -1; // sources in background pane are sorted by area
          };
 
-         return mode;
-       }
+         return source;
+       };
 
-       function operationContinue(context, selectedIDs) {
-         var _entities = selectedIDs.map(function (id) {
-           return context.graph().entity(id);
+       rendererBackgroundSource.Custom = function (template) {
+         var source = rendererBackgroundSource({
+           id: 'custom',
+           template: template
          });
 
-         var _geometries = Object.assign({
-           line: [],
-           vertex: []
-         }, utilArrayGroupBy(_entities, function (entity) {
-           return entity.geometry(context.graph());
-         }));
+         source.name = function () {
+           return _t('background.custom');
+         };
 
-         var _vertex = _geometries.vertex.length && _geometries.vertex[0];
+         source.label = function () {
+           return _t.html('background.custom');
+         };
 
-         function candidateWays() {
-           return _vertex ? context.graph().parentWays(_vertex).filter(function (parent) {
-             return parent.geometry(context.graph()) === 'line' && !parent.isClosed() && parent.affix(_vertex.id) && (_geometries.line.length === 0 || _geometries.line[0] === parent);
-           }) : [];
-         }
+         source.imageryUsed = function () {
+           // sanitize personal connection tokens - #6801
+           var cleaned = source.template(); // from query string parameters
 
-         var _candidates = candidateWays();
+           if (cleaned.indexOf('?') !== -1) {
+             var parts = cleaned.split('?', 2);
+             var qs = utilStringQs(parts[1]);
+             ['access_token', 'connectId', 'token'].forEach(function (param) {
+               if (qs[param]) {
+                 qs[param] = '{apikey}';
+               }
+             });
+             cleaned = parts[0] + '?' + utilQsString(qs, true); // true = soft encode
+           } // from wms/wmts api path parameters
 
-         var operation = function operation() {
-           var candidate = _candidates[0];
-           context.enter(modeDrawLine(context, candidate.id, context.graph(), 'line', candidate.affix(_vertex.id), true));
-         };
 
-         operation.relatedEntityIds = function () {
-           return _candidates.length ? [_candidates[0].id] : [];
+           cleaned = cleaned.replace(/token\/(\w+)/, 'token/{apikey}');
+           return 'Custom (' + cleaned + ' )';
          };
 
-         operation.available = function () {
-           return _geometries.vertex.length === 1 && _geometries.line.length <= 1 && !context.features().hasHiddenConnections(_vertex, context.graph());
+         source.area = function () {
+           return -2; // sources in background pane are sorted by area
          };
 
-         operation.disabled = function () {
-           if (_candidates.length === 0) {
-             return 'not_eligible';
-           } else if (_candidates.length > 1) {
-             return 'multiple';
-           }
+         return source;
+       };
 
-           return false;
-         };
+       function rendererTileLayer(context) {
+         var transformProp = utilPrefixCSSProperty('Transform');
+         var tiler = utilTiler();
+         var _tileSize = 256;
 
-         operation.tooltip = function () {
-           var disable = operation.disabled();
-           return disable ? _t('operations.continue.' + disable) : _t('operations.continue.description');
-         };
+         var _projection;
 
-         operation.annotation = function () {
-           return _t('operations.continue.annotation.line');
-         };
+         var _cache = {};
 
-         operation.id = 'continue';
-         operation.keys = [_t('operations.continue.key')];
-         operation.title = _t('operations.continue.title');
-         operation.behavior = behaviorOperation(context).which(operation);
-         return operation;
-       }
+         var _tileOrigin;
 
-       function operationCopy(context, selectedIDs) {
-         function getFilteredIdsToCopy() {
-           return selectedIDs.filter(function (selectedID) {
-             var entity = context.graph().hasEntity(selectedID); // don't copy untagged vertices separately from ways
+         var _zoom;
 
-             return entity.hasInterestingTags() || entity.geometry(context.graph()) !== 'vertex';
-           });
+         var _source;
+
+         function tileSizeAtZoom(d, z) {
+           var EPSILON = 0.002; // close seams
+
+           return _tileSize * Math.pow(2, z - d[2]) / _tileSize + EPSILON;
          }
 
-         var operation = function operation() {
-           var graph = context.graph();
-           var selected = groupEntities(getFilteredIdsToCopy(), graph);
-           var canCopy = [];
-           var skip = {};
-           var entity;
-           var i;
+         function atZoom(t, distance) {
+           var power = Math.pow(2, distance);
+           return [Math.floor(t[0] * power), Math.floor(t[1] * power), t[2] + distance];
+         }
 
-           for (i = 0; i < selected.relation.length; i++) {
-             entity = selected.relation[i];
+         function lookUp(d) {
+           for (var up = -1; up > -d[2]; up--) {
+             var tile = atZoom(d, up);
 
-             if (!skip[entity.id] && entity.isComplete(graph)) {
-               canCopy.push(entity.id);
-               skip = getDescendants(entity.id, graph, skip);
+             if (_cache[_source.url(tile)] !== false) {
+               return tile;
              }
            }
+         }
 
-           for (i = 0; i < selected.way.length; i++) {
-             entity = selected.way[i];
+         function uniqueBy(a, n) {
+           var o = [];
+           var seen = {};
 
-             if (!skip[entity.id]) {
-               canCopy.push(entity.id);
-               skip = getDescendants(entity.id, graph, skip);
+           for (var i = 0; i < a.length; i++) {
+             if (seen[a[i][n]] === undefined) {
+               o.push(a[i]);
+               seen[a[i][n]] = true;
              }
            }
 
-           for (i = 0; i < selected.node.length; i++) {
-             entity = selected.node[i];
+           return o;
+         }
 
-             if (!skip[entity.id]) {
-               canCopy.push(entity.id);
-             }
-           }
+         function addSource(d) {
+           d.push(_source.url(d));
+           return d;
+         } // Update tiles based on current state of `projection`.
 
-           context.copyIDs(canCopy);
 
-           if (_point && (canCopy.length !== 1 || graph.entity(canCopy[0]).type !== 'node')) {
-             // store the anchor coordinates if copying more than a single node
-             context.copyLonLat(context.projection.invert(_point));
+         function background(selection) {
+           _zoom = geoScaleToZoom(_projection.scale(), _tileSize);
+           var pixelOffset;
+
+           if (_source) {
+             pixelOffset = [_source.offset()[0] * Math.pow(2, _zoom), _source.offset()[1] * Math.pow(2, _zoom)];
            } else {
-             context.copyLonLat(null);
+             pixelOffset = [0, 0];
            }
-         };
 
-         function groupEntities(ids, graph) {
-           var entities = ids.map(function (id) {
-             return graph.entity(id);
-           });
-           return Object.assign({
-             relation: [],
-             way: [],
-             node: []
-           }, utilArrayGroupBy(entities, 'type'));
-         }
+           var translate = [_projection.translate()[0] + pixelOffset[0], _projection.translate()[1] + pixelOffset[1]];
+           tiler.scale(_projection.scale() * 2 * Math.PI).translate(translate);
+           _tileOrigin = [_projection.scale() * Math.PI - translate[0], _projection.scale() * Math.PI - translate[1]];
+           render(selection);
+         } // Derive the tiles onscreen, remove those offscreen and position them.
+         // Important that this part not depend on `_projection` because it's
+         // rentered when tiles load/error (see #644).
 
-         function getDescendants(id, graph, descendants) {
-           var entity = graph.entity(id);
-           var children;
-           descendants = descendants || {};
 
-           if (entity.type === 'relation') {
-             children = entity.members.map(function (m) {
-               return m.id;
+         function render(selection) {
+           if (!_source) return;
+           var requests = [];
+           var showDebug = context.getDebug('tile') && !_source.overlay;
+
+           if (_source.validZoom(_zoom)) {
+             tiler.skipNullIsland(!!_source.overlay);
+             tiler().forEach(function (d) {
+               addSource(d);
+               if (d[3] === '') return;
+               if (typeof d[3] !== 'string') return; // Workaround for #2295
+
+               requests.push(d);
+
+               if (_cache[d[3]] === false && lookUp(d)) {
+                 requests.push(addSource(lookUp(d)));
+               }
+             });
+             requests = uniqueBy(requests, 3).filter(function (r) {
+               // don't re-request tiles which have failed in the past
+               return _cache[r[3]] !== false;
              });
-           } else if (entity.type === 'way') {
-             children = entity.nodes;
-           } else {
-             children = [];
            }
 
-           for (var i = 0; i < children.length; i++) {
-             if (!descendants[children[i]]) {
-               descendants[children[i]] = true;
-               descendants = getDescendants(children[i], graph, descendants);
-             }
+           function load(d3_event, d) {
+             _cache[d[3]] = true;
+             select(this).on('error', null).on('load', null).classed('tile-loaded', true);
+             render(selection);
+           }
+
+           function error(d3_event, d) {
+             _cache[d[3]] = false;
+             select(this).on('error', null).on('load', null).remove();
+             render(selection);
            }
 
-           return descendants;
-         }
+           function imageTransform(d) {
+             var ts = _tileSize * Math.pow(2, _zoom - d[2]);
 
-         operation.available = function () {
-           return getFilteredIdsToCopy().length > 0;
-         };
+             var scale = tileSizeAtZoom(d, _zoom);
+             return 'translate(' + (d[0] * ts - _tileOrigin[0]) + 'px,' + (d[1] * ts - _tileOrigin[1]) + 'px) ' + 'scale(' + scale + ',' + scale + ')';
+           }
 
-         operation.disabled = function () {
-           var extent = utilTotalExtent(getFilteredIdsToCopy(), context.graph());
+           function tileCenter(d) {
+             var ts = _tileSize * Math.pow(2, _zoom - d[2]);
 
-           if (extent.percentContainedIn(context.map().extent()) < 0.8) {
-             return 'too_large';
+             return [d[0] * ts - _tileOrigin[0] + ts / 2, d[1] * ts - _tileOrigin[1] + ts / 2];
            }
 
-           return false;
-         };
+           function debugTransform(d) {
+             var coord = tileCenter(d);
+             return 'translate(' + coord[0] + 'px,' + coord[1] + 'px)';
+           } // Pick a representative tile near the center of the viewport
+           // (This is useful for sampling the imagery vintage)
 
-         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();
-         };
+           var dims = tiler.size();
+           var mapCenter = [dims[0] / 2, dims[1] / 2];
+           var minDist = Math.max(dims[0], dims[1]);
+           var nearCenter;
+           requests.forEach(function (d) {
+             var c = tileCenter(d);
+             var dist = geoVecLength(c, mapCenter);
 
-         operation.tooltip = function () {
-           var disable = operation.disabled();
-           return disable ? _t('operations.copy.' + disable, {
-             n: selectedIDs.length
-           }) : _t('operations.copy.description', {
-             n: selectedIDs.length
+             if (dist < minDist) {
+               minDist = dist;
+               nearCenter = d;
+             }
            });
-         };
-
-         operation.annotation = function () {
-           return _t('operations.copy.annotation', {
-             n: selectedIDs.length
+           var image = selection.selectAll('img').data(requests, function (d) {
+             return d[3];
            });
-         };
+           image.exit().style(transformProp, imageTransform).classed('tile-removing', true).classed('tile-center', false).each(function () {
+             var tile = select(this);
+             window.setTimeout(function () {
+               if (tile.classed('tile-removing')) {
+                 tile.remove();
+               }
+             }, 300);
+           });
+           image.enter().append('img').attr('class', 'tile').attr('draggable', 'false').style('width', _tileSize + 'px').style('height', _tileSize + 'px').attr('src', function (d) {
+             return d[3];
+           }).on('error', error).on('load', load).merge(image).style(transformProp, imageTransform).classed('tile-debug', showDebug).classed('tile-removing', false).classed('tile-center', function (d) {
+             return d === nearCenter;
+           });
+           var debug = selection.selectAll('.tile-label-debug').data(showDebug ? requests : [], function (d) {
+             return d[3];
+           });
+           debug.exit().remove();
 
-         var _point;
+           if (showDebug) {
+             var debugEnter = debug.enter().append('div').attr('class', 'tile-label-debug');
+             debugEnter.append('div').attr('class', 'tile-label-debug-coord');
+             debugEnter.append('div').attr('class', 'tile-label-debug-vintage');
+             debug = debug.merge(debugEnter);
+             debug.style(transformProp, debugTransform);
+             debug.selectAll('.tile-label-debug-coord').html(function (d) {
+               return d[2] + ' / ' + d[0] + ' / ' + d[1];
+             });
+             debug.selectAll('.tile-label-debug-vintage').each(function (d) {
+               var span = select(this);
+               var center = context.projection.invert(tileCenter(d));
 
-         operation.point = function (val) {
-           _point = val;
-           return operation;
-         };
+               _source.getMetadata(center, d, function (err, result) {
+                 span.html(result && result.vintage && result.vintage.range || _t('info_panels.background.vintage') + ': ' + _t('info_panels.background.unknown'));
+               });
+             });
+           }
+         }
 
-         operation.id = 'copy';
-         operation.keys = [uiCmd('⌘C')];
-         operation.title = _t('operations.copy.title');
-         operation.behavior = behaviorOperation(context).which(operation);
-         return operation;
-       }
+         background.projection = function (val) {
+           if (!arguments.length) return _projection;
+           _projection = val;
+           return background;
+         };
 
-       function operationDisconnect(context, selectedIDs) {
-         var _vertexIDs = [];
-         var _wayIDs = [];
-         var _otherIDs = [];
-         var _actions = [];
-         selectedIDs.forEach(function (id) {
-           var entity = context.entity(id);
+         background.dimensions = function (val) {
+           if (!arguments.length) return tiler.size();
+           tiler.size(val);
+           return background;
+         };
 
-           if (entity.type === 'way') {
-             _wayIDs.push(id);
-           } else if (entity.geometry(context.graph()) === 'vertex') {
-             _vertexIDs.push(id);
-           } else {
-             _otherIDs.push(id);
-           }
-         });
+         background.source = function (val) {
+           if (!arguments.length) return _source;
+           _source = val;
+           _tileSize = _source.tileSize;
+           _cache = {};
+           tiler.tileSize(_source.tileSize).zoomExtent(_source.zoomExtent);
+           return background;
+         };
 
-         var _coords,
-             _descriptionID = '',
-             _annotationID = 'features';
+         return background;
+       }
 
-         var _disconnectingVertexIds = [];
-         var _disconnectingWayIds = [];
+       var _imageryIndex = null;
+       function rendererBackground(context) {
+         var dispatch = dispatch$8('change');
+         var detected = utilDetect();
+         var baseLayer = rendererTileLayer(context).projection(context.projection);
+         var _isValid = true;
+         var _overlayLayers = [];
+         var _brightness = 1;
+         var _contrast = 1;
+         var _saturation = 1;
+         var _sharpness = 1;
 
-         if (_vertexIDs.length > 0) {
-           // At the selected vertices, disconnect the selected ways, if any, else
-           // disconnect all connected ways
-           _disconnectingVertexIds = _vertexIDs;
+         function ensureImageryIndex() {
+           return _mainFileFetcher.get('imagery').then(function (sources) {
+             if (_imageryIndex) return _imageryIndex;
+             _imageryIndex = {
+               imagery: sources,
+               features: {}
+             }; // use which-polygon to support efficient index and querying for imagery
 
-           _vertexIDs.forEach(function (vertexID) {
-             var action = actionDisconnect(vertexID);
+             var features = sources.map(function (source) {
+               if (!source.polygon) return null; // workaround for editor-layer-index weirdness..
+               // Add an extra array nest to each element in `source.polygon`
+               // so the rings are not treated as a bunch of holes:
+               // what we have: [ [[outer],[hole],[hole]] ]
+               // what we want: [ [[outer]],[[outer]],[[outer]] ]
 
-             if (_wayIDs.length > 0) {
-               var waysIDsForVertex = _wayIDs.filter(function (wayID) {
-                 var way = context.entity(wayID);
-                 return way.nodes.indexOf(vertexID) !== -1;
+               var rings = source.polygon.map(function (ring) {
+                 return [ring];
                });
+               var feature = {
+                 type: 'Feature',
+                 properties: {
+                   id: source.id
+                 },
+                 geometry: {
+                   type: 'MultiPolygon',
+                   coordinates: rings
+                 }
+               };
+               _imageryIndex.features[source.id] = feature;
+               return feature;
+             }).filter(Boolean);
+             _imageryIndex.query = whichPolygon_1({
+               type: 'FeatureCollection',
+               features: features
+             }); // Instantiate `rendererBackgroundSource` objects for each source
 
-               action.limitWays(waysIDsForVertex);
-             }
+             _imageryIndex.backgrounds = sources.map(function (source) {
+               if (source.type === 'bing') {
+                 return rendererBackgroundSource.Bing(source, dispatch);
+               } else if (/^EsriWorldImagery/.test(source.id)) {
+                 return rendererBackgroundSource.Esri(source);
+               } else {
+                 return rendererBackgroundSource(source);
+               }
+             }); // Add 'None'
 
-             _actions.push(action);
+             _imageryIndex.backgrounds.unshift(rendererBackgroundSource.None()); // Add 'Custom'
 
-             _disconnectingWayIds = _disconnectingWayIds.concat(context.graph().parentWays(context.graph().entity(vertexID)).map(function (d) {
-               return d.id;
-             }));
-           });
 
-           _disconnectingWayIds = utilArrayUniq(_disconnectingWayIds).filter(function (id) {
-             return _wayIDs.indexOf(id) === -1;
-           });
-           _descriptionID += _actions.length === 1 ? 'single_point.' : 'multiple_points.';
+             var template = corePreferences('background-custom-template') || '';
+             var custom = rendererBackgroundSource.Custom(template);
 
-           if (_wayIDs.length === 1) {
-             _descriptionID += 'single_way.' + context.graph().geometry(_wayIDs[0]);
-           } else {
-             _descriptionID += _wayIDs.length === 0 ? 'no_ways' : 'multiple_ways';
-           }
-         } else if (_wayIDs.length > 0) {
-           // Disconnect the selected ways from each other, if they're connected,
-           // else disconnect them from all connected ways
-           var ways = _wayIDs.map(function (id) {
-             return context.entity(id);
+             _imageryIndex.backgrounds.unshift(custom);
+
+             return _imageryIndex;
            });
+         }
 
-           var nodes = utilGetAllNodes(_wayIDs, context.graph());
-           _coords = nodes.map(function (n) {
-             return n.loc;
-           }); // actions for connected nodes shared by at least two selected ways
+         function background(selection) {
+           var currSource = baseLayer.source(); // If we are displaying an Esri basemap at high zoom,
+           // check its tilemap to see how high the zoom can go
 
-           var sharedActions = [];
-           var sharedNodes = []; // actions for connected nodes
+           if (context.map().zoom() > 18) {
+             if (currSource && /^EsriWorldImagery/.test(currSource.id)) {
+               var center = context.map().center();
+               currSource.fetchTilemap(center);
+             }
+           } // Is the imagery valid here? - #4827
 
-           var unsharedActions = [];
-           var unsharedNodes = [];
-           nodes.forEach(function (node) {
-             var action = actionDisconnect(node.id).limitWays(_wayIDs);
 
-             if (action.disabled(context.graph()) !== 'not_connected') {
-               var count = 0;
+           var sources = background.sources(context.map().extent());
+           var wasValid = _isValid;
+           _isValid = !!sources.filter(function (d) {
+             return d === currSource;
+           }).length;
 
-               for (var i in ways) {
-                 var way = ways[i];
+           if (wasValid !== _isValid) {
+             // change in valid status
+             background.updateImagery();
+           }
 
-                 if (way.nodes.indexOf(node.id) !== -1) {
-                   count += 1;
-                 }
+           var baseFilter = '';
 
-                 if (count > 1) break;
-               }
+           if (detected.cssfilters) {
+             if (_brightness !== 1) {
+               baseFilter += " brightness(".concat(_brightness, ")");
+             }
 
-               if (count > 1) {
-                 sharedActions.push(action);
-                 sharedNodes.push(node);
-               } else {
-                 unsharedActions.push(action);
-                 unsharedNodes.push(node);
-               }
+             if (_contrast !== 1) {
+               baseFilter += " contrast(".concat(_contrast, ")");
              }
-           });
-           _descriptionID += 'no_points.';
-           _descriptionID += _wayIDs.length === 1 ? 'single_way.' : 'multiple_ways.';
 
-           if (sharedActions.length) {
-             // if any nodes are shared, only disconnect the selected ways from each other
-             _actions = sharedActions;
-             _disconnectingVertexIds = sharedNodes.map(function (node) {
-               return node.id;
-             });
-             _descriptionID += 'conjoined';
-             _annotationID = 'from_each_other';
-           } else {
-             // if no nodes are shared, disconnect the selected ways from all connected ways
-             _actions = unsharedActions;
-             _disconnectingVertexIds = unsharedNodes.map(function (node) {
-               return node.id;
-             });
+             if (_saturation !== 1) {
+               baseFilter += " saturate(".concat(_saturation, ")");
+             }
 
-             if (_wayIDs.length === 1) {
-               _descriptionID += context.graph().geometry(_wayIDs[0]);
-             } else {
-               _descriptionID += 'separate';
+             if (_sharpness < 1) {
+               // gaussian blur
+               var blur = d3_interpolateNumber(0.5, 5)(1 - _sharpness);
+               baseFilter += " blur(".concat(blur, "px)");
              }
            }
-         }
 
-         var _extent = utilTotalExtent(_disconnectingVertexIds, context.graph());
-
-         var operation = function operation() {
-           context.perform(function (graph) {
-             return _actions.reduce(function (graph, action) {
-               return action(graph);
-             }, graph);
-           }, operation.annotation());
-           context.validator().validate();
-         };
+           var base = selection.selectAll('.layer-background').data([0]);
+           base = base.enter().insert('div', '.layer-data').attr('class', 'layer layer-background').merge(base);
 
-         operation.relatedEntityIds = function () {
-           if (_vertexIDs.length) {
-             return _disconnectingWayIds;
+           if (detected.cssfilters) {
+             base.style('filter', baseFilter || null);
+           } else {
+             base.style('opacity', _brightness);
            }
 
-           return _disconnectingVertexIds;
-         };
-
-         operation.available = function () {
-           if (_actions.length === 0) return false;
-           if (_otherIDs.length !== 0) return false;
-           if (_vertexIDs.length !== 0 && _wayIDs.length !== 0 && !_wayIDs.every(function (wayID) {
-             return _vertexIDs.some(function (vertexID) {
-               var way = context.entity(wayID);
-               return way.nodes.indexOf(vertexID) !== -1;
-             });
-           })) return false;
-           return true;
-         };
-
-         operation.disabled = function () {
-           var reason;
-
-           for (var actionIndex in _actions) {
-             reason = _actions[actionIndex].disabled(context.graph());
-             if (reason) return reason;
-           }
+           var imagery = base.selectAll('.layer-imagery').data([0]);
+           imagery.enter().append('div').attr('class', 'layer layer-imagery').merge(imagery).call(baseLayer);
+           var maskFilter = '';
+           var mixBlendMode = '';
 
-           if (_extent && _extent.percentContainedIn(context.map().extent()) < 0.8) {
-             return 'too_large.' + ((_vertexIDs.length ? _vertexIDs : _wayIDs).length === 1 ? 'single' : 'multiple');
-           } else if (_coords && someMissing()) {
-             return 'not_downloaded';
-           } else if (selectedIDs.some(context.hasHiddenConnections)) {
-             return 'connected_to_hidden';
+           if (detected.cssfilters && _sharpness > 1) {
+             // apply unsharp mask
+             mixBlendMode = 'overlay';
+             maskFilter = 'saturate(0) blur(3px) invert(1)';
+             var contrast = _sharpness - 1;
+             maskFilter += " contrast(".concat(contrast, ")");
+             var brightness = d3_interpolateNumber(1, 0.85)(_sharpness - 1);
+             maskFilter += " brightness(".concat(brightness, ")");
            }
 
-           return false;
+           var mask = base.selectAll('.layer-unsharp-mask').data(detected.cssfilters && _sharpness > 1 ? [0] : []);
+           mask.exit().remove();
+           mask.enter().append('div').attr('class', 'layer layer-mask layer-unsharp-mask').merge(mask).call(baseLayer).style('filter', maskFilter || null).style('mix-blend-mode', mixBlendMode || null);
+           var overlays = selection.selectAll('.layer-overlay').data(_overlayLayers, function (d) {
+             return d.source().name();
+           });
+           overlays.exit().remove();
+           overlays.enter().insert('div', '.layer-data').attr('class', 'layer layer-overlay').merge(overlays).each(function (layer, i, nodes) {
+             return select(nodes[i]).call(layer);
+           });
+         }
 
-           function someMissing() {
-             if (context.inIntro()) return false;
-             var osm = context.connection();
+         background.updateImagery = function () {
+           var currSource = baseLayer.source();
+           if (context.inIntro() || !currSource) return;
 
-             if (osm) {
-               var missing = _coords.filter(function (loc) {
-                 return !osm.isDataLoaded(loc);
-               });
+           var o = _overlayLayers.filter(function (d) {
+             return !d.source().isLocatorOverlay() && !d.source().isHidden();
+           }).map(function (d) {
+             return d.source().id;
+           }).join(',');
 
-               if (missing.length) {
-                 missing.forEach(function (loc) {
-                   context.loadTileAtLoc(loc);
-                 });
-                 return true;
-               }
-             }
+           var meters = geoOffsetToMeters(currSource.offset());
+           var EPSILON = 0.01;
+           var x = +meters[0].toFixed(2);
+           var y = +meters[1].toFixed(2);
+           var hash = utilStringQs(window.location.hash);
+           var id = currSource.id;
 
-             return false;
+           if (id === 'custom') {
+             id = "custom:".concat(currSource.template());
            }
-         };
-
-         operation.tooltip = function () {
-           var disable = operation.disabled();
 
-           if (disable) {
-             return _t('operations.disconnect.' + disable);
+           if (id) {
+             hash.background = id;
+           } else {
+             delete hash.background;
            }
 
-           return _t('operations.disconnect.description.' + _descriptionID);
-         };
-
-         operation.annotation = function () {
-           return _t('operations.disconnect.annotation.' + _annotationID);
-         };
-
-         operation.id = 'disconnect';
-         operation.keys = [_t('operations.disconnect.key')];
-         operation.title = _t('operations.disconnect.title');
-         operation.behavior = behaviorOperation(context).which(operation);
-         return operation;
-       }
-
-       function operationDowngrade(context, selectedIDs) {
-         var _affectedFeatureCount = 0;
-
-         var _downgradeType = downgradeTypeForEntityIDs(selectedIDs);
-
-         var _multi = _affectedFeatureCount === 1 ? 'single' : 'multiple';
+           if (o) {
+             hash.overlays = o;
+           } else {
+             delete hash.overlays;
+           }
 
-         function downgradeTypeForEntityIDs(entityIds) {
-           var downgradeType;
-           _affectedFeatureCount = 0;
+           if (Math.abs(x) > EPSILON || Math.abs(y) > EPSILON) {
+             hash.offset = "".concat(x, ",").concat(y);
+           } else {
+             delete hash.offset;
+           }
 
-           for (var i in entityIds) {
-             var entityID = entityIds[i];
-             var type = downgradeTypeForEntityID(entityID);
+           if (!window.mocha) {
+             window.location.replace('#' + utilQsString(hash, true));
+           }
 
-             if (type) {
-               _affectedFeatureCount += 1;
+           var imageryUsed = [];
+           var photoOverlaysUsed = [];
+           var currUsed = currSource.imageryUsed();
 
-               if (downgradeType && type !== downgradeType) {
-                 if (downgradeType !== 'generic' && type !== 'generic') {
-                   downgradeType = 'building_address';
-                 } else {
-                   downgradeType = 'generic';
-                 }
-               } else {
-                 downgradeType = type;
-               }
-             }
+           if (currUsed && _isValid) {
+             imageryUsed.push(currUsed);
            }
 
-           return downgradeType;
-         }
+           _overlayLayers.filter(function (d) {
+             return !d.source().isLocatorOverlay() && !d.source().isHidden();
+           }).forEach(function (d) {
+             return imageryUsed.push(d.source().imageryUsed());
+           });
 
-         function downgradeTypeForEntityID(entityID) {
-           var graph = context.graph();
-           var entity = graph.entity(entityID);
-           var preset = _mainPresetIndex.match(entity, graph);
-           if (!preset || preset.isFallback()) return null;
+           var dataLayer = context.layers().layer('data');
 
-           if (entity.type === 'node' && preset.id !== 'address' && Object.keys(entity.tags).some(function (key) {
-             return key.match(/^addr:.{1,}/);
-           })) {
-             return 'address';
+           if (dataLayer && dataLayer.enabled() && dataLayer.hasData()) {
+             imageryUsed.push(dataLayer.getSrc());
            }
 
-           var geometry = entity.geometry(graph);
+           var photoOverlayLayers = {
+             streetside: 'Bing Streetside',
+             mapillary: 'Mapillary Images',
+             'mapillary-map-features': 'Mapillary Map Features',
+             'mapillary-signs': 'Mapillary Signs',
+             openstreetcam: 'OpenStreetCam Images'
+           };
 
-           if (geometry === 'area' && entity.tags.building && !preset.tags.building) {
-             return 'building';
+           for (var layerID in photoOverlayLayers) {
+             var layer = context.layers().layer(layerID);
+
+             if (layer && layer.enabled()) {
+               photoOverlaysUsed.push(layerID);
+               imageryUsed.push(photoOverlayLayers[layerID]);
+             }
            }
 
-           if (geometry === 'vertex' && Object.keys(entity.tags).length) {
-             return 'generic';
-           }
+           context.history().imageryUsed(imageryUsed);
+           context.history().photoOverlaysUsed(photoOverlaysUsed);
+         };
 
-           return null;
-         }
+         var _checkedBlocklists;
 
-         var buildingKeysToKeep = ['architect', 'building', 'height', 'layer', 'source', 'type', 'wheelchair'];
-         var addressKeysToKeep = ['source'];
+         background.sources = function (extent, zoom, includeCurrent) {
+           if (!_imageryIndex) return []; // called before init()?
 
-         var operation = function operation() {
-           context.perform(function (graph) {
-             for (var i in selectedIDs) {
-               var entityID = selectedIDs[i];
-               var type = downgradeTypeForEntityID(entityID);
-               if (!type) continue;
-               var tags = Object.assign({}, graph.entity(entityID).tags); // shallow copy
+           var visible = {};
+           (_imageryIndex.query.bbox(extent.rectangle(), true) || []).forEach(function (d) {
+             return visible[d.id] = true;
+           });
+           var currSource = baseLayer.source();
+           var osm = context.connection();
+           var blocklists = osm && osm.imageryBlocklists();
 
-               for (var key in tags) {
-                 if (type === 'address' && addressKeysToKeep.indexOf(key) !== -1) continue;
+           if (blocklists && blocklists !== _checkedBlocklists) {
+             _imageryIndex.backgrounds.forEach(function (source) {
+               source.isBlocked = blocklists.some(function (blocklist) {
+                 return blocklist.test(source.template());
+               });
+             });
 
-                 if (type === 'building') {
-                   if (buildingKeysToKeep.indexOf(key) !== -1 || key.match(/^building:.{1,}/) || key.match(/^roof:.{1,}/)) continue;
-                 }
+             _checkedBlocklists = blocklists;
+           }
 
-                 if (type !== 'generic') {
-                   if (key.match(/^addr:.{1,}/) || key.match(/^source:.{1,}/)) continue;
-                 }
+           return _imageryIndex.backgrounds.filter(function (source) {
+             if (includeCurrent && currSource === source) return true; // optionally always include the current imagery
 
-                 delete tags[key];
-               }
+             if (source.isBlocked) return false; // even bundled sources may be blocked - #7905
 
-               graph = actionChangeTags(entityID, tags)(graph);
-             }
+             if (!source.polygon) return true; // always include imagery with worldwide coverage
 
-             return graph;
-           }, operation.annotation());
-           context.validator().validate(); // refresh the select mode to enable the delete operation
+             if (zoom && zoom < 6) return false; // optionally exclude local imagery at low zooms
 
-           context.enter(modeSelect(context, selectedIDs));
+             return visible[source.id]; // include imagery visible in given extent
+           });
          };
 
-         operation.available = function () {
-           return _downgradeType;
+         background.dimensions = function (val) {
+           if (!val) return;
+           baseLayer.dimensions(val);
+
+           _overlayLayers.forEach(function (layer) {
+             return layer.dimensions(val);
+           });
          };
 
-         operation.disabled = function () {
-           if (selectedIDs.some(hasWikidataTag)) {
-             return 'has_wikidata_tag';
-           }
+         background.baseLayerSource = function (d) {
+           if (!arguments.length) return baseLayer.source(); // test source against OSM imagery blocklists..
 
-           return false;
+           var osm = context.connection();
+           if (!osm) return background;
+           var blocklists = osm.imageryBlocklists();
+           var template = d.template();
+           var fail = false;
+           var tested = 0;
+           var regex;
 
-           function hasWikidataTag(id) {
-             var entity = context.entity(id);
-             return entity.tags.wikidata && entity.tags.wikidata.trim().length > 0;
+           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 = /.*\.google(apis)?\..*\/(vt|kh)[\?\/].*([xyz]=.*){3}.*/;
+             fail = regex.test(template);
            }
+
+           baseLayer.source(!fail ? d : background.findSource('none'));
+           dispatch.call('change');
+           background.updateImagery();
+           return background;
          };
 
-         operation.tooltip = function () {
-           var disable = operation.disabled();
-           return disable ? _t('operations.downgrade.' + disable + '.' + _multi) : _t('operations.downgrade.description.' + _downgradeType);
+         background.findSource = function (id) {
+           if (!id || !_imageryIndex) return null; // called before init()?
+
+           return _imageryIndex.backgrounds.find(function (d) {
+             return d.id && d.id === id;
+           });
          };
 
-         operation.annotation = function () {
-           var suffix;
+         background.bing = function () {
+           background.baseLayerSource(background.findSource('Bing'));
+         };
 
-           if (_downgradeType === 'building_address') {
-             suffix = 'generic';
-           } else {
-             suffix = _downgradeType;
-           }
+         background.showsLayer = function (d) {
+           var currSource = baseLayer.source();
+           if (!d || !currSource) return false;
+           return d.id === currSource.id || _overlayLayers.some(function (layer) {
+             return d.id === layer.source().id;
+           });
+         };
 
-           return _t('operations.downgrade.annotation.' + suffix, {
-             n: _affectedFeatureCount
+         background.overlayLayerSources = function () {
+           return _overlayLayers.map(function (layer) {
+             return layer.source();
            });
          };
 
-         operation.id = 'downgrade';
-         operation.keys = [uiCmd('⌫')];
-         operation.title = _t('operations.downgrade.title');
-         operation.behavior = behaviorOperation(context).which(operation);
-         return operation;
-       }
+         background.toggleOverlayLayer = function (d) {
+           var layer;
 
-       function operationExtract(context, selectedIDs) {
-         var _amount = selectedIDs.length === 1 ? 'single' : 'multiple';
+           for (var i = 0; i < _overlayLayers.length; i++) {
+             layer = _overlayLayers[i];
 
-         var _geometries = utilArrayUniq(selectedIDs.map(function (entityID) {
-           return context.graph().hasEntity(entityID) && context.graph().geometry(entityID);
-         }).filter(Boolean));
+             if (layer.source() === d) {
+               _overlayLayers.splice(i, 1);
 
-         var _geometryID = _geometries.length === 1 ? _geometries[0] : 'feature';
+               dispatch.call('change');
+               background.updateImagery();
+               return;
+             }
+           }
 
-         var _extent;
+           layer = rendererTileLayer(context).source(d).projection(context.projection).dimensions(baseLayer.dimensions());
 
-         var _actions = selectedIDs.map(function (entityID) {
-           var graph = context.graph();
-           var entity = graph.hasEntity(entityID);
-           if (!entity || !entity.hasInterestingTags()) return null;
-           if (entity.type === 'node' && graph.parentWays(entity).length === 0) return null;
+           _overlayLayers.push(layer);
 
-           if (entity.type !== 'node') {
-             var preset = _mainPresetIndex.match(entity, graph); // only allow extraction from ways/relations if the preset supports points
+           dispatch.call('change');
+           background.updateImagery();
+         };
 
-             if (preset.geometry.indexOf('point') === -1) return null;
+         background.nudge = function (d, zoom) {
+           var currSource = baseLayer.source();
+
+           if (currSource) {
+             currSource.nudge(d, zoom);
+             dispatch.call('change');
+             background.updateImagery();
            }
 
-           _extent = _extent ? _extent.extend(entity.extent(graph)) : entity.extent(graph);
-           return actionExtract(entityID);
-         }).filter(Boolean);
+           return background;
+         };
 
-         var operation = function operation() {
-           var combinedAction = function combinedAction(graph) {
-             _actions.forEach(function (action) {
-               graph = action(graph);
-             });
+         background.offset = function (d) {
+           var currSource = baseLayer.source();
 
-             return graph;
-           };
+           if (!arguments.length) {
+             return currSource && currSource.offset() || [0, 0];
+           }
 
-           context.perform(combinedAction, operation.annotation()); // do the extract
+           if (currSource) {
+             currSource.offset(d);
+             dispatch.call('change');
+             background.updateImagery();
+           }
 
-           var extractedNodeIDs = _actions.map(function (action) {
-             return action.getExtractedNodeID();
-           });
+           return background;
+         };
 
-           context.enter(modeSelect(context, extractedNodeIDs));
+         background.brightness = function (d) {
+           if (!arguments.length) return _brightness;
+           _brightness = d;
+           if (context.mode()) dispatch.call('change');
+           return background;
          };
 
-         operation.available = function () {
-           return _actions.length && selectedIDs.length === _actions.length;
+         background.contrast = function (d) {
+           if (!arguments.length) return _contrast;
+           _contrast = d;
+           if (context.mode()) dispatch.call('change');
+           return background;
          };
 
-         operation.disabled = function () {
-           if (_extent && _extent.percentContainedIn(context.map().extent()) < 0.8) {
-             return 'too_large';
-           } else if (selectedIDs.some(function (entityID) {
-             return context.graph().geometry(entityID) === 'vertex' && context.hasHiddenConnections(entityID);
-           })) {
-             return 'connected_to_hidden';
-           }
+         background.saturation = function (d) {
+           if (!arguments.length) return _saturation;
+           _saturation = d;
+           if (context.mode()) dispatch.call('change');
+           return background;
+         };
 
-           return false;
+         background.sharpness = function (d) {
+           if (!arguments.length) return _sharpness;
+           _sharpness = d;
+           if (context.mode()) dispatch.call('change');
+           return background;
          };
 
-         operation.tooltip = function () {
-           var disableReason = operation.disabled();
+         var _loadPromise;
 
-           if (disableReason) {
-             return _t('operations.extract.' + disableReason + '.' + _amount);
-           } else {
-             return _t('operations.extract.description.' + _geometryID + '.' + _amount);
+         background.ensureLoaded = function () {
+           if (_loadPromise) return _loadPromise;
+
+           function parseMapParams(qmap) {
+             if (!qmap) return false;
+             var params = qmap.split('/').map(Number);
+             if (params.length < 3 || params.some(isNaN)) return false;
+             return geoExtent([params[2], params[1]]); // lon,lat
            }
-         };
 
-         operation.annotation = function () {
-           return _t('operations.extract.annotation', {
-             n: selectedIDs.length
-           });
-         };
+           var hash = utilStringQs(window.location.hash);
+           var requested = hash.background || hash.layer;
+           var extent = parseMapParams(hash.map);
+           return _loadPromise = ensureImageryIndex().then(function (imageryIndex) {
+             var first = imageryIndex.backgrounds.length && imageryIndex.backgrounds[0];
+             var best;
 
-         operation.id = 'extract';
-         operation.keys = [_t('operations.extract.key')];
-         operation.title = _t('operations.extract.title');
-         operation.behavior = behaviorOperation(context).which(operation);
-         return operation;
-       }
+             if (!requested && extent) {
+               best = background.sources(extent).find(function (s) {
+                 return s.best();
+               });
+             } // Decide which background layer to display
 
-       function operationMerge(context, selectedIDs) {
-         var _action = getAction();
 
-         function getAction() {
-           // prefer a non-disabled action first
-           var join = actionJoin(selectedIDs);
-           if (!join.disabled(context.graph())) return join;
-           var merge = actionMerge(selectedIDs);
-           if (!merge.disabled(context.graph())) return merge;
-           var mergePolygon = actionMergePolygon(selectedIDs);
-           if (!mergePolygon.disabled(context.graph())) return mergePolygon;
-           var mergeNodes = actionMergeNodes(selectedIDs);
-           if (!mergeNodes.disabled(context.graph())) return mergeNodes; // otherwise prefer an action with an interesting disabled reason
+             if (requested && requested.indexOf('custom:') === 0) {
+               var template = requested.replace(/^custom:/, '');
+               var custom = background.findSource('custom');
+               background.baseLayerSource(custom.template(template));
+               corePreferences('background-custom-template', template);
+             } else {
+               background.baseLayerSource(background.findSource(requested) || best || background.findSource(corePreferences('background-last-used')) || background.findSource('Bing') || first || background.findSource('none'));
+             }
 
-           if (join.disabled(context.graph()) !== 'not_eligible') return join;
-           if (merge.disabled(context.graph()) !== 'not_eligible') return merge;
-           if (mergePolygon.disabled(context.graph()) !== 'not_eligible') return mergePolygon;
-           return mergeNodes;
-         }
+             var locator = imageryIndex.backgrounds.find(function (d) {
+               return d.overlay && d["default"];
+             });
 
-         var operation = function operation() {
-           if (operation.disabled()) return;
-           context.perform(_action, operation.annotation());
-           context.validator().validate();
-           var resultIDs = selectedIDs.filter(context.hasEntity);
+             if (locator) {
+               background.toggleOverlayLayer(locator);
+             }
 
-           if (resultIDs.length > 1) {
-             var interestingIDs = resultIDs.filter(function (id) {
-               return context.entity(id).hasInterestingTags();
+             var overlays = (hash.overlays || '').split(',');
+             overlays.forEach(function (overlay) {
+               overlay = background.findSource(overlay);
+
+               if (overlay) {
+                 background.toggleOverlayLayer(overlay);
+               }
              });
-             if (interestingIDs.length) resultIDs = interestingIDs;
-           }
 
-           context.enter(modeSelect(context, resultIDs));
-         };
+             if (hash.gpx) {
+               var gpx = context.layers().layer('data');
 
-         operation.available = function () {
-           return selectedIDs.length >= 2;
+               if (gpx) {
+                 gpx.url(hash.gpx, '.gpx');
+               }
+             }
+
+             if (hash.offset) {
+               var offset = hash.offset.replace(/;/g, ',').split(',').map(function (n) {
+                 return !isNaN(n) && n;
+               });
+
+               if (offset.length === 2) {
+                 background.offset(geoMetersToOffset(offset));
+               }
+             }
+           })["catch"](function () {
+             /* ignore */
+           });
          };
 
-         operation.disabled = function () {
-           var actionDisabled = _action.disabled(context.graph());
+         return utilRebind(background, dispatch, 'on');
+       }
 
-           if (actionDisabled) return actionDisabled;
-           var osm = context.connection();
+       function rendererFeatures(context) {
+         var dispatch = dispatch$8('change', 'redraw');
+         var features = utilRebind({}, dispatch, 'on');
 
-           if (osm && _action.resultingWayNodesLength && _action.resultingWayNodesLength(context.graph()) > osm.maxWayNodes()) {
-             return 'too_many_vertices';
-           }
+         var _deferred = new Set();
 
-           return false;
+         var traffic_roads = {
+           'motorway': true,
+           'motorway_link': true,
+           'trunk': true,
+           'trunk_link': true,
+           'primary': true,
+           'primary_link': true,
+           'secondary': true,
+           'secondary_link': true,
+           'tertiary': true,
+           'tertiary_link': true,
+           'residential': true,
+           'unclassified': true,
+           'living_street': true
+         };
+         var service_roads = {
+           'service': true,
+           'road': true,
+           'track': true
+         };
+         var paths = {
+           'path': true,
+           'footway': true,
+           'cycleway': true,
+           'bridleway': true,
+           'steps': true,
+           'pedestrian': true
+         };
+         var past_futures = {
+           'proposed': true,
+           'construction': true,
+           'abandoned': true,
+           'dismantled': true,
+           'disused': true,
+           'razed': true,
+           'demolished': true,
+           'obliterated': true
          };
+         var _cullFactor = 1;
+         var _cache = {};
+         var _rules = {};
+         var _stats = {};
+         var _keys = [];
+         var _hidden = [];
+         var _forceVisible = {};
 
-         operation.tooltip = function () {
-           var disabled = operation.disabled();
+         function update() {
+           if (!window.mocha) {
+             var hash = utilStringQs(window.location.hash);
+             var disabled = features.disabled();
 
-           if (disabled) {
-             if (disabled === 'restriction') {
-               return _t('operations.merge.restriction', {
-                 relation: _mainPresetIndex.item('type/restriction').name()
-               });
+             if (disabled.length) {
+               hash.disable_features = disabled.join(',');
+             } else {
+               delete hash.disable_features;
              }
 
-             return _t('operations.merge.' + disabled);
+             window.location.replace('#' + utilQsString(hash, true));
+             corePreferences('disabled-features', disabled.join(','));
            }
 
-           return _t('operations.merge.description');
-         };
-
-         operation.annotation = function () {
-           return _t('operations.merge.annotation', {
-             n: selectedIDs.length
-           });
-         };
+           _hidden = features.hidden();
+           dispatch.call('change');
+           dispatch.call('redraw');
+         }
 
-         operation.id = 'merge';
-         operation.keys = [_t('operations.merge.key')];
-         operation.title = _t('operations.merge.title');
-         operation.behavior = behaviorOperation(context).which(operation);
-         return operation;
-       }
+         function defineRule(k, filter, max) {
+           var isEnabled = true;
 
-       function operationPaste(context) {
-         var _pastePoint;
+           _keys.push(k);
 
-         var operation = function operation() {
-           if (!_pastePoint) return;
-           var oldIDs = context.copyIDs();
-           if (!oldIDs.length) return;
-           var projection = context.projection;
-           var extent = geoExtent();
-           var oldGraph = context.copyGraph();
-           var newIDs = [];
-           var action = actionCopyEntities(oldIDs, oldGraph);
-           context.perform(action);
-           var copies = action.copies();
-           var originals = new Set();
-           Object.values(copies).forEach(function (entity) {
-             originals.add(entity.id);
-           });
+           _rules[k] = {
+             filter: filter,
+             enabled: isEnabled,
+             // whether the user wants it enabled..
+             count: 0,
+             currentMax: max || Infinity,
+             defaultMax: max || Infinity,
+             enable: function enable() {
+               this.enabled = true;
+               this.currentMax = this.defaultMax;
+             },
+             disable: function disable() {
+               this.enabled = false;
+               this.currentMax = 0;
+             },
+             hidden: function hidden() {
+               return this.count === 0 && !this.enabled || this.count > this.currentMax * _cullFactor;
+             },
+             autoHidden: function autoHidden() {
+               return this.hidden() && this.currentMax > 0;
+             }
+           };
+         }
 
-           for (var id in copies) {
-             var oldEntity = oldGraph.entity(id);
-             var newEntity = copies[id];
+         defineRule('points', function isPoint(tags, geometry) {
+           return geometry === 'point';
+         }, 200);
+         defineRule('traffic_roads', function isTrafficRoad(tags) {
+           return traffic_roads[tags.highway];
+         });
+         defineRule('service_roads', function isServiceRoad(tags) {
+           return service_roads[tags.highway];
+         });
+         defineRule('paths', function isPath(tags) {
+           return paths[tags.highway];
+         });
+         defineRule('buildings', function isBuilding(tags) {
+           return !!tags.building && tags.building !== 'no' || tags.parking === 'multi-storey' || tags.parking === 'sheds' || tags.parking === 'carports' || tags.parking === 'garage_boxes';
+         }, 250);
+         defineRule('building_parts', function isBuildingPart(tags) {
+           return tags['building:part'];
+         });
+         defineRule('indoor', function isIndoor(tags) {
+           return tags.indoor;
+         });
+         defineRule('landuse', function isLanduse(tags, geometry) {
+           return geometry === 'area' && !_rules.buildings.filter(tags) && !_rules.building_parts.filter(tags) && !_rules.indoor.filter(tags) && !_rules.water.filter(tags) && !_rules.pistes.filter(tags);
+         });
+         defineRule('boundaries', function isBoundary(tags) {
+           return !!tags.boundary && !(traffic_roads[tags.highway] || service_roads[tags.highway] || paths[tags.highway] || tags.waterway || tags.railway || tags.landuse || tags.natural || tags.building || tags.power);
+         });
+         defineRule('water', function isWater(tags) {
+           return !!tags.waterway || tags.natural === 'water' || tags.natural === 'coastline' || tags.natural === 'bay' || tags.landuse === 'pond' || tags.landuse === 'basin' || tags.landuse === 'reservoir' || tags.landuse === 'salt_pond';
+         });
+         defineRule('rail', function isRail(tags) {
+           return (!!tags.railway || tags.landuse === 'railway') && !(traffic_roads[tags.highway] || service_roads[tags.highway] || paths[tags.highway]);
+         });
+         defineRule('pistes', function isPiste(tags) {
+           return tags['piste:type'];
+         });
+         defineRule('aerialways', function isPiste(tags) {
+           return tags.aerialway && tags.aerialway !== 'yes' && tags.aerialway !== 'station';
+         });
+         defineRule('power', function isPower(tags) {
+           return !!tags.power;
+         }); // contains a past/future tag, but not in active use as a road/path/cycleway/etc..
 
-             extent._extend(oldEntity.extent(oldGraph)); // Exclude child nodes from newIDs if their parent way was also copied.
+         defineRule('past_future', function isPastFuture(tags) {
+           if (traffic_roads[tags.highway] || service_roads[tags.highway] || paths[tags.highway]) {
+             return false;
+           }
 
+           var strings = Object.keys(tags);
 
-             var parents = context.graph().parentWays(newEntity);
-             var parentCopied = parents.some(function (parent) {
-               return originals.has(parent.id);
-             });
+           for (var i = 0; i < strings.length; i++) {
+             var s = strings[i];
 
-             if (!parentCopied) {
-               newIDs.push(newEntity.id);
+             if (past_futures[s] || past_futures[tags[s]]) {
+               return true;
              }
-           } // Use the location of the copy operation to offset the paste location,
-           // or else use the center of the pasted extent
+           }
 
+           return false;
+         }); // Lines or areas that don't match another feature filter.
+         // IMPORTANT: The 'others' feature must be the last one defined,
+         //   so that code in getMatches can skip this test if `hasMatch = true`
 
-           var copyPoint = context.copyLonLat() && projection(context.copyLonLat()) || projection(extent.center());
-           var delta = geoVecSubtract(_pastePoint, copyPoint); // Move the pasted objects to be anchored at the paste location
+         defineRule('others', function isOther(tags, geometry) {
+           return geometry === 'line' || geometry === 'area';
+         });
 
-           context.replace(actionMove(newIDs, delta, projection), operation.annotation());
-           context.enter(modeSelect(context, newIDs));
+         features.features = function () {
+           return _rules;
          };
 
-         operation.point = function (val) {
-           _pastePoint = val;
-           return operation;
+         features.keys = function () {
+           return _keys;
          };
 
-         operation.available = function () {
-           return context.mode().id === 'browse';
-         };
+         features.enabled = function (k) {
+           if (!arguments.length) {
+             return _keys.filter(function (k) {
+               return _rules[k].enabled;
+             });
+           }
 
-         operation.disabled = function () {
-           return !context.copyIDs().length;
+           return _rules[k] && _rules[k].enabled;
          };
 
-         operation.tooltip = function () {
-           var oldGraph = context.copyGraph();
-           var ids = context.copyIDs();
-
-           if (!ids.length) {
-             return _t('operations.paste.nothing_copied');
+         features.disabled = function (k) {
+           if (!arguments.length) {
+             return _keys.filter(function (k) {
+               return !_rules[k].enabled;
+             });
            }
 
-           return _t('operations.paste.description', {
-             feature: utilDisplayLabel(oldGraph.entity(ids[0]), oldGraph),
-             n: ids.length
-           });
+           return _rules[k] && !_rules[k].enabled;
          };
 
-         operation.annotation = function () {
-           var ids = context.copyIDs();
-           return _t('operations.paste.annotation', {
-             n: ids.length
-           });
-         };
+         features.hidden = function (k) {
+           if (!arguments.length) {
+             return _keys.filter(function (k) {
+               return _rules[k].hidden();
+             });
+           }
 
-         operation.id = 'paste';
-         operation.keys = [uiCmd('⌘V')];
-         operation.title = _t('operations.paste.title');
-         return operation;
-       }
+           return _rules[k] && _rules[k].hidden();
+         };
 
-       function operationReverse(context, selectedIDs) {
-         var operation = function operation() {
-           context.perform(function combinedReverseAction(graph) {
-             actions().forEach(function (action) {
-               graph = action(graph);
+         features.autoHidden = function (k) {
+           if (!arguments.length) {
+             return _keys.filter(function (k) {
+               return _rules[k].autoHidden();
              });
-             return graph;
-           }, operation.annotation());
-           context.validator().validate();
+           }
+
+           return _rules[k] && _rules[k].autoHidden();
          };
 
-         function actions(situation) {
-           return selectedIDs.map(function (entityID) {
-             var entity = context.hasEntity(entityID);
-             if (!entity) return null;
+         features.enable = function (k) {
+           if (_rules[k] && !_rules[k].enabled) {
+             _rules[k].enable();
 
-             if (situation === 'toolbar') {
-               if (entity.type === 'way' && !entity.isOneWay() && !entity.isSided()) return null;
-             }
+             update();
+           }
+         };
 
-             var geometry = entity.geometry(context.graph());
-             if (entity.type !== 'node' && geometry !== 'line') return null;
-             var action = actionReverse(entityID);
-             if (action.disabled(context.graph())) return null;
-             return action;
-           }).filter(Boolean);
-         }
+         features.enableAll = function () {
+           var didEnable = false;
 
-         function reverseTypeID() {
-           var acts = actions();
-           var nodeActionCount = acts.filter(function (act) {
-             var entity = context.hasEntity(act.entityID());
-             return entity && entity.type === 'node';
-           }).length;
-           if (nodeActionCount === 0) return 'line';
-           if (nodeActionCount === acts.length) return 'point';
-           return 'feature';
-         }
+           for (var k in _rules) {
+             if (!_rules[k].enabled) {
+               didEnable = true;
 
-         operation.available = function (situation) {
-           return actions(situation).length > 0;
-         };
+               _rules[k].enable();
+             }
+           }
 
-         operation.disabled = function () {
-           return false;
+           if (didEnable) update();
          };
 
-         operation.tooltip = function () {
-           return _t('operations.reverse.description.' + reverseTypeID());
-         };
+         features.disable = function (k) {
+           if (_rules[k] && _rules[k].enabled) {
+             _rules[k].disable();
 
-         operation.annotation = function () {
-           var acts = actions();
-           return _t('operations.reverse.annotation.' + reverseTypeID(), {
-             n: acts.length
-           });
+             update();
+           }
          };
 
-         operation.id = 'reverse';
-         operation.keys = [_t('operations.reverse.key')];
-         operation.title = _t('operations.reverse.title');
-         operation.behavior = behaviorOperation(context).which(operation);
-         return operation;
-       }
+         features.disableAll = function () {
+           var didDisable = false;
 
-       function operationSplit(context, selectedIDs) {
-         var _vertexIds = selectedIDs.filter(function (id) {
-           return context.graph().geometry(id) === 'vertex';
-         });
+           for (var k in _rules) {
+             if (_rules[k].enabled) {
+               didDisable = true;
 
-         var _selectedWayIds = selectedIDs.filter(function (id) {
-           var entity = context.graph().hasEntity(id);
-           return entity && entity.type === 'way';
-         });
+               _rules[k].disable();
+             }
+           }
 
-         var _isAvailable = _vertexIds.length > 0 && _vertexIds.length + _selectedWayIds.length === selectedIDs.length;
+           if (didDisable) update();
+         };
 
-         var _action = actionSplit(_vertexIds);
+         features.toggle = function (k) {
+           if (_rules[k]) {
+             (function (f) {
+               return f.enabled ? f.disable() : f.enable();
+             })(_rules[k]);
 
-         var _ways = [];
-         var _geometry = 'feature';
-         var _waysAmount = 'single';
+             update();
+           }
+         };
 
-         var _nodesAmount = _vertexIds.length === 1 ? 'single' : 'multiple';
+         features.resetStats = function () {
+           for (var i = 0; i < _keys.length; i++) {
+             _rules[_keys[i]].count = 0;
+           }
 
-         if (_isAvailable) {
-           if (_selectedWayIds.length) _action.limitWays(_selectedWayIds);
-           _ways = _action.ways(context.graph());
-           var geometries = {};
+           dispatch.call('change');
+         };
 
-           _ways.forEach(function (way) {
-             geometries[way.geometry(context.graph())] = true;
-           });
+         features.gatherStats = function (d, resolver, dimensions) {
+           var needsRedraw = false;
+           var types = utilArrayGroupBy(d, 'type');
+           var entities = [].concat(types.relation || [], types.way || [], types.node || []);
+           var currHidden, geometry, matches, i, j;
 
-           if (Object.keys(geometries).length === 1) {
-             _geometry = Object.keys(geometries)[0];
-           }
+           for (i = 0; i < _keys.length; i++) {
+             _rules[_keys[i]].count = 0;
+           } // adjust the threshold for point/building culling based on viewport size..
+           // a _cullFactor of 1 corresponds to a 1000x1000px viewport..
 
-           _waysAmount = _ways.length === 1 ? 'single' : 'multiple';
-         }
 
-         var operation = function operation() {
-           var difference = context.perform(_action, operation.annotation()); // select both the nodes and the ways so the mapper can immediately disconnect them if desired
+           _cullFactor = dimensions[0] * dimensions[1] / 1000000;
 
-           var idsToSelect = _vertexIds.concat(difference.extantIDs().filter(function (id) {
-             // filter out relations that may have had member additions
-             return context.entity(id).type === 'way';
-           }));
+           for (i = 0; i < entities.length; i++) {
+             geometry = entities[i].geometry(resolver);
+             matches = Object.keys(features.getMatches(entities[i], resolver, geometry));
 
-           context.enter(modeSelect(context, idsToSelect));
-         };
+             for (j = 0; j < matches.length; j++) {
+               _rules[matches[j]].count++;
+             }
+           }
 
-         operation.relatedEntityIds = function () {
-           return _selectedWayIds.length ? [] : _ways.map(function (way) {
-             return way.id;
-           });
-         };
+           currHidden = features.hidden();
 
-         operation.available = function () {
-           return _isAvailable;
-         };
+           if (currHidden !== _hidden) {
+             _hidden = currHidden;
+             needsRedraw = true;
+             dispatch.call('change');
+           }
 
-         operation.disabled = function () {
-           var reason = _action.disabled(context.graph());
+           return needsRedraw;
+         };
 
-           if (reason) {
-             return reason;
-           } else if (selectedIDs.some(context.hasHiddenConnections)) {
-             return 'connected_to_hidden';
+         features.stats = function () {
+           for (var i = 0; i < _keys.length; i++) {
+             _stats[_keys[i]] = _rules[_keys[i]].count;
            }
 
-           return false;
+           return _stats;
          };
 
-         operation.tooltip = function () {
-           var disable = operation.disabled();
-           if (disable) return _t('operations.split.' + disable);
-           return _t('operations.split.description.' + _geometry + '.' + _waysAmount + '.' + _nodesAmount + '_node');
+         features.clear = function (d) {
+           for (var i = 0; i < d.length; i++) {
+             features.clearEntity(d[i]);
+           }
          };
 
-         operation.annotation = function () {
-           return _t('operations.split.annotation.' + _geometry, {
-             n: _ways.length
-           });
+         features.clearEntity = function (entity) {
+           delete _cache[osmEntity.key(entity)];
          };
 
-         operation.id = 'split';
-         operation.keys = [_t('operations.split.key')];
-         operation.title = _t('operations.split.title');
-         operation.behavior = behaviorOperation(context).which(operation);
-         return operation;
-       }
+         features.reset = function () {
+           Array.from(_deferred).forEach(function (handle) {
+             window.cancelIdleCallback(handle);
 
-       function operationStraighten(context, selectedIDs) {
-         var _wayIDs = selectedIDs.filter(function (id) {
-           return id.charAt(0) === 'w';
-         });
+             _deferred["delete"](handle);
+           });
+           _cache = {};
+         }; // only certain relations are worth checking
 
-         var _nodeIDs = selectedIDs.filter(function (id) {
-           return id.charAt(0) === 'n';
-         });
 
-         var _amount = (_wayIDs.length ? _wayIDs : _nodeIDs).length === 1 ? 'single' : 'multiple';
+         function relationShouldBeChecked(relation) {
+           // multipolygon features have `area` geometry and aren't checked here
+           return relation.tags.type === 'boundary';
+         }
 
-         var _nodes = utilGetAllNodes(selectedIDs, context.graph());
+         features.getMatches = function (entity, resolver, geometry) {
+           if (geometry === 'vertex' || geometry === 'relation' && !relationShouldBeChecked(entity)) return {};
+           var ent = osmEntity.key(entity);
 
-         var _coords = _nodes.map(function (n) {
-           return n.loc;
-         });
+           if (!_cache[ent]) {
+             _cache[ent] = {};
+           }
 
-         var _extent = utilTotalExtent(selectedIDs, context.graph());
+           if (!_cache[ent].matches) {
+             var matches = {};
+             var hasMatch = false;
 
-         var _action = chooseAction();
+             for (var i = 0; i < _keys.length; i++) {
+               if (_keys[i] === 'others') {
+                 if (hasMatch) continue; // If an entity...
+                 //   1. is a way that hasn't matched other 'interesting' feature rules,
 
-         var _geometry;
+                 if (entity.type === 'way') {
+                   var parents = features.getParents(entity, resolver, geometry); //   2a. belongs only to a single multipolygon relation
 
-         function chooseAction() {
-           // straighten selected nodes
-           if (_wayIDs.length === 0 && _nodeIDs.length > 2) {
-             _geometry = 'point';
-             return actionStraightenNodes(_nodeIDs, context.projection); // straighten selected ways (possibly between range of 2 selected nodes)
-           } else if (_wayIDs.length > 0 && (_nodeIDs.length === 0 || _nodeIDs.length === 2)) {
-             var startNodeIDs = [];
-             var endNodeIDs = [];
+                   if (parents.length === 1 && parents[0].isMultipolygon() || // 2b. or belongs only to boundary relations
+                   parents.length > 0 && parents.every(function (parent) {
+                     return parent.tags.type === 'boundary';
+                   })) {
+                     // ...then match whatever feature rules the parent relation has matched.
+                     // see #2548, #2887
+                     //
+                     // IMPORTANT:
+                     // For this to work, getMatches must be called on relations before ways.
+                     //
+                     var pkey = osmEntity.key(parents[0]);
 
-             for (var i = 0; i < selectedIDs.length; i++) {
-               var entity = context.entity(selectedIDs[i]);
+                     if (_cache[pkey] && _cache[pkey].matches) {
+                       matches = Object.assign({}, _cache[pkey].matches); // shallow copy
 
-               if (entity.type === 'node') {
-                 continue;
-               } else if (entity.type !== 'way' || entity.isClosed()) {
-                 return null; // exit early, can't straighten these
+                       continue;
+                     }
+                   }
+                 }
                }
 
-               startNodeIDs.push(entity.first());
-               endNodeIDs.push(entity.last());
-             } // Remove duplicate end/startNodeIDs (duplicate nodes cannot be at the line end)
+               if (_rules[_keys[i]].filter(entity.tags, geometry)) {
+                 matches[_keys[i]] = hasMatch = true;
+               }
+             }
 
+             _cache[ent].matches = matches;
+           }
 
-             startNodeIDs = startNodeIDs.filter(function (n) {
-               return startNodeIDs.indexOf(n) === startNodeIDs.lastIndexOf(n);
-             });
-             endNodeIDs = endNodeIDs.filter(function (n) {
-               return endNodeIDs.indexOf(n) === endNodeIDs.lastIndexOf(n);
-             }); // Ensure all ways are connected (i.e. only 2 unique endpoints/startpoints)
+           return _cache[ent].matches;
+         };
 
-             if (utilArrayDifference(startNodeIDs, endNodeIDs).length + utilArrayDifference(endNodeIDs, startNodeIDs).length !== 2) return null; // Ensure path contains at least 3 unique nodes
+         features.getParents = function (entity, resolver, geometry) {
+           if (geometry === 'point') return [];
+           var ent = osmEntity.key(entity);
 
-             var wayNodeIDs = utilGetAllNodes(_wayIDs, context.graph()).map(function (node) {
-               return node.id;
-             });
-             if (wayNodeIDs.length <= 2) return null; // If range of 2 selected nodes is supplied, ensure nodes lie on the selected path
+           if (!_cache[ent]) {
+             _cache[ent] = {};
+           }
 
-             if (_nodeIDs.length === 2 && (wayNodeIDs.indexOf(_nodeIDs[0]) === -1 || wayNodeIDs.indexOf(_nodeIDs[1]) === -1)) return null;
+           if (!_cache[ent].parents) {
+             var parents = [];
 
-             if (_nodeIDs.length) {
-               // If we're only straightenting between two points, we only need that extent visible
-               _extent = utilTotalExtent(_nodeIDs, context.graph());
+             if (geometry === 'vertex') {
+               parents = resolver.parentWays(entity);
+             } else {
+               // 'line', 'area', 'relation'
+               parents = resolver.parentRelations(entity);
              }
 
-             _geometry = 'line';
-             return actionStraightenWay(selectedIDs, context.projection);
+             _cache[ent].parents = parents;
            }
 
-           return null;
-         }
-
-         function operation() {
-           if (!_action) return;
-           context.perform(_action, operation.annotation());
-           window.setTimeout(function () {
-             context.validator().validate();
-           }, 300); // after any transition
-         }
-
-         operation.available = function () {
-           return Boolean(_action);
+           return _cache[ent].parents;
          };
 
-         operation.disabled = function () {
-           var reason = _action.disabled(context.graph());
+         features.isHiddenPreset = function (preset, geometry) {
+           if (!_hidden.length) return false;
+           if (!preset.tags) return false;
+           var test = preset.setTags({}, geometry);
 
-           if (reason) {
-             return reason;
-           } else if (_extent.percentContainedIn(context.map().extent()) < 0.8) {
-             return 'too_large';
-           } else if (someMissing()) {
-             return 'not_downloaded';
-           } else if (selectedIDs.some(context.hasHiddenConnections)) {
-             return 'connected_to_hidden';
+           for (var key in _rules) {
+             if (_rules[key].filter(test, geometry)) {
+               if (_hidden.indexOf(key) !== -1) {
+                 return key;
+               }
+
+               return false;
+             }
            }
 
            return false;
+         };
 
-           function someMissing() {
-             if (context.inIntro()) return false;
-             var osm = context.connection();
+         features.isHiddenFeature = function (entity, resolver, geometry) {
+           if (!_hidden.length) return false;
+           if (!entity.version) return false;
+           if (_forceVisible[entity.id]) return false;
+           var matches = Object.keys(features.getMatches(entity, resolver, geometry));
+           return matches.length && matches.every(function (k) {
+             return features.hidden(k);
+           });
+         };
 
-             if (osm) {
-               var missing = _coords.filter(function (loc) {
-                 return !osm.isDataLoaded(loc);
-               });
+         features.isHiddenChild = function (entity, resolver, geometry) {
+           if (!_hidden.length) return false;
+           if (!entity.version || geometry === 'point') return false;
+           if (_forceVisible[entity.id]) return false;
+           var parents = features.getParents(entity, resolver, geometry);
+           if (!parents.length) return false;
 
-               if (missing.length) {
-                 missing.forEach(function (loc) {
-                   context.loadTileAtLoc(loc);
-                 });
-                 return true;
-               }
+           for (var i = 0; i < parents.length; i++) {
+             if (!features.isHidden(parents[i], resolver, parents[i].geometry(resolver))) {
+               return false;
              }
-
-             return false;
            }
-         };
-
-         operation.tooltip = function () {
-           var disable = operation.disabled();
-           return disable ? _t('operations.straighten.' + disable + '.' + _amount) : _t('operations.straighten.description.' + _geometry + (_wayIDs.length === 1 ? '' : 's'));
-         };
 
-         operation.annotation = function () {
-           return _t('operations.straighten.annotation.' + _geometry, {
-             n: _wayIDs.length ? _wayIDs.length : _nodeIDs.length
-           });
+           return true;
          };
 
-         operation.id = 'straighten';
-         operation.keys = [_t('operations.straighten.key')];
-         operation.title = _t('operations.straighten.title');
-         operation.behavior = behaviorOperation(context).which(operation);
-         return operation;
-       }
-
-       var Operations = /*#__PURE__*/Object.freeze({
-               __proto__: null,
-               operationCircularize: operationCircularize,
-               operationContinue: operationContinue,
-               operationCopy: operationCopy,
-               operationDelete: operationDelete,
-               operationDisconnect: operationDisconnect,
-               operationDowngrade: operationDowngrade,
-               operationExtract: operationExtract,
-               operationMerge: operationMerge,
-               operationMove: operationMove,
-               operationOrthogonalize: operationOrthogonalize,
-               operationPaste: operationPaste,
-               operationReflectShort: operationReflectShort,
-               operationReflectLong: operationReflectLong,
-               operationReverse: operationReverse,
-               operationRotate: operationRotate,
-               operationSplit: operationSplit,
-               operationStraighten: operationStraighten
-       });
+         features.hasHiddenConnections = function (entity, resolver) {
+           if (!_hidden.length) return false;
+           var childNodes, connections;
 
-       var _relatedParent;
+           if (entity.type === 'midpoint') {
+             childNodes = [resolver.entity(entity.edge[0]), resolver.entity(entity.edge[1])];
+             connections = [];
+           } else {
+             childNodes = entity.nodes ? resolver.childNodes(entity) : [];
+             connections = features.getParents(entity, resolver, entity.geometry(resolver));
+           } // gather ways connected to child nodes..
 
-       function modeSelect(context, selectedIDs) {
-         var mode = {
-           id: 'select',
-           button: 'browse'
-         };
-         var keybinding = utilKeybinding('select');
 
-         var _breatheBehavior = behaviorBreathe();
+           connections = childNodes.reduce(function (result, e) {
+             return resolver.isShared(e) ? utilArrayUnion(result, resolver.parentWays(e)) : result;
+           }, connections);
+           return connections.some(function (e) {
+             return features.isHidden(e, resolver, e.geometry(resolver));
+           });
+         };
 
-         var _modeDragNode = modeDragNode(context);
+         features.isHidden = function (entity, resolver, geometry) {
+           if (!_hidden.length) return false;
+           if (!entity.version) return false;
+           var fn = geometry === 'vertex' ? features.isHiddenChild : features.isHiddenFeature;
+           return fn(entity, resolver, geometry);
+         };
 
-         var _selectBehavior;
+         features.filter = function (d, resolver) {
+           if (!_hidden.length) return d;
+           var result = [];
 
-         var _behaviors = [];
-         var _operations = [];
-         var _newFeature = false;
-         var _follow = false;
+           for (var i = 0; i < d.length; i++) {
+             var entity = d[i];
 
-         function singular() {
-           if (selectedIDs && selectedIDs.length === 1) {
-             return context.hasEntity(selectedIDs[0]);
+             if (!features.isHidden(entity, resolver, entity.geometry(resolver))) {
+               result.push(entity);
+             }
            }
-         }
 
-         function selectedEntities() {
-           return selectedIDs.map(function (id) {
-             return context.hasEntity(id);
-           }).filter(Boolean);
-         }
+           return result;
+         };
 
-         function checkSelectedIDs() {
-           var ids = [];
+         features.forceVisible = function (entityIDs) {
+           if (!arguments.length) return Object.keys(_forceVisible);
+           _forceVisible = {};
 
-           if (Array.isArray(selectedIDs)) {
-             ids = selectedIDs.filter(function (id) {
-               return context.hasEntity(id);
-             });
-           }
+           for (var i = 0; i < entityIDs.length; i++) {
+             _forceVisible[entityIDs[i]] = true;
+             var entity = context.hasEntity(entityIDs[i]);
 
-           if (!ids.length) {
-             context.enter(modeBrowse(context));
-             return false;
-           } else if (selectedIDs.length > 1 && ids.length === 1 || selectedIDs.length === 1 && ids.length > 1) {
-             // switch between single- and multi-select UI
-             context.enter(modeSelect(context, ids));
-             return false;
+             if (entity && entity.type === 'relation') {
+               // also show relation members (one level deep)
+               for (var j in entity.members) {
+                 _forceVisible[entity.members[j].id] = true;
+               }
+             }
            }
 
-           selectedIDs = ids;
-           return true;
-         } // find the common parent ways for nextVertex, previousVertex
+           return features;
+         };
 
+         features.init = function () {
+           var storage = corePreferences('disabled-features');
 
-         function commonParents() {
-           var graph = context.graph();
-           var commonParents = [];
+           if (storage) {
+             var storageDisabled = storage.replace(/;/g, ',').split(',');
+             storageDisabled.forEach(features.disable);
+           }
 
-           for (var i = 0; i < selectedIDs.length; i++) {
-             var entity = context.hasEntity(selectedIDs[i]);
+           var hash = utilStringQs(window.location.hash);
 
-             if (!entity || entity.geometry(graph) !== 'vertex') {
-               return []; // selection includes some not vertices
-             }
+           if (hash.disable_features) {
+             var hashDisabled = hash.disable_features.replace(/;/g, ',').split(',');
+             hashDisabled.forEach(features.disable);
+           }
+         }; // warm up the feature matching cache upon merging fetched data
 
-             var currParents = graph.parentWays(entity).map(function (w) {
-               return w.id;
-             });
 
-             if (!commonParents.length) {
-               commonParents = currParents;
-               continue;
-             }
+         context.history().on('merge.features', function (newEntities) {
+           if (!newEntities) return;
+           var handle = window.requestIdleCallback(function () {
+             var graph = context.graph();
+             var types = utilArrayGroupBy(newEntities, 'type'); // ensure that getMatches is called on relations before ways
 
-             commonParents = utilArrayIntersection(commonParents, currParents);
+             var entities = [].concat(types.relation || [], types.way || [], types.node || []);
 
-             if (!commonParents.length) {
-               return [];
+             for (var i = 0; i < entities.length; i++) {
+               var geometry = entities[i].geometry(graph);
+               features.getMatches(entities[i], graph, geometry);
              }
-           }
-
-           return commonParents;
-         }
-
-         function singularParent() {
-           var parents = commonParents();
+           });
 
-           if (!parents || parents.length === 0) {
-             _relatedParent = null;
-             return null;
-           } // relatedParent is used when we visit a vertex with multiple
-           // parents, and we want to remember which parent line we started on.
+           _deferred.add(handle);
+         });
+         return features;
+       }
 
+       /** Error message constants. */
 
-           if (parents.length === 1) {
-             _relatedParent = parents[0]; // remember this parent for later
+       var FUNC_ERROR_TEXT = 'Expected a function';
+       /**
+        * Creates a throttled function that only invokes `func` at most once per
+        * every `wait` milliseconds. The throttled function comes with a `cancel`
+        * method to cancel delayed `func` invocations and a `flush` method to
+        * immediately invoke them. Provide `options` to indicate whether `func`
+        * should be invoked on the leading and/or trailing edge of the `wait`
+        * timeout. The `func` is invoked with the last arguments provided to the
+        * throttled function. Subsequent calls to the throttled function return the
+        * result of the last `func` invocation.
+        *
+        * **Note:** If `leading` and `trailing` options are `true`, `func` is
+        * invoked on the trailing edge of the timeout only if the throttled function
+        * is invoked more than once during the `wait` timeout.
+        *
+        * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
+        * until to the next tick, similar to `setTimeout` with a timeout of `0`.
+        *
+        * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
+        * for details over the differences between `_.throttle` and `_.debounce`.
+        *
+        * @static
+        * @memberOf _
+        * @since 0.1.0
+        * @category Function
+        * @param {Function} func The function to throttle.
+        * @param {number} [wait=0] The number of milliseconds to throttle invocations to.
+        * @param {Object} [options={}] The options object.
+        * @param {boolean} [options.leading=true]
+        *  Specify invoking on the leading edge of the timeout.
+        * @param {boolean} [options.trailing=true]
+        *  Specify invoking on the trailing edge of the timeout.
+        * @returns {Function} Returns the new throttled function.
+        * @example
+        *
+        * // Avoid excessively updating the position while scrolling.
+        * jQuery(window).on('scroll', _.throttle(updatePosition, 100));
+        *
+        * // Invoke `renewToken` when the click event is fired, but not more than once every 5 minutes.
+        * var throttled = _.throttle(renewToken, 300000, { 'trailing': false });
+        * jQuery(element).on('click', throttled);
+        *
+        * // Cancel the trailing throttled invocation.
+        * jQuery(window).on('popstate', throttled.cancel);
+        */
 
-             return _relatedParent;
-           }
+       function throttle(func, wait, options) {
+         var leading = true,
+             trailing = true;
 
-           if (parents.indexOf(_relatedParent) !== -1) {
-             return _relatedParent; // prefer the previously seen parent
-           }
+         if (typeof func != 'function') {
+           throw new TypeError(FUNC_ERROR_TEXT);
+         }
 
-           return parents[0];
+         if (isObject$2(options)) {
+           leading = 'leading' in options ? !!options.leading : leading;
+           trailing = 'trailing' in options ? !!options.trailing : trailing;
          }
 
-         mode.selectedIDs = function (val) {
-           if (!arguments.length) return selectedIDs;
-           selectedIDs = val;
-           return mode;
-         };
+         return debounce(func, wait, {
+           'leading': leading,
+           'maxWait': wait,
+           'trailing': trailing
+         });
+       }
 
-         mode.zoomToSelected = function () {
-           context.map().zoomToEase(selectedEntities());
-         };
+       //
+       // - the activeID - nope
+       // - 1 away (adjacent) to the activeID - yes (vertices will be merged)
+       // - 2 away from the activeID - nope (would create a self intersecting segment)
+       // - all others on a linear way - yes
+       // - all others on a closed way - nope (would create a self intersecting polygon)
+       //
+       // returns
+       // 0 = active vertex - no touch/connect
+       // 1 = passive vertex - yes touch/connect
+       // 2 = adjacent vertex - yes but pay attention segmenting a line here
+       //
 
-         mode.newFeature = function (val) {
-           if (!arguments.length) return _newFeature;
-           _newFeature = val;
-           return mode;
-         };
+       function svgPassiveVertex(node, graph, activeID) {
+         if (!activeID) return 1;
+         if (activeID === node.id) return 0;
+         var parents = graph.parentWays(node);
+         var i, j, nodes, isClosed, ix1, ix2, ix3, ix4, max;
 
-         mode.selectBehavior = function (val) {
-           if (!arguments.length) return _selectBehavior;
-           _selectBehavior = val;
-           return mode;
-         };
+         for (i = 0; i < parents.length; i++) {
+           nodes = parents[i].nodes;
+           isClosed = parents[i].isClosed();
 
-         mode.follow = function (val) {
-           if (!arguments.length) return _follow;
-           _follow = val;
-           return mode;
-         };
+           for (j = 0; j < nodes.length; j++) {
+             // find this vertex, look nearby
+             if (nodes[j] === node.id) {
+               ix1 = j - 2;
+               ix2 = j - 1;
+               ix3 = j + 1;
+               ix4 = j + 2;
 
-         function loadOperations() {
-           _operations.forEach(function (operation) {
-             if (operation.behavior) {
-               context.uninstall(operation.behavior);
+               if (isClosed) {
+                 // wraparound if needed
+                 max = nodes.length - 1;
+                 if (ix1 < 0) ix1 = max + ix1;
+                 if (ix2 < 0) ix2 = max + ix2;
+                 if (ix3 > max) ix3 = ix3 - max;
+                 if (ix4 > max) ix4 = ix4 - max;
+               }
+
+               if (nodes[ix1] === activeID) return 0; // no - prevent self intersect
+               else if (nodes[ix2] === activeID) return 2; // ok - adjacent
+                 else if (nodes[ix3] === activeID) return 2; // ok - adjacent
+                   else if (nodes[ix4] === activeID) return 0; // no - prevent self intersect
+                     else if (isClosed && nodes.indexOf(activeID) !== -1) return 0; // no - prevent self intersect
              }
-           });
+           }
+         }
 
-           _operations = Object.values(Operations).map(function (o) {
-             return o(context, selectedIDs);
-           }).filter(function (o) {
-             return o.id !== 'delete' && o.id !== 'downgrade' && o.id !== 'copy';
-           }).concat([// group copy/downgrade/delete operation together at the end of the list
-           operationCopy(context, selectedIDs), operationDowngrade(context, selectedIDs), operationDelete(context, selectedIDs)]).filter(function (operation) {
-             return operation.available();
+         return 1; // ok
+       }
+       function svgMarkerSegments(projection, graph, dt, shouldReverse, bothDirections) {
+         return function (entity) {
+           var i = 0;
+           var offset = dt;
+           var segments = [];
+           var clip = d3_geoIdentity().clipExtent(projection.clipExtent()).stream;
+           var coordinates = graph.childNodes(entity).map(function (n) {
+             return n.loc;
            });
+           var a, b;
 
-           _operations.forEach(function (operation) {
-             if (operation.behavior) {
-               context.install(operation.behavior);
-             }
-           }); // remove any displayed menu
+           if (shouldReverse(entity)) {
+             coordinates.reverse();
+           }
 
+           d3_geoStream({
+             type: 'LineString',
+             coordinates: coordinates
+           }, projection.stream(clip({
+             lineStart: function lineStart() {},
+             lineEnd: function lineEnd() {
+               a = null;
+             },
+             point: function point(x, y) {
+               b = [x, y];
 
-           context.ui().closeEditMenu();
-         }
+               if (a) {
+                 var span = geoVecLength(a, b) - offset;
 
-         mode.operations = function () {
-           return _operations;
-         };
+                 if (span >= 0) {
+                   var heading = geoVecAngle(a, b);
+                   var dx = dt * Math.cos(heading);
+                   var dy = dt * Math.sin(heading);
+                   var p = [a[0] + offset * Math.cos(heading), a[1] + offset * Math.sin(heading)]; // gather coordinates
 
-         mode.enter = function () {
-           if (!checkSelectedIDs()) return;
-           context.features().forceVisible(selectedIDs);
+                   var coord = [a, p];
 
-           _modeDragNode.restoreSelectedIDs(selectedIDs);
+                   for (span -= dt; span >= 0; span -= dt) {
+                     p = geoVecAdd(p, [dx, dy]);
+                     coord.push(p);
+                   }
 
-           loadOperations();
+                   coord.push(b); // generate svg paths
 
-           if (!_behaviors.length) {
-             if (!_selectBehavior) _selectBehavior = behaviorSelect(context);
-             _behaviors = [behaviorPaste(context), _breatheBehavior, behaviorHover(context).on('hover', context.ui().sidebar.hoverModeSelect), _selectBehavior, behaviorLasso(context), _modeDragNode.behavior, modeDragNote(context).behavior];
-           }
+                   var segment = '';
+                   var j;
 
-           _behaviors.forEach(context.install);
+                   for (j = 0; j < coord.length; j++) {
+                     segment += (j === 0 ? 'M' : 'L') + coord[j][0] + ',' + coord[j][1];
+                   }
 
-           keybinding.on(_t('inspector.zoom_to.key'), mode.zoomToSelected).on(['[', 'pgup'], previousVertex).on([']', 'pgdown'], nextVertex).on(['{', uiCmd('⌘['), 'home'], firstVertex).on(['}', uiCmd('⌘]'), 'end'], lastVertex).on(uiCmd('⇧←'), nudgeSelection([-10, 0])).on(uiCmd('⇧↑'), nudgeSelection([0, -10])).on(uiCmd('⇧→'), nudgeSelection([10, 0])).on(uiCmd('⇧↓'), nudgeSelection([0, 10])).on(uiCmd('⇧⌥←'), nudgeSelection([-100, 0])).on(uiCmd('⇧⌥↑'), nudgeSelection([0, -100])).on(uiCmd('⇧⌥→'), nudgeSelection([100, 0])).on(uiCmd('⇧⌥↓'), nudgeSelection([0, 100])).on(utilKeybinding.plusKeys.map(function (key) {
-             return uiCmd('⇧' + key);
-           }), scaleSelection(1.05)).on(utilKeybinding.plusKeys.map(function (key) {
-             return uiCmd('⇧⌥' + key);
-           }), scaleSelection(Math.pow(1.05, 5))).on(utilKeybinding.minusKeys.map(function (key) {
-             return uiCmd('⇧' + key);
-           }), scaleSelection(1 / 1.05)).on(utilKeybinding.minusKeys.map(function (key) {
-             return uiCmd('⇧⌥' + key);
-           }), scaleSelection(1 / Math.pow(1.05, 5))).on(['\\', 'pause'], nextParent).on('⎋', esc, true);
-           select(document).call(keybinding);
-           context.ui().sidebar.select(selectedIDs, _newFeature);
-           context.history().on('change.select', function () {
-             loadOperations(); // reselect after change in case relation members were removed or added
+                   segments.push({
+                     id: entity.id,
+                     index: i++,
+                     d: segment
+                   });
 
-             selectElements();
-           }).on('undone.select', checkSelectedIDs).on('redone.select', checkSelectedIDs);
-           context.map().on('drawn.select', selectElements).on('crossEditableZoom.select', function () {
-             selectElements();
+                   if (bothDirections(entity)) {
+                     segment = '';
 
-             _breatheBehavior.restartIfNeeded(context.surface());
-           });
-           context.map().doubleUpHandler().on('doubleUp.modeSelect', didDoubleUp);
-           selectElements();
+                     for (j = coord.length - 1; j >= 0; j--) {
+                       segment += (j === coord.length - 1 ? 'M' : 'L') + coord[j][0] + ',' + coord[j][1];
+                     }
 
-           if (_follow) {
-             var extent = geoExtent();
-             var graph = context.graph();
-             selectedIDs.forEach(function (id) {
-               var entity = context.entity(id);
+                     segments.push({
+                       id: entity.id,
+                       index: i++,
+                       d: segment
+                     });
+                   }
+                 }
 
-               extent._extend(entity.extent(graph));
-             });
-             var loc = extent.center();
-             context.map().centerEase(loc); // we could enter the mode multiple times, so reset follow for next time
+                 offset = -span;
+               }
 
-             _follow = false;
+               a = b;
+             }
+           })));
+           return segments;
+         };
+       }
+       function svgPath(projection, graph, isArea) {
+         // Explanation of magic numbers:
+         // "padding" here allows space for strokes to extend beyond the viewport,
+         // so that the stroke isn't drawn along the edge of the viewport when
+         // the shape is clipped.
+         //
+         // When drawing lines, pad viewport by 5px.
+         // When drawing areas, pad viewport by 65px in each direction to allow
+         // for 60px area fill stroke (see ".fill-partial path.fill" css rule)
+         var cache = {};
+         var padding = isArea ? 65 : 5;
+         var viewport = projection.clipExtent();
+         var paddedExtent = [[viewport[0][0] - padding, viewport[0][1] - padding], [viewport[1][0] + padding, viewport[1][1] + padding]];
+         var clip = d3_geoIdentity().clipExtent(paddedExtent).stream;
+         var project = projection.stream;
+         var path = d3_geoPath().projection({
+           stream: function stream(output) {
+             return project(clip(output));
            }
+         });
 
-           function nudgeSelection(delta) {
-             return function () {
-               // prevent nudging during low zoom selection
-               if (!context.map().withinEditableZoom()) return;
-               var moveOp = operationMove(context, selectedIDs);
+         var svgpath = function svgpath(entity) {
+           if (entity.id in cache) {
+             return cache[entity.id];
+           } else {
+             return cache[entity.id] = path(entity.asGeoJSON(graph));
+           }
+         };
 
-               if (moveOp.disabled()) {
-                 context.ui().flash.duration(4000).iconName('#iD-operation-' + moveOp.id).iconClass('operation disabled').label(moveOp.tooltip)();
-               } else {
-                 context.perform(actionMove(selectedIDs, delta, context.projection), moveOp.annotation());
-                 context.validator().validate();
-               }
-             };
+         svgpath.geojson = function (d) {
+           if (d.__featurehash__ !== undefined) {
+             if (d.__featurehash__ in cache) {
+               return cache[d.__featurehash__];
+             } else {
+               return cache[d.__featurehash__] = path(d);
+             }
+           } else {
+             return path(d);
            }
+         };
 
-           function scaleSelection(factor) {
-             return function () {
-               // prevent scaling during low zoom selection
-               if (!context.map().withinEditableZoom()) return;
-               var nodes = utilGetAllNodes(selectedIDs, context.graph());
-               var isUp = factor > 1; // can only scale if multiple nodes are selected
+         return svgpath;
+       }
+       function svgPointTransform(projection) {
+         var svgpoint = function svgpoint(entity) {
+           // http://jsperf.com/short-array-join
+           var pt = projection(entity.loc);
+           return 'translate(' + pt[0] + ',' + pt[1] + ')';
+         };
 
-               if (nodes.length <= 1) return;
-               var extent = utilTotalExtent(selectedIDs, context.graph()); // These disabled checks would normally be handled by an operation
-               // object, but we don't want an actual scale operation at this point.
+         svgpoint.geojson = function (d) {
+           return svgpoint(d.properties.entity);
+         };
 
-               function scalingDisabled() {
-                 if (tooSmall()) {
-                   return 'too_small';
-                 } else if (extent.percentContainedIn(context.map().extent()) < 0.8) {
-                   return 'too_large';
-                 } else if (someMissing() || selectedIDs.some(incompleteRelation)) {
-                   return 'not_downloaded';
-                 } else if (selectedIDs.some(context.hasHiddenConnections)) {
-                   return 'connected_to_hidden';
-                 }
+         return svgpoint;
+       }
+       function svgRelationMemberTags(graph) {
+         return function (entity) {
+           var tags = entity.tags;
+           var shouldCopyMultipolygonTags = !entity.hasInterestingTags();
+           graph.parentRelations(entity).forEach(function (relation) {
+             var type = relation.tags.type;
 
-                 return false;
+             if (type === 'multipolygon' && shouldCopyMultipolygonTags || type === 'boundary') {
+               tags = Object.assign({}, relation.tags, tags);
+             }
+           });
+           return tags;
+         };
+       }
+       function svgSegmentWay(way, graph, activeID) {
+         // When there is no activeID, we can memoize this expensive computation
+         if (activeID === undefined) {
+           return graph["transient"](way, 'waySegments', getWaySegments);
+         } else {
+           return getWaySegments();
+         }
 
-                 function tooSmall() {
-                   if (isUp) return false;
-                   var dLon = Math.abs(extent[1][0] - extent[0][0]);
-                   var dLat = Math.abs(extent[1][1] - extent[0][1]);
-                   return dLon < geoMetersToLon(1, extent[1][1]) && dLat < geoMetersToLat(1);
-                 }
+         function getWaySegments() {
+           var isActiveWay = way.nodes.indexOf(activeID) !== -1;
+           var features = {
+             passive: [],
+             active: []
+           };
+           var start = {};
+           var end = {};
+           var node, type;
 
-                 function someMissing() {
-                   if (context.inIntro()) return false;
-                   var osm = context.connection();
+           for (var i = 0; i < way.nodes.length; i++) {
+             node = graph.entity(way.nodes[i]);
+             type = svgPassiveVertex(node, graph, activeID);
+             end = {
+               node: node,
+               type: type
+             };
 
-                   if (osm) {
-                     var missing = nodes.filter(function (n) {
-                       return !osm.isDataLoaded(n.loc);
-                     });
+             if (start.type !== undefined) {
+               if (start.node.id === activeID || end.node.id === activeID) ; else if (isActiveWay && (start.type === 2 || end.type === 2)) {
+                 // one adjacent vertex
+                 pushActive(start, end, i);
+               } else if (start.type === 0 && end.type === 0) {
+                 // both active vertices
+                 pushActive(start, end, i);
+               } else {
+                 pushPassive(start, end, i);
+               }
+             }
 
-                     if (missing.length) {
-                       missing.forEach(function (loc) {
-                         context.loadTileAtLoc(loc);
-                       });
-                       return true;
-                     }
-                   }
+             start = end;
+           }
 
-                   return false;
-                 }
+           return features;
 
-                 function incompleteRelation(id) {
-                   var entity = context.entity(id);
-                   return entity.type === 'relation' && !entity.isComplete(context.graph());
-                 }
+           function pushActive(start, end, index) {
+             features.active.push({
+               type: 'Feature',
+               id: way.id + '-' + index + '-nope',
+               properties: {
+                 nope: true,
+                 target: true,
+                 entity: way,
+                 nodes: [start.node, end.node],
+                 index: index
+               },
+               geometry: {
+                 type: 'LineString',
+                 coordinates: [start.node.loc, end.node.loc]
                }
+             });
+           }
 
-               var disabled = scalingDisabled();
-
-               if (disabled) {
-                 var multi = selectedIDs.length === 1 ? 'single' : 'multiple';
-                 context.ui().flash.duration(4000).iconName('#iD-icon-no').iconClass('operation disabled').label(_t('operations.scale.' + disabled + '.' + multi))();
-               } else {
-                 var pivot = context.projection(extent.center());
-                 var annotation = _t('operations.scale.annotation.' + (isUp ? 'up' : 'down') + '.feature', {
-                   n: selectedIDs.length
-                 });
-                 context.perform(actionScale(selectedIDs, pivot, factor, context.projection), annotation);
-                 context.validator().validate();
+           function pushPassive(start, end, index) {
+             features.passive.push({
+               type: 'Feature',
+               id: way.id + '-' + index,
+               properties: {
+                 target: true,
+                 entity: way,
+                 nodes: [start.node, end.node],
+                 index: index
+               },
+               geometry: {
+                 type: 'LineString',
+                 coordinates: [start.node.loc, end.node.loc]
                }
-             };
+             });
            }
+         }
+       }
 
-           function didDoubleUp(d3_event, loc) {
-             if (!context.map().withinEditableZoom()) return;
-             var target = select(d3_event.target);
-             var datum = target.datum();
-             var entity = datum && datum.properties && datum.properties.entity;
-             if (!entity) return;
+       function svgTagClasses() {
+         var primaries = ['building', 'highway', 'railway', 'waterway', 'aeroway', 'aerialway', 'piste:type', 'boundary', 'power', 'amenity', 'natural', 'landuse', 'leisure', 'military', 'place', 'man_made', 'route', 'attraction', 'building:part', 'indoor'];
+         var statuses = [// nonexistent, might be built
+         'proposed', 'planned', // under maintentance or between groundbreaking and opening
+         'construction', // existent but not functional
+         'disused', // dilapidated to nonexistent
+         'abandoned', // nonexistent, still may appear in imagery
+         'dismantled', 'razed', 'demolished', 'obliterated', // existent occasionally, e.g. stormwater drainage basin
+         'intermittent'];
+         var secondaries = ['oneway', 'bridge', 'tunnel', 'embankment', 'cutting', 'barrier', 'surface', 'tracktype', 'footway', 'crossing', 'service', 'sport', 'public_transport', 'location', 'parking', 'golf', 'type', 'leisure', 'man_made', 'indoor'];
 
-             if (entity instanceof osmWay && target.classed('target')) {
-               var choice = geoChooseEdge(context.graph().childNodes(entity), loc, context.projection);
-               var prev = entity.nodes[choice.index - 1];
-               var next = entity.nodes[choice.index];
-               context.perform(actionAddMidpoint({
-                 loc: choice.loc,
-                 edge: [prev, next]
-               }, osmNode()), _t('operations.add.annotation.vertex'));
-             } else if (entity.type === 'midpoint') {
-               context.perform(actionAddMidpoint({
-                 loc: entity.loc,
-                 edge: entity.edge
-               }, osmNode()), _t('operations.add.annotation.vertex'));
-             }
-           }
+         var _tags = function _tags(entity) {
+           return entity.tags;
+         };
 
-           function selectElements() {
-             if (!checkSelectedIDs()) return;
-             var surface = context.surface();
-             surface.selectAll('.selected-member').classed('selected-member', false);
-             surface.selectAll('.selected').classed('selected', false);
-             surface.selectAll('.related').classed('related', false);
-             singularParent();
+         var tagClasses = function tagClasses(selection) {
+           selection.each(function tagClassesEach(entity) {
+             var value = this.className;
 
-             if (_relatedParent) {
-               surface.selectAll(utilEntitySelector([_relatedParent])).classed('related', true);
+             if (value.baseVal !== undefined) {
+               value = value.baseVal;
              }
 
-             if (context.map().withinEditableZoom()) {
-               // Apply selection styling if not in wide selection
-               surface.selectAll(utilDeepMemberSelector(selectedIDs, context.graph(), true
-               /* skipMultipolgonMembers */
-               )).classed('selected-member', true);
-               surface.selectAll(utilEntityOrDeepMemberSelector(selectedIDs, context.graph())).classed('selected', true);
+             var t = _tags(entity);
+
+             var computed = tagClasses.getClassesString(t, value);
+
+             if (computed !== value) {
+               select(this).attr('class', computed);
              }
-           }
+           });
+         };
 
-           function esc() {
-             if (context.container().select('.combobox').size()) return;
-             context.enter(modeBrowse(context));
-           }
+         tagClasses.getClassesString = function (t, value) {
+           var primary, status;
+           var i, j, k, v; // in some situations we want to render perimeter strokes a certain way
 
-           function firstVertex(d3_event) {
-             d3_event.preventDefault();
-             var entity = singular();
-             var parent = singularParent();
-             var way;
+           var overrideGeometry;
 
-             if (entity && entity.type === 'way') {
-               way = entity;
-             } else if (parent) {
-               way = context.entity(parent);
+           if (/\bstroke\b/.test(value)) {
+             if (!!t.barrier && t.barrier !== 'no') {
+               overrideGeometry = 'line';
              }
+           } // preserve base classes (nothing with `tag-`)
 
-             if (way) {
-               context.enter(modeSelect(context, [way.first()]).follow(true));
-             }
-           }
 
-           function lastVertex(d3_event) {
-             d3_event.preventDefault();
-             var entity = singular();
-             var parent = singularParent();
-             var way;
+           var classes = value.trim().split(/\s+/).filter(function (klass) {
+             return klass.length && !/^tag-/.test(klass);
+           }).map(function (klass) {
+             // special overrides for some perimeter strokes
+             return klass === 'line' || klass === 'area' ? overrideGeometry || klass : klass;
+           }); // pick at most one primary classification tag..
 
-             if (entity && entity.type === 'way') {
-               way = entity;
-             } else if (parent) {
-               way = context.entity(parent);
-             }
+           for (i = 0; i < primaries.length; i++) {
+             k = primaries[i];
+             v = t[k];
+             if (!v || v === 'no') continue;
 
-             if (way) {
-               context.enter(modeSelect(context, [way.last()]).follow(true));
+             if (k === 'piste:type') {
+               // avoid a ':' in the class name
+               k = 'piste';
+             } else if (k === 'building:part') {
+               // avoid a ':' in the class name
+               k = 'building_part';
              }
-           }
 
-           function previousVertex(d3_event) {
-             d3_event.preventDefault();
-             var parent = singularParent();
-             if (!parent) return;
-             var way = context.entity(parent);
-             var length = way.nodes.length;
-             var curr = way.nodes.indexOf(selectedIDs[0]);
-             var index = -1;
+             primary = k;
 
-             if (curr > 0) {
-               index = curr - 1;
-             } else if (way.isClosed()) {
-               index = length - 2;
+             if (statuses.indexOf(v) !== -1) {
+               // e.g. `railway=abandoned`
+               status = v;
+               classes.push('tag-' + k);
+             } else {
+               classes.push('tag-' + k);
+               classes.push('tag-' + k + '-' + v);
              }
 
-             if (index !== -1) {
-               context.enter(modeSelect(context, [way.nodes[index]]).follow(true));
-             }
+             break;
            }
 
-           function nextVertex(d3_event) {
-             d3_event.preventDefault();
-             var parent = singularParent();
-             if (!parent) return;
-             var way = context.entity(parent);
-             var length = way.nodes.length;
-             var curr = way.nodes.indexOf(selectedIDs[0]);
-             var index = -1;
+           if (!primary) {
+             for (i = 0; i < statuses.length; i++) {
+               for (j = 0; j < primaries.length; j++) {
+                 k = statuses[i] + ':' + primaries[j]; // e.g. `demolished:building=yes`
 
-             if (curr < length - 1) {
-               index = curr + 1;
-             } else if (way.isClosed()) {
-               index = 0;
+                 v = t[k];
+                 if (!v || v === 'no') continue;
+                 status = statuses[i];
+                 break;
+               }
              }
+           } // add at most one status tag, only if relates to primary tag..
 
-             if (index !== -1) {
-               context.enter(modeSelect(context, [way.nodes[index]]).follow(true));
-             }
-           }
 
-           function nextParent(d3_event) {
-             d3_event.preventDefault();
-             var parents = commonParents();
-             if (!parents || parents.length < 2) return;
-             var index = parents.indexOf(_relatedParent);
+           if (!status) {
+             for (i = 0; i < statuses.length; i++) {
+               k = statuses[i];
+               v = t[k];
+               if (!v || v === 'no') continue;
 
-             if (index < 0 || index > parents.length - 2) {
-               _relatedParent = parents[0];
-             } else {
-               _relatedParent = parents[index + 1];
-             }
+               if (v === 'yes') {
+                 // e.g. `railway=rail + abandoned=yes`
+                 status = k;
+               } else if (primary && primary === v) {
+                 // e.g. `railway=rail + abandoned=railway`
+                 status = k;
+               } else if (!primary && primaries.indexOf(v) !== -1) {
+                 // e.g. `abandoned=railway`
+                 status = k;
+                 primary = v;
+                 classes.push('tag-' + v);
+               } // else ignore e.g.  `highway=path + abandoned=railway`
 
-             var surface = context.surface();
-             surface.selectAll('.related').classed('related', false);
 
-             if (_relatedParent) {
-               surface.selectAll(utilEntitySelector([_relatedParent])).classed('related', true);
+               if (status) break;
              }
            }
-         };
 
-         mode.exit = function () {
-           _newFeature = false;
+           if (status) {
+             classes.push('tag-status');
+             classes.push('tag-status-' + status);
+           } // add any secondary tags
 
-           _operations.forEach(function (operation) {
-             if (operation.behavior) {
-               context.uninstall(operation.behavior);
+
+           for (i = 0; i < secondaries.length; i++) {
+             k = secondaries[i];
+             v = t[k];
+             if (!v || v === 'no' || k === primary) continue;
+             classes.push('tag-' + k);
+             classes.push('tag-' + k + '-' + v);
+           } // For highways, look for surface tagging..
+
+
+           if (primary === 'highway' && !osmPathHighwayTagValues[t.highway] || primary === 'aeroway') {
+             var surface = t.highway === 'track' ? 'unpaved' : 'paved';
+
+             for (k in t) {
+               v = t[k];
+
+               if (k in osmPavedTags) {
+                 surface = osmPavedTags[k][v] ? 'paved' : 'unpaved';
+               }
+
+               if (k in osmSemipavedTags && !!osmSemipavedTags[k][v]) {
+                 surface = 'semipaved';
+               }
              }
-           });
 
-           _operations = [];
+             classes.push('tag-' + surface);
+           } // If this is a wikidata-tagged item, add a class for that..
 
-           _behaviors.forEach(context.uninstall);
 
-           select(document).call(keybinding.unbind);
-           context.ui().closeEditMenu();
-           context.history().on('change.select', null).on('undone.select', null).on('redone.select', null);
-           var surface = context.surface();
-           surface.selectAll('.selected-member').classed('selected-member', false);
-           surface.selectAll('.selected').classed('selected', false);
-           surface.selectAll('.highlighted').classed('highlighted', false);
-           surface.selectAll('.related').classed('related', false);
-           context.map().on('drawn.select', null);
-           context.ui().sidebar.hide();
-           context.features().forceVisible([]);
-           var entity = singular();
+           var qid = t.wikidata || t['flag:wikidata'] || t['brand:wikidata'] || t['network:wikidata'] || t['operator:wikidata'];
 
-           if (_newFeature && entity && entity.type === 'relation' && // no tags
-           Object.keys(entity.tags).length === 0 && // no parent relations
-           context.graph().parentRelations(entity).length === 0 && ( // no members or one member with no role
-           entity.members.length === 0 || entity.members.length === 1 && !entity.members[0].role)) {
-             // the user added this relation but didn't edit it at all, so just delete it
-             var deleteAction = actionDeleteRelation(entity.id, true
-             /* don't delete untagged members */
-             );
-             context.perform(deleteAction, _t('operations.delete.annotation.relation'));
+           if (qid) {
+             classes.push('tag-wikidata');
            }
+
+           return classes.join(' ').trim();
          };
 
-         return mode;
-       }
+         tagClasses.tags = function (val) {
+           if (!arguments.length) return _tags;
+           _tags = val;
+           return tagClasses;
+         };
 
-       function uiLasso(context) {
-         var group, polygon;
-         lasso.coordinates = [];
+         return tagClasses;
+       }
 
-         function lasso(selection) {
-           context.container().classed('lasso', true);
-           group = selection.append('g').attr('class', 'lasso hide');
-           polygon = group.append('path').attr('class', 'lasso-path');
-           group.call(uiToggle(true));
+       // Patterns only work in Firefox when set directly on element.
+       // (This is not a bug: https://bugzilla.mozilla.org/show_bug.cgi?id=750632)
+       var patterns = {
+         // tag - pattern name
+         // -or-
+         // tag - value - pattern name
+         // -or-
+         // tag - value - rules (optional tag-values, pattern name)
+         // (matches earlier rules first, so fallback should be last entry)
+         amenity: {
+           grave_yard: 'cemetery',
+           fountain: 'water_standing'
+         },
+         landuse: {
+           cemetery: [{
+             religion: 'christian',
+             pattern: 'cemetery_christian'
+           }, {
+             religion: 'buddhist',
+             pattern: 'cemetery_buddhist'
+           }, {
+             religion: 'muslim',
+             pattern: 'cemetery_muslim'
+           }, {
+             religion: 'jewish',
+             pattern: 'cemetery_jewish'
+           }, {
+             pattern: 'cemetery'
+           }],
+           construction: 'construction',
+           farmland: 'farmland',
+           farmyard: 'farmyard',
+           forest: [{
+             leaf_type: 'broadleaved',
+             pattern: 'forest_broadleaved'
+           }, {
+             leaf_type: 'needleleaved',
+             pattern: 'forest_needleleaved'
+           }, {
+             leaf_type: 'leafless',
+             pattern: 'forest_leafless'
+           }, {
+             pattern: 'forest'
+           } // same as 'leaf_type:mixed'
+           ],
+           grave_yard: 'cemetery',
+           grass: [{
+             golf: 'green',
+             pattern: 'golf_green'
+           }, {
+             pattern: 'grass'
+           }],
+           landfill: 'landfill',
+           meadow: 'meadow',
+           military: 'construction',
+           orchard: 'orchard',
+           quarry: 'quarry',
+           vineyard: 'vineyard'
+         },
+         natural: {
+           beach: 'beach',
+           grassland: 'grass',
+           sand: 'beach',
+           scrub: 'scrub',
+           water: [{
+             water: 'pond',
+             pattern: 'pond'
+           }, {
+             water: 'reservoir',
+             pattern: 'water_standing'
+           }, {
+             pattern: 'waves'
+           }],
+           wetland: [{
+             wetland: 'marsh',
+             pattern: 'wetland_marsh'
+           }, {
+             wetland: 'swamp',
+             pattern: 'wetland_swamp'
+           }, {
+             wetland: 'bog',
+             pattern: 'wetland_bog'
+           }, {
+             wetland: 'reedbed',
+             pattern: 'wetland_reedbed'
+           }, {
+             pattern: 'wetland'
+           }],
+           wood: [{
+             leaf_type: 'broadleaved',
+             pattern: 'forest_broadleaved'
+           }, {
+             leaf_type: 'needleleaved',
+             pattern: 'forest_needleleaved'
+           }, {
+             leaf_type: 'leafless',
+             pattern: 'forest_leafless'
+           }, {
+             pattern: 'forest'
+           } // same as 'leaf_type:mixed'
+           ]
+         },
+         traffic_calming: {
+           island: [{
+             surface: 'grass',
+             pattern: 'grass'
+           }],
+           chicane: [{
+             surface: 'grass',
+             pattern: 'grass'
+           }],
+           choker: [{
+             surface: 'grass',
+             pattern: 'grass'
+           }]
          }
-
-         function draw() {
-           if (polygon) {
-             polygon.data([lasso.coordinates]).attr('d', function (d) {
-               return 'M' + d.join(' L') + ' Z';
-             });
-           }
+       };
+       function svgTagPattern(tags) {
+         // Skip pattern filling if this is a building (buildings don't get patterns applied)
+         if (tags.building && tags.building !== 'no') {
+           return null;
          }
 
-         lasso.extent = function () {
-           return lasso.coordinates.reduce(function (extent, point) {
-             return extent.extend(geoExtent(point));
-           }, geoExtent());
-         };
+         for (var tag in patterns) {
+           var entityValue = tags[tag];
+           if (!entityValue) continue;
 
-         lasso.p = function (_) {
-           if (!arguments.length) return lasso;
-           lasso.coordinates.push(_);
-           draw();
-           return lasso;
-         };
+           if (typeof patterns[tag] === 'string') {
+             // extra short syntax (just tag) - pattern name
+             return 'pattern-' + patterns[tag];
+           } else {
+             var values = patterns[tag];
 
-         lasso.close = function () {
-           if (group) {
-             group.call(uiToggle(false, function () {
-               select(this).remove();
-             }));
-           }
+             for (var value in values) {
+               if (entityValue !== value) continue;
+               var rules = values[value];
 
-           context.container().classed('lasso', false);
-         };
+               if (typeof rules === 'string') {
+                 // short syntax - pattern name
+                 return 'pattern-' + rules;
+               } // long syntax - rule array
+
+
+               for (var ruleKey in rules) {
+                 var rule = rules[ruleKey];
+                 var pass = true;
+
+                 for (var criterion in rule) {
+                   if (criterion !== 'pattern') {
+                     // reserved for pattern name
+                     // The only rule is a required tag-value pair
+                     var v = tags[criterion];
+
+                     if (!v || v !== rule[criterion]) {
+                       pass = false;
+                       break;
+                     }
+                   }
+                 }
+
+                 if (pass) {
+                   return 'pattern-' + rule.pattern;
+                 }
+               }
+             }
+           }
+         }
 
-         return lasso;
+         return null;
        }
 
-       function behaviorLasso(context) {
-         // use pointer events on supported platforms; fallback to mouse events
-         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
+       function svgAreas(projection, context) {
+         function getPatternStyle(tags) {
+           var imageID = svgTagPattern(tags);
 
-         var behavior = function behavior(selection) {
-           var lasso;
+           if (imageID) {
+             return 'url("#ideditor-' + imageID + '")';
+           }
 
-           function pointerdown(d3_event) {
-             var button = 0; // left
+           return '';
+         }
 
-             if (d3_event.button === button && d3_event.shiftKey === true) {
-               lasso = null;
-               select(window).on(_pointerPrefix + 'move.lasso', pointermove).on(_pointerPrefix + 'up.lasso', pointerup);
-               d3_event.stopPropagation();
-             }
-           }
+         function drawTargets(selection, graph, entities, filter) {
+           var targetClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
+           var nopeClass = context.getDebug('target') ? 'red ' : 'nocolor ';
+           var getPath = svgPath(projection).geojson;
+           var activeID = context.activeID();
+           var base = context.history().base(); // The targets and nopes will be MultiLineString sub-segments of the ways
 
-           function pointermove() {
-             if (!lasso) {
-               lasso = uiLasso(context);
-               context.surface().call(lasso);
-             }
+           var data = {
+             targets: [],
+             nopes: []
+           };
+           entities.forEach(function (way) {
+             var features = svgSegmentWay(way, graph, activeID);
+             data.targets.push.apply(data.targets, features.passive);
+             data.nopes.push.apply(data.nopes, features.active);
+           }); // Targets allow hover and vertex snapping
 
-             lasso.p(context.map().mouse());
-           }
+           var targetData = data.targets.filter(getPath);
+           var targets = selection.selectAll('.area.target-allowed').filter(function (d) {
+             return filter(d.properties.entity);
+           }).data(targetData, function key(d) {
+             return d.id;
+           }); // exit
 
-           function normalize(a, b) {
-             return [[Math.min(a[0], b[0]), Math.min(a[1], b[1])], [Math.max(a[0], b[0]), Math.max(a[1], b[1])]];
-           }
+           targets.exit().remove();
 
-           function lassoed() {
-             if (!lasso) return [];
-             var graph = context.graph();
-             var limitToNodes;
+           var segmentWasEdited = function segmentWasEdited(d) {
+             var wayID = d.properties.entity.id; // if the whole line was edited, don't draw segment changes
 
-             if (context.map().editableDataEnabled(true
-             /* skipZoomCheck */
-             ) && context.map().isInWideSelection()) {
-               // only select from the visible nodes
-               limitToNodes = new Set(utilGetAllNodes(context.selectedIDs(), graph));
-             } else if (!context.map().editableDataEnabled()) {
-               return [];
+             if (!base.entities[wayID] || !fastDeepEqual(graph.entities[wayID].nodes, base.entities[wayID].nodes)) {
+               return false;
              }
 
-             var bounds = lasso.extent().map(context.projection.invert);
-             var extent = geoExtent(normalize(bounds[0], bounds[1]));
-             var intersects = context.history().intersects(extent).filter(function (entity) {
-               return entity.type === 'node' && (!limitToNodes || limitToNodes.has(entity)) && geoPointInPolygon(context.projection(entity.loc), lasso.coordinates) && !context.features().isHidden(entity, graph, entity.geometry(graph));
-             }); // sort the lassoed nodes as best we can
+             return d.properties.nodes.some(function (n) {
+               return !base.entities[n.id] || !fastDeepEqual(graph.entities[n.id].loc, base.entities[n.id].loc);
+             });
+           }; // enter/update
 
-             intersects.sort(function (node1, node2) {
-               var parents1 = graph.parentWays(node1);
-               var parents2 = graph.parentWays(node2);
 
-               if (parents1.length && parents2.length) {
-                 // both nodes are vertices
-                 var sharedParents = utilArrayIntersection(parents1, parents2);
+           targets.enter().append('path').merge(targets).attr('d', getPath).attr('class', function (d) {
+             return 'way area target target-allowed ' + targetClass + d.id;
+           }).classed('segment-edited', segmentWasEdited); // NOPE
 
-                 if (sharedParents.length) {
-                   var sharedParentNodes = sharedParents[0].nodes; // vertices are members of the same way; sort them in their listed order
+           var nopeData = data.nopes.filter(getPath);
+           var nopes = selection.selectAll('.area.target-nope').filter(function (d) {
+             return filter(d.properties.entity);
+           }).data(nopeData, function key(d) {
+             return d.id;
+           }); // exit
 
-                   return sharedParentNodes.indexOf(node1.id) - sharedParentNodes.indexOf(node2.id);
-                 } else {
-                   // vertices do not share a way; group them by their respective parent ways
-                   return parseFloat(parents1[0].id.slice(1)) - parseFloat(parents2[0].id.slice(1));
-                 }
-               } else if (parents1.length || parents2.length) {
-                 // only one node is a vertex; sort standalone points before vertices
-                 return parents1.length - parents2.length;
-               } // both nodes are standalone points; sort left to right
+           nopes.exit().remove(); // enter/update
 
+           nopes.enter().append('path').merge(nopes).attr('d', getPath).attr('class', function (d) {
+             return 'way area target target-nope ' + nopeClass + d.id;
+           }).classed('segment-edited', segmentWasEdited);
+         }
 
-               return node1.loc[0] - node2.loc[0];
-             });
-             return intersects.map(function (entity) {
-               return entity.id;
-             });
-           }
+         function drawAreas(selection, graph, entities, filter) {
+           var path = svgPath(projection, graph, true);
+           var areas = {};
+           var multipolygon;
+           var base = context.history().base();
 
-           function pointerup() {
-             select(window).on(_pointerPrefix + 'move.lasso', null).on(_pointerPrefix + 'up.lasso', null);
-             if (!lasso) return;
-             var ids = lassoed();
-             lasso.close();
+           for (var i = 0; i < entities.length; i++) {
+             var entity = entities[i];
+             if (entity.geometry(graph) !== 'area') continue;
+             multipolygon = osmIsOldMultipolygonOuterMember(entity, graph);
 
-             if (ids.length) {
-               context.enter(modeSelect(context, ids));
+             if (multipolygon) {
+               areas[multipolygon.id] = {
+                 entity: multipolygon.mergeTags(entity.tags),
+                 area: Math.abs(entity.area(graph))
+               };
+             } else if (!areas[entity.id]) {
+               areas[entity.id] = {
+                 entity: entity,
+                 area: Math.abs(entity.area(graph))
+               };
              }
            }
 
-           selection.on(_pointerPrefix + 'down.lasso', pointerdown);
-         };
-
-         behavior.off = function (selection) {
-           selection.on(_pointerPrefix + 'down.lasso', null);
-         };
+           var fills = Object.values(areas).filter(function hasPath(a) {
+             return path(a.entity);
+           });
+           fills.sort(function areaSort(a, b) {
+             return b.area - a.area;
+           });
+           fills = fills.map(function (a) {
+             return a.entity;
+           });
+           var strokes = fills.filter(function (area) {
+             return area.type === 'way';
+           });
+           var data = {
+             clip: fills,
+             shadow: strokes,
+             stroke: strokes,
+             fill: fills
+           };
+           var clipPaths = context.surface().selectAll('defs').selectAll('.clipPath-osm').filter(filter).data(data.clip, osmEntity.key);
+           clipPaths.exit().remove();
+           var clipPathsEnter = clipPaths.enter().append('clipPath').attr('class', 'clipPath-osm').attr('id', function (entity) {
+             return 'ideditor-' + entity.id + '-clippath';
+           });
+           clipPathsEnter.append('path');
+           clipPaths.merge(clipPathsEnter).selectAll('path').attr('d', path);
+           var drawLayer = selection.selectAll('.layer-osm.areas');
+           var touchLayer = selection.selectAll('.layer-touch.areas'); // Draw areas..
 
-         return behavior;
-       }
+           var areagroup = drawLayer.selectAll('g.areagroup').data(['fill', 'shadow', 'stroke']);
+           areagroup = areagroup.enter().append('g').attr('class', function (d) {
+             return 'areagroup area-' + d;
+           }).merge(areagroup);
+           var paths = areagroup.selectAll('path').filter(filter).data(function (layer) {
+             return data[layer];
+           }, osmEntity.key);
+           paths.exit().remove();
+           var fillpaths = selection.selectAll('.area-fill path.area').nodes();
+           var bisect = d3_bisector(function (node) {
+             return -node.__data__.area(graph);
+           }).left;
 
-       function modeBrowse(context) {
-         var mode = {
-           button: 'browse',
-           id: 'browse',
-           title: _t('modes.browse.title'),
-           description: _t('modes.browse.description')
-         };
-         var sidebar;
+           function sortedByArea(entity) {
+             if (this._parent.__data__ === 'fill') {
+               return fillpaths[bisect(fillpaths, -entity.area(graph))];
+             }
+           }
 
-         var _selectBehavior;
+           paths = paths.enter().insert('path', sortedByArea).merge(paths).each(function (entity) {
+             var layer = this.parentNode.__data__;
+             this.setAttribute('class', entity.type + ' area ' + layer + ' ' + entity.id);
 
-         var _behaviors = [];
+             if (layer === 'fill') {
+               this.setAttribute('clip-path', 'url(#ideditor-' + entity.id + '-clippath)');
+               this.style.fill = this.style.stroke = getPatternStyle(entity.tags);
+             }
+           }).classed('added', function (d) {
+             return !base.entities[d.id];
+           }).classed('geometry-edited', function (d) {
+             return graph.entities[d.id] && base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].nodes, base.entities[d.id].nodes);
+           }).classed('retagged', function (d) {
+             return graph.entities[d.id] && base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].tags, base.entities[d.id].tags);
+           }).call(svgTagClasses()).attr('d', path); // Draw touch targets..
 
-         mode.selectBehavior = function (val) {
-           if (!arguments.length) return _selectBehavior;
-           _selectBehavior = val;
-           return mode;
-         };
+           touchLayer.call(drawTargets, graph, data.stroke, filter);
+         }
 
-         mode.enter = function () {
-           if (!_behaviors.length) {
-             if (!_selectBehavior) _selectBehavior = behaviorSelect(context);
-             _behaviors = [behaviorPaste(context), behaviorHover(context).on('hover', context.ui().sidebar.hover), _selectBehavior, behaviorLasso(context), modeDragNode(context).behavior, modeDragNote(context).behavior];
-           }
+         return drawAreas;
+       }
 
-           _behaviors.forEach(context.install); // Get focus on the body.
+       var fastJsonStableStringify = function fastJsonStableStringify(data, opts) {
+         if (!opts) opts = {};
+         if (typeof opts === 'function') opts = {
+           cmp: opts
+         };
+         var cycles = typeof opts.cycles === 'boolean' ? opts.cycles : false;
 
+         var cmp = opts.cmp && function (f) {
+           return function (node) {
+             return function (a, b) {
+               var aobj = {
+                 key: a,
+                 value: node[a]
+               };
+               var bobj = {
+                 key: b,
+                 value: node[b]
+               };
+               return f(aobj, bobj);
+             };
+           };
+         }(opts.cmp);
 
-           if (document.activeElement && document.activeElement.blur) {
-             document.activeElement.blur();
+         var seen = [];
+         return function stringify(node) {
+           if (node && node.toJSON && typeof node.toJSON === 'function') {
+             node = node.toJSON();
            }
 
-           if (sidebar) {
-             context.ui().sidebar.show(sidebar);
-           } else {
-             context.ui().sidebar.select(null);
-           }
-         };
+           if (node === undefined) return;
+           if (typeof node == 'number') return isFinite(node) ? '' + node : 'null';
+           if (_typeof(node) !== 'object') return JSON.stringify(node);
+           var i, out;
 
-         mode.exit = function () {
-           context.ui().sidebar.hover.cancel();
+           if (Array.isArray(node)) {
+             out = '[';
 
-           _behaviors.forEach(context.uninstall);
+             for (i = 0; i < node.length; i++) {
+               if (i) out += ',';
+               out += stringify(node[i]) || 'null';
+             }
 
-           if (sidebar) {
-             context.ui().sidebar.hide();
+             return out + ']';
            }
-         };
 
-         mode.sidebar = function (_) {
-           if (!arguments.length) return sidebar;
-           sidebar = _;
-           return mode;
-         };
+           if (node === null) return 'null';
 
-         mode.operations = function () {
-           return [operationPaste(context)];
-         };
+           if (seen.indexOf(node) !== -1) {
+             if (cycles) return JSON.stringify('__cycle__');
+             throw new TypeError('Converting circular structure to JSON');
+           }
 
-         return mode;
-       }
+           var seenIndex = seen.push(node) - 1;
+           var keys = Object.keys(node).sort(cmp && cmp(node));
+           out = '';
 
-       function behaviorAddWay(context) {
-         var dispatch$1 = dispatch('start', 'startFromWay', 'startFromNode');
-         var draw = behaviorDraw(context);
+           for (i = 0; i < keys.length; i++) {
+             var key = keys[i];
+             var value = stringify(node[key]);
+             if (!value) continue;
+             if (out) out += ',';
+             out += JSON.stringify(key) + ':' + value;
+           }
 
-         function behavior(surface) {
-           draw.on('click', function () {
-             dispatch$1.apply('start', this, arguments);
-           }).on('clickWay', function () {
-             dispatch$1.apply('startFromWay', this, arguments);
-           }).on('clickNode', function () {
-             dispatch$1.apply('startFromNode', this, arguments);
-           }).on('cancel', behavior.cancel).on('finish', behavior.cancel);
-           context.map().dblclickZoomEnable(false);
-           surface.call(draw);
-         }
+           seen.splice(seenIndex, 1);
+           return '{' + out + '}';
+         }(data);
+       };
 
-         behavior.off = function (surface) {
-           surface.call(draw.off);
-         };
+       //[4]           NameStartChar      ::=          ":" | [A-Z] | "_" | [a-z] | [#xC0-#xD6] | [#xD8-#xF6] | [#xF8-#x2FF] | [#x370-#x37D] | [#x37F-#x1FFF] | [#x200C-#x200D] | [#x2070-#x218F] | [#x2C00-#x2FEF] | [#x3001-#xD7FF] | [#xF900-#xFDCF] | [#xFDF0-#xFFFD] | [#x10000-#xEFFFF]
+       //[4a]          NameChar           ::=          NameStartChar | "-" | "." | [0-9] | #xB7 | [#x0300-#x036F] | [#x203F-#x2040]
+       //[5]           Name       ::=          NameStartChar (NameChar)*
+       var nameStartChar = /[A-Z_a-z\xC0-\xD6\xD8-\xF6\u00F8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD]/; //\u10000-\uEFFFF
 
-         behavior.cancel = function () {
-           window.setTimeout(function () {
-             context.map().dblclickZoomEnable(true);
-           }, 1000);
-           context.enter(modeBrowse(context));
-         };
+       var nameChar = new RegExp("[\\-\\.0-9" + nameStartChar.source.slice(1, -1) + "\\u00B7\\u0300-\\u036F\\u203F-\\u2040]");
+       var tagNamePattern = new RegExp('^' + nameStartChar.source + nameChar.source + '*(?:\:' + nameStartChar.source + nameChar.source + '*)?$'); //var tagNamePattern = /^[a-zA-Z_][\w\-\.]*(?:\:[a-zA-Z_][\w\-\.]*)?$/
+       //var handlers = 'resolveEntity,getExternalSubset,characters,endDocument,endElement,endPrefixMapping,ignorableWhitespace,processingInstruction,setDocumentLocator,skippedEntity,startDocument,startElement,startPrefixMapping,notationDecl,unparsedEntityDecl,error,fatalError,warning,attributeDecl,elementDecl,externalEntityDecl,internalEntityDecl,comment,endCDATA,endDTD,endEntity,startCDATA,startDTD,startEntity'.split(',')
+       //S_TAG,        S_ATTR, S_EQ,   S_ATTR_NOQUOT_VALUE
+       //S_ATTR_SPACE, S_ATTR_END,     S_TAG_SPACE, S_TAG_CLOSE
 
-         return utilRebind(behavior, dispatch$1, 'on');
-       }
+       var S_TAG = 0; //tag name offerring
 
-       function behaviorHash(context) {
-         // cached window.location.hash
-         var _cachedHash = null; // allowable latitude range
+       var S_ATTR = 1; //attr name offerring 
 
-         var _latitudeLimit = 90 - 1e-8;
+       var S_ATTR_SPACE = 2; //attr name end and space offer
 
-         function computedHashParameters() {
-           var map = context.map();
-           var center = map.center();
-           var zoom = map.zoom();
-           var precision = Math.max(0, Math.ceil(Math.log(zoom) / Math.LN2));
-           var oldParams = utilObjectOmit(utilStringQs(window.location.hash), ['comment', 'source', 'hashtags', 'walkthrough']);
-           var newParams = {};
-           delete oldParams.id;
-           var selected = context.selectedIDs().filter(function (id) {
-             return context.hasEntity(id);
-           });
+       var S_EQ = 3; //=space?
 
-           if (selected.length) {
-             newParams.id = selected.join(',');
-           }
+       var S_ATTR_NOQUOT_VALUE = 4; //attr value(no quot value only)
 
-           newParams.map = zoom.toFixed(2) + '/' + center[1].toFixed(precision) + '/' + center[0].toFixed(precision);
-           return Object.assign(oldParams, newParams);
-         }
+       var S_ATTR_END = 5; //attr value end and no space(quot end)
 
-         function computedHash() {
-           return '#' + utilQsString(computedHashParameters(), true);
-         }
+       var S_TAG_SPACE = 6; //(attr value end || tag end ) && (space offer)
 
-         function computedTitle(includeChangeCount) {
-           var baseTitle = context.documentTitleBase() || 'iD';
-           var contextual;
-           var changeCount;
-           var titleID;
-           var selected = context.selectedIDs().filter(function (id) {
-             return context.hasEntity(id);
-           });
+       var S_TAG_CLOSE = 7; //closed el<el />
 
-           if (selected.length) {
-             var firstLabel = utilDisplayLabel(context.entity(selected[0]), context.graph());
+       function XMLReader() {}
 
-             if (selected.length > 1) {
-               contextual = _t('title.labeled_and_more', {
-                 labeled: firstLabel,
-                 count: selected.length - 1
-               });
-             } else {
-               contextual = firstLabel;
-             }
+       XMLReader.prototype = {
+         parse: function parse(source, defaultNSMap, entityMap) {
+           var domBuilder = this.domBuilder;
+           domBuilder.startDocument();
 
-             titleID = 'context';
-           }
+           _copy(defaultNSMap, defaultNSMap = {});
 
-           if (includeChangeCount) {
-             changeCount = context.history().difference().summary().length;
+           _parse(source, defaultNSMap, entityMap, domBuilder, this.errorHandler);
 
-             if (changeCount > 0) {
-               titleID = contextual ? 'changes_context' : 'changes';
-             }
-           }
+           domBuilder.endDocument();
+         }
+       };
 
-           if (titleID) {
-             return _t('title.format.' + titleID, {
-               changes: changeCount,
-               base: baseTitle,
-               context: contextual
-             });
+       function _parse(source, defaultNSMapCopy, entityMap, domBuilder, errorHandler) {
+         function fixedFromCharCode(code) {
+           // String.prototype.fromCharCode does not supports
+           // > 2 bytes unicode chars directly
+           if (code > 0xffff) {
+             code -= 0x10000;
+             var surrogate1 = 0xd800 + (code >> 10),
+                 surrogate2 = 0xdc00 + (code & 0x3ff);
+             return String.fromCharCode(surrogate1, surrogate2);
+           } else {
+             return String.fromCharCode(code);
            }
-
-           return baseTitle;
          }
 
-         function updateTitle(includeChangeCount) {
-           if (!context.setsDocumentTitle()) return;
-           var newTitle = computedTitle(includeChangeCount);
+         function entityReplacer(a) {
+           var k = a.slice(1, -1);
 
-           if (document.title !== newTitle) {
-             document.title = newTitle;
+           if (k in entityMap) {
+             return entityMap[k];
+           } else if (k.charAt(0) === '#') {
+             return fixedFromCharCode(parseInt(k.substr(1).replace('x', '0x')));
+           } else {
+             errorHandler.error('entity not found:' + a);
+             return a;
            }
          }
 
-         function updateHashIfNeeded() {
-           if (context.inIntro()) return;
-           var latestHash = computedHash();
-
-           if (_cachedHash !== latestHash) {
-             _cachedHash = latestHash; // Update the URL hash without affecting the browser navigation stack,
-             // though unavoidably creating a browser history entry
-
-             window.history.replaceState(null, computedTitle(false
-             /* includeChangeCount */
-             ), latestHash); // set the title we want displayed for the browser tab/window
-
-             updateTitle(true
-             /* includeChangeCount */
-             );
+         function appendText(end) {
+           //has some bugs
+           if (end > start) {
+             var xt = source.substring(start, end).replace(/&#?\w+;/g, entityReplacer);
+             locator && position(start);
+             domBuilder.characters(xt, 0, end - start);
+             start = end;
            }
          }
 
-         var _throttledUpdate = throttle(updateHashIfNeeded, 500);
+         function position(p, m) {
+           while (p >= lineEnd && (m = linePattern.exec(source))) {
+             lineStart = m.index;
+             lineEnd = lineStart + m[0].length;
+             locator.lineNumber++; //console.log('line++:',locator,startPos,endPos)
+           }
 
-         var _throttledUpdateTitle = throttle(function () {
-           updateTitle(true
-           /* includeChangeCount */
-           );
-         }, 500);
+           locator.columnNumber = p - lineStart + 1;
+         }
 
-         function hashchange() {
-           // ignore spurious hashchange events
-           if (window.location.hash === _cachedHash) return;
-           _cachedHash = window.location.hash;
-           var q = utilStringQs(_cachedHash);
-           var mapArgs = (q.map || '').split('/').map(Number);
+         var lineStart = 0;
+         var lineEnd = 0;
+         var linePattern = /.*(?:\r\n?|\n)|.*$/g;
+         var locator = domBuilder.locator;
+         var parseStack = [{
+           currentNSMap: defaultNSMapCopy
+         }];
+         var closeMap = {};
+         var start = 0;
 
-           if (mapArgs.length < 3 || mapArgs.some(isNaN)) {
-             // replace bogus hash
-             updateHashIfNeeded();
-           } else {
-             // don't update if the new hash already reflects the state of iD
-             if (_cachedHash === computedHash()) return;
-             var mode = context.mode();
-             context.map().centerZoom([mapArgs[2], Math.min(_latitudeLimit, Math.max(-_latitudeLimit, mapArgs[1]))], mapArgs[0]);
+         while (true) {
+           try {
+             var tagStart = source.indexOf('<', start);
+
+             if (tagStart < 0) {
+               if (!source.substr(start).match(/^\s*$/)) {
+                 var doc = domBuilder.doc;
+                 var text = doc.createTextNode(source.substr(start));
+                 doc.appendChild(text);
+                 domBuilder.currentElement = text;
+               }
 
-             if (q.id && mode) {
-               var ids = q.id.split(',').filter(function (id) {
-                 return context.hasEntity(id);
-               });
+               return;
+             }
 
-               if (ids.length && (mode.id === 'browse' || mode.id === 'select' && !utilArrayIdentical(mode.selectedIDs(), ids))) {
-                 context.enter(modeSelect(context, ids));
-                 return;
-               }
+             if (tagStart > start) {
+               appendText(tagStart);
              }
 
-             var center = context.map().center();
-             var dist = geoSphericalDistance(center, [mapArgs[2], mapArgs[1]]);
-             var maxdist = 500; // Don't allow the hash location to change too much while drawing
-             // This can happen if the user accidentally hit the back button.  #3996
+             switch (source.charAt(tagStart + 1)) {
+               case '/':
+                 var end = source.indexOf('>', tagStart + 3);
+                 var tagName = source.substring(tagStart + 2, end);
+                 var config = parseStack.pop();
 
-             if (mode && mode.id.match(/^draw/) !== null && dist > maxdist) {
-               context.enter(modeBrowse(context));
-               return;
-             }
-           }
-         }
+                 if (end < 0) {
+                   tagName = source.substring(tagStart + 2).replace(/[\s<].*/, ''); //console.error('#@@@@@@'+tagName)
 
-         function behavior() {
-           context.map().on('move.behaviorHash', _throttledUpdate);
-           context.history().on('change.behaviorHash', _throttledUpdateTitle);
-           context.on('enter.behaviorHash', _throttledUpdate);
-           select(window).on('hashchange.behaviorHash', hashchange);
+                   errorHandler.error("end tag name: " + tagName + ' is not complete:' + config.tagName);
+                   end = tagStart + 1 + tagName.length;
+                 } else if (tagName.match(/\s</)) {
+                   tagName = tagName.replace(/[\s<].*/, '');
+                   errorHandler.error("end tag name: " + tagName + ' maybe not complete');
+                   end = tagStart + 1 + tagName.length;
+                 } //console.error(parseStack.length,parseStack)
+                 //console.error(config);
 
-           if (window.location.hash) {
-             var q = utilStringQs(window.location.hash);
 
-             if (q.id) {
-               //if (!context.history().hasRestorableChanges()) {
-               // targeting specific features: download, select, and zoom to them
-               context.zoomToEntity(q.id.split(',')[0], !q.map); //}
-             }
+                 var localNSMap = config.localNSMap;
+                 var endMatch = config.tagName == tagName;
+                 var endIgnoreCaseMach = endMatch || config.tagName && config.tagName.toLowerCase() == tagName.toLowerCase();
 
-             if (q.walkthrough === 'true') {
-               behavior.startWalkthrough = true;
-             }
+                 if (endIgnoreCaseMach) {
+                   domBuilder.endElement(config.uri, config.localName, tagName);
 
-             if (q.map) {
-               behavior.hadHash = true;
-             }
+                   if (localNSMap) {
+                     for (var prefix in localNSMap) {
+                       domBuilder.endPrefixMapping(prefix);
+                     }
+                   }
 
-             hashchange();
-             updateTitle(false);
-           }
-         }
+                   if (!endMatch) {
+                     errorHandler.fatalError("end tag name: " + tagName + ' is not match the current start tagName:' + config.tagName);
+                   }
+                 } else {
+                   parseStack.push(config);
+                 }
 
-         behavior.off = function () {
-           _throttledUpdate.cancel();
+                 end++;
+                 break;
+               // end elment
 
-           _throttledUpdateTitle.cancel();
+               case '?':
+                 // <?...?>
+                 locator && position(tagStart);
+                 end = parseInstruction(source, tagStart, domBuilder);
+                 break;
 
-           context.map().on('move.behaviorHash', null);
-           context.on('enter.behaviorHash', null);
-           select(window).on('hashchange.behaviorHash', null);
-           window.location.hash = '';
-         };
+               case '!':
+                 // <!doctype,<![CDATA,<!--
+                 locator && position(tagStart);
+                 end = parseDCC(source, tagStart, domBuilder, errorHandler);
+                 break;
 
-         return behavior;
-       }
+               default:
+                 locator && position(tagStart);
+                 var el = new ElementAttributes();
+                 var currentNSMap = parseStack[parseStack.length - 1].currentNSMap; //elStartEnd
 
-       /*
-           iD.coreDifference represents the difference between two graphs.
-           It knows how to calculate the set of entities that were
-           created, modified, or deleted, and also contains the logic
-           for recursively extending a difference to the complete set
-           of entities that will require a redraw, taking into account
-           child and parent relationships.
-        */
+                 var end = parseElementStartPart(source, tagStart, el, currentNSMap, entityReplacer, errorHandler);
+                 var len = el.length;
 
-       function coreDifference(base, head) {
-         var _changes = {};
-         var _didChange = {}; // 'addition', 'deletion', 'geometry', 'properties'
+                 if (!el.closed && fixSelfClosed(source, end, el.tagName, closeMap)) {
+                   el.closed = true;
 
-         var _diff = {};
+                   if (!entityMap.nbsp) {
+                     errorHandler.warning('unclosed xml attribute');
+                   }
+                 }
 
-         function checkEntityID(id) {
-           var h = head.entities[id];
-           var b = base.entities[id];
-           if (h === b) return;
-           if (_changes[id]) return;
+                 if (locator && len) {
+                   var locator2 = copyLocator(locator, {}); //try{//attribute position fixed
 
-           if (!h && b) {
-             _changes[id] = {
-               base: b,
-               head: h
-             };
-             _didChange.deletion = true;
-             return;
-           }
+                   for (var i = 0; i < len; i++) {
+                     var a = el[i];
+                     position(a.offset);
+                     a.locator = copyLocator(locator, {});
+                   } //}catch(e){console.error('@@@@@'+e)}
 
-           if (h && !b) {
-             _changes[id] = {
-               base: b,
-               head: h
-             };
-             _didChange.addition = true;
-             return;
-           }
 
-           if (h && b) {
-             if (h.members && b.members && !fastDeepEqual(h.members, b.members)) {
-               _changes[id] = {
-                 base: b,
-                 head: h
-               };
-               _didChange.geometry = true;
-               _didChange.properties = true;
-               return;
-             }
+                   domBuilder.locator = locator2;
 
-             if (h.loc && b.loc && !geoVecEqual(h.loc, b.loc)) {
-               _changes[id] = {
-                 base: b,
-                 head: h
-               };
-               _didChange.geometry = true;
-             }
+                   if (appendElement(el, domBuilder, currentNSMap)) {
+                     parseStack.push(el);
+                   }
 
-             if (h.nodes && b.nodes && !fastDeepEqual(h.nodes, b.nodes)) {
-               _changes[id] = {
-                 base: b,
-                 head: h
-               };
-               _didChange.geometry = true;
-             }
+                   domBuilder.locator = locator;
+                 } else {
+                   if (appendElement(el, domBuilder, currentNSMap)) {
+                     parseStack.push(el);
+                   }
+                 }
+
+                 if (el.uri === 'http://www.w3.org/1999/xhtml' && !el.closed) {
+                   end = parseHtmlSpecialContent(source, end, el.tagName, entityReplacer, domBuilder);
+                 } else {
+                   end++;
+                 }
 
-             if (h.tags && b.tags && !fastDeepEqual(h.tags, b.tags)) {
-               _changes[id] = {
-                 base: b,
-                 head: h
-               };
-               _didChange.properties = true;
              }
-           }
-         }
+           } catch (e) {
+             errorHandler.error('element parse error: ' + e); //errorHandler.error('element parse error: '+e);
 
-         function load() {
-           // HOT CODE: there can be many thousands of downloaded entities, so looping
-           // through them all can become a performance bottleneck. Optimize by
-           // resolving duplicates and using a basic `for` loop
-           var ids = utilArrayUniq(Object.keys(head.entities).concat(Object.keys(base.entities)));
+             end = -1; //throw e;
+           }
 
-           for (var i = 0; i < ids.length; i++) {
-             checkEntityID(ids[i]);
+           if (end > start) {
+             start = end;
+           } else {
+             //TODO: 这里有可能sax回退,有位置错误风险
+             appendText(Math.max(tagStart, start) + 1);
            }
          }
+       }
 
-         load();
+       function copyLocator(f, t) {
+         t.lineNumber = f.lineNumber;
+         t.columnNumber = f.columnNumber;
+         return t;
+       }
+       /**
+        * @see #appendElement(source,elStartEnd,el,selfClosed,entityReplacer,domBuilder,parseStack);
+        * @return end of the elementStartPart(end of elementEndPart for selfClosed el)
+        */
 
-         _diff.length = function length() {
-           return Object.keys(_changes).length;
-         };
 
-         _diff.changes = function changes() {
-           return _changes;
-         };
+       function parseElementStartPart(source, start, el, currentNSMap, entityReplacer, errorHandler) {
+         var attrName;
+         var value;
+         var p = ++start;
+         var s = S_TAG; //status
 
-         _diff.didChange = _didChange; // pass true to include affected relation members
+         while (true) {
+           var c = source.charAt(p);
 
-         _diff.extantIDs = function extantIDs(includeRelMembers) {
-           var result = new Set();
-           Object.keys(_changes).forEach(function (id) {
-             if (_changes[id].head) {
-               result.add(id);
-             }
+           switch (c) {
+             case '=':
+               if (s === S_ATTR) {
+                 //attrName
+                 attrName = source.slice(start, p);
+                 s = S_EQ;
+               } else if (s === S_ATTR_SPACE) {
+                 s = S_EQ;
+               } else {
+                 //fatalError: equal must after attrName or space after attrName
+                 throw new Error('attribute equal must after attrName');
+               }
 
-             var h = _changes[id].head;
-             var b = _changes[id].base;
-             var entity = h || b;
+               break;
 
-             if (includeRelMembers && entity.type === 'relation') {
-               var mh = h ? h.members.map(function (m) {
-                 return m.id;
-               }) : [];
-               var mb = b ? b.members.map(function (m) {
-                 return m.id;
-               }) : [];
-               utilArrayUnion(mh, mb).forEach(function (memberID) {
-                 if (head.hasEntity(memberID)) {
-                   result.add(memberID);
-                 }
-               });
-             }
-           });
-           return Array.from(result);
-         };
+             case '\'':
+             case '"':
+               if (s === S_EQ || s === S_ATTR //|| s == S_ATTR_SPACE
+               ) {
+                   //equal
+                   if (s === S_ATTR) {
+                     errorHandler.warning('attribute value must after "="');
+                     attrName = source.slice(start, p);
+                   }
 
-         _diff.modified = function modified() {
-           var result = [];
-           Object.values(_changes).forEach(function (change) {
-             if (change.base && change.head) {
-               result.push(change.head);
-             }
-           });
-           return result;
-         };
+                   start = p + 1;
+                   p = source.indexOf(c, start);
 
-         _diff.created = function created() {
-           var result = [];
-           Object.values(_changes).forEach(function (change) {
-             if (!change.base && change.head) {
-               result.push(change.head);
-             }
-           });
-           return result;
-         };
+                   if (p > 0) {
+                     value = source.slice(start, p).replace(/&#?\w+;/g, entityReplacer);
+                     el.add(attrName, value, start - 1);
+                     s = S_ATTR_END;
+                   } else {
+                     //fatalError: no end quot match
+                     throw new Error('attribute value no end \'' + c + '\' match');
+                   }
+                 } else if (s == S_ATTR_NOQUOT_VALUE) {
+                 value = source.slice(start, p).replace(/&#?\w+;/g, entityReplacer); //console.log(attrName,value,start,p)
 
-         _diff.deleted = function deleted() {
-           var result = [];
-           Object.values(_changes).forEach(function (change) {
-             if (change.base && !change.head) {
-               result.push(change.base);
-             }
-           });
-           return result;
-         };
+                 el.add(attrName, value, start); //console.dir(el)
 
-         _diff.summary = function summary() {
-           var relevant = {};
-           var keys = Object.keys(_changes);
+                 errorHandler.warning('attribute "' + attrName + '" missed start quot(' + c + ')!!');
+                 start = p + 1;
+                 s = S_ATTR_END;
+               } else {
+                 //fatalError: no equal before
+                 throw new Error('attribute value must after "="');
+               }
 
-           for (var i = 0; i < keys.length; i++) {
-             var change = _changes[keys[i]];
+               break;
 
-             if (change.head && change.head.geometry(head) !== 'vertex') {
-               addEntity(change.head, head, change.base ? 'modified' : 'created');
-             } else if (change.base && change.base.geometry(base) !== 'vertex') {
-               addEntity(change.base, base, 'deleted');
-             } else if (change.base && change.head) {
-               // modified vertex
-               var moved = !fastDeepEqual(change.base.loc, change.head.loc);
-               var retagged = !fastDeepEqual(change.base.tags, change.head.tags);
+             case '/':
+               switch (s) {
+                 case S_TAG:
+                   el.setTagName(source.slice(start, p));
 
-               if (moved) {
-                 addParents(change.head);
-               }
+                 case S_ATTR_END:
+                 case S_TAG_SPACE:
+                 case S_TAG_CLOSE:
+                   s = S_TAG_CLOSE;
+                   el.closed = true;
 
-               if (retagged || moved && change.head.hasInterestingTags()) {
-                 addEntity(change.head, head, 'modified');
+                 case S_ATTR_NOQUOT_VALUE:
+                 case S_ATTR:
+                 case S_ATTR_SPACE:
+                   break;
+                 //case S_EQ:
+
+                 default:
+                   throw new Error("attribute invalid close char('/')");
                }
-             } else if (change.head && change.head.hasInterestingTags()) {
-               // created vertex
-               addEntity(change.head, head, 'created');
-             } else if (change.base && change.base.hasInterestingTags()) {
-               // deleted vertex
-               addEntity(change.base, base, 'deleted');
-             }
-           }
 
-           return Object.values(relevant);
+               break;
 
-           function addEntity(entity, graph, changeType) {
-             relevant[entity.id] = {
-               entity: entity,
-               graph: graph,
-               changeType: changeType
-             };
-           }
+             case '':
+               //end document
+               //throw new Error('unexpected end of input')
+               errorHandler.error('unexpected end of input');
 
-           function addParents(entity) {
-             var parents = head.parentWays(entity);
+               if (s == S_TAG) {
+                 el.setTagName(source.slice(start, p));
+               }
 
-             for (var j = parents.length - 1; j >= 0; j--) {
-               var parent = parents[j];
+               return p;
 
-               if (!(parent.id in relevant)) {
-                 addEntity(parent, head, 'modified');
-               }
-             }
-           }
-         }; // returns complete set of entities that require a redraw
-         //  (optionally within given `extent`)
+             case '>':
+               switch (s) {
+                 case S_TAG:
+                   el.setTagName(source.slice(start, p));
 
+                 case S_ATTR_END:
+                 case S_TAG_SPACE:
+                 case S_TAG_CLOSE:
+                   break;
+                 //normal
 
-         _diff.complete = function complete(extent) {
-           var result = {};
-           var id, change;
+                 case S_ATTR_NOQUOT_VALUE: //Compatible state
 
-           for (id in _changes) {
-             change = _changes[id];
-             var h = change.head;
-             var b = change.base;
-             var entity = h || b;
-             var i;
-             if (extent && (!h || !h.intersects(extent, head)) && (!b || !b.intersects(extent, base))) continue;
-             result[id] = h;
+                 case S_ATTR:
+                   value = source.slice(start, p);
 
-             if (entity.type === 'way') {
-               var nh = h ? h.nodes : [];
-               var nb = b ? b.nodes : [];
-               var diff;
-               diff = utilArrayDifference(nh, nb);
+                   if (value.slice(-1) === '/') {
+                     el.closed = true;
+                     value = value.slice(0, -1);
+                   }
 
-               for (i = 0; i < diff.length; i++) {
-                 result[diff[i]] = head.hasEntity(diff[i]);
-               }
+                 case S_ATTR_SPACE:
+                   if (s === S_ATTR_SPACE) {
+                     value = attrName;
+                   }
 
-               diff = utilArrayDifference(nb, nh);
+                   if (s == S_ATTR_NOQUOT_VALUE) {
+                     errorHandler.warning('attribute "' + value + '" missed quot(")!!');
+                     el.add(attrName, value.replace(/&#?\w+;/g, entityReplacer), start);
+                   } else {
+                     if (currentNSMap[''] !== 'http://www.w3.org/1999/xhtml' || !value.match(/^(?:disabled|checked|selected)$/i)) {
+                       errorHandler.warning('attribute "' + value + '" missed value!! "' + value + '" instead!!');
+                     }
 
-               for (i = 0; i < diff.length; i++) {
-                 result[diff[i]] = head.hasEntity(diff[i]);
-               }
-             }
+                     el.add(value, value, start);
+                   }
 
-             if (entity.type === 'relation' && entity.isMultipolygon()) {
-               var mh = h ? h.members.map(function (m) {
-                 return m.id;
-               }) : [];
-               var mb = b ? b.members.map(function (m) {
-                 return m.id;
-               }) : [];
-               var ids = utilArrayUnion(mh, mb);
+                   break;
 
-               for (i = 0; i < ids.length; i++) {
-                 var member = head.hasEntity(ids[i]);
-                 if (!member) continue; // not downloaded
+                 case S_EQ:
+                   throw new Error('attribute value missed!!');
+               } //                    console.log(tagName,tagNamePattern,tagNamePattern.test(tagName))
 
-                 if (extent && !member.intersects(extent, head)) continue; // not visible
 
-                 result[ids[i]] = member;
-               }
-             }
+               return p;
 
-             addParents(head.parentWays(entity), result);
-             addParents(head.parentRelations(entity), result);
-           }
+             /*xml space '\x20' | #x9 | #xD | #xA; */
 
-           return result;
+             case "\x80":
+               c = ' ';
 
-           function addParents(parents, result) {
-             for (var i = 0; i < parents.length; i++) {
-               var parent = parents[i];
-               if (parent.id in result) continue;
-               result[parent.id] = parent;
-               addParents(head.parentRelations(parent), result);
-             }
-           }
-         };
+             default:
+               if (c <= ' ') {
+                 //space
+                 switch (s) {
+                   case S_TAG:
+                     el.setTagName(source.slice(start, p)); //tagName
 
-         return _diff;
-       }
+                     s = S_TAG_SPACE;
+                     break;
 
-       function coreTree(head) {
-         // tree for entities
-         var _rtree = new RBush();
+                   case S_ATTR:
+                     attrName = source.slice(start, p);
+                     s = S_ATTR_SPACE;
+                     break;
 
-         var _bboxes = {}; // maintain a separate tree for granular way segments
+                   case S_ATTR_NOQUOT_VALUE:
+                     var value = source.slice(start, p).replace(/&#?\w+;/g, entityReplacer);
+                     errorHandler.warning('attribute "' + value + '" missed quot(")!!');
+                     el.add(attrName, value, start);
 
-         var _segmentsRTree = new RBush();
+                   case S_ATTR_END:
+                     s = S_TAG_SPACE;
+                     break;
+                   //case S_TAG_SPACE:
+                   //case S_EQ:
+                   //case S_ATTR_SPACE:
+                   //  void();break;
+                   //case S_TAG_CLOSE:
+                   //ignore warning
+                 }
+               } else {
+                 //not space
+                 //S_TAG,      S_ATTR, S_EQ,   S_ATTR_NOQUOT_VALUE
+                 //S_ATTR_SPACE,       S_ATTR_END,     S_TAG_SPACE, S_TAG_CLOSE
+                 switch (s) {
+                   //case S_TAG:void();break;
+                   //case S_ATTR:void();break;
+                   //case S_ATTR_NOQUOT_VALUE:void();break;
+                   case S_ATTR_SPACE:
+                     el.tagName;
 
-         var _segmentsBBoxes = {};
-         var _segmentsByWayId = {};
-         var tree = {};
+                     if (currentNSMap[''] !== 'http://www.w3.org/1999/xhtml' || !attrName.match(/^(?:disabled|checked|selected)$/i)) {
+                       errorHandler.warning('attribute "' + attrName + '" missed value!! "' + attrName + '" instead2!!');
+                     }
 
-         function entityBBox(entity) {
-           var bbox = entity.extent(head).bbox();
-           bbox.id = entity.id;
-           _bboxes[entity.id] = bbox;
-           return bbox;
-         }
+                     el.add(attrName, attrName, start);
+                     start = p;
+                     s = S_ATTR;
+                     break;
 
-         function segmentBBox(segment) {
-           var extent = segment.extent(head); // extent can be null if the node entities aren't in the graph for some reason
+                   case S_ATTR_END:
+                     errorHandler.warning('attribute space is required"' + attrName + '"!!');
 
-           if (!extent) return null;
-           var bbox = extent.bbox();
-           bbox.segment = segment;
-           _segmentsBBoxes[segment.id] = bbox;
-           return bbox;
-         }
+                   case S_TAG_SPACE:
+                     s = S_ATTR;
+                     start = p;
+                     break;
 
-         function removeEntity(entity) {
-           _rtree.remove(_bboxes[entity.id]);
+                   case S_EQ:
+                     s = S_ATTR_NOQUOT_VALUE;
+                     start = p;
+                     break;
 
-           delete _bboxes[entity.id];
+                   case S_TAG_CLOSE:
+                     throw new Error("elements closed character '/' and '>' must be connected to");
+                 }
+               }
 
-           if (_segmentsByWayId[entity.id]) {
-             _segmentsByWayId[entity.id].forEach(function (segment) {
-               _segmentsRTree.remove(_segmentsBBoxes[segment.id]);
+           } //end outer switch
+           //console.log('p++',p)
 
-               delete _segmentsBBoxes[segment.id];
-             });
 
-             delete _segmentsByWayId[entity.id];
-           }
+           p++;
          }
+       }
+       /**
+        * @return true if has new namespace define
+        */
 
-         function loadEntities(entities) {
-           _rtree.load(entities.map(entityBBox));
 
-           var segments = [];
-           entities.forEach(function (entity) {
-             if (entity.segments) {
-               var entitySegments = entity.segments(head); // cache these to make them easy to remove later
+       function appendElement(el, domBuilder, currentNSMap) {
+         var tagName = el.tagName;
+         var localNSMap = null; //var currentNSMap = parseStack[parseStack.length-1].currentNSMap;
 
-               _segmentsByWayId[entity.id] = entitySegments;
-               segments = segments.concat(entitySegments);
-             }
-           });
-           if (segments.length) _segmentsRTree.load(segments.map(segmentBBox).filter(Boolean));
-         }
+         var i = el.length;
 
-         function updateParents(entity, insertions, memo) {
-           head.parentWays(entity).forEach(function (way) {
-             if (_bboxes[way.id]) {
-               removeEntity(way);
-               insertions[way.id] = way;
-             }
+         while (i--) {
+           var a = el[i];
+           var qName = a.qName;
+           var value = a.value;
+           var nsp = qName.indexOf(':');
 
-             updateParents(way, insertions, memo);
-           });
-           head.parentRelations(entity).forEach(function (relation) {
-             if (memo[entity.id]) return;
-             memo[entity.id] = true;
+           if (nsp > 0) {
+             var prefix = a.prefix = qName.slice(0, nsp);
+             var localName = qName.slice(nsp + 1);
+             var nsPrefix = prefix === 'xmlns' && localName;
+           } else {
+             localName = qName;
+             prefix = null;
+             nsPrefix = qName === 'xmlns' && '';
+           } //can not set prefix,because prefix !== ''
 
-             if (_bboxes[relation.id]) {
-               removeEntity(relation);
-               insertions[relation.id] = relation;
-             }
 
-             updateParents(relation, insertions, memo);
-           });
-         }
+           a.localName = localName; //prefix == null for no ns prefix attribute 
 
-         tree.rebase = function (entities, force) {
-           var insertions = {};
+           if (nsPrefix !== false) {
+             //hack!!
+             if (localNSMap == null) {
+               localNSMap = {}; //console.log(currentNSMap,0)
 
-           for (var i = 0; i < entities.length; i++) {
-             var entity = entities[i];
-             if (!entity.visible) continue;
+               _copy(currentNSMap, currentNSMap = {}); //console.log(currentNSMap,1)
 
-             if (head.entities.hasOwnProperty(entity.id) || _bboxes[entity.id]) {
-               if (!force) {
-                 continue;
-               } else if (_bboxes[entity.id]) {
-                 removeEntity(entity);
-               }
              }
 
-             insertions[entity.id] = entity;
-             updateParents(entity, insertions, {});
+             currentNSMap[nsPrefix] = localNSMap[nsPrefix] = value;
+             a.uri = 'http://www.w3.org/2000/xmlns/';
+             domBuilder.startPrefixMapping(nsPrefix, value);
            }
+         }
 
-           loadEntities(Object.values(insertions));
-           return tree;
-         };
+         var i = el.length;
 
-         function updateToGraph(graph) {
-           if (graph === head) return;
-           var diff = coreDifference(head, graph);
-           head = graph;
-           var changed = diff.didChange;
-           if (!changed.addition && !changed.deletion && !changed.geometry) return;
-           var insertions = {};
+         while (i--) {
+           a = el[i];
+           var prefix = a.prefix;
 
-           if (changed.deletion) {
-             diff.deleted().forEach(function (entity) {
-               removeEntity(entity);
-             });
-           }
+           if (prefix) {
+             //no prefix attribute has no namespace
+             if (prefix === 'xml') {
+               a.uri = 'http://www.w3.org/XML/1998/namespace';
+             }
 
-           if (changed.geometry) {
-             diff.modified().forEach(function (entity) {
-               removeEntity(entity);
-               insertions[entity.id] = entity;
-               updateParents(entity, insertions, {});
-             });
+             if (prefix !== 'xmlns') {
+               a.uri = currentNSMap[prefix || '']; //{console.log('###'+a.qName,domBuilder.locator.systemId+'',currentNSMap,a.uri)}
+             }
            }
+         }
 
-           if (changed.addition) {
-             diff.created().forEach(function (entity) {
-               insertions[entity.id] = entity;
-             });
-           }
+         var nsp = tagName.indexOf(':');
 
-           loadEntities(Object.values(insertions));
-         } // returns an array of entities with bounding boxes overlapping `extent` for the given `graph`
+         if (nsp > 0) {
+           prefix = el.prefix = tagName.slice(0, nsp);
+           localName = el.localName = tagName.slice(nsp + 1);
+         } else {
+           prefix = null; //important!!
 
+           localName = el.localName = tagName;
+         } //no prefix element has default namespace
 
-         tree.intersects = function (extent, graph) {
-           updateToGraph(graph);
-           return _rtree.search(extent.bbox()).map(function (bbox) {
-             return graph.entity(bbox.id);
-           });
-         }; // returns an array of segment objects with bounding boxes overlapping `extent` for the given `graph`
 
+         var ns = el.uri = currentNSMap[prefix || ''];
+         domBuilder.startElement(ns, localName, tagName, el); //endPrefixMapping and startPrefixMapping have not any help for dom builder
+         //localNSMap = null
 
-         tree.waySegments = function (extent, graph) {
-           updateToGraph(graph);
-           return _segmentsRTree.search(extent.bbox()).map(function (bbox) {
-             return bbox.segment;
-           });
-         };
+         if (el.closed) {
+           domBuilder.endElement(ns, localName, tagName);
 
-         return tree;
+           if (localNSMap) {
+             for (prefix in localNSMap) {
+               domBuilder.endPrefixMapping(prefix);
+             }
+           }
+         } else {
+           el.currentNSMap = currentNSMap;
+           el.localNSMap = localNSMap; //parseStack.push(el);
+
+           return true;
+         }
        }
 
-       function uiModal(selection, blocking) {
-         var _this = this;
+       function parseHtmlSpecialContent(source, elStartEnd, tagName, entityReplacer, domBuilder) {
+         if (/^(?:script|textarea)$/i.test(tagName)) {
+           var elEndStart = source.indexOf('</' + tagName + '>', elStartEnd);
+           var text = source.substring(elStartEnd + 1, elEndStart);
 
-         var keybinding = utilKeybinding('modal');
-         var previous = selection.select('div.modal');
-         var animate = previous.empty();
-         previous.transition().duration(200).style('opacity', 0).remove();
-         var shaded = selection.append('div').attr('class', 'shaded').style('opacity', 0);
+           if (/[&<]/.test(text)) {
+             if (/^script$/i.test(tagName)) {
+               //if(!/\]\]>/.test(text)){
+               //lexHandler.startCDATA();
+               domBuilder.characters(text, 0, text.length); //lexHandler.endCDATA();
 
-         shaded.close = function () {
-           shaded.transition().duration(200).style('opacity', 0).remove();
-           modal.transition().duration(200).style('top', '0px');
-           select(document).call(keybinding.unbind);
-         };
+               return elEndStart; //}
+             } //}else{//text area
 
-         var modal = shaded.append('div').attr('class', 'modal fillL');
-         modal.append('input').attr('class', 'keytrap keytrap-first').on('focus.keytrap', moveFocusToLast);
 
-         if (!blocking) {
-           shaded.on('click.remove-modal', function (d3_event) {
-             if (d3_event.target === _this) {
-               shaded.close();
-             }
-           });
-           modal.append('button').attr('class', 'close').on('click', shaded.close).call(svgIcon('#iD-icon-close'));
-           keybinding.on('⌫', shaded.close).on('⎋', shaded.close);
-           select(document).call(keybinding);
+             text = text.replace(/&#?\w+;/g, entityReplacer);
+             domBuilder.characters(text, 0, text.length);
+             return elEndStart; //}
+           }
          }
 
-         modal.append('div').attr('class', 'content');
-         modal.append('input').attr('class', 'keytrap keytrap-last').on('focus.keytrap', moveFocusToFirst);
-
-         if (animate) {
-           shaded.transition().style('opacity', 1);
-         } else {
-           shaded.style('opacity', 1);
-         }
+         return elStartEnd + 1;
+       }
 
-         return shaded;
+       function fixSelfClosed(source, elStartEnd, tagName, closeMap) {
+         //if(tagName in closeMap){
+         var pos = closeMap[tagName];
 
-         function moveFocusToFirst() {
-           var node = modal // there are additional rules about what's focusable, but this suits our purposes
-           .select('a, button, input:not(.keytrap), select, textarea').node();
+         if (pos == null) {
+           //console.log(tagName)
+           pos = source.lastIndexOf('</' + tagName + '>');
 
-           if (node) {
-             node.focus();
-           } else {
-             select(this).node().blur();
+           if (pos < elStartEnd) {
+             //忘记闭合
+             pos = source.lastIndexOf('</' + tagName);
            }
+
+           closeMap[tagName] = pos;
          }
 
-         function moveFocusToLast() {
-           var nodes = modal.selectAll('a, button, input:not(.keytrap), select, textarea').nodes();
+         return pos < elStartEnd; //} 
+       }
 
-           if (nodes.length) {
-             nodes[nodes.length - 1].focus();
-           } else {
-             select(this).node().blur();
-           }
+       function _copy(source, target) {
+         for (var n in source) {
+           target[n] = source[n];
          }
        }
 
-       function uiLoading(context) {
-         var _modalSelection = select(null);
-
-         var _message = '';
-         var _blocking = false;
-
-         var loading = function loading(selection) {
-           _modalSelection = uiModal(selection, _blocking);
-
-           var loadertext = _modalSelection.select('.content').classed('loading-modal', true).append('div').attr('class', 'modal-section fillL');
+       function parseDCC(source, start, domBuilder, errorHandler) {
+         //sure start with '<!'
+         var next = source.charAt(start + 2);
 
-           loadertext.append('img').attr('class', 'loader').attr('src', context.imagePath('loader-white.gif'));
-           loadertext.append('h3').html(_message);
+         switch (next) {
+           case '-':
+             if (source.charAt(start + 3) === '-') {
+               var end = source.indexOf('-->', start + 4); //append comment source.substring(4,end)//<!--
 
-           _modalSelection.select('button.close').attr('class', 'hide');
+               if (end > start) {
+                 domBuilder.comment(source, start + 4, end - start - 4);
+                 return end + 3;
+               } else {
+                 errorHandler.error("Unclosed comment");
+                 return -1;
+               }
+             } else {
+               //error
+               return -1;
+             }
 
-           return loading;
-         };
+           default:
+             if (source.substr(start + 3, 6) == 'CDATA[') {
+               var end = source.indexOf(']]>', start + 9);
+               domBuilder.startCDATA();
+               domBuilder.characters(source, start + 9, end - start - 9);
+               domBuilder.endCDATA();
+               return end + 3;
+             } //<!DOCTYPE
+             //startDTD(java.lang.String name, java.lang.String publicId, java.lang.String systemId) 
 
-         loading.message = function (val) {
-           if (!arguments.length) return _message;
-           _message = val;
-           return loading;
-         };
 
-         loading.blocking = function (val) {
-           if (!arguments.length) return _blocking;
-           _blocking = val;
-           return loading;
-         };
+             var matchs = split(source, start);
+             var len = matchs.length;
 
-         loading.close = function () {
-           _modalSelection.remove();
-         };
+             if (len > 1 && /!doctype/i.test(matchs[0][0])) {
+               var name = matchs[1][0];
+               var pubid = len > 3 && /^public$/i.test(matchs[2][0]) && matchs[3][0];
+               var sysid = len > 4 && matchs[4][0];
+               var lastMatch = matchs[len - 1];
+               domBuilder.startDTD(name, pubid && pubid.replace(/^(['"])(.*?)\1$/, '$2'), sysid && sysid.replace(/^(['"])(.*?)\1$/, '$2'));
+               domBuilder.endDTD();
+               return lastMatch.index + lastMatch[0].length;
+             }
 
-         loading.isShown = function () {
-           return _modalSelection && !_modalSelection.empty() && _modalSelection.node().parentNode;
-         };
+         }
 
-         return loading;
+         return -1;
        }
 
-       function coreHistory(context) {
-         var dispatch$1 = dispatch('reset', 'change', 'merge', 'restore', 'undone', 'redone');
+       function parseInstruction(source, start, domBuilder) {
+         var end = source.indexOf('?>', start);
 
-         var _lock = utilSessionMutex('lock'); // restorable if iD not open in another window/tab and a saved history exists in localStorage
+         if (end) {
+           var match = source.substring(start, end).match(/^<\?(\S*)\s*([\s\S]*?)\s*$/);
 
+           if (match) {
+             match[0].length;
+             domBuilder.processingInstruction(match[1], match[2]);
+             return end + 2;
+           } else {
+             //error
+             return -1;
+           }
+         }
 
-         var _hasUnresolvedRestorableChanges = _lock.lock() && !!corePreferences(getKey('saved_history'));
+         return -1;
+       }
+       /**
+        * @param source
+        */
 
-         var duration = 150;
-         var _imageryUsed = [];
-         var _photoOverlaysUsed = [];
-         var _checkpoints = {};
 
-         var _pausedGraph;
+       function ElementAttributes(source) {}
 
-         var _stack;
+       ElementAttributes.prototype = {
+         setTagName: function setTagName(tagName) {
+           if (!tagNamePattern.test(tagName)) {
+             throw new Error('invalid tagName:' + tagName);
+           }
 
-         var _index;
+           this.tagName = tagName;
+         },
+         add: function add(qName, value, offset) {
+           if (!tagNamePattern.test(qName)) {
+             throw new Error('invalid attribute:' + qName);
+           }
 
-         var _tree; // internal _act, accepts list of actions and eased time
+           this[this.length++] = {
+             qName: qName,
+             value: value,
+             offset: offset
+           };
+         },
+         length: 0,
+         getLocalName: function getLocalName(i) {
+           return this[i].localName;
+         },
+         getLocator: function getLocator(i) {
+           return this[i].locator;
+         },
+         getQName: function getQName(i) {
+           return this[i].qName;
+         },
+         getURI: function getURI(i) {
+           return this[i].uri;
+         },
+         getValue: function getValue(i) {
+           return this[i].value;
+         } //  ,getIndex:function(uri, localName)){
+         //            if(localName){
+         //                    
+         //            }else{
+         //                    var qName = uri
+         //            }
+         //    },
+         //    getValue:function(){return this.getValue(this.getIndex.apply(this,arguments))},
+         //    getType:function(uri,localName){}
+         //    getType:function(i){},
 
+       };
 
-         function _act(actions, t) {
-           actions = Array.prototype.slice.call(actions);
-           var annotation;
+       function _set_proto_(thiz, parent) {
+         thiz.__proto__ = parent;
+         return thiz;
+       }
 
-           if (typeof actions[actions.length - 1] !== 'function') {
-             annotation = actions.pop();
+       if (!(_set_proto_({}, _set_proto_.prototype) instanceof _set_proto_)) {
+         _set_proto_ = function _set_proto_(thiz, parent) {
+           function p() {}
+           p.prototype = parent;
+           p = new p();
+
+           for (parent in thiz) {
+             p[parent] = thiz[parent];
            }
 
-           var graph = _stack[_index].graph;
+           return p;
+         };
+       }
 
-           for (var i = 0; i < actions.length; i++) {
-             graph = actions[i](graph, t);
-           }
+       function split(source, start) {
+         var match;
+         var buf = [];
+         var reg = /'[^']+'|"[^"]+"|[^\s<>\/=]+=?|(\/?\s*>|<)/g;
+         reg.lastIndex = start;
+         reg.exec(source); //skip <
 
-           return {
-             graph: graph,
-             annotation: annotation,
-             imageryUsed: _imageryUsed,
-             photoOverlaysUsed: _photoOverlaysUsed,
-             transform: context.projection.transform(),
-             selectedIDs: context.selectedIDs()
-           };
-         } // internal _perform with eased time
+         while (match = reg.exec(source)) {
+           buf.push(match);
+           if (match[1]) return buf;
+         }
+       }
+
+       var XMLReader_1 = XMLReader;
+       var sax = {
+         XMLReader: XMLReader_1
+       };
 
+       /*
+        * DOM Level 2
+        * Object DOMException
+        * @see http://www.w3.org/TR/REC-DOM-Level-1/ecma-script-language-binding.html
+        * @see http://www.w3.org/TR/2000/REC-DOM-Level-2-Core-20001113/ecma-script-binding.html
+        */
+       function copy(src, dest) {
+         for (var p in src) {
+           dest[p] = src[p];
+         }
+       }
+       /**
+       ^\w+\.prototype\.([_\w]+)\s*=\s*((?:.*\{\s*?[\r\n][\s\S]*?^})|\S.*?(?=[;\r\n]));?
+       ^\w+\.prototype\.([_\w]+)\s*=\s*(\S.*?(?=[;\r\n]));?
+        */
 
-         function _perform(args, t) {
-           var previous = _stack[_index].graph;
-           _stack = _stack.slice(0, _index + 1);
 
-           var actionResult = _act(args, t);
+       function _extends(Class, Super) {
+         var pt = Class.prototype;
 
-           _stack.push(actionResult);
+         if (Object.create) {
+           var ppt = Object.create(Super.prototype);
+           pt.__proto__ = ppt;
+         }
 
-           _index++;
-           return change(previous);
-         } // internal _replace with eased time
+         if (!(pt instanceof Super)) {
+           var t = function t() {};
+           t.prototype = Super.prototype;
+           t = new t();
+           copy(pt, t);
+           Class.prototype = pt = t;
+         }
 
+         if (pt.constructor != Class) {
+           if (typeof Class != 'function') {
+             console.error("unknow Class:" + Class);
+           }
 
-         function _replace(args, t) {
-           var previous = _stack[_index].graph; // assert(_index == _stack.length - 1)
+           pt.constructor = Class;
+         }
+       }
 
-           var actionResult = _act(args, t);
+       var htmlns = 'http://www.w3.org/1999/xhtml'; // Node Types
 
-           _stack[_index] = actionResult;
-           return change(previous);
-         } // internal _overwrite with eased time
+       var NodeType = {};
+       var ELEMENT_NODE = NodeType.ELEMENT_NODE = 1;
+       var ATTRIBUTE_NODE = NodeType.ATTRIBUTE_NODE = 2;
+       var TEXT_NODE = NodeType.TEXT_NODE = 3;
+       var CDATA_SECTION_NODE = NodeType.CDATA_SECTION_NODE = 4;
+       var ENTITY_REFERENCE_NODE = NodeType.ENTITY_REFERENCE_NODE = 5;
+       var ENTITY_NODE = NodeType.ENTITY_NODE = 6;
+       var PROCESSING_INSTRUCTION_NODE = NodeType.PROCESSING_INSTRUCTION_NODE = 7;
+       var COMMENT_NODE = NodeType.COMMENT_NODE = 8;
+       var DOCUMENT_NODE = NodeType.DOCUMENT_NODE = 9;
+       var DOCUMENT_TYPE_NODE = NodeType.DOCUMENT_TYPE_NODE = 10;
+       var DOCUMENT_FRAGMENT_NODE = NodeType.DOCUMENT_FRAGMENT_NODE = 11;
+       var NOTATION_NODE = NodeType.NOTATION_NODE = 12; // ExceptionCode
 
+       var ExceptionCode = {};
+       var ExceptionMessage = {};
+       ExceptionCode.INDEX_SIZE_ERR = (ExceptionMessage[1] = "Index size error", 1);
+       ExceptionCode.DOMSTRING_SIZE_ERR = (ExceptionMessage[2] = "DOMString size error", 2);
+       var HIERARCHY_REQUEST_ERR = ExceptionCode.HIERARCHY_REQUEST_ERR = (ExceptionMessage[3] = "Hierarchy request error", 3);
+       ExceptionCode.WRONG_DOCUMENT_ERR = (ExceptionMessage[4] = "Wrong document", 4);
+       ExceptionCode.INVALID_CHARACTER_ERR = (ExceptionMessage[5] = "Invalid character", 5);
+       ExceptionCode.NO_DATA_ALLOWED_ERR = (ExceptionMessage[6] = "No data allowed", 6);
+       ExceptionCode.NO_MODIFICATION_ALLOWED_ERR = (ExceptionMessage[7] = "No modification allowed", 7);
+       var NOT_FOUND_ERR = ExceptionCode.NOT_FOUND_ERR = (ExceptionMessage[8] = "Not found", 8);
+       ExceptionCode.NOT_SUPPORTED_ERR = (ExceptionMessage[9] = "Not supported", 9);
+       var INUSE_ATTRIBUTE_ERR = ExceptionCode.INUSE_ATTRIBUTE_ERR = (ExceptionMessage[10] = "Attribute in use", 10); //level2
 
-         function _overwrite(args, t) {
-           var previous = _stack[_index].graph;
+       ExceptionCode.INVALID_STATE_ERR = (ExceptionMessage[11] = "Invalid state", 11);
+       ExceptionCode.SYNTAX_ERR = (ExceptionMessage[12] = "Syntax error", 12);
+       ExceptionCode.INVALID_MODIFICATION_ERR = (ExceptionMessage[13] = "Invalid modification", 13);
+       ExceptionCode.NAMESPACE_ERR = (ExceptionMessage[14] = "Invalid namespace", 14);
+       ExceptionCode.INVALID_ACCESS_ERR = (ExceptionMessage[15] = "Invalid access", 15);
 
-           if (_index > 0) {
-             _index--;
+       function DOMException$1(code, message) {
+         if (message instanceof Error) {
+           var error = message;
+         } else {
+           error = this;
+           Error.call(this, ExceptionMessage[code]);
+           this.message = ExceptionMessage[code];
+           if (Error.captureStackTrace) Error.captureStackTrace(this, DOMException$1);
+         }
 
-             _stack.pop();
-           }
+         error.code = code;
+         if (message) this.message = this.message + ": " + message;
+         return error;
+       }
+       DOMException$1.prototype = Error.prototype;
+       copy(ExceptionCode, DOMException$1);
+       /**
+        * @see http://www.w3.org/TR/2000/REC-DOM-Level-2-Core-20001113/core.html#ID-536297177
+        * The NodeList interface provides the abstraction of an ordered collection of nodes, without defining or constraining how this collection is implemented. NodeList objects in the DOM are live.
+        * The items in the NodeList are accessible via an integral index, starting from 0.
+        */
 
-           _stack = _stack.slice(0, _index + 1);
+       function NodeList() {}
+       NodeList.prototype = {
+         /**
+          * The number of nodes in the list. The range of valid child node indices is 0 to length-1 inclusive.
+          * @standard level1
+          */
+         length: 0,
 
-           var actionResult = _act(args, t);
+         /**
+          * Returns the indexth item in the collection. If index is greater than or equal to the number of nodes in the list, this returns null.
+          * @standard level1
+          * @param index  unsigned long 
+          *   Index into the collection.
+          * @return Node
+          *    The node at the indexth position in the NodeList, or null if that is not a valid index. 
+          */
+         item: function item(index) {
+           return this[index] || null;
+         },
+         toString: function toString(isHTML, nodeFilter) {
+           for (var buf = [], i = 0; i < this.length; i++) {
+             serializeToString(this[i], buf, isHTML, nodeFilter);
+           }
 
-           _stack.push(actionResult);
+           return buf.join('');
+         }
+       };
 
-           _index++;
-           return change(previous);
-         } // determine difference and dispatch a change event
+       function LiveNodeList(node, refresh) {
+         this._node = node;
+         this._refresh = refresh;
 
+         _updateLiveList(this);
+       }
 
-         function change(previous) {
-           var difference = coreDifference(previous, history.graph());
+       function _updateLiveList(list) {
+         var inc = list._node._inc || list._node.ownerDocument._inc;
 
-           if (!_pausedGraph) {
-             dispatch$1.call('change', this, difference);
-           }
+         if (list._inc != inc) {
+           var ls = list._refresh(list._node); //console.log(ls.length)
 
-           return difference;
-         } // iD uses namespaced keys so multiple installations do not conflict
 
+           __set__(list, 'length', ls.length);
 
-         function getKey(n) {
-           return 'iD_' + window.location.origin + '_' + n;
+           copy(ls, list);
+           list._inc = inc;
          }
+       }
 
-         var history = {
-           graph: function graph() {
-             return _stack[_index].graph;
-           },
-           tree: function tree() {
-             return _tree;
-           },
-           base: function base() {
-             return _stack[0].graph;
-           },
-           merge: function merge(entities
-           /*, extent*/
-           ) {
-             var stack = _stack.map(function (state) {
-               return state.graph;
-             });
+       LiveNodeList.prototype.item = function (i) {
+         _updateLiveList(this);
 
-             _stack[0].graph.rebase(entities, stack, false);
+         return this[i];
+       };
 
-             _tree.rebase(entities, false);
+       _extends(LiveNodeList, NodeList);
+       /**
+        * 
+        * Objects implementing the NamedNodeMap interface are used to represent collections of nodes that can be accessed by name. Note that NamedNodeMap does not inherit from NodeList; NamedNodeMaps are not maintained in any particular order. Objects contained in an object implementing NamedNodeMap may also be accessed by an ordinal index, but this is simply to allow convenient enumeration of the contents of a NamedNodeMap, and does not imply that the DOM specifies an order to these Nodes.
+        * NamedNodeMap objects in the DOM are live.
+        * used for attributes or DocumentType entities 
+        */
 
-             dispatch$1.call('merge', this, entities);
-           },
-           perform: function perform() {
-             // complete any transition already in progress
-             select(document).interrupt('history.perform');
-             var transitionable = false;
-             var action0 = arguments[0];
 
-             if (arguments.length === 1 || arguments.length === 2 && typeof arguments[1] !== 'function') {
-               transitionable = !!action0.transitionable;
-             }
+       function NamedNodeMap() {}
 
-             if (transitionable) {
-               var origArguments = arguments;
-               select(document).transition('history.perform').duration(duration).ease(linear$1).tween('history.tween', function () {
-                 return function (t) {
-                   if (t < 1) _overwrite([action0], t);
-                 };
-               }).on('start', function () {
-                 _perform([action0], 0);
-               }).on('end interrupt', function () {
-                 _overwrite(origArguments, 1);
-               });
-             } else {
-               return _perform(arguments);
-             }
-           },
-           replace: function replace() {
-             select(document).interrupt('history.perform');
-             return _replace(arguments, 1);
-           },
-           // Same as calling pop and then perform
-           overwrite: function overwrite() {
-             select(document).interrupt('history.perform');
-             return _overwrite(arguments, 1);
-           },
-           pop: function pop(n) {
-             select(document).interrupt('history.perform');
-             var previous = _stack[_index].graph;
+       function _findNodeIndex(list, node) {
+         var i = list.length;
 
-             if (isNaN(+n) || +n < 0) {
-               n = 1;
-             }
+         while (i--) {
+           if (list[i] === node) {
+             return i;
+           }
+         }
+       }
 
-             while (n-- > 0 && _index > 0) {
-               _index--;
+       function _addNamedNode(el, list, newAttr, oldAttr) {
+         if (oldAttr) {
+           list[_findNodeIndex(list, oldAttr)] = newAttr;
+         } else {
+           list[list.length++] = newAttr;
+         }
 
-               _stack.pop();
-             }
+         if (el) {
+           newAttr.ownerElement = el;
+           var doc = el.ownerDocument;
 
-             return change(previous);
-           },
-           // Back to the previous annotated state or _index = 0.
-           undo: function undo() {
-             select(document).interrupt('history.perform');
-             var previousStack = _stack[_index];
-             var previous = previousStack.graph;
+           if (doc) {
+             oldAttr && _onRemoveAttribute(doc, el, oldAttr);
 
-             while (_index > 0) {
-               _index--;
-               if (_stack[_index].annotation) break;
-             }
+             _onAddAttribute(doc, el, newAttr);
+           }
+         }
+       }
 
-             dispatch$1.call('undone', this, _stack[_index], previousStack);
-             return change(previous);
-           },
-           // Forward to the next annotated state.
-           redo: function redo() {
-             select(document).interrupt('history.perform');
-             var previousStack = _stack[_index];
-             var previous = previousStack.graph;
-             var tryIndex = _index;
+       function _removeNamedNode(el, list, attr) {
+         //console.log('remove attr:'+attr)
+         var i = _findNodeIndex(list, attr);
 
-             while (tryIndex < _stack.length - 1) {
-               tryIndex++;
+         if (i >= 0) {
+           var lastIndex = list.length - 1;
 
-               if (_stack[tryIndex].annotation) {
-                 _index = tryIndex;
-                 dispatch$1.call('redone', this, _stack[_index], previousStack);
-                 break;
-               }
-             }
+           while (i < lastIndex) {
+             list[i] = list[++i];
+           }
 
-             return change(previous);
-           },
-           pauseChangeDispatch: function pauseChangeDispatch() {
-             if (!_pausedGraph) {
-               _pausedGraph = _stack[_index].graph;
-             }
-           },
-           resumeChangeDispatch: function resumeChangeDispatch() {
-             if (_pausedGraph) {
-               var previous = _pausedGraph;
-               _pausedGraph = null;
-               return change(previous);
-             }
-           },
-           undoAnnotation: function undoAnnotation() {
-             var i = _index;
+           list.length = lastIndex;
 
-             while (i >= 0) {
-               if (_stack[i].annotation) return _stack[i].annotation;
-               i--;
-             }
-           },
-           redoAnnotation: function redoAnnotation() {
-             var i = _index + 1;
+           if (el) {
+             var doc = el.ownerDocument;
 
-             while (i <= _stack.length - 1) {
-               if (_stack[i].annotation) return _stack[i].annotation;
-               i++;
-             }
-           },
-           // Returns the entities from the active graph with bounding boxes
-           // overlapping the given `extent`.
-           intersects: function intersects(extent) {
-             return _tree.intersects(extent, _stack[_index].graph);
-           },
-           difference: function difference() {
-             var base = _stack[0].graph;
-             var head = _stack[_index].graph;
-             return coreDifference(base, head);
-           },
-           changes: function changes(action) {
-             var base = _stack[0].graph;
-             var head = _stack[_index].graph;
+             if (doc) {
+               _onRemoveAttribute(doc, el, attr);
 
-             if (action) {
-               head = action(head);
+               attr.ownerElement = null;
              }
+           }
+         } else {
+           throw DOMException$1(NOT_FOUND_ERR, new Error(el.tagName + '@' + attr));
+         }
+       }
 
-             var difference = coreDifference(base, head);
-             return {
-               modified: difference.modified(),
-               created: difference.created(),
-               deleted: difference.deleted()
-             };
-           },
-           hasChanges: function hasChanges() {
-             return this.difference().length() > 0;
-           },
-           imageryUsed: function imageryUsed(sources) {
-             if (sources) {
-               _imageryUsed = sources;
-               return history;
-             } else {
-               var s = new Set();
-
-               _stack.slice(1, _index + 1).forEach(function (state) {
-                 state.imageryUsed.forEach(function (source) {
-                   if (source !== 'Custom') {
-                     s.add(source);
-                   }
-                 });
-               });
-
-               return Array.from(s);
-             }
-           },
-           photoOverlaysUsed: function photoOverlaysUsed(sources) {
-             if (sources) {
-               _photoOverlaysUsed = sources;
-               return history;
-             } else {
-               var s = new Set();
+       NamedNodeMap.prototype = {
+         length: 0,
+         item: NodeList.prototype.item,
+         getNamedItem: function getNamedItem(key) {
+           //          if(key.indexOf(':')>0 || key == 'xmlns'){
+           //                  return null;
+           //          }
+           //console.log()
+           var i = this.length;
 
-               _stack.slice(1, _index + 1).forEach(function (state) {
-                 if (state.photoOverlaysUsed && Array.isArray(state.photoOverlaysUsed)) {
-                   state.photoOverlaysUsed.forEach(function (photoOverlay) {
-                     s.add(photoOverlay);
-                   });
-                 }
-               });
+           while (i--) {
+             var attr = this[i]; //console.log(attr.nodeName,key)
 
-               return Array.from(s);
-             }
-           },
-           // save the current history state
-           checkpoint: function checkpoint(key) {
-             _checkpoints[key] = {
-               stack: _stack,
-               index: _index
-             };
-             return history;
-           },
-           // restore history state to a given checkpoint or reset completely
-           reset: function reset(key) {
-             if (key !== undefined && _checkpoints.hasOwnProperty(key)) {
-               _stack = _checkpoints[key].stack;
-               _index = _checkpoints[key].index;
-             } else {
-               _stack = [{
-                 graph: coreGraph()
-               }];
-               _index = 0;
-               _tree = coreTree(_stack[0].graph);
-               _checkpoints = {};
+             if (attr.nodeName == key) {
+               return attr;
              }
+           }
+         },
+         setNamedItem: function setNamedItem(attr) {
+           var el = attr.ownerElement;
 
-             dispatch$1.call('reset');
-             dispatch$1.call('change');
-             return history;
-           },
-           // `toIntroGraph()` is used to export the intro graph used by the walkthrough.
-           //
-           // To use it:
-           //  1. Start the walkthrough.
-           //  2. Get to a "free editing" tutorial step
-           //  3. Make your edits to the walkthrough map
-           //  4. In your browser dev console run:
-           //        `id.history().toIntroGraph()`
-           //  5. This outputs stringified JSON to the browser console
-           //  6. Copy it to `data/intro_graph.json` and prettify it in your code editor
-           toIntroGraph: function toIntroGraph() {
-             var nextID = {
-               n: 0,
-               r: 0,
-               w: 0
-             };
-             var permIDs = {};
-             var graph = this.graph();
-             var baseEntities = {}; // clone base entities..
-
-             Object.values(graph.base().entities).forEach(function (entity) {
-               var copy = copyIntroEntity(entity);
-               baseEntities[copy.id] = copy;
-             }); // replace base entities with head entities..
+           if (el && el != this._ownerElement) {
+             throw new DOMException$1(INUSE_ATTRIBUTE_ERR);
+           }
 
-             Object.keys(graph.entities).forEach(function (id) {
-               var entity = graph.entities[id];
+           var oldAttr = this.getNamedItem(attr.nodeName);
 
-               if (entity) {
-                 var copy = copyIntroEntity(entity);
-                 baseEntities[copy.id] = copy;
-               } else {
-                 delete baseEntities[id];
-               }
-             }); // swap temporary for permanent ids..
+           _addNamedNode(this._ownerElement, this, attr, oldAttr);
 
-             Object.values(baseEntities).forEach(function (entity) {
-               if (Array.isArray(entity.nodes)) {
-                 entity.nodes = entity.nodes.map(function (node) {
-                   return permIDs[node] || node;
-                 });
-               }
+           return oldAttr;
+         },
 
-               if (Array.isArray(entity.members)) {
-                 entity.members = entity.members.map(function (member) {
-                   member.id = permIDs[member.id] || member.id;
-                   return member;
-                 });
-               }
-             });
-             return JSON.stringify({
-               dataIntroGraph: baseEntities
-             });
+         /* returns Node */
+         setNamedItemNS: function setNamedItemNS(attr) {
+           // raises: WRONG_DOCUMENT_ERR,NO_MODIFICATION_ALLOWED_ERR,INUSE_ATTRIBUTE_ERR
+           var el = attr.ownerElement,
+               oldAttr;
 
-             function copyIntroEntity(source) {
-               var copy = utilObjectOmit(source, ['type', 'user', 'v', 'version', 'visible']); // Note: the copy is no longer an osmEntity, so it might not have `tags`
+           if (el && el != this._ownerElement) {
+             throw new DOMException$1(INUSE_ATTRIBUTE_ERR);
+           }
 
-               if (copy.tags && !Object.keys(copy.tags)) {
-                 delete copy.tags;
-               }
+           oldAttr = this.getNamedItemNS(attr.namespaceURI, attr.localName);
 
-               if (Array.isArray(copy.loc)) {
-                 copy.loc[0] = +copy.loc[0].toFixed(6);
-                 copy.loc[1] = +copy.loc[1].toFixed(6);
-               }
+           _addNamedNode(this._ownerElement, this, attr, oldAttr);
 
-               var match = source.id.match(/([nrw])-\d*/); // temporary id
+           return oldAttr;
+         },
 
-               if (match !== null) {
-                 var nrw = match[1];
-                 var permID;
+         /* returns Node */
+         removeNamedItem: function removeNamedItem(key) {
+           var attr = this.getNamedItem(key);
 
-                 do {
-                   permID = nrw + ++nextID[nrw];
-                 } while (baseEntities.hasOwnProperty(permID));
+           _removeNamedNode(this._ownerElement, this, attr);
 
-                 copy.id = permIDs[source.id] = permID;
-               }
+           return attr;
+         },
+         // raises: NOT_FOUND_ERR,NO_MODIFICATION_ALLOWED_ERR
+         //for level2
+         removeNamedItemNS: function removeNamedItemNS(namespaceURI, localName) {
+           var attr = this.getNamedItemNS(namespaceURI, localName);
 
-               return copy;
-             }
-           },
-           toJSON: function toJSON() {
-             if (!this.hasChanges()) return;
-             var allEntities = {};
-             var baseEntities = {};
-             var base = _stack[0];
+           _removeNamedNode(this._ownerElement, this, attr);
 
-             var s = _stack.map(function (i) {
-               var modified = [];
-               var deleted = [];
-               Object.keys(i.graph.entities).forEach(function (id) {
-                 var entity = i.graph.entities[id];
+           return attr;
+         },
+         getNamedItemNS: function getNamedItemNS(namespaceURI, localName) {
+           var i = this.length;
 
-                 if (entity) {
-                   var key = osmEntity.key(entity);
-                   allEntities[key] = entity;
-                   modified.push(key);
-                 } else {
-                   deleted.push(id);
-                 } // make sure that the originals of changed or deleted entities get merged
-                 // into the base of the _stack after restoring the data from JSON.
+           while (i--) {
+             var node = this[i];
 
+             if (node.localName == localName && node.namespaceURI == namespaceURI) {
+               return node;
+             }
+           }
 
-                 if (id in base.graph.entities) {
-                   baseEntities[id] = base.graph.entities[id];
-                 }
+           return null;
+         }
+       };
+       /**
+        * @see http://www.w3.org/TR/REC-DOM-Level-1/level-one-core.html#ID-102161490
+        */
 
-                 if (entity && entity.nodes) {
-                   // get originals of pre-existing child nodes
-                   entity.nodes.forEach(function (nodeID) {
-                     if (nodeID in base.graph.entities) {
-                       baseEntities[nodeID] = base.graph.entities[nodeID];
-                     }
-                   });
-                 } // get originals of parent entities too
+       function DOMImplementation(
+       /* Object */
+       features) {
+         this._features = {};
 
+         if (features) {
+           for (var feature in features) {
+             this._features = features[feature];
+           }
+         }
+       }
+       DOMImplementation.prototype = {
+         hasFeature: function hasFeature(
+         /* string */
+         feature,
+         /* string */
+         version) {
+           var versions = this._features[feature.toLowerCase()];
 
-                 var baseParents = base.graph._parentWays[id];
+           if (versions && (!version || version in versions)) {
+             return true;
+           } else {
+             return false;
+           }
+         },
+         // Introduced in DOM Level 2:
+         createDocument: function createDocument(namespaceURI, qualifiedName, doctype) {
+           // raises:INVALID_CHARACTER_ERR,NAMESPACE_ERR,WRONG_DOCUMENT_ERR
+           var doc = new Document();
+           doc.implementation = this;
+           doc.childNodes = new NodeList();
+           doc.doctype = doctype;
 
-                 if (baseParents) {
-                   baseParents.forEach(function (parentID) {
-                     if (parentID in base.graph.entities) {
-                       baseEntities[parentID] = base.graph.entities[parentID];
-                     }
-                   });
-                 }
-               });
-               var x = {};
-               if (modified.length) x.modified = modified;
-               if (deleted.length) x.deleted = deleted;
-               if (i.imageryUsed) x.imageryUsed = i.imageryUsed;
-               if (i.photoOverlaysUsed) x.photoOverlaysUsed = i.photoOverlaysUsed;
-               if (i.annotation) x.annotation = i.annotation;
-               if (i.transform) x.transform = i.transform;
-               if (i.selectedIDs) x.selectedIDs = i.selectedIDs;
-               return x;
-             });
+           if (doctype) {
+             doc.appendChild(doctype);
+           }
 
-             return JSON.stringify({
-               version: 3,
-               entities: Object.values(allEntities),
-               baseEntities: Object.values(baseEntities),
-               stack: s,
-               nextIDs: osmEntity.id.next,
-               index: _index,
-               // note the time the changes were saved
-               timestamp: new Date().getTime()
-             });
-           },
-           fromJSON: function fromJSON(json, loadChildNodes) {
-             var h = JSON.parse(json);
-             var loadComplete = true;
-             osmEntity.id.next = h.nextIDs;
-             _index = h.index;
+           if (qualifiedName) {
+             var root = doc.createElementNS(namespaceURI, qualifiedName);
+             doc.appendChild(root);
+           }
 
-             if (h.version === 2 || h.version === 3) {
-               var allEntities = {};
-               h.entities.forEach(function (entity) {
-                 allEntities[osmEntity.key(entity)] = osmEntity(entity);
-               });
+           return doc;
+         },
+         // Introduced in DOM Level 2:
+         createDocumentType: function createDocumentType(qualifiedName, publicId, systemId) {
+           // raises:INVALID_CHARACTER_ERR,NAMESPACE_ERR
+           var node = new DocumentType();
+           node.name = qualifiedName;
+           node.nodeName = qualifiedName;
+           node.publicId = publicId;
+           node.systemId = systemId; // Introduced in DOM Level 2:
+           //readonly attribute DOMString        internalSubset;
+           //TODO:..
+           //  readonly attribute NamedNodeMap     entities;
+           //  readonly attribute NamedNodeMap     notations;
 
-               if (h.version === 3) {
-                 // This merges originals for changed entities into the base of
-                 // the _stack even if the current _stack doesn't have them (for
-                 // example when iD has been restarted in a different region)
-                 var baseEntities = h.baseEntities.map(function (d) {
-                   return osmEntity(d);
-                 });
+           return node;
+         }
+       };
+       /**
+        * @see http://www.w3.org/TR/2000/REC-DOM-Level-2-Core-20001113/core.html#ID-1950641247
+        */
 
-                 var stack = _stack.map(function (state) {
-                   return state.graph;
-                 });
+       function Node() {}
+       Node.prototype = {
+         firstChild: null,
+         lastChild: null,
+         previousSibling: null,
+         nextSibling: null,
+         attributes: null,
+         parentNode: null,
+         childNodes: null,
+         ownerDocument: null,
+         nodeValue: null,
+         namespaceURI: null,
+         prefix: null,
+         localName: null,
+         // Modified in DOM Level 2:
+         insertBefore: function insertBefore(newChild, refChild) {
+           //raises 
+           return _insertBefore(this, newChild, refChild);
+         },
+         replaceChild: function replaceChild(newChild, oldChild) {
+           //raises 
+           this.insertBefore(newChild, oldChild);
 
-                 _stack[0].graph.rebase(baseEntities, stack, true);
+           if (oldChild) {
+             this.removeChild(oldChild);
+           }
+         },
+         removeChild: function removeChild(oldChild) {
+           return _removeChild(this, oldChild);
+         },
+         appendChild: function appendChild(newChild) {
+           return this.insertBefore(newChild, null);
+         },
+         hasChildNodes: function hasChildNodes() {
+           return this.firstChild != null;
+         },
+         cloneNode: function cloneNode(deep) {
+           return _cloneNode(this.ownerDocument || this, this, deep);
+         },
+         // Modified in DOM Level 2:
+         normalize: function normalize() {
+           var child = this.firstChild;
 
-                 _tree.rebase(baseEntities, true); // When we restore a modified way, we also need to fetch any missing
-                 // childnodes that would normally have been downloaded with it.. #2142
+           while (child) {
+             var next = child.nextSibling;
 
+             if (next && next.nodeType == TEXT_NODE && child.nodeType == TEXT_NODE) {
+               this.removeChild(next);
+               child.appendData(next.data);
+             } else {
+               child.normalize();
+               child = next;
+             }
+           }
+         },
+         // Introduced in DOM Level 2:
+         isSupported: function isSupported(feature, version) {
+           return this.ownerDocument.implementation.hasFeature(feature, version);
+         },
+         // Introduced in DOM Level 2:
+         hasAttributes: function hasAttributes() {
+           return this.attributes.length > 0;
+         },
+         lookupPrefix: function lookupPrefix(namespaceURI) {
+           var el = this;
 
-                 if (loadChildNodes) {
-                   var osm = context.connection();
-                   var baseWays = baseEntities.filter(function (e) {
-                     return e.type === 'way';
-                   });
-                   var nodeIDs = baseWays.reduce(function (acc, way) {
-                     return utilArrayUnion(acc, way.nodes);
-                   }, []);
-                   var missing = nodeIDs.filter(function (n) {
-                     return !_stack[0].graph.hasEntity(n);
-                   });
+           while (el) {
+             var map = el._nsMap; //console.dir(map)
 
-                   if (missing.length && osm) {
-                     loadComplete = false;
-                     context.map().redrawEnable(false);
-                     var loading = uiLoading(context).blocking(true);
-                     context.container().call(loading);
+             if (map) {
+               for (var n in map) {
+                 if (map[n] == namespaceURI) {
+                   return n;
+                 }
+               }
+             }
 
-                     var childNodesLoaded = function childNodesLoaded(err, result) {
-                       if (!err) {
-                         var visibleGroups = utilArrayGroupBy(result.data, 'visible');
-                         var visibles = visibleGroups["true"] || []; // alive nodes
+             el = el.nodeType == ATTRIBUTE_NODE ? el.ownerDocument : el.parentNode;
+           }
 
-                         var invisibles = visibleGroups["false"] || []; // deleted nodes
+           return null;
+         },
+         // Introduced in DOM Level 3:
+         lookupNamespaceURI: function lookupNamespaceURI(prefix) {
+           var el = this;
 
-                         if (visibles.length) {
-                           var visibleIDs = visibles.map(function (entity) {
-                             return entity.id;
-                           });
+           while (el) {
+             var map = el._nsMap; //console.dir(map)
 
-                           var stack = _stack.map(function (state) {
-                             return state.graph;
-                           });
+             if (map) {
+               if (prefix in map) {
+                 return map[prefix];
+               }
+             }
 
-                           missing = utilArrayDifference(missing, visibleIDs);
+             el = el.nodeType == ATTRIBUTE_NODE ? el.ownerDocument : el.parentNode;
+           }
 
-                           _stack[0].graph.rebase(visibles, stack, true);
+           return null;
+         },
+         // Introduced in DOM Level 3:
+         isDefaultNamespace: function isDefaultNamespace(namespaceURI) {
+           var prefix = this.lookupPrefix(namespaceURI);
+           return prefix == null;
+         }
+       };
 
-                           _tree.rebase(visibles, true);
-                         } // fetch older versions of nodes that were deleted..
+       function _xmlEncoder(c) {
+         return c == '<' && '&lt;' || c == '>' && '&gt;' || c == '&' && '&amp;' || c == '"' && '&quot;' || '&#' + c.charCodeAt() + ';';
+       }
 
+       copy(NodeType, Node);
+       copy(NodeType, Node.prototype);
+       /**
+        * @param callback return true for continue,false for break
+        * @return boolean true: break visit;
+        */
 
-                         invisibles.forEach(function (entity) {
-                           osm.loadEntityVersion(entity.id, +entity.version - 1, childNodesLoaded);
-                         });
-                       }
+       function _visitNode(node, callback) {
+         if (callback(node)) {
+           return true;
+         }
 
-                       if (err || !missing.length) {
-                         loading.close();
-                         context.map().redrawEnable(true);
-                         dispatch$1.call('change');
-                         dispatch$1.call('restore', this);
-                       }
-                     };
+         if (node = node.firstChild) {
+           do {
+             if (_visitNode(node, callback)) {
+               return true;
+             }
+           } while (node = node.nextSibling);
+         }
+       }
 
-                     osm.loadMultiple(missing, childNodesLoaded);
-                   }
-                 }
-               }
+       function Document() {}
 
-               _stack = h.stack.map(function (d) {
-                 var entities = {},
-                     entity;
+       function _onAddAttribute(doc, el, newAttr) {
+         doc && doc._inc++;
+         var ns = newAttr.namespaceURI;
 
-                 if (d.modified) {
-                   d.modified.forEach(function (key) {
-                     entity = allEntities[key];
-                     entities[entity.id] = entity;
-                   });
-                 }
+         if (ns == 'http://www.w3.org/2000/xmlns/') {
+           //update namespace
+           el._nsMap[newAttr.prefix ? newAttr.localName : ''] = newAttr.value;
+         }
+       }
 
-                 if (d.deleted) {
-                   d.deleted.forEach(function (id) {
-                     entities[id] = undefined;
-                   });
-                 }
+       function _onRemoveAttribute(doc, el, newAttr, remove) {
+         doc && doc._inc++;
+         var ns = newAttr.namespaceURI;
 
-                 return {
-                   graph: coreGraph(_stack[0].graph).load(entities),
-                   annotation: d.annotation,
-                   imageryUsed: d.imageryUsed,
-                   photoOverlaysUsed: d.photoOverlaysUsed,
-                   transform: d.transform,
-                   selectedIDs: d.selectedIDs
-                 };
-               });
-             } else {
-               // original version
-               _stack = h.stack.map(function (d) {
-                 var entities = {};
+         if (ns == 'http://www.w3.org/2000/xmlns/') {
+           //update namespace
+           delete el._nsMap[newAttr.prefix ? newAttr.localName : ''];
+         }
+       }
 
-                 for (var i in d.entities) {
-                   var entity = d.entities[i];
-                   entities[i] = entity === 'undefined' ? undefined : osmEntity(entity);
-                 }
+       function _onUpdateChild(doc, el, newChild) {
+         if (doc && doc._inc) {
+           doc._inc++; //update childNodes
 
-                 d.graph = coreGraph(_stack[0].graph).load(entities);
-                 return d;
-               });
-             }
+           var cs = el.childNodes;
 
-             var transform = _stack[_index].transform;
+           if (newChild) {
+             cs[cs.length++] = newChild;
+           } else {
+             //console.log(1)
+             var child = el.firstChild;
+             var i = 0;
 
-             if (transform) {
-               context.map().transformEase(transform, 0); // 0 = immediate, no easing
+             while (child) {
+               cs[i++] = child;
+               child = child.nextSibling;
              }
 
-             if (loadComplete) {
-               dispatch$1.call('change');
-               dispatch$1.call('restore', this);
-             }
+             cs.length = i;
+           }
+         }
+       }
+       /**
+        * attributes;
+        * children;
+        * 
+        * writeable properties:
+        * nodeValue,Attr:value,CharacterData:data
+        * prefix
+        */
 
-             return history;
-           },
-           lock: function lock() {
-             return _lock.lock();
-           },
-           unlock: function unlock() {
-             _lock.unlock();
-           },
-           save: function save() {
-             if (_lock.locked() && // don't overwrite existing, unresolved changes
-             !_hasUnresolvedRestorableChanges) {
-               corePreferences(getKey('saved_history'), history.toJSON() || null);
-             }
 
-             return history;
-           },
-           // delete the history version saved in localStorage
-           clearSaved: function clearSaved() {
-             context.debouncedSave.cancel();
+       function _removeChild(parentNode, child) {
+         var previous = child.previousSibling;
+         var next = child.nextSibling;
 
-             if (_lock.locked()) {
-               _hasUnresolvedRestorableChanges = false;
-               corePreferences(getKey('saved_history'), null); // clear the changeset metadata associated with the saved history
+         if (previous) {
+           previous.nextSibling = next;
+         } else {
+           parentNode.firstChild = next;
+         }
 
-               corePreferences('comment', null);
-               corePreferences('hashtags', null);
-               corePreferences('source', null);
-             }
+         if (next) {
+           next.previousSibling = previous;
+         } else {
+           parentNode.lastChild = previous;
+         }
 
-             return history;
-           },
-           savedHistoryJSON: function savedHistoryJSON() {
-             return corePreferences(getKey('saved_history'));
-           },
-           hasRestorableChanges: function hasRestorableChanges() {
-             return _hasUnresolvedRestorableChanges;
-           },
-           // load history from a version stored in localStorage
-           restore: function restore() {
-             if (_lock.locked()) {
-               _hasUnresolvedRestorableChanges = false;
-               var json = this.savedHistoryJSON();
-               if (json) history.fromJSON(json, true);
-             }
-           },
-           _getKey: getKey
-         };
-         history.reset();
-         return utilRebind(history, dispatch$1, 'on');
-       }
+         _onUpdateChild(parentNode.ownerDocument, parentNode);
 
+         return child;
+       }
        /**
-        * Look for roads that can be connected to other roads with a short extension
+        * preformance key(refChild == null)
         */
 
-       function validationAlmostJunction(context) {
-         var type = 'almost_junction';
-         var EXTEND_TH_METERS = 5;
-         var WELD_TH_METERS = 0.75; // Comes from considering bounding case of parallel ways
-
-         var CLOSE_NODE_TH = EXTEND_TH_METERS - WELD_TH_METERS; // Comes from considering bounding case of perpendicular ways
 
-         var SIG_ANGLE_TH = Math.atan(WELD_TH_METERS / EXTEND_TH_METERS);
+       function _insertBefore(parentNode, newChild, nextChild) {
+         var cp = newChild.parentNode;
 
-         function isHighway(entity) {
-           return entity.type === 'way' && osmRoutableHighwayTagValues[entity.tags.highway];
+         if (cp) {
+           cp.removeChild(newChild); //remove and update
          }
 
-         function isTaggedAsNotContinuing(node) {
-           return node.tags.noexit === 'yes' || node.tags.amenity === 'parking_entrance' || node.tags.entrance && node.tags.entrance !== 'no';
+         if (newChild.nodeType === DOCUMENT_FRAGMENT_NODE) {
+           var newFirst = newChild.firstChild;
+
+           if (newFirst == null) {
+             return newChild;
+           }
+
+           var newLast = newChild.lastChild;
+         } else {
+           newFirst = newLast = newChild;
          }
 
-         var validation = function checkAlmostJunction(entity, graph) {
-           if (!isHighway(entity)) return [];
-           if (entity.isDegenerate()) return [];
-           var tree = context.history().tree();
-           var extendableNodeInfos = findConnectableEndNodesByExtension(entity);
-           var issues = [];
-           extendableNodeInfos.forEach(function (extendableNodeInfo) {
-             issues.push(new validationIssue({
-               type: type,
-               subtype: 'highway-highway',
-               severity: 'warning',
-               message: function message(context) {
-                 var entity1 = context.hasEntity(this.entityIds[0]);
+         var pre = nextChild ? nextChild.previousSibling : parentNode.lastChild;
+         newFirst.previousSibling = pre;
+         newLast.nextSibling = nextChild;
 
-                 if (this.entityIds[0] === this.entityIds[2]) {
-                   return entity1 ? _t.html('issues.almost_junction.self.message', {
-                     feature: utilDisplayLabel(entity1, context.graph())
-                   }) : '';
-                 } else {
-                   var entity2 = context.hasEntity(this.entityIds[2]);
-                   return entity1 && entity2 ? _t.html('issues.almost_junction.message', {
-                     feature: utilDisplayLabel(entity1, context.graph()),
-                     feature2: utilDisplayLabel(entity2, context.graph())
-                   }) : '';
-                 }
-               },
-               reference: showReference,
-               entityIds: [entity.id, extendableNodeInfo.node.id, extendableNodeInfo.wid],
-               loc: extendableNodeInfo.node.loc,
-               hash: JSON.stringify(extendableNodeInfo.node.loc),
-               data: {
-                 midId: extendableNodeInfo.mid.id,
-                 edge: extendableNodeInfo.edge,
-                 cross_loc: extendableNodeInfo.cross_loc
-               },
-               dynamicFixes: makeFixes
-             }));
-           });
-           return issues;
+         if (pre) {
+           pre.nextSibling = newFirst;
+         } else {
+           parentNode.firstChild = newFirst;
+         }
+
+         if (nextChild == null) {
+           parentNode.lastChild = newLast;
+         } else {
+           nextChild.previousSibling = newLast;
+         }
 
-           function makeFixes(context) {
-             var fixes = [new validationIssueFix({
-               icon: 'iD-icon-abutment',
-               title: _t.html('issues.fix.connect_features.title'),
-               onClick: function onClick(context) {
-                 var annotation = _t('issues.fix.connect_almost_junction.annotation');
+         do {
+           newFirst.parentNode = parentNode;
+         } while (newFirst !== newLast && (newFirst = newFirst.nextSibling));
 
-                 var _this$issue$entityIds = _slicedToArray(this.issue.entityIds, 3),
-                     endNodeId = _this$issue$entityIds[1],
-                     crossWayId = _this$issue$entityIds[2];
+         _onUpdateChild(parentNode.ownerDocument || parentNode, parentNode); //console.log(parentNode.lastChild.nextSibling == null)
 
-                 var midNode = context.entity(this.issue.data.midId);
-                 var endNode = context.entity(endNodeId);
-                 var crossWay = context.entity(crossWayId); // When endpoints are close, just join if resulting small change in angle (#7201)
 
-                 var nearEndNodes = findNearbyEndNodes(endNode, crossWay);
+         if (newChild.nodeType == DOCUMENT_FRAGMENT_NODE) {
+           newChild.firstChild = newChild.lastChild = null;
+         }
 
-                 if (nearEndNodes.length > 0) {
-                   var collinear = findSmallJoinAngle(midNode, endNode, nearEndNodes);
+         return newChild;
+       }
 
-                   if (collinear) {
-                     context.perform(actionMergeNodes([collinear.id, endNode.id], collinear.loc), annotation);
-                     return;
-                   }
-                 }
+       function _appendSingleChild(parentNode, newChild) {
+         var cp = newChild.parentNode;
 
-                 var targetEdge = this.issue.data.edge;
-                 var crossLoc = this.issue.data.cross_loc;
-                 var edgeNodes = [context.entity(targetEdge[0]), context.entity(targetEdge[1])];
-                 var closestNodeInfo = geoSphericalClosestNode(edgeNodes, crossLoc); // already a point nearby, just connect to that
+         if (cp) {
+           var pre = parentNode.lastChild;
+           cp.removeChild(newChild); //remove and update
 
-                 if (closestNodeInfo.distance < WELD_TH_METERS) {
-                   context.perform(actionMergeNodes([closestNodeInfo.node.id, endNode.id], closestNodeInfo.node.loc), annotation); // else add the end node to the edge way
-                 } else {
-                   context.perform(actionAddMidpoint({
-                     loc: crossLoc,
-                     edge: targetEdge
-                   }, endNode), annotation);
-                 }
-               }
-             })];
-             var node = context.hasEntity(this.entityIds[1]);
+           var pre = parentNode.lastChild;
+         }
 
-             if (node && !node.hasInterestingTags()) {
-               // node has no descriptive tags, suggest noexit fix
-               fixes.push(new validationIssueFix({
-                 icon: 'maki-barrier',
-                 title: _t.html('issues.fix.tag_as_disconnected.title'),
-                 onClick: function onClick(context) {
-                   var nodeID = this.issue.entityIds[1];
-                   var tags = Object.assign({}, context.entity(nodeID).tags);
-                   tags.noexit = 'yes';
-                   context.perform(actionChangeTags(nodeID, tags), _t('issues.fix.tag_as_disconnected.annotation'));
-                 }
-               }));
+         var pre = parentNode.lastChild;
+         newChild.parentNode = parentNode;
+         newChild.previousSibling = pre;
+         newChild.nextSibling = null;
+
+         if (pre) {
+           pre.nextSibling = newChild;
+         } else {
+           parentNode.firstChild = newChild;
+         }
+
+         parentNode.lastChild = newChild;
+
+         _onUpdateChild(parentNode.ownerDocument, parentNode, newChild);
+
+         return newChild; //console.log("__aa",parentNode.lastChild.nextSibling == null)
+       }
+
+       Document.prototype = {
+         //implementation : null,
+         nodeName: '#document',
+         nodeType: DOCUMENT_NODE,
+         doctype: null,
+         documentElement: null,
+         _inc: 1,
+         insertBefore: function insertBefore(newChild, refChild) {
+           //raises 
+           if (newChild.nodeType == DOCUMENT_FRAGMENT_NODE) {
+             var child = newChild.firstChild;
+
+             while (child) {
+               var next = child.nextSibling;
+               this.insertBefore(child, refChild);
+               child = next;
              }
 
-             return fixes;
+             return newChild;
            }
 
-           function showReference(selection) {
-             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.almost_junction.highway-highway.reference'));
+           if (this.documentElement == null && newChild.nodeType == ELEMENT_NODE) {
+             this.documentElement = newChild;
            }
 
-           function isExtendableCandidate(node, way) {
-             // can not accurately test vertices on tiles not downloaded from osm - #5938
-             var osm = services.osm;
+           return _insertBefore(this, newChild, refChild), newChild.ownerDocument = this, newChild;
+         },
+         removeChild: function removeChild(oldChild) {
+           if (this.documentElement == oldChild) {
+             this.documentElement = null;
+           }
 
-             if (osm && !osm.isDataLoaded(node.loc)) {
-               return false;
-             }
+           return _removeChild(this, oldChild);
+         },
+         // Introduced in DOM Level 2:
+         importNode: function importNode(importedNode, deep) {
+           return _importNode(this, importedNode, deep);
+         },
+         // Introduced in DOM Level 2:
+         getElementById: function getElementById(id) {
+           var rtv = null;
 
-             if (isTaggedAsNotContinuing(node) || graph.parentWays(node).length !== 1) {
-               return false;
+           _visitNode(this.documentElement, function (node) {
+             if (node.nodeType == ELEMENT_NODE) {
+               if (node.getAttribute('id') == id) {
+                 rtv = node;
+                 return true;
+               }
              }
+           });
 
-             var occurrences = 0;
+           return rtv;
+         },
+         //document factory method:
+         createElement: function createElement(tagName) {
+           var node = new Element();
+           node.ownerDocument = this;
+           node.nodeName = tagName;
+           node.tagName = tagName;
+           node.childNodes = new NodeList();
+           var attrs = node.attributes = new NamedNodeMap();
+           attrs._ownerElement = node;
+           return node;
+         },
+         createDocumentFragment: function createDocumentFragment() {
+           var node = new DocumentFragment();
+           node.ownerDocument = this;
+           node.childNodes = new NodeList();
+           return node;
+         },
+         createTextNode: function createTextNode(data) {
+           var node = new Text();
+           node.ownerDocument = this;
+           node.appendData(data);
+           return node;
+         },
+         createComment: function createComment(data) {
+           var node = new Comment();
+           node.ownerDocument = this;
+           node.appendData(data);
+           return node;
+         },
+         createCDATASection: function createCDATASection(data) {
+           var node = new CDATASection();
+           node.ownerDocument = this;
+           node.appendData(data);
+           return node;
+         },
+         createProcessingInstruction: function createProcessingInstruction(target, data) {
+           var node = new ProcessingInstruction();
+           node.ownerDocument = this;
+           node.tagName = node.target = target;
+           node.nodeValue = node.data = data;
+           return node;
+         },
+         createAttribute: function createAttribute(name) {
+           var node = new Attr();
+           node.ownerDocument = this;
+           node.name = name;
+           node.nodeName = name;
+           node.localName = name;
+           node.specified = true;
+           return node;
+         },
+         createEntityReference: function createEntityReference(name) {
+           var node = new EntityReference();
+           node.ownerDocument = this;
+           node.nodeName = name;
+           return node;
+         },
+         // Introduced in DOM Level 2:
+         createElementNS: function createElementNS(namespaceURI, qualifiedName) {
+           var node = new Element();
+           var pl = qualifiedName.split(':');
+           var attrs = node.attributes = new NamedNodeMap();
+           node.childNodes = new NodeList();
+           node.ownerDocument = this;
+           node.nodeName = qualifiedName;
+           node.tagName = qualifiedName;
+           node.namespaceURI = namespaceURI;
 
-             for (var index in way.nodes) {
-               if (way.nodes[index] === node.id) {
-                 occurrences += 1;
+           if (pl.length == 2) {
+             node.prefix = pl[0];
+             node.localName = pl[1];
+           } else {
+             //el.prefix = null;
+             node.localName = qualifiedName;
+           }
 
-                 if (occurrences > 1) {
-                   return false;
-                 }
-               }
-             }
+           attrs._ownerElement = node;
+           return node;
+         },
+         // Introduced in DOM Level 2:
+         createAttributeNS: function createAttributeNS(namespaceURI, qualifiedName) {
+           var node = new Attr();
+           var pl = qualifiedName.split(':');
+           node.ownerDocument = this;
+           node.nodeName = qualifiedName;
+           node.name = qualifiedName;
+           node.namespaceURI = namespaceURI;
+           node.specified = true;
 
-             return true;
+           if (pl.length == 2) {
+             node.prefix = pl[0];
+             node.localName = pl[1];
+           } else {
+             //el.prefix = null;
+             node.localName = qualifiedName;
            }
 
-           function findConnectableEndNodesByExtension(way) {
-             var results = [];
-             if (way.isClosed()) return results;
-             var testNodes;
-             var indices = [0, way.nodes.length - 1];
-             indices.forEach(function (nodeIndex) {
-               var nodeID = way.nodes[nodeIndex];
-               var node = graph.entity(nodeID);
-               if (!isExtendableCandidate(node, way)) return;
-               var connectionInfo = canConnectByExtend(way, nodeIndex);
-               if (!connectionInfo) return;
-               testNodes = graph.childNodes(way).slice(); // shallow copy
+           return node;
+         }
+       };
 
-               testNodes[nodeIndex] = testNodes[nodeIndex].move(connectionInfo.cross_loc); // don't flag issue if connecting the ways would cause self-intersection
+       _extends(Document, Node);
 
-               if (geoHasSelfIntersections(testNodes, nodeID)) return;
-               results.push(connectionInfo);
-             });
-             return results;
+       function Element() {
+         this._nsMap = {};
+       }
+       Element.prototype = {
+         nodeType: ELEMENT_NODE,
+         hasAttribute: function hasAttribute(name) {
+           return this.getAttributeNode(name) != null;
+         },
+         getAttribute: function getAttribute(name) {
+           var attr = this.getAttributeNode(name);
+           return attr && attr.value || '';
+         },
+         getAttributeNode: function getAttributeNode(name) {
+           return this.attributes.getNamedItem(name);
+         },
+         setAttribute: function setAttribute(name, value) {
+           var attr = this.ownerDocument.createAttribute(name);
+           attr.value = attr.nodeValue = "" + value;
+           this.setAttributeNode(attr);
+         },
+         removeAttribute: function removeAttribute(name) {
+           var attr = this.getAttributeNode(name);
+           attr && this.removeAttributeNode(attr);
+         },
+         //four real opeartion method
+         appendChild: function appendChild(newChild) {
+           if (newChild.nodeType === DOCUMENT_FRAGMENT_NODE) {
+             return this.insertBefore(newChild, null);
+           } else {
+             return _appendSingleChild(this, newChild);
            }
+         },
+         setAttributeNode: function setAttributeNode(newAttr) {
+           return this.attributes.setNamedItem(newAttr);
+         },
+         setAttributeNodeNS: function setAttributeNodeNS(newAttr) {
+           return this.attributes.setNamedItemNS(newAttr);
+         },
+         removeAttributeNode: function removeAttributeNode(oldAttr) {
+           //console.log(this == oldAttr.ownerElement)
+           return this.attributes.removeNamedItem(oldAttr.nodeName);
+         },
+         //get real attribute name,and remove it by removeAttributeNode
+         removeAttributeNS: function removeAttributeNS(namespaceURI, localName) {
+           var old = this.getAttributeNodeNS(namespaceURI, localName);
+           old && this.removeAttributeNode(old);
+         },
+         hasAttributeNS: function hasAttributeNS(namespaceURI, localName) {
+           return this.getAttributeNodeNS(namespaceURI, localName) != null;
+         },
+         getAttributeNS: function getAttributeNS(namespaceURI, localName) {
+           var attr = this.getAttributeNodeNS(namespaceURI, localName);
+           return attr && attr.value || '';
+         },
+         setAttributeNS: function setAttributeNS(namespaceURI, qualifiedName, value) {
+           var attr = this.ownerDocument.createAttributeNS(namespaceURI, qualifiedName);
+           attr.value = attr.nodeValue = "" + value;
+           this.setAttributeNode(attr);
+         },
+         getAttributeNodeNS: function getAttributeNodeNS(namespaceURI, localName) {
+           return this.attributes.getNamedItemNS(namespaceURI, localName);
+         },
+         getElementsByTagName: function getElementsByTagName(tagName) {
+           return new LiveNodeList(this, function (base) {
+             var ls = [];
 
-           function findNearbyEndNodes(node, way) {
-             return [way.nodes[0], way.nodes[way.nodes.length - 1]].map(function (d) {
-               return graph.entity(d);
-             }).filter(function (d) {
-               // Node cannot be near to itself, but other endnode of same way could be
-               return d.id !== node.id && geoSphericalDistance(node.loc, d.loc) <= CLOSE_NODE_TH;
+             _visitNode(base, function (node) {
+               if (node !== base && node.nodeType == ELEMENT_NODE && (tagName === '*' || node.tagName == tagName)) {
+                 ls.push(node);
+               }
              });
-           }
-
-           function findSmallJoinAngle(midNode, tipNode, endNodes) {
-             // Both nodes could be close, so want to join whichever is closest to collinear
-             var joinTo;
-             var minAngle = Infinity; // Checks midNode -> tipNode -> endNode for collinearity
 
-             endNodes.forEach(function (endNode) {
-               var a1 = geoAngle(midNode, tipNode, context.projection) + Math.PI;
-               var a2 = geoAngle(midNode, endNode, context.projection) + Math.PI;
-               var diff = Math.max(a1, a2) - Math.min(a1, a2);
+             return ls;
+           });
+         },
+         getElementsByTagNameNS: function getElementsByTagNameNS(namespaceURI, localName) {
+           return new LiveNodeList(this, function (base) {
+             var ls = [];
 
-               if (diff < minAngle) {
-                 joinTo = endNode;
-                 minAngle = diff;
+             _visitNode(base, function (node) {
+               if (node !== base && node.nodeType === ELEMENT_NODE && (namespaceURI === '*' || node.namespaceURI === namespaceURI) && (localName === '*' || node.localName == localName)) {
+                 ls.push(node);
                }
              });
-             /* Threshold set by considering right angle triangle
-             based on node joining threshold and extension distance */
 
-             if (minAngle <= SIG_ANGLE_TH) return joinTo;
-             return null;
-           }
+             return ls;
+           });
+         }
+       };
+       Document.prototype.getElementsByTagName = Element.prototype.getElementsByTagName;
+       Document.prototype.getElementsByTagNameNS = Element.prototype.getElementsByTagNameNS;
 
-           function hasTag(tags, key) {
-             return tags[key] !== undefined && tags[key] !== 'no';
-           }
+       _extends(Element, Node);
 
-           function canConnectWays(way, way2) {
-             // allow self-connections
-             if (way.id === way2.id) return true; // if one is bridge or tunnel, both must be bridge or tunnel
+       function Attr() {}
+       Attr.prototype.nodeType = ATTRIBUTE_NODE;
 
-             if ((hasTag(way.tags, 'bridge') || hasTag(way2.tags, 'bridge')) && !(hasTag(way.tags, 'bridge') && hasTag(way2.tags, 'bridge'))) return false;
-             if ((hasTag(way.tags, 'tunnel') || hasTag(way2.tags, 'tunnel')) && !(hasTag(way.tags, 'tunnel') && hasTag(way2.tags, 'tunnel'))) return false; // must have equivalent layers and levels
+       _extends(Attr, Node);
 
-             var layer1 = way.tags.layer || '0',
-                 layer2 = way2.tags.layer || '0';
-             if (layer1 !== layer2) return false;
-             var level1 = way.tags.level || '0',
-                 level2 = way2.tags.level || '0';
-             if (level1 !== level2) return false;
-             return true;
+       function CharacterData() {}
+       CharacterData.prototype = {
+         data: '',
+         substringData: function substringData(offset, count) {
+           return this.data.substring(offset, offset + count);
+         },
+         appendData: function appendData(text) {
+           text = this.data + text;
+           this.nodeValue = this.data = text;
+           this.length = text.length;
+         },
+         insertData: function insertData(offset, text) {
+           this.replaceData(offset, 0, text);
+         },
+         appendChild: function appendChild(newChild) {
+           throw new Error(ExceptionMessage[HIERARCHY_REQUEST_ERR]);
+         },
+         deleteData: function deleteData(offset, count) {
+           this.replaceData(offset, count, "");
+         },
+         replaceData: function replaceData(offset, count, text) {
+           var start = this.data.substring(0, offset);
+           var end = this.data.substring(offset + count);
+           text = start + text + end;
+           this.nodeValue = this.data = text;
+           this.length = text.length;
+         }
+       };
+
+       _extends(CharacterData, Node);
+
+       function Text() {}
+       Text.prototype = {
+         nodeName: "#text",
+         nodeType: TEXT_NODE,
+         splitText: function splitText(offset) {
+           var text = this.data;
+           var newText = text.substring(offset);
+           text = text.substring(0, offset);
+           this.data = this.nodeValue = text;
+           this.length = text.length;
+           var newNode = this.ownerDocument.createTextNode(newText);
+
+           if (this.parentNode) {
+             this.parentNode.insertBefore(newNode, this.nextSibling);
            }
 
-           function canConnectByExtend(way, endNodeIdx) {
-             var tipNid = way.nodes[endNodeIdx]; // the 'tip' node for extension point
+           return newNode;
+         }
+       };
 
-             var midNid = endNodeIdx === 0 ? way.nodes[1] : way.nodes[way.nodes.length - 2]; // the other node of the edge
+       _extends(Text, CharacterData);
 
-             var tipNode = graph.entity(tipNid);
-             var midNode = graph.entity(midNid);
-             var lon = tipNode.loc[0];
-             var lat = tipNode.loc[1];
-             var lon_range = geoMetersToLon(EXTEND_TH_METERS, lat) / 2;
-             var lat_range = geoMetersToLat(EXTEND_TH_METERS) / 2;
-             var queryExtent = geoExtent([[lon - lon_range, lat - lat_range], [lon + lon_range, lat + lat_range]]); // first, extend the edge of [midNode -> tipNode] by EXTEND_TH_METERS and find the "extended tip" location
+       function Comment() {}
+       Comment.prototype = {
+         nodeName: "#comment",
+         nodeType: COMMENT_NODE
+       };
 
-             var edgeLen = geoSphericalDistance(midNode.loc, tipNode.loc);
-             var t = EXTEND_TH_METERS / edgeLen + 1.0;
-             var extTipLoc = geoVecInterp(midNode.loc, tipNode.loc, t); // then, check if the extension part [tipNode.loc -> extTipLoc] intersects any other ways
+       _extends(Comment, CharacterData);
 
-             var segmentInfos = tree.waySegments(queryExtent, graph);
+       function CDATASection() {}
+       CDATASection.prototype = {
+         nodeName: "#cdata-section",
+         nodeType: CDATA_SECTION_NODE
+       };
 
-             for (var i = 0; i < segmentInfos.length; i++) {
-               var segmentInfo = segmentInfos[i];
-               var way2 = graph.entity(segmentInfo.wayId);
-               if (!isHighway(way2)) continue;
-               if (!canConnectWays(way, way2)) continue;
-               var nAid = segmentInfo.nodes[0],
-                   nBid = segmentInfo.nodes[1];
-               if (nAid === tipNid || nBid === tipNid) continue;
-               var nA = graph.entity(nAid),
-                   nB = graph.entity(nBid);
-               var crossLoc = geoLineIntersection([tipNode.loc, extTipLoc], [nA.loc, nB.loc]);
+       _extends(CDATASection, CharacterData);
 
-               if (crossLoc) {
-                 return {
-                   mid: midNode,
-                   node: tipNode,
-                   wid: way2.id,
-                   edge: [nA.id, nB.id],
-                   cross_loc: crossLoc
-                 };
-               }
-             }
+       function DocumentType() {}
+       DocumentType.prototype.nodeType = DOCUMENT_TYPE_NODE;
 
-             return null;
-           }
-         };
+       _extends(DocumentType, Node);
 
-         validation.type = type;
-         return validation;
-       }
+       function Notation() {}
+       Notation.prototype.nodeType = NOTATION_NODE;
 
-       function validationCloseNodes(context) {
-         var type = 'close_nodes';
-         var pointThresholdMeters = 0.2;
+       _extends(Notation, Node);
 
-         var validation = function validation(entity, graph) {
-           if (entity.type === 'node') {
-             return getIssuesForNode(entity);
-           } else if (entity.type === 'way') {
-             return getIssuesForWay(entity);
-           }
+       function Entity() {}
+       Entity.prototype.nodeType = ENTITY_NODE;
 
-           return [];
+       _extends(Entity, Node);
+
+       function EntityReference() {}
+       EntityReference.prototype.nodeType = ENTITY_REFERENCE_NODE;
+
+       _extends(EntityReference, Node);
 
-           function getIssuesForNode(node) {
-             var parentWays = graph.parentWays(node);
+       function DocumentFragment() {}
+       DocumentFragment.prototype.nodeName = "#document-fragment";
+       DocumentFragment.prototype.nodeType = DOCUMENT_FRAGMENT_NODE;
 
-             if (parentWays.length) {
-               return getIssuesForVertex(node, parentWays);
-             } else {
-               return getIssuesForDetachedPoint(node);
-             }
-           }
+       _extends(DocumentFragment, Node);
 
-           function wayTypeFor(way) {
-             if (way.tags.boundary && way.tags.boundary !== 'no') return 'boundary';
-             if (way.tags.indoor && way.tags.indoor !== 'no') return 'indoor';
-             if (way.tags.building && way.tags.building !== 'no' || way.tags['building:part'] && way.tags['building:part'] !== 'no') return 'building';
-             if (osmPathHighwayTagValues[way.tags.highway]) return 'path';
-             var parentRelations = graph.parentRelations(way);
+       function ProcessingInstruction() {}
 
-             for (var i in parentRelations) {
-               var relation = parentRelations[i];
-               if (relation.tags.type === 'boundary') return 'boundary';
+       ProcessingInstruction.prototype.nodeType = PROCESSING_INSTRUCTION_NODE;
 
-               if (relation.isMultipolygon()) {
-                 if (relation.tags.indoor && relation.tags.indoor !== 'no') return 'indoor';
-                 if (relation.tags.building && relation.tags.building !== 'no' || relation.tags['building:part'] && relation.tags['building:part'] !== 'no') return 'building';
-               }
-             }
+       _extends(ProcessingInstruction, Node);
 
-             return 'other';
-           }
+       function XMLSerializer$1() {}
 
-           function shouldCheckWay(way) {
-             // don't flag issues where merging would create degenerate ways
-             if (way.nodes.length <= 2 || way.isClosed() && way.nodes.length <= 4) return false;
-             var bbox = way.extent(graph).bbox();
-             var hypotenuseMeters = geoSphericalDistance([bbox.minX, bbox.minY], [bbox.maxX, bbox.maxY]); // don't flag close nodes in very small ways
+       XMLSerializer$1.prototype.serializeToString = function (node, isHtml, nodeFilter) {
+         return nodeSerializeToString.call(node, isHtml, nodeFilter);
+       };
 
-             if (hypotenuseMeters < 1.5) return false;
-             return true;
-           }
+       Node.prototype.toString = nodeSerializeToString;
 
-           function getIssuesForWay(way) {
-             if (!shouldCheckWay(way)) return [];
-             var issues = [],
-                 nodes = graph.childNodes(way);
+       function nodeSerializeToString(isHtml, nodeFilter) {
+         var buf = [];
+         var refNode = this.nodeType == 9 ? this.documentElement : this;
+         var prefix = refNode.prefix;
+         var uri = refNode.namespaceURI;
 
-             for (var i = 0; i < nodes.length - 1; i++) {
-               var node1 = nodes[i];
-               var node2 = nodes[i + 1];
-               var issue = getWayIssueIfAny(node1, node2, way);
-               if (issue) issues.push(issue);
-             }
+         if (uri && prefix == null) {
+           //console.log(prefix)
+           var prefix = refNode.lookupPrefix(uri);
 
-             return issues;
+           if (prefix == null) {
+             //isHTML = true;
+             var visibleNamespaces = [{
+               namespace: uri,
+               prefix: null
+             } //{namespace:uri,prefix:''}
+             ];
            }
+         }
 
-           function getIssuesForVertex(node, parentWays) {
-             var issues = [];
-
-             function checkForCloseness(node1, node2, way) {
-               var issue = getWayIssueIfAny(node1, node2, way);
-               if (issue) issues.push(issue);
-             }
+         serializeToString(this, buf, isHtml, nodeFilter, visibleNamespaces); //console.log('###',this.nodeType,uri,prefix,buf.join(''))
 
-             for (var i = 0; i < parentWays.length; i++) {
-               var parentWay = parentWays[i];
-               if (!shouldCheckWay(parentWay)) continue;
-               var lastIndex = parentWay.nodes.length - 1;
+         return buf.join('');
+       }
 
-               for (var j = 0; j < parentWay.nodes.length; j++) {
-                 if (j !== 0) {
-                   if (parentWay.nodes[j - 1] === node.id) {
-                     checkForCloseness(node, graph.entity(parentWay.nodes[j]), parentWay);
-                   }
-                 }
+       function needNamespaceDefine(node, isHTML, visibleNamespaces) {
+         var prefix = node.prefix || '';
+         var uri = node.namespaceURI;
 
-                 if (j !== lastIndex) {
-                   if (parentWay.nodes[j + 1] === node.id) {
-                     checkForCloseness(graph.entity(parentWay.nodes[j]), node, parentWay);
-                   }
-                 }
-               }
-             }
+         if (!prefix && !uri) {
+           return false;
+         }
 
-             return issues;
-           }
+         if (prefix === "xml" && uri === "http://www.w3.org/XML/1998/namespace" || uri == 'http://www.w3.org/2000/xmlns/') {
+           return false;
+         }
 
-           function thresholdMetersForWay(way) {
-             if (!shouldCheckWay(way)) return 0;
-             var wayType = wayTypeFor(way); // don't flag boundaries since they might be highly detailed and can't be easily verified
+         var i = visibleNamespaces.length; //console.log('@@@@',node.tagName,prefix,uri,visibleNamespaces)
 
-             if (wayType === 'boundary') return 0; // expect some features to be mapped with higher levels of detail
+         while (i--) {
+           var ns = visibleNamespaces[i]; // get namespace prefix
+           //console.log(node.nodeType,node.tagName,ns.prefix,prefix)
 
-             if (wayType === 'indoor') return 0.01;
-             if (wayType === 'building') return 0.05;
-             if (wayType === 'path') return 0.1;
-             return 0.2;
+           if (ns.prefix == prefix) {
+             return ns.namespace != uri;
            }
+         } //console.log(isHTML,uri,prefix=='')
+         //if(isHTML && prefix ==null && uri == 'http://www.w3.org/1999/xhtml'){
+         //    return false;
+         //}
+         //node.flag = '11111'
+         //console.error(3,true,node.flag,node.prefix,node.namespaceURI)
 
-           function getIssuesForDetachedPoint(node) {
-             var issues = [];
-             var lon = node.loc[0];
-             var lat = node.loc[1];
-             var lon_range = geoMetersToLon(pointThresholdMeters, lat) / 2;
-             var lat_range = geoMetersToLat(pointThresholdMeters) / 2;
-             var queryExtent = geoExtent([[lon - lon_range, lat - lat_range], [lon + lon_range, lat + lat_range]]);
-             var intersected = context.history().tree().intersects(queryExtent, graph);
 
-             for (var j = 0; j < intersected.length; j++) {
-               var nearby = intersected[j];
-               if (nearby.id === node.id) continue;
-               if (nearby.type !== 'node' || nearby.geometry(graph) !== 'point') continue;
+         return true;
+       }
 
-               if (nearby.loc === node.loc || geoSphericalDistance(node.loc, nearby.loc) < pointThresholdMeters) {
-                 // allow very close points if tags indicate the z-axis might vary
-                 var zAxisKeys = {
-                   layer: true,
-                   level: true,
-                   'addr:housenumber': true,
-                   'addr:unit': true
-                 };
-                 var zAxisDifferentiates = false;
+       function serializeToString(node, buf, isHTML, nodeFilter, visibleNamespaces) {
+         if (nodeFilter) {
+           node = nodeFilter(node);
 
-                 for (var key in zAxisKeys) {
-                   var nodeValue = node.tags[key] || '0';
-                   var nearbyValue = nearby.tags[key] || '0';
+           if (node) {
+             if (typeof node == 'string') {
+               buf.push(node);
+               return;
+             }
+           } else {
+             return;
+           } //buf.sort.apply(attrs, attributeSorter);
 
-                   if (nodeValue !== nearbyValue) {
-                     zAxisDifferentiates = true;
-                     break;
-                   }
-                 }
+         }
 
-                 if (zAxisDifferentiates) continue;
-                 issues.push(new validationIssue({
-                   type: type,
-                   subtype: 'detached',
-                   severity: 'warning',
-                   message: function message(context) {
-                     var entity = context.hasEntity(this.entityIds[0]),
-                         entity2 = context.hasEntity(this.entityIds[1]);
-                     return entity && entity2 ? _t.html('issues.close_nodes.detached.message', {
-                       feature: utilDisplayLabel(entity, context.graph()),
-                       feature2: utilDisplayLabel(entity2, context.graph())
-                     }) : '';
-                   },
-                   reference: showReference,
-                   entityIds: [node.id, nearby.id],
-                   dynamicFixes: function dynamicFixes() {
-                     return [new validationIssueFix({
-                       icon: 'iD-operation-disconnect',
-                       title: _t.html('issues.fix.move_points_apart.title')
-                     }), new validationIssueFix({
-                       icon: 'iD-icon-layers',
-                       title: _t.html('issues.fix.use_different_layers_or_levels.title')
-                     })];
-                   }
-                 }));
+         switch (node.nodeType) {
+           case ELEMENT_NODE:
+             if (!visibleNamespaces) visibleNamespaces = [];
+             visibleNamespaces.length;
+             var attrs = node.attributes;
+             var len = attrs.length;
+             var child = node.firstChild;
+             var nodeName = node.tagName;
+             isHTML = htmlns === node.namespaceURI || isHTML;
+             buf.push('<', nodeName);
+
+             for (var i = 0; i < len; i++) {
+               // add namespaces for attributes
+               var attr = attrs.item(i);
+
+               if (attr.prefix == 'xmlns') {
+                 visibleNamespaces.push({
+                   prefix: attr.localName,
+                   namespace: attr.value
+                 });
+               } else if (attr.nodeName == 'xmlns') {
+                 visibleNamespaces.push({
+                   prefix: '',
+                   namespace: attr.value
+                 });
                }
              }
 
-             return issues;
+             for (var i = 0; i < len; i++) {
+               var attr = attrs.item(i);
 
-             function showReference(selection) {
-               var referenceText = _t('issues.close_nodes.detached.reference');
-               selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(referenceText);
-             }
-           }
+               if (needNamespaceDefine(attr, isHTML, visibleNamespaces)) {
+                 var prefix = attr.prefix || '';
+                 var uri = attr.namespaceURI;
+                 var ns = prefix ? ' xmlns:' + prefix : " xmlns";
+                 buf.push(ns, '="', uri, '"');
+                 visibleNamespaces.push({
+                   prefix: prefix,
+                   namespace: uri
+                 });
+               }
 
-           function getWayIssueIfAny(node1, node2, way) {
-             if (node1.id === node2.id || node1.hasInterestingTags() && node2.hasInterestingTags()) {
-               return null;
-             }
+               serializeToString(attr, buf, isHTML, nodeFilter, visibleNamespaces);
+             } // add namespace for current node               
 
-             if (node1.loc !== node2.loc) {
-               var parentWays1 = graph.parentWays(node1);
-               var parentWays2 = new Set(graph.parentWays(node2));
-               var sharedWays = parentWays1.filter(function (parentWay) {
-                 return parentWays2.has(parentWay);
-               });
-               var thresholds = sharedWays.map(function (parentWay) {
-                 return thresholdMetersForWay(parentWay);
+
+             if (needNamespaceDefine(node, isHTML, visibleNamespaces)) {
+               var prefix = node.prefix || '';
+               var uri = node.namespaceURI;
+               var ns = prefix ? ' xmlns:' + prefix : " xmlns";
+               buf.push(ns, '="', uri, '"');
+               visibleNamespaces.push({
+                 prefix: prefix,
+                 namespace: uri
                });
-               var threshold = Math.min.apply(Math, _toConsumableArray(thresholds));
-               var distance = geoSphericalDistance(node1.loc, node2.loc);
-               if (distance > threshold) return null;
              }
 
-             return new validationIssue({
-               type: type,
-               subtype: 'vertices',
-               severity: 'warning',
-               message: function message(context) {
-                 var entity = context.hasEntity(this.entityIds[0]);
-                 return entity ? _t.html('issues.close_nodes.message', {
-                   way: utilDisplayLabel(entity, context.graph())
-                 }) : '';
-               },
-               reference: showReference,
-               entityIds: [way.id, node1.id, node2.id],
-               loc: node1.loc,
-               dynamicFixes: function dynamicFixes() {
-                 return [new validationIssueFix({
-                   icon: 'iD-icon-plus',
-                   title: _t.html('issues.fix.merge_points.title'),
-                   onClick: function onClick(context) {
-                     var entityIds = this.issue.entityIds;
-                     var action = actionMergeNodes([entityIds[1], entityIds[2]]);
-                     context.perform(action, _t('issues.fix.merge_close_vertices.annotation'));
+             if (child || isHTML && !/^(?:meta|link|img|br|hr|input)$/i.test(nodeName)) {
+               buf.push('>'); //if is cdata child node
+
+               if (isHTML && /^script$/i.test(nodeName)) {
+                 while (child) {
+                   if (child.data) {
+                     buf.push(child.data);
+                   } else {
+                     serializeToString(child, buf, isHTML, nodeFilter, visibleNamespaces);
                    }
-                 }), new validationIssueFix({
-                   icon: 'iD-operation-disconnect',
-                   title: _t.html('issues.fix.move_points_apart.title')
-                 })];
-               }
-             });
 
-             function showReference(selection) {
-               var referenceText = _t('issues.close_nodes.reference');
-               selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(referenceText);
-             }
-           }
-         };
+                   child = child.nextSibling;
+                 }
+               } else {
+                 while (child) {
+                   serializeToString(child, buf, isHTML, nodeFilter, visibleNamespaces);
+                   child = child.nextSibling;
+                 }
+               }
 
-         validation.type = type;
-         return validation;
-       }
+               buf.push('</', nodeName, '>');
+             } else {
+               buf.push('/>');
+             } // remove added visible namespaces
+             //visibleNamespaces.length = startVisibleNamespaces;
 
-       function validationCrossingWays(context) {
-         var type = 'crossing_ways'; // returns the way or its parent relation, whichever has a useful feature type
 
-         function getFeatureWithFeatureTypeTagsForWay(way, graph) {
-           if (getFeatureType(way, graph) === null) {
-             // if the way doesn't match a feature type, check its parent relations
-             var parentRels = graph.parentRelations(way);
+             return;
 
-             for (var i = 0; i < parentRels.length; i++) {
-               var rel = parentRels[i];
+           case DOCUMENT_NODE:
+           case DOCUMENT_FRAGMENT_NODE:
+             var child = node.firstChild;
 
-               if (getFeatureType(rel, graph) !== null) {
-                 return rel;
-               }
+             while (child) {
+               serializeToString(child, buf, isHTML, nodeFilter, visibleNamespaces);
+               child = child.nextSibling;
              }
-           }
 
-           return way;
-         }
+             return;
 
-         function hasTag(tags, key) {
-           return tags[key] !== undefined && tags[key] !== 'no';
-         }
+           case ATTRIBUTE_NODE:
+             return buf.push(' ', node.name, '="', node.value.replace(/[<&"]/g, _xmlEncoder), '"');
 
-         function taggedAsIndoor(tags) {
-           return hasTag(tags, 'indoor') || hasTag(tags, 'level') || tags.highway === 'corridor';
-         }
+           case TEXT_NODE:
+             return buf.push(node.data.replace(/[<&]/g, _xmlEncoder));
 
-         function allowsBridge(featureType) {
-           return featureType === 'highway' || featureType === 'railway' || featureType === 'waterway';
-         }
+           case CDATA_SECTION_NODE:
+             return buf.push('<![CDATA[', node.data, ']]>');
 
-         function allowsTunnel(featureType) {
-           return featureType === 'highway' || featureType === 'railway' || featureType === 'waterway';
-         } // discard
+           case COMMENT_NODE:
+             return buf.push("<!--", node.data, "-->");
 
+           case DOCUMENT_TYPE_NODE:
+             var pubid = node.publicId;
+             var sysid = node.systemId;
+             buf.push('<!DOCTYPE ', node.name);
 
-         var ignoredBuildings = {
-           demolished: true,
-           dismantled: true,
-           proposed: true,
-           razed: true
-         };
+             if (pubid) {
+               buf.push(' PUBLIC "', pubid);
 
-         function getFeatureType(entity, graph) {
-           var geometry = entity.geometry(graph);
-           if (geometry !== 'line' && geometry !== 'area') return null;
-           var tags = entity.tags;
-           if (hasTag(tags, 'building') && !ignoredBuildings[tags.building]) return 'building';
-           if (hasTag(tags, 'highway') && osmRoutableHighwayTagValues[tags.highway]) return 'highway'; // don't check railway or waterway areas
+               if (sysid && sysid != '.') {
+                 buf.push('" "', sysid);
+               }
 
-           if (geometry !== 'line') return null;
-           if (hasTag(tags, 'railway') && osmRailwayTrackTagValues[tags.railway]) return 'railway';
-           if (hasTag(tags, 'waterway') && osmFlowingWaterwayTagValues[tags.waterway]) return 'waterway';
-           return null;
-         }
+               buf.push('">');
+             } else if (sysid && sysid != '.') {
+               buf.push(' SYSTEM "', sysid, '">');
+             } else {
+               var sub = node.internalSubset;
 
-         function isLegitCrossing(tags1, featureType1, tags2, featureType2) {
-           // assume 0 by default
-           var level1 = tags1.level || '0';
-           var level2 = tags2.level || '0';
+               if (sub) {
+                 buf.push(" [", sub, "]");
+               }
 
-           if (taggedAsIndoor(tags1) && taggedAsIndoor(tags2) && level1 !== level2) {
-             // assume features don't interact if they're indoor on different levels
-             return true;
-           } // assume 0 by default; don't use way.layer() since we account for structures here
+               buf.push(">");
+             }
 
+             return;
 
-           var layer1 = tags1.layer || '0';
-           var layer2 = tags2.layer || '0';
+           case PROCESSING_INSTRUCTION_NODE:
+             return buf.push("<?", node.target, " ", node.data, "?>");
 
-           if (allowsBridge(featureType1) && allowsBridge(featureType2)) {
-             if (hasTag(tags1, 'bridge') && !hasTag(tags2, 'bridge')) return true;
-             if (!hasTag(tags1, 'bridge') && hasTag(tags2, 'bridge')) return true; // crossing bridges must use different layers
+           case ENTITY_REFERENCE_NODE:
+             return buf.push('&', node.nodeName, ';');
+           //case ENTITY_NODE:
+           //case NOTATION_NODE:
 
-             if (hasTag(tags1, 'bridge') && hasTag(tags2, 'bridge') && layer1 !== layer2) return true;
-           } else if (allowsBridge(featureType1) && hasTag(tags1, 'bridge')) return true;else if (allowsBridge(featureType2) && hasTag(tags2, 'bridge')) return true;
+           default:
+             buf.push('??', node.nodeName);
+         }
+       }
 
-           if (allowsTunnel(featureType1) && allowsTunnel(featureType2)) {
-             if (hasTag(tags1, 'tunnel') && !hasTag(tags2, 'tunnel')) return true;
-             if (!hasTag(tags1, 'tunnel') && hasTag(tags2, 'tunnel')) return true; // crossing tunnels must use different layers
+       function _importNode(doc, node, deep) {
+         var node2;
 
-             if (hasTag(tags1, 'tunnel') && hasTag(tags2, 'tunnel') && layer1 !== layer2) return true;
-           } else if (allowsTunnel(featureType1) && hasTag(tags1, 'tunnel')) return true;else if (allowsTunnel(featureType2) && hasTag(tags2, 'tunnel')) return true; // don't flag crossing waterways and pier/highways
+         switch (node.nodeType) {
+           case ELEMENT_NODE:
+             node2 = node.cloneNode(false);
+             node2.ownerDocument = doc;
+           //var attrs = node2.attributes;
+           //var len = attrs.length;
+           //for(var i=0;i<len;i++){
+           //node2.setAttributeNodeNS(importNode(doc,attrs.item(i),deep));
+           //}
 
+           case DOCUMENT_FRAGMENT_NODE:
+             break;
 
-           if (featureType1 === 'waterway' && featureType2 === 'highway' && tags2.man_made === 'pier') return true;
-           if (featureType2 === 'waterway' && featureType1 === 'highway' && tags1.man_made === 'pier') return true;
+           case ATTRIBUTE_NODE:
+             deep = true;
+             break;
+           //case ENTITY_REFERENCE_NODE:
+           //case PROCESSING_INSTRUCTION_NODE:
+           ////case TEXT_NODE:
+           //case CDATA_SECTION_NODE:
+           //case COMMENT_NODE:
+           //  deep = false;
+           //  break;
+           //case DOCUMENT_NODE:
+           //case DOCUMENT_TYPE_NODE:
+           //cannot be imported.
+           //case ENTITY_NODE:
+           //case NOTATION_NODE:
+           //can not hit in level3
+           //default:throw e;
+         }
 
-           if (featureType1 === 'building' || featureType2 === 'building') {
-             // for building crossings, different layers are enough
-             if (layer1 !== layer2) return true;
+         if (!node2) {
+           node2 = node.cloneNode(false); //false
+         }
+
+         node2.ownerDocument = doc;
+         node2.parentNode = null;
+
+         if (deep) {
+           var child = node.firstChild;
+
+           while (child) {
+             node2.appendChild(_importNode(doc, child, deep));
+             child = child.nextSibling;
            }
+         }
 
-           return false;
-         } // highway values for which we shouldn't recommend connecting to waterways
+         return node2;
+       } //
+       //var _relationMap = {firstChild:1,lastChild:1,previousSibling:1,nextSibling:1,
+       //                                      attributes:1,childNodes:1,parentNode:1,documentElement:1,doctype,};
 
 
-         var highwaysDisallowingFords = {
-           motorway: true,
-           motorway_link: true,
-           trunk: true,
-           trunk_link: true,
-           primary: true,
-           primary_link: true,
-           secondary: true,
-           secondary_link: true
-         };
-         var nonCrossingHighways = {
-           track: true
-         };
+       function _cloneNode(doc, node, deep) {
+         var node2 = new node.constructor();
+
+         for (var n in node) {
+           var v = node[n];
+
+           if (_typeof(v) != 'object') {
+             if (v != node2[n]) {
+               node2[n] = v;
+             }
+           }
+         }
 
-         function tagsForConnectionNodeIfAllowed(entity1, entity2, graph) {
-           var featureType1 = getFeatureType(entity1, graph);
-           var featureType2 = getFeatureType(entity2, graph);
-           var geometry1 = entity1.geometry(graph);
-           var geometry2 = entity2.geometry(graph);
-           var bothLines = geometry1 === 'line' && geometry2 === 'line';
+         if (node.childNodes) {
+           node2.childNodes = new NodeList();
+         }
 
-           if (featureType1 === featureType2) {
-             if (featureType1 === 'highway') {
-               var entity1IsPath = osmPathHighwayTagValues[entity1.tags.highway];
-               var entity2IsPath = osmPathHighwayTagValues[entity2.tags.highway];
+         node2.ownerDocument = doc;
 
-               if ((entity1IsPath || entity2IsPath) && entity1IsPath !== entity2IsPath) {
-                 // one feature is a path but not both
-                 var roadFeature = entity1IsPath ? entity2 : entity1;
+         switch (node2.nodeType) {
+           case ELEMENT_NODE:
+             var attrs = node.attributes;
+             var attrs2 = node2.attributes = new NamedNodeMap();
+             var len = attrs.length;
+             attrs2._ownerElement = node2;
 
-                 if (nonCrossingHighways[roadFeature.tags.highway]) {
-                   // don't mark path connections with certain roads as crossings
-                   return {};
-                 }
+             for (var i = 0; i < len; i++) {
+               node2.setAttributeNode(_cloneNode(doc, attrs.item(i), true));
+             }
 
-                 var pathFeature = entity1IsPath ? entity1 : entity2;
+             break;
 
-                 if (['marked', 'unmarked'].indexOf(pathFeature.tags.crossing) !== -1) {
-                   // if the path is a crossing, match the crossing type
-                   return bothLines ? {
-                     highway: 'crossing',
-                     crossing: pathFeature.tags.crossing
-                   } : {};
-                 } // don't add a `crossing` subtag to ambiguous crossings
+           case ATTRIBUTE_NODE:
+             deep = true;
+         }
 
+         if (deep) {
+           var child = node.firstChild;
 
-                 return bothLines ? {
-                   highway: 'crossing'
-                 } : {};
-               }
+           while (child) {
+             node2.appendChild(_cloneNode(doc, child, deep));
+             child = child.nextSibling;
+           }
+         }
 
-               return {};
-             }
+         return node2;
+       }
 
-             if (featureType1 === 'waterway') return {};
-             if (featureType1 === 'railway') return {};
-           } else {
-             var featureTypes = [featureType1, featureType2];
+       function __set__(object, key, value) {
+         object[key] = value;
+       } //do dynamic
 
-             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-tram connections use this tag
-                   if (isTram) return {
-                     railway: 'tram_crossing'
-                   }; // other path-rail connections use this tag
+       try {
+         if (Object.defineProperty) {
+           var getTextContent = function getTextContent(node) {
+             switch (node.nodeType) {
+               case ELEMENT_NODE:
+               case DOCUMENT_FRAGMENT_NODE:
+                 var buf = [];
+                 node = node.firstChild;
 
-                   return {
-                     railway: 'crossing'
-                   };
-                 } else {
-                   // path-tram connections use this tag
-                   if (isTram) return {
-                     railway: 'tram_level_crossing'
-                   }; // other road-rail connections use this tag
+                 while (node) {
+                   if (node.nodeType !== 7 && node.nodeType !== 8) {
+                     buf.push(getTextContent(node));
+                   }
 
-                   return {
-                     railway: 'level_crossing'
-                   };
+                   node = node.nextSibling;
                  }
-               }
 
-               if (featureTypes.indexOf('waterway') !== -1) {
-                 // do not allow fords on structures
-                 if (hasTag(entity1.tags, 'tunnel') && hasTag(entity2.tags, 'tunnel')) return null;
-                 if (hasTag(entity1.tags, 'bridge') && hasTag(entity2.tags, 'bridge')) return null;
+                 return buf.join('');
 
-                 if (highwaysDisallowingFords[entity1.tags.highway] || highwaysDisallowingFords[entity2.tags.highway]) {
-                   // do not allow fords on major highways
-                   return null;
-                 }
+               default:
+                 return node.nodeValue;
+             }
+           };
 
-                 return bothLines ? {
-                   ford: 'yes'
-                 } : {};
+           Object.defineProperty(LiveNodeList.prototype, 'length', {
+             get: function get() {
+               _updateLiveList(this);
+
+               return this.$$length;
+             }
+           });
+           Object.defineProperty(Node.prototype, 'textContent', {
+             get: function get() {
+               return getTextContent(this);
+             },
+             set: function set(data) {
+               switch (this.nodeType) {
+                 case ELEMENT_NODE:
+                 case DOCUMENT_FRAGMENT_NODE:
+                   while (this.firstChild) {
+                     this.removeChild(this.firstChild);
+                   }
+
+                   if (data || String(data)) {
+                     this.appendChild(this.ownerDocument.createTextNode(data));
+                   }
+
+                   break;
+
+                 default:
+                   //TODO:
+                   this.data = data;
+                   this.value = data;
+                   this.nodeValue = data;
                }
              }
-           }
+           });
 
-           return null;
+           __set__ = function __set__(object, key, value) {
+             //console.log(value)
+             object['$$' + key] = value;
+           };
          }
+       } catch (e) {//ie8
+       } //if(typeof require == 'function'){
 
-         function findCrossingsByWay(way1, graph, tree) {
-           var edgeCrossInfos = [];
-           if (way1.type !== 'way') return edgeCrossInfos;
-           var taggedFeature1 = getFeatureWithFeatureTypeTagsForWay(way1, graph);
-           var way1FeatureType = getFeatureType(taggedFeature1, graph);
-           if (way1FeatureType === null) return edgeCrossInfos;
-           var checkedSingleCrossingWays = {}; // declare vars ahead of time to reduce garbage collection
 
-           var i, j;
-           var extent;
-           var n1, n2, nA, nB, nAId, nBId;
-           var segment1, segment2;
-           var oneOnly;
-           var segmentInfos, segment2Info, way2, taggedFeature2, way2FeatureType;
-           var way1Nodes = graph.childNodes(way1);
-           var comparedWays = {};
+       var DOMImplementation_1 = DOMImplementation;
+       var XMLSerializer_1 = XMLSerializer$1; //}
 
-           for (i = 0; i < way1Nodes.length - 1; i++) {
-             n1 = way1Nodes[i];
-             n2 = way1Nodes[i + 1];
-             extent = geoExtent([[Math.min(n1.loc[0], n2.loc[0]), Math.min(n1.loc[1], n2.loc[1])], [Math.max(n1.loc[0], n2.loc[0]), Math.max(n1.loc[1], n2.loc[1])]]); // Optimize by only checking overlapping segments, not every segment
-             // of overlapping ways
+       var dom = {
+         DOMImplementation: DOMImplementation_1,
+         XMLSerializer: XMLSerializer_1
+       };
 
-             segmentInfos = tree.waySegments(extent, graph);
+       var domParser = createCommonjsModule(function (module, exports) {
+         function DOMParser(options) {
+           this.options = options || {
+             locator: {}
+           };
+         }
 
-             for (j = 0; j < segmentInfos.length; j++) {
-               segment2Info = segmentInfos[j]; // don't check for self-intersection in this validation
+         DOMParser.prototype.parseFromString = function (source, mimeType) {
+           var options = this.options;
+           var sax = new XMLReader();
+           var domBuilder = options.domBuilder || new DOMHandler(); //contentHandler and LexicalHandler
 
-               if (segment2Info.wayId === way1.id) continue; // skip if this way was already checked and only one issue is needed
+           var errorHandler = options.errorHandler;
+           var locator = options.locator;
+           var defaultNSMap = options.xmlns || {};
+           var entityMap = {
+             'lt': '<',
+             'gt': '>',
+             'amp': '&',
+             'quot': '"',
+             'apos': "'"
+           };
 
-               if (checkedSingleCrossingWays[segment2Info.wayId]) continue; // mark this way as checked even if there are no crossings
+           if (locator) {
+             domBuilder.setDocumentLocator(locator);
+           }
 
-               comparedWays[segment2Info.wayId] = true;
-               way2 = graph.hasEntity(segment2Info.wayId);
-               if (!way2) continue;
-               taggedFeature2 = getFeatureWithFeatureTypeTagsForWay(way2, graph); // only check crossing highway, waterway, building, and railway
+           sax.errorHandler = buildErrorHandler(errorHandler, domBuilder, locator);
+           sax.domBuilder = options.domBuilder || domBuilder;
 
-               way2FeatureType = getFeatureType(taggedFeature2, graph);
+           if (/\/x?html?$/.test(mimeType)) {
+             entityMap.nbsp = '\xa0';
+             entityMap.copy = '\xa9';
+             defaultNSMap[''] = 'http://www.w3.org/1999/xhtml';
+           }
 
-               if (way2FeatureType === null || isLegitCrossing(taggedFeature1.tags, way1FeatureType, taggedFeature2.tags, way2FeatureType)) {
-                 continue;
-               } // create only one issue for building crossings
+           defaultNSMap.xml = defaultNSMap.xml || 'http://www.w3.org/XML/1998/namespace';
 
+           if (source) {
+             sax.parse(source, defaultNSMap, entityMap);
+           } else {
+             sax.errorHandler.error("invalid doc source");
+           }
 
-               oneOnly = way1FeatureType === 'building' || way2FeatureType === 'building';
-               nAId = segment2Info.nodes[0];
-               nBId = segment2Info.nodes[1];
+           return domBuilder.doc;
+         };
 
-               if (nAId === n1.id || nAId === n2.id || nBId === n1.id || nBId === n2.id) {
-                 // n1 or n2 is a connection node; skip
-                 continue;
-               }
+         function buildErrorHandler(errorImpl, domBuilder, locator) {
+           if (!errorImpl) {
+             if (domBuilder instanceof DOMHandler) {
+               return domBuilder;
+             }
 
-               nA = graph.hasEntity(nAId);
-               if (!nA) continue;
-               nB = graph.hasEntity(nBId);
-               if (!nB) continue;
-               segment1 = [n1.loc, n2.loc];
-               segment2 = [nA.loc, nB.loc];
-               var point = geoLineIntersection(segment1, segment2);
+             errorImpl = domBuilder;
+           }
 
-               if (point) {
-                 edgeCrossInfos.push({
-                   wayInfos: [{
-                     way: way1,
-                     featureType: way1FeatureType,
-                     edge: [n1.id, n2.id]
-                   }, {
-                     way: way2,
-                     featureType: way2FeatureType,
-                     edge: [nA.id, nB.id]
-                   }],
-                   crossPoint: point
-                 });
+           var errorHandler = {};
+           var isCallback = errorImpl instanceof Function;
+           locator = locator || {};
 
-                 if (oneOnly) {
-                   checkedSingleCrossingWays[way2.id] = true;
-                   break;
-                 }
-               }
+           function build(key) {
+             var fn = errorImpl[key];
+
+             if (!fn && isCallback) {
+               fn = errorImpl.length == 2 ? function (msg) {
+                 errorImpl(key, msg);
+               } : errorImpl;
              }
+
+             errorHandler[key] = fn && function (msg) {
+               fn('[xmldom ' + key + ']\t' + msg + _locator(locator));
+             } || function () {};
            }
 
-           return edgeCrossInfos;
+           build('warning');
+           build('error');
+           build('fatalError');
+           return errorHandler;
+         } //console.log('#\n\n\n\n\n\n\n####')
+
+         /**
+          * +ContentHandler+ErrorHandler
+          * +LexicalHandler+EntityResolver2
+          * -DeclHandler-DTDHandler 
+          * 
+          * DefaultHandler:EntityResolver, DTDHandler, ContentHandler, ErrorHandler
+          * DefaultHandler2:DefaultHandler,LexicalHandler, DeclHandler, EntityResolver2
+          * @link http://www.saxproject.org/apidoc/org/xml/sax/helpers/DefaultHandler.html
+          */
+
+
+         function DOMHandler() {
+           this.cdata = false;
          }
 
-         function waysToCheck(entity, graph) {
-           var featureType = getFeatureType(entity, graph);
-           if (!featureType) return [];
+         function position(locator, node) {
+           node.lineNumber = locator.lineNumber;
+           node.columnNumber = locator.columnNumber;
+         }
+         /**
+          * @see org.xml.sax.ContentHandler#startDocument
+          * @link http://www.saxproject.org/apidoc/org/xml/sax/ContentHandler.html
+          */
 
-           if (entity.type === 'way') {
-             return [entity];
-           } else if (entity.type === 'relation') {
-             return entity.members.reduce(function (array, member) {
-               if (member.type === 'way' && ( // only look at geometry ways
-               !member.role || member.role === 'outer' || member.role === 'inner')) {
-                 var entity = graph.hasEntity(member.id); // don't add duplicates
 
-                 if (entity && array.indexOf(entity) === -1) {
-                   array.push(entity);
-                 }
-               }
+         DOMHandler.prototype = {
+           startDocument: function startDocument() {
+             this.doc = new DOMImplementation().createDocument(null, null, null);
 
-               return array;
-             }, []);
-           }
+             if (this.locator) {
+               this.doc.documentURI = this.locator.systemId;
+             }
+           },
+           startElement: function startElement(namespaceURI, localName, qName, attrs) {
+             var doc = this.doc;
+             var el = doc.createElementNS(namespaceURI, qName || localName);
+             var len = attrs.length;
+             appendElement(this, el);
+             this.currentElement = el;
+             this.locator && position(this.locator, el);
 
-           return [];
-         }
+             for (var i = 0; i < len; i++) {
+               var namespaceURI = attrs.getURI(i);
+               var value = attrs.getValue(i);
+               var qName = attrs.getQName(i);
+               var attr = doc.createAttributeNS(namespaceURI, qName);
+               this.locator && position(attrs.getLocator(i), attr);
+               attr.value = attr.nodeValue = value;
+               el.setAttributeNode(attr);
+             }
+           },
+           endElement: function endElement(namespaceURI, localName, qName) {
+             var current = this.currentElement;
+             current.tagName;
+             this.currentElement = current.parentNode;
+           },
+           startPrefixMapping: function startPrefixMapping(prefix, uri) {},
+           endPrefixMapping: function endPrefixMapping(prefix) {},
+           processingInstruction: function processingInstruction(target, data) {
+             var ins = this.doc.createProcessingInstruction(target, data);
+             this.locator && position(this.locator, ins);
+             appendElement(this, ins);
+           },
+           ignorableWhitespace: function ignorableWhitespace(ch, start, length) {},
+           characters: function characters(chars, start, length) {
+             chars = _toString.apply(this, arguments); //console.log(chars)
 
-         var validation = function checkCrossingWays(entity, graph) {
-           var tree = context.history().tree();
-           var ways = waysToCheck(entity, graph);
-           var issues = []; // declare these here to reduce garbage collection
+             if (chars) {
+               if (this.cdata) {
+                 var charNode = this.doc.createCDATASection(chars);
+               } else {
+                 var charNode = this.doc.createTextNode(chars);
+               }
 
-           var wayIndex, crossingIndex, crossings;
+               if (this.currentElement) {
+                 this.currentElement.appendChild(charNode);
+               } else if (/^\s*$/.test(chars)) {
+                 this.doc.appendChild(charNode); //process xml
+               }
 
-           for (wayIndex in ways) {
-             crossings = findCrossingsByWay(ways[wayIndex], graph, tree);
+               this.locator && position(this.locator, charNode);
+             }
+           },
+           skippedEntity: function skippedEntity(name) {},
+           endDocument: function endDocument() {
+             this.doc.normalize();
+           },
+           setDocumentLocator: function setDocumentLocator(locator) {
+             if (this.locator = locator) {
+               // && !('lineNumber' in locator)){
+               locator.lineNumber = 0;
+             }
+           },
+           //LexicalHandler
+           comment: function comment(chars, start, length) {
+             chars = _toString.apply(this, arguments);
+             var comm = this.doc.createComment(chars);
+             this.locator && position(this.locator, comm);
+             appendElement(this, comm);
+           },
+           startCDATA: function startCDATA() {
+             //used in characters() methods
+             this.cdata = true;
+           },
+           endCDATA: function endCDATA() {
+             this.cdata = false;
+           },
+           startDTD: function startDTD(name, publicId, systemId) {
+             var impl = this.doc.implementation;
 
-             for (crossingIndex in crossings) {
-               issues.push(createIssue(crossings[crossingIndex], graph));
+             if (impl && impl.createDocumentType) {
+               var dt = impl.createDocumentType(name, publicId, systemId);
+               this.locator && position(this.locator, dt);
+               appendElement(this, dt);
              }
-           }
+           },
 
-           return issues;
+           /**
+            * @see org.xml.sax.ErrorHandler
+            * @link http://www.saxproject.org/apidoc/org/xml/sax/ErrorHandler.html
+            */
+           warning: function warning(error) {
+             console.warn('[xmldom warning]\t' + error, _locator(this.locator));
+           },
+           error: function error(_error) {
+             console.error('[xmldom error]\t' + _error, _locator(this.locator));
+           },
+           fatalError: function fatalError(error) {
+             console.error('[xmldom fatalError]\t' + error, _locator(this.locator));
+             throw error;
+           }
          };
 
-         function createIssue(crossing, graph) {
-           // use the entities with the tags that define the feature type
-           crossing.wayInfos.sort(function (way1Info, way2Info) {
-             var type1 = way1Info.featureType;
-             var type2 = way2Info.featureType;
+         function _locator(l) {
+           if (l) {
+             return '\n@' + (l.systemId || '') + '#[line:' + l.lineNumber + ',col:' + l.columnNumber + ']';
+           }
+         }
 
-             if (type1 === type2) {
-               return utilDisplayLabel(way1Info.way, graph) > utilDisplayLabel(way2Info.way, graph);
-             } else if (type1 === 'waterway') {
-               return true;
-             } else if (type2 === 'waterway') {
-               return false;
+         function _toString(chars, start, length) {
+           if (typeof chars == 'string') {
+             return chars.substr(start, length);
+           } else {
+             //java sax connect width xmldom on rhino(what about: "? && !(chars instanceof String)")
+             if (chars.length >= start + length || start) {
+               return new java.lang.String(chars, start, length) + '';
              }
 
-             return type1 < type2;
-           });
-           var entities = crossing.wayInfos.map(function (wayInfo) {
-             return getFeatureWithFeatureTypeTagsForWay(wayInfo.way, graph);
-           });
-           var edges = [crossing.wayInfos[0].edge, crossing.wayInfos[1].edge];
-           var featureTypes = [crossing.wayInfos[0].featureType, crossing.wayInfos[1].featureType];
-           var connectionTags = tagsForConnectionNodeIfAllowed(entities[0], entities[1], graph);
-           var featureType1 = crossing.wayInfos[0].featureType;
-           var featureType2 = crossing.wayInfos[1].featureType;
-           var isCrossingIndoors = taggedAsIndoor(entities[0].tags) && taggedAsIndoor(entities[1].tags);
-           var isCrossingTunnels = allowsTunnel(featureType1) && hasTag(entities[0].tags, 'tunnel') && allowsTunnel(featureType2) && hasTag(entities[1].tags, 'tunnel');
-           var isCrossingBridges = allowsBridge(featureType1) && hasTag(entities[0].tags, 'bridge') && allowsBridge(featureType2) && hasTag(entities[1].tags, 'bridge');
-           var subtype = [featureType1, featureType2].sort().join('-');
-           var crossingTypeID = subtype;
-
-           if (isCrossingIndoors) {
-             crossingTypeID = 'indoor-indoor';
-           } else if (isCrossingTunnels) {
-             crossingTypeID = 'tunnel-tunnel';
-           } else if (isCrossingBridges) {
-             crossingTypeID = 'bridge-bridge';
+             return chars;
            }
+         }
+         /*
+          * @link http://www.saxproject.org/apidoc/org/xml/sax/ext/LexicalHandler.html
+          * used method of org.xml.sax.ext.LexicalHandler:
+          *  #comment(chars, start, length)
+          *  #startCDATA()
+          *  #endCDATA()
+          *  #startDTD(name, publicId, systemId)
+          *
+          *
+          * IGNORED method of org.xml.sax.ext.LexicalHandler:
+          *  #endDTD()
+          *  #startEntity(name)
+          *  #endEntity(name)
+          *
+          *
+          * @link http://www.saxproject.org/apidoc/org/xml/sax/ext/DeclHandler.html
+          * IGNORED method of org.xml.sax.ext.DeclHandler
+          *    #attributeDecl(eName, aName, type, mode, value)
+          *  #elementDecl(name, model)
+          *  #externalEntityDecl(name, publicId, systemId)
+          *  #internalEntityDecl(name, value)
+          * @link http://www.saxproject.org/apidoc/org/xml/sax/ext/EntityResolver2.html
+          * IGNORED method of org.xml.sax.EntityResolver2
+          *  #resolveEntity(String name,String publicId,String baseURI,String systemId)
+          *  #resolveEntity(publicId, systemId)
+          *  #getExternalSubset(name, baseURI)
+          * @link http://www.saxproject.org/apidoc/org/xml/sax/DTDHandler.html
+          * IGNORED method of org.xml.sax.DTDHandler
+          *  #notationDecl(name, publicId, systemId) {};
+          *  #unparsedEntityDecl(name, publicId, systemId, notationName) {};
+          */
 
-           if (connectionTags && (isCrossingIndoors || isCrossingTunnels || isCrossingBridges)) {
-             crossingTypeID += '_connectable';
-           }
 
-           return new validationIssue({
-             type: type,
-             subtype: subtype,
-             severity: 'warning',
-             message: function message(context) {
-               var graph = context.graph();
-               var entity1 = graph.hasEntity(this.entityIds[0]),
-                   entity2 = graph.hasEntity(this.entityIds[1]);
-               return entity1 && entity2 ? _t.html('issues.crossing_ways.message', {
-                 feature: utilDisplayLabel(entity1, graph),
-                 feature2: utilDisplayLabel(entity2, graph)
-               }) : '';
-             },
-             reference: showReference,
-             entityIds: entities.map(function (entity) {
-               return entity.id;
-             }),
-             data: {
-               edges: edges,
-               featureTypes: featureTypes,
-               connectionTags: connectionTags
-             },
-             // differentiate based on the loc since two ways can cross multiple times
-             hash: crossing.crossPoint.toString() + // if the edges change then so does the fix
-             edges.slice().sort(function (edge1, edge2) {
-               // order to assure hash is deterministic
-               return edge1[0] < edge2[0] ? -1 : 1;
-             }).toString() + // ensure the correct connection tags are added in the fix
-             JSON.stringify(connectionTags),
-             loc: crossing.crossPoint,
-             dynamicFixes: function dynamicFixes(context) {
-               var mode = context.mode();
-               if (!mode || mode.id !== 'select' || mode.selectedIDs().length !== 1) return [];
-               var selectedIndex = this.entityIds[0] === mode.selectedIDs()[0] ? 0 : 1;
-               var selectedFeatureType = this.data.featureTypes[selectedIndex];
-               var otherFeatureType = this.data.featureTypes[selectedIndex === 0 ? 1 : 0];
-               var fixes = [];
+         "endDTD,startEntity,endEntity,attributeDecl,elementDecl,externalEntityDecl,internalEntityDecl,resolveEntity,getExternalSubset,notationDecl,unparsedEntityDecl".replace(/\w+/g, function (key) {
+           DOMHandler.prototype[key] = function () {
+             return null;
+           };
+         });
+         /* Private static helpers treated below as private instance methods, so don't need to add these to the public API; we might use a Relator to also get rid of non-standard public properties */
 
-               if (connectionTags) {
-                 fixes.push(makeConnectWaysFix(this.data.connectionTags));
-               }
+         function appendElement(hander, node) {
+           if (!hander.currentElement) {
+             hander.doc.appendChild(node);
+           } else {
+             hander.currentElement.appendChild(node);
+           }
+         } //appendChild and setAttributeNS are preformance key
+         //if(typeof require == 'function'){
 
-               if (isCrossingIndoors) {
-                 fixes.push(new validationIssueFix({
-                   icon: 'iD-icon-layers',
-                   title: _t.html('issues.fix.use_different_levels.title')
-                 }));
-               } else if (isCrossingTunnels || isCrossingBridges || featureType1 === 'building' || featureType2 === 'building') {
-                 fixes.push(makeChangeLayerFix('higher'));
-                 fixes.push(makeChangeLayerFix('lower')); // can only add bridge/tunnel if both features are lines
-               } 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 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 uncommon
 
+         var XMLReader = sax.XMLReader;
+         var DOMImplementation = exports.DOMImplementation = dom.DOMImplementation;
+         exports.XMLSerializer = dom.XMLSerializer;
+         exports.DOMParser = DOMParser; //}
+       });
 
-                 var skipTunnelFix = otherFeatureType === 'waterway' && selectedFeatureType !== 'waterway';
+       var togeojson = createCommonjsModule(function (module, exports) {
+         var toGeoJSON = function () {
 
-                 if (allowsTunnel(selectedFeatureType) && !skipTunnelFix) {
-                   fixes.push(makeAddBridgeOrTunnelFix('add_a_tunnel', 'temaki-tunnel', 'tunnel'));
-                 }
-               } // repositioning the features is always an option
+           var removeSpace = /\s*/g,
+               trimSpace = /^\s*|\s*$/g,
+               splitSpace = /\s+/; // generate a short, numeric hash of a string
 
+           function okhash(x) {
+             if (!x || !x.length) return 0;
 
-               fixes.push(new validationIssueFix({
-                 icon: 'iD-operation-move',
-                 title: _t.html('issues.fix.reposition_features.title')
-               }));
-               return fixes;
+             for (var i = 0, h = 0; i < x.length; i++) {
+               h = (h << 5) - h + x.charCodeAt(i) | 0;
              }
-           });
-
-           function showReference(selection) {
-             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.crossing_ways.' + crossingTypeID + '.reference'));
-           }
-         }
 
-         function makeAddBridgeOrTunnelFix(fixTitleID, iconName, bridgeOrTunnel) {
-           return new validationIssueFix({
-             icon: iconName,
-             title: _t.html('issues.fix.' + fixTitleID + '.title'),
-             onClick: function onClick(context) {
-               var mode = context.mode();
-               if (!mode || mode.id !== 'select') return;
-               var selectedIDs = mode.selectedIDs();
-               if (selectedIDs.length !== 1) return;
-               var selectedWayID = selectedIDs[0];
-               if (!context.hasEntity(selectedWayID)) return;
-               var resultWayIDs = [selectedWayID];
-               var edge, crossedEdge, crossedWayID;
+             return h;
+           } // all Y children of X
 
-               if (this.issue.entityIds[0] === selectedWayID) {
-                 edge = this.issue.data.edges[0];
-                 crossedEdge = this.issue.data.edges[1];
-                 crossedWayID = this.issue.entityIds[1];
-               } else {
-                 edge = this.issue.data.edges[1];
-                 crossedEdge = this.issue.data.edges[0];
-                 crossedWayID = this.issue.entityIds[0];
-               }
 
-               var crossingLoc = this.issue.loc;
-               var projection = context.projection;
+           function get(x, y) {
+             return x.getElementsByTagName(y);
+           }
 
-               var action = function actionAddStructure(graph) {
-                 var edgeNodes = [graph.entity(edge[0]), graph.entity(edge[1])];
-                 var crossedWay = graph.hasEntity(crossedWayID); // use the explicit width of the crossed feature as the structure length, if available
+           function attr(x, y) {
+             return x.getAttribute(y);
+           }
 
-                 var structLengthMeters = crossedWay && crossedWay.tags.width && parseFloat(crossedWay.tags.width);
+           function attrf(x, y) {
+             return parseFloat(attr(x, y));
+           } // one Y child of X, if any, otherwise null
 
-                 if (!structLengthMeters) {
-                   // if no explicit width is set, approximate the width based on the tags
-                   structLengthMeters = crossedWay && crossedWay.impliedLineWidthMeters();
-                 }
 
-                 if (structLengthMeters) {
-                   if (getFeatureType(crossedWay, graph) === 'railway') {
-                     // bridges over railways are generally much longer than the rail bed itself, compensate
-                     structLengthMeters *= 2;
-                   }
-                 } else {
-                   // should ideally never land here since all rail/water/road tags should have an implied width
-                   structLengthMeters = 8;
-                 }
+           function get1(x, y) {
+             var n = get(x, y);
+             return n.length ? n[0] : null;
+           } // https://developer.mozilla.org/en-US/docs/Web/API/Node.normalize
 
-                 var a1 = geoAngle(edgeNodes[0], edgeNodes[1], projection) + Math.PI;
-                 var a2 = geoAngle(graph.entity(crossedEdge[0]), graph.entity(crossedEdge[1]), projection) + Math.PI;
-                 var crossingAngle = Math.max(a1, a2) - Math.min(a1, a2);
-                 if (crossingAngle > Math.PI) crossingAngle -= Math.PI; // lengthen the structure to account for the angle of the crossing
 
-                 structLengthMeters = structLengthMeters / 2 / Math.sin(crossingAngle) * 2; // add padding since the structure must extend past the edges of the crossed feature
+           function norm(el) {
+             if (el.normalize) {
+               el.normalize();
+             }
 
-                 structLengthMeters += 4; // clamp the length to a reasonable range
+             return el;
+           } // cast array x into numbers
 
-                 structLengthMeters = Math.min(Math.max(structLengthMeters, 4), 50);
 
-                 function geomToProj(geoPoint) {
-                   return [geoLonToMeters(geoPoint[0], geoPoint[1]), geoLatToMeters(geoPoint[1])];
-                 }
+           function numarray(x) {
+             for (var j = 0, o = []; j < x.length; j++) {
+               o[j] = parseFloat(x[j]);
+             }
 
-                 function projToGeom(projPoint) {
-                   var lat = geoMetersToLat(projPoint[1]);
-                   return [geoMetersToLon(projPoint[0], lat), lat];
-                 }
+             return o;
+           } // get the content of a text node, if any
 
-                 var projEdgeNode1 = geomToProj(edgeNodes[0].loc);
-                 var projEdgeNode2 = geomToProj(edgeNodes[1].loc);
-                 var projectedAngle = geoVecAngle(projEdgeNode1, projEdgeNode2);
-                 var projectedCrossingLoc = geomToProj(crossingLoc);
-                 var linearToSphericalMetersRatio = geoVecLength(projEdgeNode1, projEdgeNode2) / geoSphericalDistance(edgeNodes[0].loc, edgeNodes[1].loc);
 
-                 function locSphericalDistanceFromCrossingLoc(angle, distanceMeters) {
-                   var lengthSphericalMeters = distanceMeters * linearToSphericalMetersRatio;
-                   return projToGeom([projectedCrossingLoc[0] + Math.cos(angle) * lengthSphericalMeters, projectedCrossingLoc[1] + Math.sin(angle) * lengthSphericalMeters]);
-                 }
+           function nodeVal(x) {
+             if (x) {
+               norm(x);
+             }
 
-                 var endpointLocGetter1 = function endpointLocGetter1(lengthMeters) {
-                   return locSphericalDistanceFromCrossingLoc(projectedAngle, lengthMeters);
-                 };
+             return x && x.textContent || '';
+           } // get the contents of multiple text nodes, if present
 
-                 var endpointLocGetter2 = function endpointLocGetter2(lengthMeters) {
-                   return locSphericalDistanceFromCrossingLoc(projectedAngle + Math.PI, lengthMeters);
-                 }; // avoid creating very short edges from splitting too close to another node
 
+           function getMulti(x, ys) {
+             var o = {},
+                 n,
+                 k;
 
-                 var minEdgeLengthMeters = 0.55; // decide where to bound the structure along the way, splitting as necessary
+             for (k = 0; k < ys.length; k++) {
+               n = get1(x, ys[k]);
+               if (n) o[ys[k]] = nodeVal(n);
+             }
 
-                 function determineEndpoint(edge, endNode, locGetter) {
-                   var newNode;
-                   var idealLengthMeters = structLengthMeters / 2; // distance between the crossing location and the end of the edge,
-                   // the maximum length of this side of the structure
+             return o;
+           } // add properties of Y to X, overwriting if present in both
 
-                   var crossingToEdgeEndDistance = geoSphericalDistance(crossingLoc, endNode.loc);
 
-                   if (crossingToEdgeEndDistance - idealLengthMeters > minEdgeLengthMeters) {
-                     // the edge is long enough to insert a new node
-                     // the loc that would result in the full expected length
-                     var idealNodeLoc = locGetter(idealLengthMeters);
-                     newNode = osmNode();
-                     graph = actionAddMidpoint({
-                       loc: idealNodeLoc,
-                       edge: edge
-                     }, newNode)(graph);
-                   } else {
-                     var edgeCount = 0;
-                     endNode.parentIntersectionWays(graph).forEach(function (way) {
-                       way.nodes.forEach(function (nodeID) {
-                         if (nodeID === endNode.id) {
-                           if (endNode.id === way.first() && endNode.id !== way.last() || endNode.id === way.last() && endNode.id !== way.first()) {
-                             edgeCount += 1;
-                           } else {
-                             edgeCount += 2;
-                           }
-                         }
-                       });
-                     });
+           function extend(x, y) {
+             for (var k in y) {
+               x[k] = y[k];
+             }
+           } // get one coordinate from a coordinate array, if any
 
-                     if (edgeCount >= 3) {
-                       // the end node is a junction, try to leave a segment
-                       // between it and the structure - #7202
-                       var insetLength = crossingToEdgeEndDistance - minEdgeLengthMeters;
 
-                       if (insetLength > minEdgeLengthMeters) {
-                         var insetNodeLoc = locGetter(insetLength);
-                         newNode = osmNode();
-                         graph = actionAddMidpoint({
-                           loc: insetNodeLoc,
-                           edge: edge
-                         }, newNode)(graph);
-                       }
-                     }
-                   } // if the edge is too short to subdivide as desired, then
-                   // just bound the structure at the existing end node
+           function coord1(v) {
+             return numarray(v.replace(removeSpace, '').split(','));
+           } // get all coordinates from a coordinate array as [[],[]]
 
 
-                   if (!newNode) newNode = endNode;
-                   var splitAction = actionSplit([newNode.id]).limitWays(resultWayIDs); // only split selected or created ways
-                   // do the split
+           function coord(v) {
+             var coords = v.replace(trimSpace, '').split(splitSpace),
+                 o = [];
 
-                   graph = splitAction(graph);
+             for (var i = 0; i < coords.length; i++) {
+               o.push(coord1(coords[i]));
+             }
 
-                   if (splitAction.getCreatedWayIDs().length) {
-                     resultWayIDs.push(splitAction.getCreatedWayIDs()[0]);
-                   }
+             return o;
+           }
 
-                   return newNode;
-                 }
+           function coordPair(x) {
+             var ll = [attrf(x, 'lon'), attrf(x, 'lat')],
+                 ele = get1(x, 'ele'),
+                 // handle namespaced attribute in browser
+             heartRate = get1(x, 'gpxtpx:hr') || get1(x, 'hr'),
+                 time = get1(x, 'time'),
+                 e;
 
-                 var structEndNode1 = determineEndpoint(edge, edgeNodes[1], endpointLocGetter1);
-                 var structEndNode2 = determineEndpoint([edgeNodes[0].id, structEndNode1.id], edgeNodes[0], endpointLocGetter2);
-                 var structureWay = resultWayIDs.map(function (id) {
-                   return graph.entity(id);
-                 }).find(function (way) {
-                   return way.nodes.indexOf(structEndNode1.id) !== -1 && way.nodes.indexOf(structEndNode2.id) !== -1;
-                 });
-                 var tags = Object.assign({}, structureWay.tags); // copy tags
+             if (ele) {
+               e = parseFloat(nodeVal(ele));
 
-                 if (bridgeOrTunnel === 'bridge') {
-                   tags.bridge = 'yes';
-                   tags.layer = '1';
-                 } else {
-                   var tunnelValue = 'yes';
+               if (!isNaN(e)) {
+                 ll.push(e);
+               }
+             }
 
-                   if (getFeatureType(structureWay, graph) === 'waterway') {
-                     // use `tunnel=culvert` for waterways by default
-                     tunnelValue = 'culvert';
-                   }
+             return {
+               coordinates: ll,
+               time: time ? nodeVal(time) : null,
+               heartRate: heartRate ? parseFloat(nodeVal(heartRate)) : null
+             };
+           } // create a new feature collection parent object
 
-                   tags.tunnel = tunnelValue;
-                   tags.layer = '-1';
-                 } // apply the structure tags to the way
 
+           function fc() {
+             return {
+               type: 'FeatureCollection',
+               features: []
+             };
+           }
 
-                 graph = actionChangeTags(structureWay.id, tags)(graph);
-                 return graph;
-               };
+           var serializer;
 
-               context.perform(action, _t('issues.fix.' + fixTitleID + '.annotation'));
-               context.enter(modeSelect(context, resultWayIDs));
-             }
-           });
-         }
+           if (typeof XMLSerializer !== 'undefined') {
+             /* istanbul ignore next */
+             serializer = new XMLSerializer(); // only require xmldom in a node environment
+           } else if ((typeof process === "undefined" ? "undefined" : _typeof(process)) === 'object' && !process.browser) {
+             serializer = new domParser.XMLSerializer();
+           }
 
-         function makeConnectWaysFix(connectionTags) {
-           var fixTitleID = 'connect_features';
+           function xml2str(str) {
+             // IE9 will create a new XMLSerializer but it'll crash immediately.
+             // This line is ignored because we don't run coverage tests in IE9
 
-           if (connectionTags.ford) {
-             fixTitleID = 'connect_using_ford';
+             /* istanbul ignore next */
+             if (str.xml !== undefined) return str.xml;
+             return serializer.serializeToString(str);
            }
 
-           return new validationIssueFix({
-             icon: 'iD-icon-crossing',
-             title: _t.html('issues.fix.' + fixTitleID + '.title'),
-             onClick: function onClick(context) {
-               var loc = this.issue.loc;
-               var connectionTags = this.issue.data.connectionTags;
-               var edges = this.issue.data.edges;
-               context.perform(function actionConnectCrossingWays(graph) {
-                 // create the new node for the points
-                 var node = osmNode({
-                   loc: loc,
-                   tags: connectionTags
-                 });
-                 graph = graph.replace(node);
-                 var nodesToMerge = [node.id];
-                 var mergeThresholdInMeters = 0.75;
-                 edges.forEach(function (edge) {
-                   var edgeNodes = [graph.entity(edge[0]), graph.entity(edge[1])];
-                   var closestNodeInfo = geoSphericalClosestNode(edgeNodes, loc); // if there is already a point nearby, use that
+           var t = {
+             kml: function kml(doc) {
+               var gj = fc(),
+                   // styleindex keeps track of hashed styles in order to match features
+               styleIndex = {},
+                   styleByHash = {},
+                   // stylemapindex keeps track of style maps to expose in properties
+               styleMapIndex = {},
+                   // atomic geospatial types supported by KML - MultiGeometry is
+               // handled separately
+               geotypes = ['Polygon', 'LineString', 'Point', 'Track', 'gx:Track'],
+                   // all root placemarks in the file
+               placemarks = get(doc, 'Placemark'),
+                   styles = get(doc, 'Style'),
+                   styleMaps = get(doc, 'StyleMap');
 
-                   if (closestNodeInfo.distance < mergeThresholdInMeters) {
-                     nodesToMerge.push(closestNodeInfo.node.id); // else add the new node to the way
-                   } else {
-                     graph = actionAddMidpoint({
-                       loc: loc,
-                       edge: edge
-                     }, node)(graph);
-                   }
-                 });
+               for (var k = 0; k < styles.length; k++) {
+                 var hash = okhash(xml2str(styles[k])).toString(16);
+                 styleIndex['#' + attr(styles[k], 'id')] = hash;
+                 styleByHash[hash] = styles[k];
+               }
 
-                 if (nodesToMerge.length > 1) {
-                   // if we're using nearby nodes, merge them with the new node
-                   graph = actionMergeNodes(nodesToMerge, loc)(graph);
+               for (var l = 0; l < styleMaps.length; l++) {
+                 styleIndex['#' + attr(styleMaps[l], 'id')] = okhash(xml2str(styleMaps[l])).toString(16);
+                 var pairs = get(styleMaps[l], 'Pair');
+                 var pairsMap = {};
+
+                 for (var m = 0; m < pairs.length; m++) {
+                   pairsMap[nodeVal(get1(pairs[m], 'key'))] = nodeVal(get1(pairs[m], 'styleUrl'));
                  }
 
-                 return graph;
-               }, _t('issues.fix.connect_crossing_features.annotation'));
-             }
-           });
-         }
+                 styleMapIndex['#' + attr(styleMaps[l], 'id')] = pairsMap;
+               }
 
-         function makeChangeLayerFix(higherOrLower) {
-           return new validationIssueFix({
-             icon: 'iD-icon-' + (higherOrLower === 'higher' ? 'up' : 'down'),
-             title: _t.html('issues.fix.tag_this_as_' + higherOrLower + '.title'),
-             onClick: function onClick(context) {
-               var mode = context.mode();
-               if (!mode || mode.id !== 'select') return;
-               var selectedIDs = mode.selectedIDs();
-               if (selectedIDs.length !== 1) return;
-               var selectedID = selectedIDs[0];
-               if (!this.issue.entityIds.some(function (entityId) {
-                 return entityId === selectedID;
-               })) return;
-               var entity = context.hasEntity(selectedID);
-               if (!entity) return;
-               var tags = Object.assign({}, entity.tags); // shallow copy
+               for (var j = 0; j < placemarks.length; j++) {
+                 gj.features = gj.features.concat(getPlacemark(placemarks[j]));
+               }
 
-               var layer = tags.layer && Number(tags.layer);
+               function kmlColor(v) {
+                 var color, opacity;
+                 v = v || '';
 
-               if (layer && !isNaN(layer)) {
-                 if (higherOrLower === 'higher') {
-                   layer += 1;
-                 } else {
-                   layer -= 1;
+                 if (v.substr(0, 1) === '#') {
+                   v = v.substr(1);
                  }
-               } else {
-                 if (higherOrLower === 'higher') {
-                   layer = 1;
-                 } else {
-                   layer = -1;
+
+                 if (v.length === 6 || v.length === 3) {
+                   color = v;
                  }
-               }
 
-               tags.layer = layer.toString();
-               context.perform(actionChangeTags(entity.id, tags), _t('operations.change_tags.annotation'));
-             }
-           });
-         }
+                 if (v.length === 8) {
+                   opacity = parseInt(v.substr(0, 2), 16) / 255;
+                   color = '#' + v.substr(6, 2) + v.substr(4, 2) + v.substr(2, 2);
+                 }
 
-         validation.type = type;
-         return validation;
-       }
+                 return [color, isNaN(opacity) ? undefined : opacity];
+               }
 
-       function validationDisconnectedWay() {
-         var type = 'disconnected_way';
+               function gxCoord(v) {
+                 return numarray(v.split(' '));
+               }
 
-         function isTaggedAsHighway(entity) {
-           return osmRoutableHighwayTagValues[entity.tags.highway];
-         }
+               function gxCoords(root) {
+                 var elems = get(root, 'coord'),
+                     coords = [],
+                     times = [];
+                 if (elems.length === 0) elems = get(root, 'gx:coord');
 
-         var validation = function checkDisconnectedWay(entity, graph) {
-           var routingIslandWays = routingIslandForEntity(entity);
-           if (!routingIslandWays) return [];
-           return [new validationIssue({
-             type: type,
-             subtype: 'highway',
-             severity: 'warning',
-             message: function message(context) {
-               var entity = this.entityIds.length && context.hasEntity(this.entityIds[0]);
-               var label = entity && utilDisplayLabel(entity, context.graph());
-               return _t.html('issues.disconnected_way.routable.message', {
-                 count: this.entityIds.length,
-                 highway: label
-               });
-             },
-             reference: showReference,
-             entityIds: Array.from(routingIslandWays).map(function (way) {
-               return way.id;
-             }),
-             dynamicFixes: makeFixes
-           })];
+                 for (var i = 0; i < elems.length; i++) {
+                   coords.push(gxCoord(nodeVal(elems[i])));
+                 }
 
-           function makeFixes(context) {
-             var fixes = [];
-             var singleEntity = this.entityIds.length === 1 && context.hasEntity(this.entityIds[0]);
+                 var timeElems = get(root, 'when');
+
+                 for (var j = 0; j < timeElems.length; j++) {
+                   times.push(nodeVal(timeElems[j]));
+                 }
 
-             if (singleEntity) {
-               if (singleEntity.type === 'way' && !singleEntity.isClosed()) {
-                 var textDirection = _mainLocalizer.textDirection();
-                 var startFix = makeContinueDrawingFixIfAllowed(textDirection, singleEntity.first(), 'start');
-                 if (startFix) fixes.push(startFix);
-                 var endFix = makeContinueDrawingFixIfAllowed(textDirection, singleEntity.last(), 'end');
-                 if (endFix) fixes.push(endFix);
+                 return {
+                   coords: coords,
+                   times: times
+                 };
                }
 
-               if (!fixes.length) {
-                 fixes.push(new validationIssueFix({
-                   title: _t.html('issues.fix.connect_feature.title')
-                 }));
-               }
+               function getGeometry(root) {
+                 var geomNode,
+                     geomNodes,
+                     i,
+                     j,
+                     k,
+                     geoms = [],
+                     coordTimes = [];
 
-               fixes.push(new validationIssueFix({
-                 icon: 'iD-operation-delete',
-                 title: _t.html('issues.fix.delete_feature.title'),
-                 entityIds: [singleEntity.id],
-                 onClick: function onClick(context) {
-                   var id = this.issue.entityIds[0];
-                   var operation = operationDelete(context, [id]);
+                 if (get1(root, 'MultiGeometry')) {
+                   return getGeometry(get1(root, 'MultiGeometry'));
+                 }
 
-                   if (!operation.disabled()) {
-                     operation();
-                   }
+                 if (get1(root, 'MultiTrack')) {
+                   return getGeometry(get1(root, 'MultiTrack'));
                  }
-               }));
-             } else {
-               fixes.push(new validationIssueFix({
-                 title: _t.html('issues.fix.connect_features.title')
-               }));
-             }
 
-             return fixes;
-           }
+                 if (get1(root, 'gx:MultiTrack')) {
+                   return getGeometry(get1(root, 'gx:MultiTrack'));
+                 }
 
-           function showReference(selection) {
-             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.disconnected_way.routable.reference'));
-           }
+                 for (i = 0; i < geotypes.length; i++) {
+                   geomNodes = get(root, geotypes[i]);
 
-           function routingIslandForEntity(entity) {
-             var routingIsland = new Set(); // the interconnected routable features
+                   if (geomNodes) {
+                     for (j = 0; j < geomNodes.length; j++) {
+                       geomNode = geomNodes[j];
 
-             var waysToCheck = []; // the queue of remaining routable ways to traverse
+                       if (geotypes[i] === 'Point') {
+                         geoms.push({
+                           type: 'Point',
+                           coordinates: coord1(nodeVal(get1(geomNode, 'coordinates')))
+                         });
+                       } else if (geotypes[i] === 'LineString') {
+                         geoms.push({
+                           type: 'LineString',
+                           coordinates: coord(nodeVal(get1(geomNode, 'coordinates')))
+                         });
+                       } else if (geotypes[i] === 'Polygon') {
+                         var rings = get(geomNode, 'LinearRing'),
+                             coords = [];
 
-             function queueParentWays(node) {
-               graph.parentWays(node).forEach(function (parentWay) {
-                 if (!routingIsland.has(parentWay) && // only check each feature once
-                 isRoutableWay(parentWay, false)) {
-                   // only check routable features
-                   routingIsland.add(parentWay);
-                   waysToCheck.push(parentWay);
+                         for (k = 0; k < rings.length; k++) {
+                           coords.push(coord(nodeVal(get1(rings[k], 'coordinates'))));
+                         }
+
+                         geoms.push({
+                           type: 'Polygon',
+                           coordinates: coords
+                         });
+                       } else if (geotypes[i] === 'Track' || geotypes[i] === 'gx:Track') {
+                         var track = gxCoords(geomNode);
+                         geoms.push({
+                           type: 'LineString',
+                           coordinates: track.coords
+                         });
+                         if (track.times.length) coordTimes.push(track.times);
+                       }
+                     }
+                   }
                  }
-               });
-             }
 
-             if (entity.type === 'way' && isRoutableWay(entity, true)) {
-               routingIsland.add(entity);
-               waysToCheck.push(entity);
-             } else if (entity.type === 'node' && isRoutableNode(entity)) {
-               routingIsland.add(entity);
-               queueParentWays(entity);
-             } else {
-               // this feature isn't routable, cannot be a routing island
-               return null;
-             }
+                 return {
+                   geoms: geoms,
+                   coordTimes: coordTimes
+                 };
+               }
 
-             while (waysToCheck.length) {
-               var wayToCheck = waysToCheck.pop();
-               var childNodes = graph.childNodes(wayToCheck);
+               function getPlacemark(root) {
+                 var geomsAndTimes = getGeometry(root),
+                     i,
+                     properties = {},
+                     name = nodeVal(get1(root, 'name')),
+                     address = nodeVal(get1(root, 'address')),
+                     styleUrl = nodeVal(get1(root, 'styleUrl')),
+                     description = nodeVal(get1(root, 'description')),
+                     timeSpan = get1(root, 'TimeSpan'),
+                     timeStamp = get1(root, 'TimeStamp'),
+                     extendedData = get1(root, 'ExtendedData'),
+                     lineStyle = get1(root, 'LineStyle'),
+                     polyStyle = get1(root, 'PolyStyle'),
+                     visibility = get1(root, 'visibility');
+                 if (!geomsAndTimes.geoms.length) return [];
+                 if (name) properties.name = name;
+                 if (address) properties.address = address;
 
-               for (var i in childNodes) {
-                 var vertex = childNodes[i];
+                 if (styleUrl) {
+                   if (styleUrl[0] !== '#') {
+                     styleUrl = '#' + styleUrl;
+                   }
 
-                 if (isConnectedVertex(vertex)) {
-                   // found a link to the wider network, not a routing island
-                   return null;
-                 }
+                   properties.styleUrl = styleUrl;
 
-                 if (isRoutableNode(vertex)) {
-                   routingIsland.add(vertex);
+                   if (styleIndex[styleUrl]) {
+                     properties.styleHash = styleIndex[styleUrl];
+                   }
+
+                   if (styleMapIndex[styleUrl]) {
+                     properties.styleMapHash = styleMapIndex[styleUrl];
+                     properties.styleHash = styleIndex[styleMapIndex[styleUrl].normal];
+                   } // Try to populate the lineStyle or polyStyle since we got the style hash
+
+
+                   var style = styleByHash[properties.styleHash];
+
+                   if (style) {
+                     if (!lineStyle) lineStyle = get1(style, 'LineStyle');
+                     if (!polyStyle) polyStyle = get1(style, 'PolyStyle');
+                   }
                  }
 
-                 queueParentWays(vertex);
-               }
-             } // no network link found, this is a routing island, return its members
+                 if (description) properties.description = description;
 
+                 if (timeSpan) {
+                   var begin = nodeVal(get1(timeSpan, 'begin'));
+                   var end = nodeVal(get1(timeSpan, 'end'));
+                   properties.timespan = {
+                     begin: begin,
+                     end: end
+                   };
+                 }
 
-             return routingIsland;
-           }
+                 if (timeStamp) {
+                   properties.timestamp = nodeVal(get1(timeStamp, 'when'));
+                 }
 
-           function isConnectedVertex(vertex) {
-             // assume ways overlapping unloaded tiles are connected to the wider road network  - #5938
-             var osm = services.osm;
-             if (osm && !osm.isDataLoaded(vertex.loc)) return true; // entrances are considered connected
+                 if (lineStyle) {
+                   var linestyles = kmlColor(nodeVal(get1(lineStyle, 'color'))),
+                       color = linestyles[0],
+                       opacity = linestyles[1],
+                       width = parseFloat(nodeVal(get1(lineStyle, 'width')));
+                   if (color) properties.stroke = color;
+                   if (!isNaN(opacity)) properties['stroke-opacity'] = opacity;
+                   if (!isNaN(width)) properties['stroke-width'] = width;
+                 }
 
-             if (vertex.tags.entrance && vertex.tags.entrance !== 'no') return true;
-             if (vertex.tags.amenity === 'parking_entrance') return true;
-             return false;
-           }
+                 if (polyStyle) {
+                   var polystyles = kmlColor(nodeVal(get1(polyStyle, 'color'))),
+                       pcolor = polystyles[0],
+                       popacity = polystyles[1],
+                       fill = nodeVal(get1(polyStyle, 'fill')),
+                       outline = nodeVal(get1(polyStyle, 'outline'));
+                   if (pcolor) properties.fill = pcolor;
+                   if (!isNaN(popacity)) properties['fill-opacity'] = popacity;
+                   if (fill) properties['fill-opacity'] = fill === '1' ? properties['fill-opacity'] || 1 : 0;
+                   if (outline) properties['stroke-opacity'] = outline === '1' ? properties['stroke-opacity'] || 1 : 0;
+                 }
 
-           function isRoutableNode(node) {
-             // treat elevators as distinct features in the highway network
-             if (node.tags.highway === 'elevator') return true;
-             return false;
-           }
+                 if (extendedData) {
+                   var datas = get(extendedData, 'Data'),
+                       simpleDatas = get(extendedData, 'SimpleData');
 
-           function isRoutableWay(way, ignoreInnerWays) {
-             if (isTaggedAsHighway(way) || way.tags.route === 'ferry') return true;
-             return graph.parentRelations(way).some(function (parentRelation) {
-               if (parentRelation.tags.type === 'route' && parentRelation.tags.route === 'ferry') return true;
-               if (parentRelation.isMultipolygon() && isTaggedAsHighway(parentRelation) && (!ignoreInnerWays || parentRelation.memberById(way.id).role !== 'inner')) return true;
-               return false;
-             });
-           }
+                   for (i = 0; i < datas.length; i++) {
+                     properties[datas[i].getAttribute('name')] = nodeVal(get1(datas[i], 'value'));
+                   }
 
-           function makeContinueDrawingFixIfAllowed(textDirection, vertexID, whichEnd) {
-             var vertex = graph.hasEntity(vertexID);
-             if (!vertex || vertex.tags.noexit === 'yes') return null;
-             var useLeftContinue = whichEnd === 'start' && textDirection === 'ltr' || whichEnd === 'end' && textDirection === 'rtl';
-             return new validationIssueFix({
-               icon: 'iD-operation-continue' + (useLeftContinue ? '-left' : ''),
-               title: _t.html('issues.fix.continue_from_' + whichEnd + '.title'),
-               entityIds: [vertexID],
-               onClick: function onClick(context) {
-                 var wayId = this.issue.entityIds[0];
-                 var way = context.hasEntity(wayId);
-                 var vertexId = this.entityIds[0];
-                 var vertex = context.hasEntity(vertexId);
-                 if (!way || !vertex) return; // make sure the vertex is actually visible and editable
+                   for (i = 0; i < simpleDatas.length; i++) {
+                     properties[simpleDatas[i].getAttribute('name')] = nodeVal(simpleDatas[i]);
+                   }
+                 }
 
-                 var map = context.map();
+                 if (visibility) {
+                   properties.visibility = nodeVal(visibility);
+                 }
 
-                 if (!context.editable() || !map.trimmedExtent().contains(vertex.loc)) {
-                   map.zoomToEase(vertex);
+                 if (geomsAndTimes.coordTimes.length) {
+                   properties.coordTimes = geomsAndTimes.coordTimes.length === 1 ? geomsAndTimes.coordTimes[0] : geomsAndTimes.coordTimes;
                  }
 
-                 context.enter(modeDrawLine(context, wayId, context.graph(), 'line', way.affix(vertexId), true));
+                 var feature = {
+                   type: 'Feature',
+                   geometry: geomsAndTimes.geoms.length === 1 ? geomsAndTimes.geoms[0] : {
+                     type: 'GeometryCollection',
+                     geometries: geomsAndTimes.geoms
+                   },
+                   properties: properties
+                 };
+                 if (attr(root, 'id')) feature.id = attr(root, 'id');
+                 return [feature];
                }
-             });
-           }
-         };
 
-         validation.type = type;
-         return validation;
-       }
+               return gj;
+             },
+             gpx: function gpx(doc) {
+               var i,
+                   tracks = get(doc, 'trk'),
+                   routes = get(doc, 'rte'),
+                   waypoints = get(doc, 'wpt'),
+                   // a feature collection
+               gj = fc(),
+                   feature;
 
-       function validationFormatting() {
-         var type = 'invalid_format';
+               for (i = 0; i < tracks.length; i++) {
+                 feature = getTrack(tracks[i]);
+                 if (feature) gj.features.push(feature);
+               }
 
-         var validation = function validation(entity) {
-           var issues = [];
+               for (i = 0; i < routes.length; i++) {
+                 feature = getRoute(routes[i]);
+                 if (feature) gj.features.push(feature);
+               }
 
-           function isValidEmail(email) {
-             // Emails in OSM are going to be official so they should be pretty simple
-             // Using negated lists to better support all possible unicode characters (#6494)
-             var valid_email = /^[^\(\)\\,":;<>@\[\]]+@[^\(\)\\,":;<>@\[\]\.]+(?:\.[a-z0-9-]+)*$/i; // An empty value is also acceptable
+               for (i = 0; i < waypoints.length; i++) {
+                 gj.features.push(getPoint(waypoints[i]));
+               }
 
-             return !email || valid_email.test(email);
-           }
-           /*
-           function isSchemePresent(url) {
-               var valid_scheme = /^https?:\/\//i;
-               return (!url || valid_scheme.test(url));
-           }
-           */
+               function getPoints(node, pointname) {
+                 var pts = get(node, pointname),
+                     line = [],
+                     times = [],
+                     heartRates = [],
+                     l = pts.length;
+                 if (l < 2) return {}; // Invalid line in GeoJSON
 
+                 for (var i = 0; i < l; i++) {
+                   var c = coordPair(pts[i]);
+                   line.push(c.coordinates);
+                   if (c.time) times.push(c.time);
+                   if (c.heartRate) heartRates.push(c.heartRate);
+                 }
 
-           function showReferenceEmail(selection) {
-             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.invalid_format.email.reference'));
-           }
-           /*
-           function showReferenceWebsite(selection) {
-               selection.selectAll('.issue-reference')
-                   .data([0])
-                   .enter()
-                   .append('div')
-                   .attr('class', 'issue-reference')
-                   .html(t.html('issues.invalid_format.website.reference'));
-           }
-            if (entity.tags.website) {
-               // Multiple websites are possible
-               // If ever we support ES6, arrow functions make this nicer
-               var websites = entity.tags.website
-                   .split(';')
-                   .map(function(s) { return s.trim(); })
-                   .filter(function(x) { return !isSchemePresent(x); });
-                if (websites.length) {
-                   issues.push(new validationIssue({
-                       type: type,
-                       subtype: 'website',
-                       severity: 'warning',
-                       message: function(context) {
-                           var entity = context.hasEntity(this.entityIds[0]);
-                           return entity ? t.html('issues.invalid_format.website.message' + this.data,
-                               { feature: utilDisplayLabel(entity, context.graph()), site: websites.join(', ') }) : '';
-                       },
-                       reference: showReferenceWebsite,
-                       entityIds: [entity.id],
-                       hash: websites.join(),
-                       data: (websites.length > 1) ? '_multi' : ''
-                   }));
+                 return {
+                   line: line,
+                   times: times,
+                   heartRates: heartRates
+                 };
                }
-           }
-           */
 
+               function getTrack(node) {
+                 var segments = get(node, 'trkseg'),
+                     track = [],
+                     times = [],
+                     heartRates = [],
+                     line;
 
-           if (entity.tags.email) {
-             // Multiple emails are possible
-             var emails = entity.tags.email.split(';').map(function (s) {
-               return s.trim();
-             }).filter(function (x) {
-               return !isValidEmail(x);
-             });
+                 for (var i = 0; i < segments.length; i++) {
+                   line = getPoints(segments[i], 'trkpt');
 
-             if (emails.length) {
-               issues.push(new validationIssue({
-                 type: type,
-                 subtype: 'email',
-                 severity: 'warning',
-                 message: function message(context) {
-                   var entity = context.hasEntity(this.entityIds[0]);
-                   return entity ? _t.html('issues.invalid_format.email.message' + this.data, {
-                     feature: utilDisplayLabel(entity, context.graph()),
-                     email: emails.join(', ')
-                   }) : '';
-                 },
-                 reference: showReferenceEmail,
-                 entityIds: [entity.id],
-                 hash: emails.join(),
-                 data: emails.length > 1 ? '_multi' : ''
-               }));
-             }
-           }
+                   if (line) {
+                     if (line.line) track.push(line.line);
+                     if (line.times && line.times.length) times.push(line.times);
+                     if (line.heartRates && line.heartRates.length) heartRates.push(line.heartRates);
+                   }
+                 }
 
-           return issues;
-         };
+                 if (track.length === 0) return;
+                 var properties = getProperties(node);
+                 extend(properties, getLineStyle(get1(node, 'extensions')));
+                 if (times.length) properties.coordTimes = track.length === 1 ? times[0] : times;
+                 if (heartRates.length) properties.heartRates = track.length === 1 ? heartRates[0] : heartRates;
+                 return {
+                   type: 'Feature',
+                   properties: properties,
+                   geometry: {
+                     type: track.length === 1 ? 'LineString' : 'MultiLineString',
+                     coordinates: track.length === 1 ? track[0] : track
+                   }
+                 };
+               }
 
-         validation.type = type;
-         return validation;
-       }
+               function getRoute(node) {
+                 var line = getPoints(node, 'rtept');
+                 if (!line.line) return;
+                 var prop = getProperties(node);
+                 extend(prop, getLineStyle(get1(node, 'extensions')));
+                 var routeObj = {
+                   type: 'Feature',
+                   properties: prop,
+                   geometry: {
+                     type: 'LineString',
+                     coordinates: line.line
+                   }
+                 };
+                 return routeObj;
+               }
 
-       function validationHelpRequest(context) {
-         var type = 'help_request';
+               function getPoint(node) {
+                 var prop = getProperties(node);
+                 extend(prop, getMulti(node, ['sym']));
+                 return {
+                   type: 'Feature',
+                   properties: prop,
+                   geometry: {
+                     type: 'Point',
+                     coordinates: coordPair(node).coordinates
+                   }
+                 };
+               }
 
-         var validation = function checkFixmeTag(entity) {
-           if (!entity.tags.fixme) return []; // don't flag fixmes on features added by the user
+               function getLineStyle(extensions) {
+                 var style = {};
 
-           if (entity.version === undefined) return [];
+                 if (extensions) {
+                   var lineStyle = get1(extensions, 'line');
 
-           if (entity.v !== undefined) {
-             var baseEntity = context.history().base().hasEntity(entity.id); // don't flag fixmes added by the user on existing features
+                   if (lineStyle) {
+                     var color = nodeVal(get1(lineStyle, 'color')),
+                         opacity = parseFloat(nodeVal(get1(lineStyle, 'opacity'))),
+                         width = parseFloat(nodeVal(get1(lineStyle, 'width')));
+                     if (color) style.stroke = color;
+                     if (!isNaN(opacity)) style['stroke-opacity'] = opacity; // GPX width is in mm, convert to px with 96 px per inch
 
-             if (!baseEntity || !baseEntity.tags.fixme) return [];
-           }
+                     if (!isNaN(width)) style['stroke-width'] = width * 96 / 25.4;
+                   }
+                 }
 
-           return [new validationIssue({
-             type: type,
-             subtype: 'fixme_tag',
-             severity: 'warning',
-             message: function message(context) {
-               var entity = context.hasEntity(this.entityIds[0]);
-               return entity ? _t.html('issues.fixme_tag.message', {
-                 feature: utilDisplayLabel(entity, context.graph())
-               }) : '';
-             },
-             dynamicFixes: function dynamicFixes() {
-               return [new validationIssueFix({
-                 title: _t.html('issues.fix.address_the_concern.title')
-               })];
-             },
-             reference: showReference,
-             entityIds: [entity.id]
-           })];
+                 return style;
+               }
 
-           function showReference(selection) {
-             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.fixme_tag.reference'));
-           }
-         };
+               function getProperties(node) {
+                 var prop = getMulti(node, ['name', 'cmt', 'desc', 'type', 'time', 'keywords']),
+                     links = get(node, 'link');
+                 if (links.length) prop.links = [];
 
-         validation.type = type;
-         return validation;
-       }
+                 for (var i = 0, link; i < links.length; i++) {
+                   link = {
+                     href: attr(links[i], 'href')
+                   };
+                   extend(link, getMulti(links[i], ['text', 'type']));
+                   prop.links.push(link);
+                 }
 
-       function validationImpossibleOneway() {
-         var type = 'impossible_oneway';
+                 return prop;
+               }
 
-         var validation = function checkImpossibleOneway(entity, graph) {
-           if (entity.type !== 'way' || entity.geometry(graph) !== 'line') return [];
-           if (entity.isClosed()) return [];
-           if (!typeForWay(entity)) return [];
-           if (!isOneway(entity)) return [];
-           var firstIssues = issuesForNode(entity, entity.first());
-           var lastIssues = issuesForNode(entity, entity.last());
-           return firstIssues.concat(lastIssues);
+               return gj;
+             }
+           };
+           return t;
+         }();
 
-           function typeForWay(way) {
-             if (way.geometry(graph) !== 'line') return null;
-             if (osmRoutableHighwayTagValues[way.tags.highway]) return 'highway';
-             if (osmFlowingWaterwayTagValues[way.tags.waterway]) return 'waterway';
-             return null;
-           }
+         module.exports = toGeoJSON;
+       });
 
-           function isOneway(way) {
-             if (way.tags.oneway === 'yes') return true;
-             if (way.tags.oneway) return false;
+       var _initialized = false;
+       var _enabled = false;
 
-             for (var key in way.tags) {
-               if (osmOneWayTags[key] && osmOneWayTags[key][way.tags[key]]) {
-                 return true;
-               }
-             }
+       var _geojson;
 
-             return false;
-           }
+       function svgData(projection, context, dispatch) {
+         var throttledRedraw = throttle(function () {
+           dispatch.call('change');
+         }, 1000);
 
-           function nodeOccursMoreThanOnce(way, nodeID) {
-             var occurrences = 0;
+         var _showLabels = true;
+         var detected = utilDetect();
+         var layer = select(null);
 
-             for (var index in way.nodes) {
-               if (way.nodes[index] === nodeID) {
-                 occurrences += 1;
-                 if (occurrences > 1) return true;
-               }
-             }
+         var _vtService;
 
-             return false;
-           }
+         var _fileList;
 
-           function isConnectedViaOtherTypes(way, node) {
-             var wayType = typeForWay(way);
+         var _template;
 
-             if (wayType === 'highway') {
-               // entrances are considered connected
-               if (node.tags.entrance && node.tags.entrance !== 'no') return true;
-               if (node.tags.amenity === 'parking_entrance') return true;
-             } else if (wayType === 'waterway') {
-               if (node.id === way.first()) {
-                 // multiple waterways may start at the same spring
-                 if (node.tags.natural === 'spring') return true;
-               } else {
-                 // multiple waterways may end at the same drain
-                 if (node.tags.manhole === 'drain') return true;
-               }
-             }
+         var _src;
 
-             return graph.parentWays(node).some(function (parentWay) {
-               if (parentWay.id === way.id) return false;
+         function init() {
+           if (_initialized) return; // run once
 
-               if (wayType === 'highway') {
-                 // allow connections to highway areas
-                 if (parentWay.geometry(graph) === 'area' && osmRoutableHighwayTagValues[parentWay.tags.highway]) return true; // count connections to ferry routes as connected
+           _geojson = {};
+           _enabled = true;
 
-                 if (parentWay.tags.route === 'ferry') return true;
-                 return graph.parentRelations(parentWay).some(function (parentRelation) {
-                   if (parentRelation.tags.type === 'route' && parentRelation.tags.route === 'ferry') return true; // allow connections to highway multipolygons
+           function over(d3_event) {
+             d3_event.stopPropagation();
+             d3_event.preventDefault();
+             d3_event.dataTransfer.dropEffect = 'copy';
+           }
 
-                   return parentRelation.isMultipolygon() && osmRoutableHighwayTagValues[parentRelation.tags.highway];
-                 });
-               } else if (wayType === 'waterway') {
-                 // multiple waterways may start or end at a water body at the same node
-                 if (parentWay.tags.natural === 'water' || parentWay.tags.natural === 'coastline') return true;
-               }
+           context.container().attr('dropzone', 'copy').on('drop.svgData', function (d3_event) {
+             d3_event.stopPropagation();
+             d3_event.preventDefault();
+             if (!detected.filedrop) return;
+             drawData.fileList(d3_event.dataTransfer.files);
+           }).on('dragenter.svgData', over).on('dragexit.svgData', over).on('dragover.svgData', over);
+           _initialized = true;
+         }
 
-               return false;
-             });
+         function getService() {
+           if (services.vectorTile && !_vtService) {
+             _vtService = services.vectorTile;
+
+             _vtService.event.on('loadedData', throttledRedraw);
+           } else if (!services.vectorTile && _vtService) {
+             _vtService = null;
            }
 
-           function issuesForNode(way, nodeID) {
-             var isFirst = nodeID === way.first();
-             var wayType = typeForWay(way); // ignore if this way is self-connected at this node
+           return _vtService;
+         }
 
-             if (nodeOccursMoreThanOnce(way, nodeID)) return [];
-             var osm = services.osm;
-             if (!osm) return [];
-             var node = graph.hasEntity(nodeID); // ignore if this node or its tile are unloaded
+         function showLayer() {
+           layerOn();
+           layer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end', function () {
+             dispatch.call('change');
+           });
+         }
 
-             if (!node || !osm.isDataLoaded(node.loc)) return [];
-             if (isConnectedViaOtherTypes(way, node)) return [];
-             var attachedWaysOfSameType = graph.parentWays(node).filter(function (parentWay) {
-               if (parentWay.id === way.id) return false;
-               return typeForWay(parentWay) === wayType;
-             }); // assume it's okay for waterways to start or end disconnected for now
+         function hideLayer() {
+           throttledRedraw.cancel();
+           layer.transition().duration(250).style('opacity', 0).on('end', layerOff);
+         }
 
-             if (wayType === 'waterway' && attachedWaysOfSameType.length === 0) return [];
-             var attachedOneways = attachedWaysOfSameType.filter(function (attachedWay) {
-               return isOneway(attachedWay);
-             }); // ignore if the way is connected to some non-oneway features
+         function layerOn() {
+           layer.style('display', 'block');
+         }
 
-             if (attachedOneways.length < attachedWaysOfSameType.length) return [];
+         function layerOff() {
+           layer.selectAll('.viewfield-group').remove();
+           layer.style('display', 'none');
+         } // ensure that all geojson features in a collection have IDs
 
-             if (attachedOneways.length) {
-               var connectedEndpointsOkay = attachedOneways.some(function (attachedOneway) {
-                 if ((isFirst ? attachedOneway.first() : attachedOneway.last()) !== nodeID) return true;
-                 if (nodeOccursMoreThanOnce(attachedOneway, nodeID)) return true;
-                 return false;
-               });
-               if (connectedEndpointsOkay) return [];
-             }
 
-             var placement = isFirst ? 'start' : 'end',
-                 messageID = wayType + '.',
-                 referenceID = wayType + '.';
+         function ensureIDs(gj) {
+           if (!gj) return null;
 
-             if (wayType === 'waterway') {
-               messageID += 'connected.' + placement;
-               referenceID += 'connected';
-             } else {
-               messageID += placement;
-               referenceID += placement;
+           if (gj.type === 'FeatureCollection') {
+             for (var i = 0; i < gj.features.length; i++) {
+               ensureFeatureID(gj.features[i]);
              }
+           } else {
+             ensureFeatureID(gj);
+           }
 
-             return [new validationIssue({
-               type: type,
-               subtype: wayType,
-               severity: 'warning',
-               message: function message(context) {
-                 var entity = context.hasEntity(this.entityIds[0]);
-                 return entity ? _t.html('issues.impossible_oneway.' + messageID + '.message', {
-                   feature: utilDisplayLabel(entity, context.graph())
-                 }) : '';
-               },
-               reference: getReference(referenceID),
-               entityIds: [way.id, node.id],
-               dynamicFixes: function dynamicFixes() {
-                 var fixes = [];
+           return gj;
+         } // ensure that each single Feature object has a unique ID
 
-                 if (attachedOneways.length) {
-                   fixes.push(new validationIssueFix({
-                     icon: 'iD-operation-reverse',
-                     title: _t.html('issues.fix.reverse_feature.title'),
-                     entityIds: [way.id],
-                     onClick: function onClick(context) {
-                       var id = this.issue.entityIds[0];
-                       context.perform(actionReverse(id), _t('operations.reverse.annotation.line', {
-                         n: 1
-                       }));
-                     }
-                   }));
-                 }
 
-                 if (node.tags.noexit !== 'yes') {
-                   var textDirection = _mainLocalizer.textDirection();
-                   var useLeftContinue = isFirst && textDirection === 'ltr' || !isFirst && textDirection === 'rtl';
-                   fixes.push(new validationIssueFix({
-                     icon: 'iD-operation-continue' + (useLeftContinue ? '-left' : ''),
-                     title: _t.html('issues.fix.continue_from_' + (isFirst ? 'start' : 'end') + '.title'),
-                     onClick: function onClick(context) {
-                       var entityID = this.issue.entityIds[0];
-                       var vertexID = this.issue.entityIds[1];
-                       var way = context.entity(entityID);
-                       var vertex = context.entity(vertexID);
-                       continueDrawing(way, vertex, context);
-                     }
-                   }));
-                 }
+         function ensureFeatureID(feature) {
+           if (!feature) return;
+           feature.__featurehash__ = utilHashcode(fastJsonStableStringify(feature));
+           return feature;
+         } // Prefer an array of Features instead of a FeatureCollection
 
-                 return fixes;
-               },
-               loc: node.loc
-             })];
 
-             function getReference(referenceID) {
-               return function showReference(selection) {
-                 selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.impossible_oneway.' + referenceID + '.reference'));
-               };
-             }
+         function getFeatures(gj) {
+           if (!gj) return [];
+
+           if (gj.type === 'FeatureCollection') {
+             return gj.features;
+           } else {
+             return [gj];
            }
-         };
+         }
 
-         function continueDrawing(way, vertex, context) {
-           // make sure the vertex is actually visible and editable
-           var map = context.map();
+         function featureKey(d) {
+           return d.__featurehash__;
+         }
 
-           if (!context.editable() || !map.trimmedExtent().contains(vertex.loc)) {
-             map.zoomToEase(vertex);
-           }
+         function isPolygon(d) {
+           return d.geometry.type === 'Polygon' || d.geometry.type === 'MultiPolygon';
+         }
 
-           context.enter(modeDrawLine(context, way.id, context.graph(), 'line', way.affix(vertex.id), true));
+         function clipPathID(d) {
+           return 'ideditor-data-' + d.__featurehash__ + '-clippath';
          }
 
-         validation.type = type;
-         return validation;
-       }
+         function featureClasses(d) {
+           return ['data' + d.__featurehash__, d.geometry.type, isPolygon(d) ? 'area' : '', d.__layerID__ || ''].filter(Boolean).join(' ');
+         }
 
-       function validationIncompatibleSource() {
-         var type = 'incompatible_source';
-         var invalidSources = [{
-           id: 'google',
-           regex: 'google',
-           exceptRegex: 'books.google|Google Books|drive.google|googledrive|Google Drive'
-         }];
+         function drawData(selection) {
+           var vtService = getService();
+           var getPath = svgPath(projection).geojson;
+           var getAreaPath = svgPath(projection, null, true).geojson;
+           var hasData = drawData.hasData();
+           layer = selection.selectAll('.layer-mapdata').data(_enabled && hasData ? [0] : []);
+           layer.exit().remove();
+           layer = layer.enter().append('g').attr('class', 'layer-mapdata').merge(layer);
+           var surface = context.surface();
+           if (!surface || surface.empty()) return; // not ready to draw yet, starting up
+           // Gather data
 
-         var validation = function checkIncompatibleSource(entity) {
-           var entitySources = entity.tags && entity.tags.source && entity.tags.source.split(';');
-           if (!entitySources) return [];
-           var issues = [];
-           invalidSources.forEach(function (invalidSource) {
-             var hasInvalidSource = entitySources.some(function (source) {
-               if (!source.match(new RegExp(invalidSource.regex, 'i'))) return false;
-               if (invalidSource.exceptRegex && source.match(new RegExp(invalidSource.exceptRegex, 'i'))) return false;
-               return true;
-             });
-             if (!hasInvalidSource) return;
-             issues.push(new validationIssue({
-               type: type,
-               severity: 'warning',
-               message: function message(context) {
-                 var entity = context.hasEntity(this.entityIds[0]);
-                 return entity ? _t.html('issues.incompatible_source.' + invalidSource.id + '.feature.message', {
-                   feature: utilDisplayLabel(entity, context.graph())
-                 }) : '';
-               },
-               reference: getReference(invalidSource.id),
-               entityIds: [entity.id],
-               dynamicFixes: function dynamicFixes() {
-                 return [new validationIssueFix({
-                   title: _t.html('issues.fix.remove_proprietary_data.title')
-                 })];
-               }
-             }));
-           });
-           return issues;
+           var geoData, polygonData;
 
-           function getReference(id) {
-             return function showReference(selection) {
-               selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.incompatible_source.' + id + '.reference'));
-             };
+           if (_template && vtService) {
+             // fetch data from vector tile service
+             var sourceID = _template;
+             vtService.loadTiles(sourceID, _template, projection);
+             geoData = vtService.data(sourceID, projection);
+           } else {
+             geoData = getFeatures(_geojson);
            }
-         };
-
-         validation.type = type;
-         return validation;
-       }
 
-       function validationMaprules() {
-         var type = 'maprules';
+           geoData = geoData.filter(getPath);
+           polygonData = geoData.filter(isPolygon); // Draw clip paths for polygons
 
-         var validation = function checkMaprules(entity, graph) {
-           if (!services.maprules) return [];
-           var rules = services.maprules.validationRules();
-           var issues = [];
+           var clipPaths = surface.selectAll('defs').selectAll('.clipPath-data').data(polygonData, featureKey);
+           clipPaths.exit().remove();
+           var clipPathsEnter = clipPaths.enter().append('clipPath').attr('class', 'clipPath-data').attr('id', clipPathID);
+           clipPathsEnter.append('path');
+           clipPaths.merge(clipPathsEnter).selectAll('path').attr('d', getAreaPath); // Draw fill, shadow, stroke layers
 
-           for (var i = 0; i < rules.length; i++) {
-             var rule = rules[i];
-             rule.findIssues(entity, graph, issues);
-           }
+           var datagroups = layer.selectAll('g.datagroup').data(['fill', 'shadow', 'stroke']);
+           datagroups = datagroups.enter().append('g').attr('class', function (d) {
+             return 'datagroup datagroup-' + d;
+           }).merge(datagroups); // Draw paths
 
-           return issues;
-         };
+           var pathData = {
+             fill: polygonData,
+             shadow: geoData,
+             stroke: geoData
+           };
+           var paths = datagroups.selectAll('path').data(function (layer) {
+             return pathData[layer];
+           }, featureKey); // exit
 
-         validation.type = type;
-         return validation;
-       }
+           paths.exit().remove(); // enter/update
 
-       function validationMismatchedGeometry() {
-         var type = 'mismatched_geometry';
+           paths = paths.enter().append('path').attr('class', function (d) {
+             var datagroup = this.parentNode.__data__;
+             return 'pathdata ' + datagroup + ' ' + featureClasses(d);
+           }).attr('clip-path', function (d) {
+             var datagroup = this.parentNode.__data__;
+             return datagroup === 'fill' ? 'url(#' + clipPathID(d) + ')' : null;
+           }).merge(paths).attr('d', function (d) {
+             var datagroup = this.parentNode.__data__;
+             return datagroup === 'fill' ? getAreaPath(d) : getPath(d);
+           }); // Draw labels
 
-         function tagSuggestingLineIsArea(entity) {
-           if (entity.type !== 'way' || entity.isClosed()) return null;
-           var tagSuggestingArea = entity.tagSuggestingArea();
+           layer.call(drawLabels, 'label-halo', geoData).call(drawLabels, 'label', geoData);
 
-           if (!tagSuggestingArea) {
-             return null;
-           }
+           function drawLabels(selection, textClass, data) {
+             var labelPath = d3_geoPath(projection);
+             var labelData = data.filter(function (d) {
+               return _showLabels && d.properties && (d.properties.desc || d.properties.name);
+             });
+             var labels = selection.selectAll('text.' + textClass).data(labelData, featureKey); // exit
 
-           var asLine = _mainPresetIndex.matchTags(tagSuggestingArea, 'line');
-           var asArea = _mainPresetIndex.matchTags(tagSuggestingArea, 'area');
+             labels.exit().remove(); // enter/update
 
-           if (asLine && asArea && asLine === asArea) {
-             // these tags also allow lines and making this an area wouldn't matter
-             return null;
+             labels = labels.enter().append('text').attr('class', function (d) {
+               return textClass + ' ' + featureClasses(d);
+             }).merge(labels).text(function (d) {
+               return d.properties.desc || d.properties.name;
+             }).attr('x', function (d) {
+               var centroid = labelPath.centroid(d);
+               return centroid[0] + 11;
+             }).attr('y', function (d) {
+               var centroid = labelPath.centroid(d);
+               return centroid[1];
+             });
            }
-
-           return tagSuggestingArea;
          }
 
-         function makeConnectEndpointsFixOnClick(way, graph) {
-           // must have at least three nodes to close this automatically
-           if (way.nodes.length < 3) return null;
-           var nodes = graph.childNodes(way),
-               testNodes;
-           var firstToLastDistanceMeters = geoSphericalDistance(nodes[0].loc, nodes[nodes.length - 1].loc); // if the distance is very small, attempt to merge the endpoints
+         function getExtension(fileName) {
+           if (!fileName) return;
+           var re = /\.(gpx|kml|(geo)?json)$/i;
+           var match = fileName.toLowerCase().match(re);
+           return match && match.length && match[0];
+         }
 
-           if (firstToLastDistanceMeters < 0.75) {
-             testNodes = nodes.slice(); // shallow copy
+         function xmlToDom(textdata) {
+           return new DOMParser().parseFromString(textdata, 'text/xml');
+         }
 
-             testNodes.pop();
-             testNodes.push(testNodes[0]); // make sure this will not create a self-intersection
+         drawData.setFile = function (extension, data) {
+           _template = null;
+           _fileList = null;
+           _geojson = null;
+           _src = null;
+           var gj;
 
-             if (!geoHasSelfIntersections(testNodes, testNodes[0].id)) {
-               return function (context) {
-                 var way = context.entity(this.issue.entityIds[0]);
-                 context.perform(actionMergeNodes([way.nodes[0], way.nodes[way.nodes.length - 1]], nodes[0].loc), _t('issues.fix.connect_endpoints.annotation'));
-               };
-             }
-           } // if the points were not merged, attempt to close the way
+           switch (extension) {
+             case '.gpx':
+               gj = togeojson.gpx(xmlToDom(data));
+               break;
 
+             case '.kml':
+               gj = togeojson.kml(xmlToDom(data));
+               break;
 
-           testNodes = nodes.slice(); // shallow copy
+             case '.geojson':
+             case '.json':
+               gj = JSON.parse(data);
+               break;
+           }
 
-           testNodes.push(testNodes[0]); // make sure this will not create a self-intersection
+           gj = gj || {};
 
-           if (!geoHasSelfIntersections(testNodes, testNodes[0].id)) {
-             return function (context) {
-               var wayId = this.issue.entityIds[0];
-               var way = context.entity(wayId);
-               var nodeId = way.nodes[0];
-               var index = way.nodes.length;
-               context.perform(actionAddVertex(wayId, nodeId, index), _t('issues.fix.connect_endpoints.annotation'));
-             };
+           if (Object.keys(gj).length) {
+             _geojson = ensureIDs(gj);
+             _src = extension + ' data file';
+             this.fitZoom();
            }
-         }
 
-         function lineTaggedAsAreaIssue(entity) {
-           var tagSuggestingArea = tagSuggestingLineIsArea(entity);
-           if (!tagSuggestingArea) return null;
-           return new validationIssue({
-             type: type,
-             subtype: 'area_as_line',
-             severity: 'warning',
-             message: function message(context) {
-               var entity = context.hasEntity(this.entityIds[0]);
-               return entity ? _t.html('issues.tag_suggests_area.message', {
-                 feature: utilDisplayLabel(entity, 'area'),
-                 tag: utilTagText({
-                   tags: tagSuggestingArea
-                 })
-               }) : '';
-             },
-             reference: showReference,
-             entityIds: [entity.id],
-             hash: JSON.stringify(tagSuggestingArea),
-             dynamicFixes: function dynamicFixes(context) {
-               var fixes = [];
-               var entity = context.entity(this.entityIds[0]);
-               var connectEndsOnClick = makeConnectEndpointsFixOnClick(entity, context.graph());
-               fixes.push(new validationIssueFix({
-                 title: _t.html('issues.fix.connect_endpoints.title'),
-                 onClick: connectEndsOnClick
-               }));
-               fixes.push(new validationIssueFix({
-                 icon: 'iD-operation-delete',
-                 title: _t.html('issues.fix.remove_tag.title'),
-                 onClick: function onClick(context) {
-                   var entityId = this.issue.entityIds[0];
-                   var entity = context.entity(entityId);
-                   var tags = Object.assign({}, entity.tags); // shallow copy
+           dispatch.call('change');
+           return this;
+         };
 
-                   for (var key in tagSuggestingArea) {
-                     delete tags[key];
-                   }
+         drawData.showLabels = function (val) {
+           if (!arguments.length) return _showLabels;
+           _showLabels = val;
+           return this;
+         };
 
-                   context.perform(actionChangeTags(entityId, tags), _t('issues.fix.remove_tag.annotation'));
-                 }
-               }));
-               return fixes;
-             }
-           });
+         drawData.enabled = function (val) {
+           if (!arguments.length) return _enabled;
+           _enabled = val;
 
-           function showReference(selection) {
-             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.tag_suggests_area.reference'));
+           if (_enabled) {
+             showLayer();
+           } else {
+             hideLayer();
            }
-         }
 
-         function vertexTaggedAsPointIssue(entity, graph) {
-           // we only care about nodes
-           if (entity.type !== 'node') return null; // ignore tagless points
-
-           if (Object.keys(entity.tags).length === 0) return null; // address lines are special so just ignore them
+           dispatch.call('change');
+           return this;
+         };
 
-           if (entity.isOnAddressLine(graph)) return null;
-           var geometry = entity.geometry(graph);
-           var allowedGeometries = osmNodeGeometriesForTags(entity.tags);
+         drawData.hasData = function () {
+           var gj = _geojson || {};
+           return !!(_template || Object.keys(gj).length);
+         };
 
-           if (geometry === 'point' && !allowedGeometries.point && allowedGeometries.vertex) {
-             return new validationIssue({
-               type: type,
-               subtype: 'vertex_as_point',
-               severity: 'warning',
-               message: function message(context) {
-                 var entity = context.hasEntity(this.entityIds[0]);
-                 return entity ? _t.html('issues.vertex_as_point.message', {
-                   feature: utilDisplayLabel(entity, 'vertex')
-                 }) : '';
-               },
-               reference: function showReference(selection) {
-                 selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.vertex_as_point.reference'));
-               },
-               entityIds: [entity.id]
-             });
-           } else if (geometry === 'vertex' && !allowedGeometries.vertex && allowedGeometries.point) {
-             return new validationIssue({
-               type: type,
-               subtype: 'point_as_vertex',
-               severity: 'warning',
-               message: function message(context) {
-                 var entity = context.hasEntity(this.entityIds[0]);
-                 return entity ? _t.html('issues.point_as_vertex.message', {
-                   feature: utilDisplayLabel(entity, 'point')
-                 }) : '';
-               },
-               reference: function showReference(selection) {
-                 selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.point_as_vertex.reference'));
-               },
-               entityIds: [entity.id],
-               dynamicFixes: function dynamicFixes(context) {
-                 var entityId = this.entityIds[0];
-                 var extractOnClick = null;
-
-                 if (!context.hasHiddenConnections(entityId)) {
-                   extractOnClick = function extractOnClick(context) {
-                     var entityId = this.issue.entityIds[0];
-                     var action = actionExtract(entityId);
-                     context.perform(action, _t('operations.extract.annotation', {
-                       n: 1
-                     })); // re-enter mode to trigger updates
-
-                     context.enter(modeSelect(context, [action.getExtractedNodeID()]));
-                   };
-                 }
+         drawData.template = function (val, src) {
+           if (!arguments.length) return _template; // test source against OSM imagery blocklists..
 
-                 return [new validationIssueFix({
-                   icon: 'iD-operation-extract',
-                   title: _t.html('issues.fix.extract_point.title'),
-                   onClick: extractOnClick
-                 })];
-               }
-             });
-           }
+           var osm = context.connection();
 
-           return null;
-         }
+           if (osm) {
+             var blocklists = osm.imageryBlocklists();
+             var fail = false;
+             var tested = 0;
+             var regex;
 
-         function unclosedMultipolygonPartIssues(entity, graph) {
-           if (entity.type !== 'relation' || !entity.isMultipolygon() || entity.isDegenerate() || // cannot determine issues for incompletely-downloaded relations
-           !entity.isComplete(graph)) return null;
-           var sequences = osmJoinWays(entity.members, graph);
-           var issues = [];
+             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.
 
-           for (var i in sequences) {
-             var sequence = sequences[i];
-             if (!sequence.nodes) continue;
-             var firstNode = sequence.nodes[0];
-             var lastNode = sequence.nodes[sequence.nodes.length - 1]; // part is closed if the first and last nodes are the same
 
-             if (firstNode === lastNode) continue;
-             var issue = new validationIssue({
-               type: type,
-               subtype: 'unclosed_multipolygon_part',
-               severity: 'warning',
-               message: function message(context) {
-                 var entity = context.hasEntity(this.entityIds[0]);
-                 return entity ? _t.html('issues.unclosed_multipolygon_part.message', {
-                   feature: utilDisplayLabel(entity, context.graph())
-                 }) : '';
-               },
-               reference: showReference,
-               loc: sequence.nodes[0].loc,
-               entityIds: [entity.id],
-               hash: sequence.map(function (way) {
-                 return way.id;
-               }).join()
-             });
-             issues.push(issue);
+             if (!tested) {
+               regex = /.*\.google(apis)?\..*\/(vt|kh)[\?\/].*([xyz]=.*){3}.*/;
+               fail = regex.test(val);
+             }
            }
 
-           return issues;
+           _template = val;
+           _fileList = null;
+           _geojson = null; // strip off the querystring/hash from the template,
+           // it often includes the access token
 
-           function showReference(selection) {
-             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.unclosed_multipolygon_part.reference'));
+           _src = src || 'vectortile:' + val.split(/[?#]/)[0];
+           dispatch.call('change');
+           return this;
+         };
+
+         drawData.geojson = function (gj, src) {
+           if (!arguments.length) return _geojson;
+           _template = null;
+           _fileList = null;
+           _geojson = null;
+           _src = null;
+           gj = gj || {};
+
+           if (Object.keys(gj).length) {
+             _geojson = ensureIDs(gj);
+             _src = src || 'unknown.geojson';
            }
-         }
 
-         var validation = function checkMismatchedGeometry(entity, graph) {
-           var issues = [vertexTaggedAsPointIssue(entity, graph), lineTaggedAsAreaIssue(entity)];
-           issues = issues.concat(unclosedMultipolygonPartIssues(entity, graph));
-           return issues.filter(Boolean);
+           dispatch.call('change');
+           return this;
          };
 
-         validation.type = type;
-         return validation;
-       }
+         drawData.fileList = function (fileList) {
+           if (!arguments.length) return _fileList;
+           _template = null;
+           _fileList = fileList;
+           _geojson = null;
+           _src = null;
+           if (!fileList || !fileList.length) return this;
+           var f = fileList[0];
+           var extension = getExtension(f.name);
+           var reader = new FileReader();
 
-       function validationMissingRole() {
-         var type = 'missing_role';
+           reader.onload = function () {
+             return function (e) {
+               drawData.setFile(extension, e.target.result);
+             };
+           }();
 
-         var validation = function checkMissingRole(entity, graph) {
-           var issues = [];
+           reader.readAsText(f);
+           return this;
+         };
 
-           if (entity.type === 'way') {
-             graph.parentRelations(entity).forEach(function (relation) {
-               if (!relation.isMultipolygon()) return;
-               var member = relation.memberById(entity.id);
+         drawData.url = function (url, defaultExtension) {
+           _template = null;
+           _fileList = null;
+           _geojson = null;
+           _src = null; // strip off any querystring/hash from the url before checking extension
 
-               if (member && isMissingRole(member)) {
-                 issues.push(makeIssue(entity, relation, member));
-               }
-             });
-           } else if (entity.type === 'relation' && entity.isMultipolygon()) {
-             entity.indexedMembers().forEach(function (member) {
-               var way = graph.hasEntity(member.id);
+           var testUrl = url.split(/[?#]/)[0];
+           var extension = getExtension(testUrl) || defaultExtension;
 
-               if (way && isMissingRole(member)) {
-                 issues.push(makeIssue(way, entity, member));
-               }
+           if (extension) {
+             _template = null;
+             d3_text(url).then(function (data) {
+               drawData.setFile(extension, data);
+             })["catch"](function () {
+               /* ignore */
              });
+           } else {
+             drawData.template(url);
            }
 
-           return issues;
+           return this;
          };
 
-         function isMissingRole(member) {
-           return !member.role || !member.role.trim().length;
-         }
-
-         function makeIssue(way, relation, member) {
-           return new validationIssue({
-             type: type,
-             severity: 'warning',
-             message: function message(context) {
-               var member = context.hasEntity(this.entityIds[1]),
-                   relation = context.hasEntity(this.entityIds[0]);
-               return member && relation ? _t.html('issues.missing_role.message', {
-                 member: utilDisplayLabel(member, context.graph()),
-                 relation: utilDisplayLabel(relation, context.graph())
-               }) : '';
-             },
-             reference: showReference,
-             entityIds: [relation.id, way.id],
-             data: {
-               member: member
-             },
-             hash: member.index.toString(),
-             dynamicFixes: function dynamicFixes() {
-               return [makeAddRoleFix('inner'), makeAddRoleFix('outer'), new validationIssueFix({
-                 icon: 'iD-operation-delete',
-                 title: _t.html('issues.fix.remove_from_relation.title'),
-                 onClick: function onClick(context) {
-                   context.perform(actionDeleteMember(this.issue.entityIds[0], this.issue.data.member.index), _t('operations.delete_member.annotation', {
-                     n: 1
-                   }));
-                 }
-               })];
-             }
-           });
+         drawData.getSrc = function () {
+           return _src || '';
+         };
 
-           function showReference(selection) {
-             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.missing_role.multipolygon.reference'));
-           }
-         }
+         drawData.fitZoom = function () {
+           var features = getFeatures(_geojson);
+           if (!features.length) return;
+           var map = context.map();
+           var viewport = map.trimmedExtent().polygon();
+           var coords = features.reduce(function (coords, feature) {
+             var geom = feature.geometry;
+             if (!geom) return coords;
+             var c = geom.coordinates;
+             /* eslint-disable no-fallthrough */
 
-         function makeAddRoleFix(role) {
-           return new validationIssueFix({
-             title: _t.html('issues.fix.set_as_' + role + '.title'),
-             onClick: function onClick(context) {
-               var oldMember = this.issue.data.member;
-               var member = {
-                 id: this.issue.entityIds[1],
-                 type: oldMember.type,
-                 role: role
-               };
-               context.perform(actionChangeMember(this.issue.entityIds[0], member, oldMember.index), _t('operations.change_role.annotation', {
-                 n: 1
-               }));
-             }
-           });
-         }
+             switch (geom.type) {
+               case 'Point':
+                 c = [c];
 
-         validation.type = type;
-         return validation;
-       }
+               case 'MultiPoint':
+               case 'LineString':
+                 break;
 
-       function validationMissingTag(context) {
-         var type = 'missing_tag';
+               case 'MultiPolygon':
+                 c = utilArrayFlatten(c);
 
-         function hasDescriptiveTags(entity, graph) {
-           var keys = Object.keys(entity.tags).filter(function (k) {
-             if (k === 'area' || k === 'name') {
-               return false;
-             } else {
-               return osmIsInterestingTag(k);
+               case 'Polygon':
+               case 'MultiLineString':
+                 c = utilArrayFlatten(c);
+                 break;
              }
-           });
+             /* eslint-enable no-fallthrough */
 
-           if (entity.type === 'relation' && keys.length === 1 && entity.tags.type === 'multipolygon') {
-             // this relation's only interesting tag just says its a multipolygon,
-             // which is not descriptive enough
-             // It's okay for a simple multipolygon to have no descriptive tags
-             // if its outer way has them (old model, see `outdated_tags.js`)
-             return osmOldMultipolygonOuterMemberOfRelation(entity, graph);
+
+             return utilArrayUnion(coords, c);
+           }, []);
+
+           if (!geoPolygonIntersectsPolygon(viewport, coords, true)) {
+             var extent = geoExtent(d3_geoBounds({
+               type: 'LineString',
+               coordinates: coords
+             }));
+             map.centerZoom(extent.center(), map.trimmedExtentZoom(extent));
            }
 
-           return keys.length > 0;
-         }
+           return this;
+         };
 
-         function isUnknownRoad(entity) {
-           return entity.type === 'way' && entity.tags.highway === 'road';
-         }
+         init();
+         return drawData;
+       }
 
-         function isUntypedRelation(entity) {
-           return entity.type === 'relation' && !entity.tags.type;
-         }
+       function svgDebug(projection, context) {
+         function drawDebug(selection) {
+           var showTile = context.getDebug('tile');
+           var showCollision = context.getDebug('collision');
+           var showImagery = context.getDebug('imagery');
+           var showTouchTargets = context.getDebug('target');
+           var showDownloaded = context.getDebug('downloaded');
+           var debugData = [];
 
-         var validation = function checkMissingTag(entity, graph) {
-           var subtype;
-           var osm = context.connection();
-           var isUnloadedNode = entity.type === 'node' && osm && !osm.isDataLoaded(entity.loc); // we can't know if the node is a vertex if the tile is undownloaded
+           if (showTile) {
+             debugData.push({
+               "class": 'red',
+               label: 'tile'
+             });
+           }
 
-           if (!isUnloadedNode && // allow untagged nodes that are part of ways
-           entity.geometry(graph) !== 'vertex' && // allow untagged entities that are part of relations
-           !entity.hasParentRelations(graph)) {
-             if (Object.keys(entity.tags).length === 0) {
-               subtype = 'any';
-             } else if (!hasDescriptiveTags(entity, graph)) {
-               subtype = 'descriptive';
-             } else if (isUntypedRelation(entity)) {
-               subtype = 'relation_type';
-             }
-           } // flag an unknown road even if it's a member of a relation
+           if (showCollision) {
+             debugData.push({
+               "class": 'yellow',
+               label: 'collision'
+             });
+           }
 
+           if (showImagery) {
+             debugData.push({
+               "class": 'orange',
+               label: 'imagery'
+             });
+           }
 
-           if (!subtype && isUnknownRoad(entity)) {
-             subtype = 'highway_classification';
+           if (showTouchTargets) {
+             debugData.push({
+               "class": 'pink',
+               label: 'touchTargets'
+             });
            }
 
-           if (!subtype) return [];
-           var messageID = subtype === 'highway_classification' ? 'unknown_road' : 'missing_tag.' + subtype;
-           var referenceID = subtype === 'highway_classification' ? 'unknown_road' : 'missing_tag'; // can always delete if the user created it in the first place..
+           if (showDownloaded) {
+             debugData.push({
+               "class": 'purple',
+               label: 'downloaded'
+             });
+           }
 
-           var canDelete = entity.version === undefined || entity.v !== undefined;
-           var severity = canDelete && subtype !== 'highway_classification' ? 'error' : 'warning';
-           return [new validationIssue({
-             type: type,
-             subtype: subtype,
-             severity: severity,
-             message: function message(context) {
-               var entity = context.hasEntity(this.entityIds[0]);
-               return entity ? _t.html('issues.' + messageID + '.message', {
-                 feature: utilDisplayLabel(entity, context.graph())
-               }) : '';
-             },
-             reference: showReference,
-             entityIds: [entity.id],
-             dynamicFixes: function dynamicFixes(context) {
-               var fixes = [];
-               var selectFixType = subtype === 'highway_classification' ? 'select_road_type' : 'select_preset';
-               fixes.push(new validationIssueFix({
-                 icon: 'iD-icon-search',
-                 title: _t.html('issues.fix.' + selectFixType + '.title'),
-                 onClick: function onClick(context) {
-                   context.ui().sidebar.showPresetList();
+           var legend = context.container().select('.main-content').selectAll('.debug-legend').data(debugData.length ? [0] : []);
+           legend.exit().remove();
+           legend = legend.enter().append('div').attr('class', 'fillD debug-legend').merge(legend);
+           var legendItems = legend.selectAll('.debug-legend-item').data(debugData, function (d) {
+             return d.label;
+           });
+           legendItems.exit().remove();
+           legendItems.enter().append('span').attr('class', function (d) {
+             return "debug-legend-item ".concat(d["class"]);
+           }).text(function (d) {
+             return d.label;
+           });
+           var layer = selection.selectAll('.layer-debug').data(showImagery || showDownloaded ? [0] : []);
+           layer.exit().remove();
+           layer = layer.enter().append('g').attr('class', 'layer-debug').merge(layer); // imagery
+
+           var extent = context.map().extent();
+           _mainFileFetcher.get('imagery').then(function (d) {
+             var hits = showImagery && d.query.bbox(extent.rectangle(), true) || [];
+             var features = hits.map(function (d) {
+               return d.features[d.id];
+             });
+             var imagery = layer.selectAll('path.debug-imagery').data(features);
+             imagery.exit().remove();
+             imagery.enter().append('path').attr('class', 'debug-imagery debug orange');
+           })["catch"](function () {
+             /* ignore */
+           }); // downloaded
+
+           var osm = context.connection();
+           var dataDownloaded = [];
+
+           if (osm && showDownloaded) {
+             var rtree = osm.caches('get').tile.rtree;
+             dataDownloaded = rtree.all().map(function (bbox) {
+               return {
+                 type: 'Feature',
+                 properties: {
+                   id: bbox.id
+                 },
+                 geometry: {
+                   type: 'Polygon',
+                   coordinates: [[[bbox.minX, bbox.minY], [bbox.minX, bbox.maxY], [bbox.maxX, bbox.maxY], [bbox.maxX, bbox.minY], [bbox.minX, bbox.minY]]]
                  }
-               }));
-               var deleteOnClick;
-               var id = this.entityIds[0];
-               var operation = operationDelete(context, [id]);
-               var disabledReasonID = operation.disabled();
+               };
+             });
+           }
 
-               if (!disabledReasonID) {
-                 deleteOnClick = function deleteOnClick(context) {
-                   var id = this.issue.entityIds[0];
-                   var operation = operationDelete(context, [id]);
+           var downloaded = layer.selectAll('path.debug-downloaded').data(showDownloaded ? dataDownloaded : []);
+           downloaded.exit().remove();
+           downloaded.enter().append('path').attr('class', 'debug-downloaded debug purple'); // update
 
-                   if (!operation.disabled()) {
-                     operation();
-                   }
-                 };
-               }
+           layer.selectAll('path').attr('d', svgPath(projection).geojson);
+         } // This looks strange because `enabled` methods on other layers are
+         // chainable getter/setters, and this one is just a getter.
 
-               fixes.push(new validationIssueFix({
-                 icon: 'iD-operation-delete',
-                 title: _t.html('issues.fix.delete_feature.title'),
-                 disabledReason: disabledReasonID ? _t('operations.delete.' + disabledReasonID + '.single') : undefined,
-                 onClick: deleteOnClick
-               }));
-               return fixes;
-             }
-           })];
 
-           function showReference(selection) {
-             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.' + referenceID + '.reference'));
+         drawDebug.enabled = function () {
+           if (!arguments.length) {
+             return context.getDebug('tile') || context.getDebug('collision') || context.getDebug('imagery') || context.getDebug('target') || context.getDebug('downloaded');
+           } else {
+             return this;
            }
          };
 
-         validation.type = type;
-         return validation;
+         return drawDebug;
        }
 
-       var simplify = function simplify(str) {
-         return diacritics.remove(str.replace(/&/g, 'and').replace(/[\s\-=_!"#%'*{},.\/:;?\(\)\[\]@\\$\^*+<>~`’\u00a1\u00a7\u00b6\u00b7\u00bf\u037e\u0387\u055a-\u055f\u0589\u05c0\u05c3\u05c6\u05f3\u05f4\u0609\u060a\u060c\u060d\u061b\u061e\u061f\u066a-\u066d\u06d4\u0700-\u070d\u07f7-\u07f9\u0830-\u083e\u085e\u0964\u0965\u0970\u0af0\u0df4\u0e4f\u0e5a\u0e5b\u0f04-\u0f12\u0f14\u0f85\u0fd0-\u0fd4\u0fd9\u0fda\u104a-\u104f\u10fb\u1360-\u1368\u166d\u166e\u16eb-\u16ed\u1735\u1736\u17d4-\u17d6\u17d8-\u17da\u1800-\u1805\u1807-\u180a\u1944\u1945\u1a1e\u1a1f\u1aa0-\u1aa6\u1aa8-\u1aad\u1b5a-\u1b60\u1bfc-\u1bff\u1c3b-\u1c3f\u1c7e\u1c7f\u1cc0-\u1cc7\u1cd3\u2016\u2017\u2020-\u2027\u2030-\u2038\u203b-\u203e\u2041-\u2043\u2047-\u2051\u2053\u2055-\u205e\u2cf9-\u2cfc\u2cfe\u2cff\u2d70\u2e00\u2e01\u2e06-\u2e08\u2e0b\u2e0e-\u2e16\u2e18\u2e19\u2e1b\u2e1e\u2e1f\u2e2a-\u2e2e\u2e30-\u2e39\u3001-\u3003\u303d\u30fb\ua4fe\ua4ff\ua60d-\ua60f\ua673\ua67e\ua6f2-\ua6f7\ua874-\ua877\ua8ce\ua8cf\ua8f8-\ua8fa\ua92e\ua92f\ua95f\ua9c1-\ua9cd\ua9de\ua9df\uaa5c-\uaa5f\uaade\uaadf\uaaf0\uaaf1\uabeb\ufe10-\ufe16\ufe19\ufe30\ufe45\ufe46\ufe49-\ufe4c\ufe50-\ufe52\ufe54-\ufe57\ufe5f-\ufe61\ufe68\ufe6a\ufe6b\uff01-\uff03\uff05-\uff07\uff0a\uff0c\uff0e\uff0f\uff1a\uff1b\uff1f\uff20\uff3c\uff61\uff64\uff65]+/g, '').toLowerCase());
-       };
+       /*
+           A standalone SVG element that contains only a `defs` sub-element. To be
+           used once globally, since defs IDs must be unique within a document.
+       */
 
-       // {
-       //   kvnd:        "amenity/fast_food|Thaï Express~(North America)",
-       //   kvn:         "amenity/fast_food|Thaï Express",
-       //   kv:          "amenity/fast_food",
-       //   k:           "amenity",
-       //   v:           "fast_food",
-       //   n:           "Thaï Express",
-       //   d:           "(North America)",
-       //   nsimple:     "thaiexpress",
-       //   kvnnsimple:  "amenity/fast_food|thaiexpress"
-       // }
+       function svgDefs(context) {
+         var _defsSelection = select(null);
 
-       var to_parts = function to_parts(kvnd) {
-         var parts = {};
-         parts.kvnd = kvnd;
-         var kvndparts = kvnd.split('~', 2);
-         if (kvndparts.length > 1) parts.d = kvndparts[1];
-         parts.kvn = kvndparts[0];
-         var kvnparts = parts.kvn.split('|', 2);
-         if (kvnparts.length > 1) parts.n = kvnparts[1];
-         parts.kv = kvnparts[0];
-         var kvparts = parts.kv.split('/', 2);
-         parts.k = kvparts[0];
-         parts.v = kvparts[1];
-         parts.nsimple = simplify(parts.n);
-         parts.kvnsimple = parts.kv + '|' + parts.nsimple;
-         return parts;
-       };
+         var _spritesheetIds = ['iD-sprite', 'maki-sprite', 'temaki-sprite', 'fa-sprite', 'community-sprite'];
 
-       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 require$$0 = {
-       matchGroups: matchGroups
-       };
+         function drawDefs(selection) {
+           _defsSelection = selection.append('defs'); // add markers
+
+           _defsSelection.append('marker').attr('id', 'ideditor-oneway-marker').attr('viewBox', '0 0 10 5').attr('refX', 2.5).attr('refY', 2.5).attr('markerWidth', 2).attr('markerHeight', 2).attr('markerUnits', 'strokeWidth').attr('orient', 'auto').append('path').attr('class', 'oneway-marker-path').attr('d', 'M 5,3 L 0,3 L 0,2 L 5,2 L 5,0 L 10,2.5 L 5,5 z').attr('stroke', 'none').attr('fill', '#000').attr('opacity', '0.75'); // SVG markers have to be given a colour where they're defined
+           // (they can't inherit it from the line they're attached to),
+           // so we need to manually define markers for each color of tag
+           // (also, it's slightly nicer if we can control the
+           // positioning for different tags)
+
+
+           function addSidedMarker(name, color, offset) {
+             _defsSelection.append('marker').attr('id', 'ideditor-sided-marker-' + name).attr('viewBox', '0 0 2 2').attr('refX', 1).attr('refY', -offset).attr('markerWidth', 1.5).attr('markerHeight', 1.5).attr('markerUnits', 'strokeWidth').attr('orient', 'auto').append('path').attr('class', 'sided-marker-path sided-marker-' + name + '-path').attr('d', 'M 0,0 L 1,1 L 2,0 z').attr('stroke', 'none').attr('fill', color);
+           }
+
+           addSidedMarker('natural', 'rgb(170, 170, 170)', 0); // for a coastline, the arrows are (somewhat unintuitively) on
+           // the water side, so let's color them blue (with a gap) to
+           // give a stronger indication
+
+           addSidedMarker('coastline', '#77dede', 1);
+           addSidedMarker('waterway', '#77dede', 1); // barriers have a dashed line, and separating the triangle
+           // from the line visually suits that
 
-       var matchGroups$1 = require$$0.matchGroups;
+           addSidedMarker('barrier', '#ddd', 1);
+           addSidedMarker('man_made', '#fff', 0);
 
-       var matcher$1 = function matcher() {
-         var _warnings = []; // array of match conflict pairs
+           _defsSelection.append('marker').attr('id', 'ideditor-viewfield-marker').attr('viewBox', '0 0 16 16').attr('refX', 8).attr('refY', 16).attr('markerWidth', 4).attr('markerHeight', 4).attr('markerUnits', 'strokeWidth').attr('orient', 'auto').append('path').attr('class', 'viewfield-marker-path').attr('d', 'M 6,14 C 8,13.4 8,13.4 10,14 L 16,3 C 12,0 4,0 0,3 z').attr('fill', '#333').attr('fill-opacity', '0.75').attr('stroke', '#fff').attr('stroke-width', '0.5px').attr('stroke-opacity', '0.75');
 
-         var _ambiguous = {};
-         var _matchIndex = {};
-         var matcher = {}; // Create an index of all the keys/simplenames for fast matching
+           _defsSelection.append('marker').attr('id', 'ideditor-viewfield-marker-wireframe').attr('viewBox', '0 0 16 16').attr('refX', 8).attr('refY', 16).attr('markerWidth', 4).attr('markerHeight', 4).attr('markerUnits', 'strokeWidth').attr('orient', 'auto').append('path').attr('class', 'viewfield-marker-path').attr('d', 'M 6,14 C 8,13.4 8,13.4 10,14 L 16,3 C 12,0 4,0 0,3 z').attr('fill', 'none').attr('stroke', '#fff').attr('stroke-width', '0.5px').attr('stroke-opacity', '0.75'); // add patterns
 
-         matcher.buildMatchIndex = function (brands) {
-           // two passes - once for primary names, once for secondary/alternate names
-           Object.keys(brands).forEach(function (kvnd) {
-             return insertNames(kvnd, 'primary');
-           });
-           Object.keys(brands).forEach(function (kvnd) {
-             return insertNames(kvnd, 'secondary');
-           });
 
-           function insertNames(kvnd, which) {
-             var obj = brands[kvnd];
-             var parts = to_parts(kvnd); // Exit early for ambiguous names in the second pass.
-             // They were collected in the first pass and we don't gather alt names for them.
+           var patterns = _defsSelection.selectAll('pattern').data([// pattern name, pattern image name
+           ['beach', 'dots'], ['construction', 'construction'], ['cemetery', 'cemetery'], ['cemetery_christian', 'cemetery_christian'], ['cemetery_buddhist', 'cemetery_buddhist'], ['cemetery_muslim', 'cemetery_muslim'], ['cemetery_jewish', 'cemetery_jewish'], ['farmland', 'farmland'], ['farmyard', 'farmyard'], ['forest', 'forest'], ['forest_broadleaved', 'forest_broadleaved'], ['forest_needleleaved', 'forest_needleleaved'], ['forest_leafless', 'forest_leafless'], ['golf_green', 'grass'], ['grass', 'grass'], ['landfill', 'landfill'], ['meadow', 'grass'], ['orchard', 'orchard'], ['pond', 'pond'], ['quarry', 'quarry'], ['scrub', 'bushes'], ['vineyard', 'vineyard'], ['water_standing', 'lines'], ['waves', 'waves'], ['wetland', 'wetland'], ['wetland_marsh', 'wetland_marsh'], ['wetland_swamp', 'wetland_swamp'], ['wetland_bog', 'wetland_bog'], ['wetland_reedbed', 'wetland_reedbed']]).enter().append('pattern').attr('id', function (d) {
+             return 'ideditor-pattern-' + d[0];
+           }).attr('width', 32).attr('height', 32).attr('patternUnits', 'userSpaceOnUse');
 
-             if (which === 'secondary' && parts.d) return;
+           patterns.append('rect').attr('x', 0).attr('y', 0).attr('width', 32).attr('height', 32).attr('class', function (d) {
+             return 'pattern-color-' + d[0];
+           });
+           patterns.append('image').attr('x', 0).attr('y', 0).attr('width', 32).attr('height', 32).attr('xlink:href', function (d) {
+             return context.imagePath('pattern/' + d[1] + '.png');
+           }); // add clip paths
 
-             if (obj.countryCodes) {
-               parts.countryCodes = obj.countryCodes.slice(); // copy
-             }
+           _defsSelection.selectAll('clipPath').data([12, 18, 20, 32, 45]).enter().append('clipPath').attr('id', function (d) {
+             return 'ideditor-clip-square-' + d;
+           }).append('rect').attr('x', 0).attr('y', 0).attr('width', function (d) {
+             return d;
+           }).attr('height', function (d) {
+             return d;
+           }); // add symbol spritesheets
 
-             var nomatches = obj.nomatch || [];
 
-             if (nomatches.some(function (s) {
-               return s === kvnd;
-             })) {
-               console.log("WARNING match/nomatch conflict for ".concat(kvnd));
-               return;
-             }
+           addSprites(_spritesheetIds, true);
+         }
 
-             var match_kv = [parts.kv].concat(obj.matchTags || []).concat(["".concat(parts.k, "/yes"), "building/yes"]) // #3454 - match some generic tags
-             .map(function (s) {
-               return s.toLowerCase();
-             });
-             var match_nsimple = [];
+         function addSprites(ids, overrideColors) {
+           _spritesheetIds = utilArrayUniq(_spritesheetIds.concat(ids));
 
-             if (which === 'primary') {
-               match_nsimple = [parts.n].concat(obj.matchNames || []).concat(obj.tags.official_name || []) // #2732 - match alternate names
-               .map(simplify);
-             } else if (which === 'secondary') {
-               match_nsimple = [].concat(obj.tags.alt_name || []) // #2732 - match alternate names
-               .concat(obj.tags.short_name || []) // #2732 - match alternate names
-               .map(simplify);
-             }
+           var spritesheets = _defsSelection.selectAll('.spritesheet').data(_spritesheetIds);
 
-             if (!match_nsimple.length) return; // nothing to do
+           spritesheets.enter().append('g').attr('class', function (d) {
+             return 'spritesheet spritesheet-' + d;
+           }).each(function (d) {
+             var url = context.imagePath(d + '.svg');
+             var node = select(this).node();
+             svg(url).then(function (svg) {
+               node.appendChild(select(svg.documentElement).attr('id', 'ideditor-' + d).node());
 
-             match_kv.forEach(function (kv) {
-               match_nsimple.forEach(function (nsimple) {
-                 if (parts.d) {
-                   // Known ambiguous names with disambiguation string ~(USA) / ~(Canada)
-                   // FIXME: Name collisions will overwrite the initial entry (ok for now)
-                   if (!_ambiguous[kv]) _ambiguous[kv] = {};
-                   _ambiguous[kv][nsimple] = parts;
-                 } else {
-                   // Names we mostly expect to be unique..
-                   if (!_matchIndex[kv]) _matchIndex[kv] = {};
-                   var m = _matchIndex[kv][nsimple];
-
-                   if (m) {
-                     // There already is a match for this name, skip it
-                     // Warn if we detect collisions in a primary name.
-                     // Skip warning if a secondary name or a generic `*=yes` tag - #2972 / #3454
-                     if (which === 'primary' && !/\/yes$/.test(kv)) {
-                       _warnings.push([m.kvnd, "".concat(kvnd, " (").concat(kv, "/").concat(nsimple, ")")]);
-                     }
-                   } else {
-                     _matchIndex[kv][nsimple] = parts; // insert
-                   }
-                 }
-               });
+               if (overrideColors && d !== 'iD-sprite') {
+                 // allow icon colors to be overridden..
+                 select(node).selectAll('path').attr('fill', 'currentColor');
+               }
+             })["catch"](function () {
+               /* ignore */
              });
-           }
-         }; // pass a `key`, `value`, `name` and return the best match,
-         // `countryCode` optional (if supplied, must match that too)
-
+           });
+           spritesheets.exit().remove();
+         }
 
-         matcher.matchKVN = function (key, value, name, countryCode) {
-           return matcher.matchParts(to_parts("".concat(key, "/").concat(value, "|").concat(name)), countryCode);
-         }; // pass a parts object and return the best match,
-         // `countryCode` optional (if supplied, must match that too)
+         drawDefs.addSprites = addSprites;
+         return drawDefs;
+       }
 
+       var _layerEnabled$2 = false;
 
-         matcher.matchParts = function (parts, countryCode) {
-           var match = null;
-           var inGroup = false; // fixme: we currently return a single match for ambiguous
+       var _qaService$2;
 
-           match = _ambiguous[parts.kv] && _ambiguous[parts.kv][parts.nsimple];
-           if (match && matchesCountryCode(match)) return match; // try to return an exact match
+       function svgKeepRight(projection, context, dispatch) {
+         var throttledRedraw = throttle(function () {
+           return dispatch.call('change');
+         }, 1000);
 
-           match = _matchIndex[parts.kv] && _matchIndex[parts.kv][parts.nsimple];
-           if (match && matchesCountryCode(match)) return match; // look in match groups
+         var minZoom = 12;
+         var touchLayer = select(null);
+         var drawLayer = select(null);
+         var layerVisible = false;
 
-           for (var mg in matchGroups$1) {
-             var matchGroup = matchGroups$1[mg];
-             match = null;
-             inGroup = false;
+         function markerPath(selection, klass) {
+           selection.attr('class', klass).attr('transform', 'translate(-4, -24)').attr('d', 'M11.6,6.2H7.1l1.4-5.1C8.6,0.6,8.1,0,7.5,0H2.2C1.7,0,1.3,0.3,1.3,0.8L0,10.2c-0.1,0.6,0.4,1.1,0.9,1.1h4.6l-1.8,7.6C3.6,19.4,4.1,20,4.7,20c0.3,0,0.6-0.2,0.8-0.5l6.9-11.9C12.7,7,12.3,6.2,11.6,6.2z');
+         } // Loosely-coupled keepRight service for fetching issues.
 
-             for (var i = 0; i < matchGroup.length; i++) {
-               var otherkv = matchGroup[i].toLowerCase();
 
-               if (!inGroup) {
-                 inGroup = otherkv === parts.kv;
-               }
+         function getService() {
+           if (services.keepRight && !_qaService$2) {
+             _qaService$2 = services.keepRight;
 
-               if (!match) {
-                 // fixme: we currently return a single match for ambiguous
-                 match = _ambiguous[otherkv] && _ambiguous[otherkv][parts.nsimple];
-               }
+             _qaService$2.on('loaded', throttledRedraw);
+           } else if (!services.keepRight && _qaService$2) {
+             _qaService$2 = null;
+           }
 
-               if (!match) {
-                 match = _matchIndex[otherkv] && _matchIndex[otherkv][parts.nsimple];
-               }
+           return _qaService$2;
+         } // Show the markers
 
-               if (match && !matchesCountryCode(match)) {
-                 match = null;
-               }
 
-               if (inGroup && match) {
-                 return match;
-               }
-             }
+         function editOn() {
+           if (!layerVisible) {
+             layerVisible = true;
+             drawLayer.style('display', 'block');
            }
+         } // Immediately remove the markers and their touch targets
 
-           return null;
 
-           function matchesCountryCode(match) {
-             if (!countryCode) return true;
-             if (!match.countryCodes) return true;
-             return match.countryCodes.indexOf(countryCode) !== -1;
+         function editOff() {
+           if (layerVisible) {
+             layerVisible = false;
+             drawLayer.style('display', 'none');
+             drawLayer.selectAll('.qaItem.keepRight').remove();
+             touchLayer.selectAll('.qaItem.keepRight').remove();
            }
-         };
+         } // Enable the layer.  This shows the markers and transitions them to visible.
 
-         matcher.getWarnings = function () {
-           return _warnings;
-         };
 
-         return matcher;
-       };
+         function layerOn() {
+           editOn();
+           drawLayer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end interrupt', function () {
+             return dispatch.call('change');
+           });
+         } // Disable the layer.  This transitions the layer invisible and then hides the markers.
 
-       var fromCharCode = String.fromCharCode;
-       var nativeFromCodePoint = String.fromCodePoint;
 
-       // length should be 1, old FF problem
-       var INCORRECT_LENGTH = !!nativeFromCodePoint && nativeFromCodePoint.length != 1;
+         function layerOff() {
+           throttledRedraw.cancel();
+           drawLayer.interrupt();
+           touchLayer.selectAll('.qaItem.keepRight').remove();
+           drawLayer.transition().duration(250).style('opacity', 0).on('end interrupt', function () {
+             editOff();
+             dispatch.call('change');
+           });
+         } // Update the issue markers
 
-       // `String.fromCodePoint` method
-       // https://tc39.es/ecma262/#sec-string.fromcodepoint
-       _export({ target: 'String', stat: true, forced: INCORRECT_LENGTH }, {
-         // eslint-disable-next-line no-unused-vars -- required for `.length`
-         fromCodePoint: function fromCodePoint(x) {
-           var elements = [];
-           var length = arguments.length;
-           var i = 0;
-           var code;
-           while (length > i) {
-             code = +arguments[i++];
-             if (toAbsoluteIndex(code, 0x10FFFF) !== code) throw RangeError(code + ' is not a valid code point');
-             elements.push(code < 0x10000
-               ? fromCharCode(code)
-               : fromCharCode(((code -= 0x10000) >> 10) + 0xD800, code % 0x400 + 0xDC00)
-             );
-           } return elements.join('');
-         }
-       });
 
-       var quickselect$2 = createCommonjsModule(function (module, exports) {
-         (function (global, factory) {
-            module.exports = factory() ;
-         })(commonjsGlobal, function () {
+         function updateMarkers() {
+           if (!layerVisible || !_layerEnabled$2) return;
+           var service = getService();
+           var selectedID = context.selectedErrorID();
+           var data = service ? service.getItems(projection) : [];
+           var getTransform = svgPointTransform(projection); // Draw markers..
 
-           function quickselect(arr, k, left, right, compare) {
-             quickselectStep(arr, k, left || 0, right || arr.length - 1, compare || defaultCompare);
-           }
+           var markers = drawLayer.selectAll('.qaItem.keepRight').data(data, function (d) {
+             return d.id;
+           }); // exit
 
-           function quickselectStep(arr, k, left, right, compare) {
-             while (right > left) {
-               if (right - left > 600) {
-                 var n = right - left + 1;
-                 var m = k - left + 1;
-                 var z = Math.log(n);
-                 var s = 0.5 * Math.exp(2 * z / 3);
-                 var sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1);
-                 var newLeft = Math.max(left, Math.floor(k - m * s / n + sd));
-                 var newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd));
-                 quickselectStep(arr, k, newLeft, newRight, compare);
-               }
+           markers.exit().remove(); // enter
 
-               var t = arr[k];
-               var i = left;
-               var j = right;
-               swap(arr, left, k);
-               if (compare(arr[right], t) > 0) swap(arr, left, right);
+           var markersEnter = markers.enter().append('g').attr('class', function (d) {
+             return "qaItem ".concat(d.service, " itemId-").concat(d.id, " itemType-").concat(d.parentIssueType);
+           });
+           markersEnter.append('ellipse').attr('cx', 0.5).attr('cy', 1).attr('rx', 6.5).attr('ry', 3).attr('class', 'stroke');
+           markersEnter.append('path').call(markerPath, 'shadow');
+           markersEnter.append('use').attr('class', 'qaItem-fill').attr('width', '20px').attr('height', '20px').attr('x', '-8px').attr('y', '-22px').attr('xlink:href', '#iD-icon-bolt'); // update
 
-               while (i < j) {
-                 swap(arr, i, j);
-                 i++;
-                 j--;
+           markers.merge(markersEnter).sort(sortY).classed('selected', function (d) {
+             return d.id === selectedID;
+           }).attr('transform', getTransform); // Draw targets..
 
-                 while (compare(arr[i], t) < 0) {
-                   i++;
-                 }
+           if (touchLayer.empty()) return;
+           var fillClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
+           var targets = touchLayer.selectAll('.qaItem.keepRight').data(data, function (d) {
+             return d.id;
+           }); // exit
 
-                 while (compare(arr[j], t) > 0) {
-                   j--;
-                 }
-               }
+           targets.exit().remove(); // enter/update
 
-               if (compare(arr[left], t) === 0) swap(arr, left, j);else {
-                 j++;
-                 swap(arr, j, right);
-               }
-               if (j <= k) left = j + 1;
-               if (k <= j) right = j - 1;
-             }
+           targets.enter().append('rect').attr('width', '20px').attr('height', '20px').attr('x', '-8px').attr('y', '-22px').merge(targets).sort(sortY).attr('class', function (d) {
+             return "qaItem ".concat(d.service, " target ").concat(fillClass, " itemId-").concat(d.id);
+           }).attr('transform', getTransform);
+
+           function sortY(a, b) {
+             return a.id === selectedID ? 1 : b.id === selectedID ? -1 : a.severity === 'error' && b.severity !== 'error' ? 1 : b.severity === 'error' && a.severity !== 'error' ? -1 : b.loc[1] - a.loc[1];
            }
+         } // Draw the keepRight layer and schedule loading issues and updating markers.
 
-           function swap(arr, i, j) {
-             var tmp = arr[i];
-             arr[i] = arr[j];
-             arr[j] = tmp;
+
+         function drawKeepRight(selection) {
+           var service = getService();
+           var surface = context.surface();
+
+           if (surface && !surface.empty()) {
+             touchLayer = surface.selectAll('.data-layer.touch .layer-touch.markers');
            }
 
-           function defaultCompare(a, b) {
-             return a < b ? -1 : a > b ? 1 : 0;
+           drawLayer = selection.selectAll('.layer-keepRight').data(service ? [0] : []);
+           drawLayer.exit().remove();
+           drawLayer = drawLayer.enter().append('g').attr('class', 'layer-keepRight').style('display', _layerEnabled$2 ? 'block' : 'none').merge(drawLayer);
+
+           if (_layerEnabled$2) {
+             if (service && ~~context.map().zoom() >= minZoom) {
+               editOn();
+               service.loadIssues(projection);
+               updateMarkers();
+             } else {
+               editOff();
+             }
            }
+         } // Toggles the layer on and off
 
-           return quickselect;
-         });
-       });
 
-       var rbush_1 = rbush;
-       var _default$1 = rbush;
+         drawKeepRight.enabled = function (val) {
+           if (!arguments.length) return _layerEnabled$2;
+           _layerEnabled$2 = val;
 
-       function rbush(maxEntries, format) {
-         if (!(this instanceof rbush)) return new rbush(maxEntries, format); // max entries in a node is 9 by default; min node fill is 40% for best performance
+           if (_layerEnabled$2) {
+             layerOn();
+           } else {
+             layerOff();
 
-         this._maxEntries = Math.max(4, maxEntries || 9);
-         this._minEntries = Math.max(2, Math.ceil(this._maxEntries * 0.4));
+             if (context.selectedErrorID()) {
+               context.enter(modeBrowse(context));
+             }
+           }
 
-         if (format) {
-           this._initFormat(format);
-         }
+           dispatch.call('change');
+           return this;
+         };
 
-         this.clear();
+         drawKeepRight.supported = function () {
+           return !!getService();
+         };
+
+         return drawKeepRight;
        }
 
-       rbush.prototype = {
-         all: function all() {
-           return this._all(this.data, []);
-         },
-         search: function search(bbox) {
-           var node = this.data,
-               result = [],
-               toBBox = this.toBBox;
-           if (!intersects$1(bbox, node)) return result;
-           var nodesToSearch = [],
-               i,
-               len,
-               child,
-               childBBox;
+       function svgGeolocate(projection) {
+         var layer = select(null);
 
-           while (node) {
-             for (i = 0, len = node.children.length; i < len; i++) {
-               child = node.children[i];
-               childBBox = node.leaf ? toBBox(child) : child;
+         var _position;
 
-               if (intersects$1(bbox, childBBox)) {
-                 if (node.leaf) result.push(child);else if (contains$1(bbox, childBBox)) this._all(child, result);else nodesToSearch.push(child);
-               }
-             }
+         function init() {
+           if (svgGeolocate.initialized) return; // run once
 
-             node = nodesToSearch.pop();
-           }
+           svgGeolocate.enabled = false;
+           svgGeolocate.initialized = true;
+         }
 
-           return result;
-         },
-         collides: function collides(bbox) {
-           var node = this.data,
-               toBBox = this.toBBox;
-           if (!intersects$1(bbox, node)) return false;
-           var nodesToSearch = [],
-               i,
-               len,
-               child,
-               childBBox;
+         function showLayer() {
+           layer.style('display', 'block');
+         }
 
-           while (node) {
-             for (i = 0, len = node.children.length; i < len; i++) {
-               child = node.children[i];
-               childBBox = node.leaf ? toBBox(child) : child;
+         function hideLayer() {
+           layer.transition().duration(250).style('opacity', 0);
+         }
 
-               if (intersects$1(bbox, childBBox)) {
-                 if (node.leaf || contains$1(bbox, childBBox)) return true;
-                 nodesToSearch.push(child);
-               }
-             }
+         function layerOn() {
+           layer.style('opacity', 0).transition().duration(250).style('opacity', 1);
+         }
 
-             node = nodesToSearch.pop();
-           }
+         function layerOff() {
+           layer.style('display', 'none');
+         }
 
-           return false;
-         },
-         load: function load(data) {
-           if (!(data && data.length)) return this;
+         function transform(d) {
+           return svgPointTransform(projection)(d);
+         }
 
-           if (data.length < this._minEntries) {
-             for (var i = 0, len = data.length; i < len; i++) {
-               this.insert(data[i]);
-             }
+         function accuracy(accuracy, loc) {
+           // converts accuracy to pixels...
+           var degreesRadius = geoMetersToLat(accuracy),
+               tangentLoc = [loc[0], loc[1] + degreesRadius],
+               projectedTangent = projection(tangentLoc),
+               projectedLoc = projection([loc[0], loc[1]]); // southern most point will have higher pixel value...
 
-             return this;
-           } // recursively build the tree with the given data from scratch using OMT algorithm
+           return Math.round(projectedLoc[1] - projectedTangent[1]).toString();
+         }
 
+         function update() {
+           var geolocation = {
+             loc: [_position.coords.longitude, _position.coords.latitude]
+           };
+           var groups = layer.selectAll('.geolocations').selectAll('.geolocation').data([geolocation]);
+           groups.exit().remove();
+           var pointsEnter = groups.enter().append('g').attr('class', 'geolocation');
+           pointsEnter.append('circle').attr('class', 'geolocate-radius').attr('dx', '0').attr('dy', '0').attr('fill', 'rgb(15,128,225)').attr('fill-opacity', '0.3').attr('r', '0');
+           pointsEnter.append('circle').attr('dx', '0').attr('dy', '0').attr('fill', 'rgb(15,128,225)').attr('stroke', 'white').attr('stroke-width', '1.5').attr('r', '6');
+           groups.merge(pointsEnter).attr('transform', transform);
+           layer.select('.geolocate-radius').attr('r', accuracy(_position.coords.accuracy, geolocation.loc));
+         }
 
-           var node = this._build(data.slice(), 0, data.length - 1, 0);
+         function drawLocation(selection) {
+           var enabled = svgGeolocate.enabled;
+           layer = selection.selectAll('.layer-geolocate').data([0]);
+           layer.exit().remove();
+           var layerEnter = layer.enter().append('g').attr('class', 'layer-geolocate').style('display', enabled ? 'block' : 'none');
+           layerEnter.append('g').attr('class', 'geolocations');
+           layer = layerEnter.merge(layer);
 
-           if (!this.data.children.length) {
-             // save as is if tree is empty
-             this.data = node;
-           } else if (this.data.height === node.height) {
-             // split root if trees have the same height
-             this._splitRoot(this.data, node);
+           if (enabled) {
+             update();
            } else {
-             if (this.data.height < node.height) {
-               // swap trees if inserted one is bigger
-               var tmpNode = this.data;
-               this.data = node;
-               node = tmpNode;
-             } // insert the small tree into the large tree at appropriate level
+             layerOff();
+           }
+         }
 
+         drawLocation.enabled = function (position, enabled) {
+           if (!arguments.length) return svgGeolocate.enabled;
+           _position = position;
+           svgGeolocate.enabled = enabled;
 
-             this._insert(node, this.data.height - node.height - 1, true);
+           if (svgGeolocate.enabled) {
+             showLayer();
+             layerOn();
+           } else {
+             hideLayer();
            }
 
            return this;
-         },
-         insert: function insert(item) {
-           if (item) this._insert(item, this.data.height - 1);
-           return this;
-         },
-         clear: function clear() {
-           this.data = createNode$1([]);
-           return this;
-         },
-         remove: function remove(item, equalsFn) {
-           if (!item) return this;
-           var node = this.data,
-               bbox = this.toBBox(item),
-               path = [],
-               indexes = [],
-               i,
-               parent,
-               index,
-               goingUp; // depth-first iterative tree traversal
+         };
 
-           while (node || path.length) {
-             if (!node) {
-               // go up
-               node = path.pop();
-               parent = path[path.length - 1];
-               i = indexes.pop();
-               goingUp = true;
-             }
+         init();
+         return drawLocation;
+       }
 
-             if (node.leaf) {
-               // check current node
-               index = findItem$1(item, node.children, equalsFn);
+       function svgLabels(projection, context) {
+         var path = d3_geoPath(projection);
+         var detected = utilDetect();
+         var baselineHack = detected.ie || detected.browser.toLowerCase() === 'edge' || detected.browser.toLowerCase() === 'firefox' && detected.version >= 70;
 
-               if (index !== -1) {
-                 // item found, remove the item and condense tree upwards
-                 node.children.splice(index, 1);
-                 path.push(node);
+         var _rdrawn = new RBush();
 
-                 this._condense(path);
+         var _rskipped = new RBush();
 
-                 return this;
-               }
-             }
+         var _textWidthCache = {};
+         var _entitybboxes = {}; // Listed from highest to lowest priority
 
-             if (!goingUp && !node.leaf && contains$1(node, bbox)) {
-               // go down
-               path.push(node);
-               indexes.push(i);
-               i = 0;
-               parent = node;
-               node = node.children[0];
-             } else if (parent) {
-               // go right
-               i++;
-               node = parent.children[i];
-               goingUp = false;
-             } else node = null; // nothing found
+         var labelStack = [['line', 'aeroway', '*', 12], ['line', 'highway', 'motorway', 12], ['line', 'highway', 'trunk', 12], ['line', 'highway', 'primary', 12], ['line', 'highway', 'secondary', 12], ['line', 'highway', 'tertiary', 12], ['line', 'highway', '*', 12], ['line', 'railway', '*', 12], ['line', 'waterway', '*', 12], ['area', 'aeroway', '*', 12], ['area', 'amenity', '*', 12], ['area', 'building', '*', 12], ['area', 'historic', '*', 12], ['area', 'leisure', '*', 12], ['area', 'man_made', '*', 12], ['area', 'natural', '*', 12], ['area', 'shop', '*', 12], ['area', 'tourism', '*', 12], ['area', 'camp_site', '*', 12], ['point', 'aeroway', '*', 10], ['point', 'amenity', '*', 10], ['point', 'building', '*', 10], ['point', 'historic', '*', 10], ['point', 'leisure', '*', 10], ['point', 'man_made', '*', 10], ['point', 'natural', '*', 10], ['point', 'shop', '*', 10], ['point', 'tourism', '*', 10], ['point', 'camp_site', '*', 10], ['line', 'name', '*', 12], ['area', 'name', '*', 12], ['point', 'name', '*', 10]];
 
-           }
+         function shouldSkipIcon(preset) {
+           var noIcons = ['building', 'landuse', 'natural'];
+           return noIcons.some(function (s) {
+             return preset.id.indexOf(s) >= 0;
+           });
+         }
 
-           return this;
-         },
-         toBBox: function toBBox(item) {
-           return item;
-         },
-         compareMinX: compareNodeMinX$1,
-         compareMinY: compareNodeMinY$1,
-         toJSON: function toJSON() {
-           return this.data;
-         },
-         fromJSON: function fromJSON(data) {
-           this.data = data;
-           return this;
-         },
-         _all: function _all(node, result) {
-           var nodesToSearch = [];
+         function get(array, prop) {
+           return function (d, i) {
+             return array[i][prop];
+           };
+         }
 
-           while (node) {
-             if (node.leaf) result.push.apply(result, node.children);else nodesToSearch.push.apply(nodesToSearch, node.children);
-             node = nodesToSearch.pop();
+         function textWidth(text, size, elem) {
+           var c = _textWidthCache[size];
+           if (!c) c = _textWidthCache[size] = {};
+
+           if (c[text]) {
+             return c[text];
+           } else if (elem) {
+             c[text] = elem.getComputedTextLength();
+             return c[text];
+           } else {
+             var str = encodeURIComponent(text).match(/%[CDEFcdef]/g);
+
+             if (str === null) {
+               return size / 3 * 2 * text.length;
+             } else {
+               return size / 3 * (2 * text.length + str.length);
+             }
            }
+         }
 
-           return result;
-         },
-         _build: function _build(items, left, right, height) {
-           var N = right - left + 1,
-               M = this._maxEntries,
-               node;
+         function drawLinePaths(selection, entities, filter, classes, labels) {
+           var paths = selection.selectAll('path').filter(filter).data(entities, osmEntity.key); // exit
 
-           if (N <= M) {
-             // reached leaf level; return leaf
-             node = createNode$1(items.slice(left, right + 1));
-             calcBBox$1(node, this.toBBox);
-             return node;
-           }
+           paths.exit().remove(); // enter/update
 
-           if (!height) {
-             // target height of the bulk-loaded tree
-             height = Math.ceil(Math.log(N) / Math.log(M)); // target number of root entries to maximize storage utilization
+           paths.enter().append('path').style('stroke-width', get(labels, 'font-size')).attr('id', function (d) {
+             return 'ideditor-labelpath-' + d.id;
+           }).attr('class', classes).merge(paths).attr('d', get(labels, 'lineString'));
+         }
 
-             M = Math.ceil(N / Math.pow(M, height - 1));
-           }
+         function drawLineLabels(selection, entities, filter, classes, labels) {
+           var texts = selection.selectAll('text.' + classes).filter(filter).data(entities, osmEntity.key); // exit
 
-           node = createNode$1([]);
-           node.leaf = false;
-           node.height = height; // split the items into M mostly square tiles
+           texts.exit().remove(); // enter
 
-           var N2 = Math.ceil(N / M),
-               N1 = N2 * Math.ceil(Math.sqrt(M)),
-               i,
-               j,
-               right2,
-               right3;
-           multiSelect$1(items, left, right, N1, this.compareMinX);
+           texts.enter().append('text').attr('class', function (d, i) {
+             return classes + ' ' + labels[i].classes + ' ' + d.id;
+           }).attr('dy', baselineHack ? '0.35em' : null).append('textPath').attr('class', 'textpath'); // update
 
-           for (i = left; i <= right; i += N1) {
-             right2 = Math.min(i + N1 - 1, right);
-             multiSelect$1(items, i, right2, N2, this.compareMinY);
+           selection.selectAll('text.' + classes).selectAll('.textpath').filter(filter).data(entities, osmEntity.key).attr('startOffset', '50%').attr('xlink:href', function (d) {
+             return '#ideditor-labelpath-' + d.id;
+           }).text(utilDisplayNameForPath);
+         }
 
-             for (j = i; j <= right2; j += N2) {
-               right3 = Math.min(j + N2 - 1, right2); // pack each entry recursively
+         function drawPointLabels(selection, entities, filter, classes, labels) {
+           var texts = selection.selectAll('text.' + classes).filter(filter).data(entities, osmEntity.key); // exit
 
-               node.children.push(this._build(items, j, right3, height - 1));
-             }
+           texts.exit().remove(); // enter/update
+
+           texts.enter().append('text').attr('class', function (d, i) {
+             return classes + ' ' + labels[i].classes + ' ' + d.id;
+           }).merge(texts).attr('x', get(labels, 'x')).attr('y', get(labels, 'y')).style('text-anchor', get(labels, 'textAnchor')).text(utilDisplayName).each(function (d, i) {
+             textWidth(utilDisplayName(d), labels[i].height, this);
+           });
+         }
+
+         function drawAreaLabels(selection, entities, filter, classes, labels) {
+           entities = entities.filter(hasText);
+           labels = labels.filter(hasText);
+           drawPointLabels(selection, entities, filter, classes, labels);
+
+           function hasText(d, i) {
+             return labels[i].hasOwnProperty('x') && labels[i].hasOwnProperty('y');
            }
+         }
 
-           calcBBox$1(node, this.toBBox);
-           return node;
-         },
-         _chooseSubtree: function _chooseSubtree(bbox, node, level, path) {
-           var i, len, child, targetNode, area, enlargement, minArea, minEnlargement;
+         function drawAreaIcons(selection, entities, filter, classes, labels) {
+           var icons = selection.selectAll('use.' + classes).filter(filter).data(entities, osmEntity.key); // exit
 
-           while (true) {
-             path.push(node);
-             if (node.leaf || path.length - 1 === level) break;
-             minArea = minEnlargement = Infinity;
+           icons.exit().remove(); // enter/update
 
-             for (i = 0, len = node.children.length; i < len; i++) {
-               child = node.children[i];
-               area = bboxArea$1(child);
-               enlargement = enlargedArea$1(bbox, child) - area; // choose entry with the least area enlargement
+           icons.enter().append('use').attr('class', 'icon ' + classes).attr('width', '17px').attr('height', '17px').merge(icons).attr('transform', get(labels, 'transform')).attr('xlink:href', function (d) {
+             var preset = _mainPresetIndex.match(d, context.graph());
+             var picon = preset && preset.icon;
 
-               if (enlargement < minEnlargement) {
-                 minEnlargement = enlargement;
-                 minArea = area < minArea ? area : minArea;
-                 targetNode = child;
-               } else if (enlargement === minEnlargement) {
-                 // otherwise choose one with the smallest area
-                 if (area < minArea) {
-                   minArea = area;
-                   targetNode = child;
-                 }
-               }
+             if (!picon) {
+               return '';
+             } else {
+               var isMaki = /^maki-/.test(picon);
+               return '#' + picon + (isMaki ? '-15' : '');
              }
+           });
+         }
 
-             node = targetNode || node.children[0];
+         function drawCollisionBoxes(selection, rtree, which) {
+           var classes = 'debug ' + which + ' ' + (which === 'debug-skipped' ? 'orange' : 'yellow');
+           var gj = [];
+
+           if (context.getDebug('collision')) {
+             gj = rtree.all().map(function (d) {
+               return {
+                 type: 'Polygon',
+                 coordinates: [[[d.minX, d.minY], [d.maxX, d.minY], [d.maxX, d.maxY], [d.minX, d.maxY], [d.minX, d.minY]]]
+               };
+             });
            }
 
-           return node;
-         },
-         _insert: function _insert(item, level, isNode) {
-           var toBBox = this.toBBox,
-               bbox = isNode ? item : toBBox(item),
-               insertPath = []; // find the best node for accommodating the item, saving all nodes along the path too
+           var boxes = selection.selectAll('.' + which).data(gj); // exit
 
-           var node = this._chooseSubtree(bbox, this.data, level, insertPath); // put the item into the node
+           boxes.exit().remove(); // enter/update
 
+           boxes.enter().append('path').attr('class', classes).merge(boxes).attr('d', d3_geoPath());
+         }
 
-           node.children.push(item);
-           extend$3(node, bbox); // split on node overflow; propagate upwards if necessary
+         function drawLabels(selection, graph, entities, filter, dimensions, fullRedraw) {
+           var wireframe = context.surface().classed('fill-wireframe');
+           var zoom = geoScaleToZoom(projection.scale());
+           var labelable = [];
+           var renderNodeAs = {};
+           var i, j, k, entity, geometry;
 
-           while (level >= 0) {
-             if (insertPath[level].children.length > this._maxEntries) {
-               this._split(insertPath, level);
+           for (i = 0; i < labelStack.length; i++) {
+             labelable.push([]);
+           }
 
-               level--;
-             } else break;
-           } // adjust bboxes along the insertion path
+           if (fullRedraw) {
+             _rdrawn.clear();
 
+             _rskipped.clear();
 
-           this._adjustParentBBoxes(bbox, insertPath, level);
-         },
-         // split overflowed node into two
-         _split: function _split(insertPath, level) {
-           var node = insertPath[level],
-               M = node.children.length,
-               m = this._minEntries;
+             _entitybboxes = {};
+           } else {
+             for (i = 0; i < entities.length; i++) {
+               entity = entities[i];
+               var toRemove = [].concat(_entitybboxes[entity.id] || []).concat(_entitybboxes[entity.id + 'I'] || []);
 
-           this._chooseSplitAxis(node, m, M);
+               for (j = 0; j < toRemove.length; j++) {
+                 _rdrawn.remove(toRemove[j]);
 
-           var splitIndex = this._chooseSplitIndex(node, m, M);
+                 _rskipped.remove(toRemove[j]);
+               }
+             }
+           } // Loop through all the entities to do some preprocessing
 
-           var newNode = createNode$1(node.children.splice(splitIndex, node.children.length - splitIndex));
-           newNode.height = node.height;
-           newNode.leaf = node.leaf;
-           calcBBox$1(node, this.toBBox);
-           calcBBox$1(newNode, this.toBBox);
-           if (level) insertPath[level - 1].children.push(newNode);else this._splitRoot(node, newNode);
-         },
-         _splitRoot: function _splitRoot(node, newNode) {
-           // split root node
-           this.data = createNode$1([node, newNode]);
-           this.data.height = node.height + 1;
-           this.data.leaf = false;
-           calcBBox$1(this.data, this.toBBox);
-         },
-         _chooseSplitIndex: function _chooseSplitIndex(node, m, M) {
-           var i, bbox1, bbox2, overlap, area, minOverlap, minArea, index;
-           minOverlap = minArea = Infinity;
 
-           for (i = m; i <= M - m; i++) {
-             bbox1 = distBBox$1(node, 0, i, this.toBBox);
-             bbox2 = distBBox$1(node, i, M, this.toBBox);
-             overlap = intersectionArea$1(bbox1, bbox2);
-             area = bboxArea$1(bbox1) + bboxArea$1(bbox2); // choose distribution with minimum overlap
+           for (i = 0; i < entities.length; i++) {
+             entity = entities[i];
+             geometry = entity.geometry(graph); // Insert collision boxes around interesting points/vertices
 
-             if (overlap < minOverlap) {
-               minOverlap = overlap;
-               index = i;
-               minArea = area < minArea ? area : minArea;
-             } else if (overlap === minOverlap) {
-               // otherwise choose distribution with minimum area
-               if (area < minArea) {
-                 minArea = area;
-                 index = i;
+             if (geometry === 'point' || geometry === 'vertex' && isInterestingVertex(entity)) {
+               var hasDirections = entity.directions(graph, projection).length;
+               var markerPadding;
+
+               if (!wireframe && geometry === 'point' && !(zoom >= 18 && hasDirections)) {
+                 renderNodeAs[entity.id] = 'point';
+                 markerPadding = 20; // extra y for marker height
+               } else {
+                 renderNodeAs[entity.id] = 'vertex';
+                 markerPadding = 0;
                }
-             }
-           }
 
-           return index;
-         },
-         // sorts node children by the best axis for split
-         _chooseSplitAxis: function _chooseSplitAxis(node, m, M) {
-           var compareMinX = node.leaf ? this.compareMinX : compareNodeMinX$1,
-               compareMinY = node.leaf ? this.compareMinY : compareNodeMinY$1,
-               xMargin = this._allDistMargin(node, m, M, compareMinX),
-               yMargin = this._allDistMargin(node, m, M, compareMinY); // if total distributions margin value is minimal for x, sort by minX,
-           // otherwise it's already sorted by minY
+               var coord = projection(entity.loc);
+               var nodePadding = 10;
+               var bbox = {
+                 minX: coord[0] - nodePadding,
+                 minY: coord[1] - nodePadding - markerPadding,
+                 maxX: coord[0] + nodePadding,
+                 maxY: coord[1] + nodePadding
+               };
+               doInsert(bbox, entity.id + 'P');
+             } // From here on, treat vertices like points
 
 
-           if (xMargin < yMargin) node.children.sort(compareMinX);
-         },
-         // total margin of all possible split distributions where each node is at least m full
-         _allDistMargin: function _allDistMargin(node, m, M, compare) {
-           node.children.sort(compare);
-           var toBBox = this.toBBox,
-               leftBBox = distBBox$1(node, 0, m, toBBox),
-               rightBBox = distBBox$1(node, M - m, M, toBBox),
-               margin = bboxMargin$1(leftBBox) + bboxMargin$1(rightBBox),
-               i,
-               child;
+             if (geometry === 'vertex') {
+               geometry = 'point';
+             } // Determine which entities are label-able
 
-           for (i = m; i < M - m; i++) {
-             child = node.children[i];
-             extend$3(leftBBox, node.leaf ? toBBox(child) : child);
-             margin += bboxMargin$1(leftBBox);
-           }
 
-           for (i = M - m - 1; i >= m; i--) {
-             child = node.children[i];
-             extend$3(rightBBox, node.leaf ? toBBox(child) : child);
-             margin += bboxMargin$1(rightBBox);
-           }
+             var preset = geometry === 'area' && _mainPresetIndex.match(entity, graph);
+             var icon = preset && !shouldSkipIcon(preset) && preset.icon;
+             if (!icon && !utilDisplayName(entity)) continue;
 
-           return margin;
-         },
-         _adjustParentBBoxes: function _adjustParentBBoxes(bbox, path, level) {
-           // adjust bboxes along the given tree path
-           for (var i = level; i >= 0; i--) {
-             extend$3(path[i], bbox);
-           }
-         },
-         _condense: function _condense(path) {
-           // go through the path, removing empty nodes and updating bboxes
-           for (var i = path.length - 1, siblings; i >= 0; i--) {
-             if (path[i].children.length === 0) {
-               if (i > 0) {
-                 siblings = path[i - 1].children;
-                 siblings.splice(siblings.indexOf(path[i]), 1);
-               } else this.clear();
-             } else calcBBox$1(path[i], this.toBBox);
+             for (k = 0; k < labelStack.length; k++) {
+               var matchGeom = labelStack[k][0];
+               var matchKey = labelStack[k][1];
+               var matchVal = labelStack[k][2];
+               var hasVal = entity.tags[matchKey];
+
+               if (geometry === matchGeom && hasVal && (matchVal === '*' || matchVal === hasVal)) {
+                 labelable[k].push(entity);
+                 break;
+               }
+             }
            }
-         },
-         _initFormat: function _initFormat(format) {
-           // data format (minX, minY, maxX, maxY accessors)
-           // uses eval-type function compilation instead of just accepting a toBBox function
-           // because the algorithms are very sensitive to sorting functions performance,
-           // so they should be dead simple and without inner calls
-           var compareArr = ['return a', ' - b', ';'];
-           this.compareMinX = new Function('a', 'b', compareArr.join(format[0]));
-           this.compareMinY = new Function('a', 'b', compareArr.join(format[1]));
-           this.toBBox = new Function('a', 'return {minX: a' + format[0] + ', minY: a' + format[1] + ', maxX: a' + format[2] + ', maxY: a' + format[3] + '};');
-         }
-       };
 
-       function findItem$1(item, items, equalsFn) {
-         if (!equalsFn) return items.indexOf(item);
+           var positions = {
+             point: [],
+             line: [],
+             area: []
+           };
+           var labelled = {
+             point: [],
+             line: [],
+             area: []
+           }; // Try and find a valid label for labellable entities
 
-         for (var i = 0; i < items.length; i++) {
-           if (equalsFn(item, items[i])) return i;
-         }
+           for (k = 0; k < labelable.length; k++) {
+             var fontSize = labelStack[k][3];
 
-         return -1;
-       } // calculate node's bbox from bboxes of its children
+             for (i = 0; i < labelable[k].length; i++) {
+               entity = labelable[k][i];
+               geometry = entity.geometry(graph);
+               var getName = geometry === 'line' ? utilDisplayNameForPath : utilDisplayName;
+               var name = getName(entity);
+               var width = name && textWidth(name, fontSize);
+               var p = null;
 
+               if (geometry === 'point' || geometry === 'vertex') {
+                 // no point or vertex labels in wireframe mode
+                 // no vertex labels at low zooms (vertices have no icons)
+                 if (wireframe) continue;
+                 var renderAs = renderNodeAs[entity.id];
+                 if (renderAs === 'vertex' && zoom < 17) continue;
+                 p = getPointLabel(entity, width, fontSize, renderAs);
+               } else if (geometry === 'line') {
+                 p = getLineLabel(entity, width, fontSize);
+               } else if (geometry === 'area') {
+                 p = getAreaLabel(entity, width, fontSize);
+               }
 
-       function calcBBox$1(node, toBBox) {
-         distBBox$1(node, 0, node.children.length, toBBox, node);
-       } // min bounding rectangle of node children from k to p-1
+               if (p) {
+                 if (geometry === 'vertex') {
+                   geometry = 'point';
+                 } // treat vertex like point
 
 
-       function distBBox$1(node, k, p, toBBox, destNode) {
-         if (!destNode) destNode = createNode$1(null);
-         destNode.minX = Infinity;
-         destNode.minY = Infinity;
-         destNode.maxX = -Infinity;
-         destNode.maxY = -Infinity;
+                 p.classes = geometry + ' tag-' + labelStack[k][1];
+                 positions[geometry].push(p);
+                 labelled[geometry].push(entity);
+               }
+             }
+           }
 
-         for (var i = k, child; i < p; i++) {
-           child = node.children[i];
-           extend$3(destNode, node.leaf ? toBBox(child) : child);
-         }
+           function isInterestingVertex(entity) {
+             var selectedIDs = context.selectedIDs();
+             return entity.hasInterestingTags() || entity.isEndpoint(graph) || entity.isConnected(graph) || selectedIDs.indexOf(entity.id) !== -1 || graph.parentWays(entity).some(function (parent) {
+               return selectedIDs.indexOf(parent.id) !== -1;
+             });
+           }
 
-         return destNode;
-       }
+           function getPointLabel(entity, width, height, geometry) {
+             var y = geometry === 'point' ? -12 : 0;
+             var pointOffsets = {
+               ltr: [15, y, 'start'],
+               rtl: [-15, y, 'end']
+             };
+             var textDirection = _mainLocalizer.textDirection();
+             var coord = projection(entity.loc);
+             var textPadding = 2;
+             var offset = pointOffsets[textDirection];
+             var p = {
+               height: height,
+               width: width,
+               x: coord[0] + offset[0],
+               y: coord[1] + offset[1],
+               textAnchor: offset[2]
+             }; // insert a collision box for the text label..
 
-       function extend$3(a, b) {
-         a.minX = Math.min(a.minX, b.minX);
-         a.minY = Math.min(a.minY, b.minY);
-         a.maxX = Math.max(a.maxX, b.maxX);
-         a.maxY = Math.max(a.maxY, b.maxY);
-         return a;
-       }
+             var bbox;
 
-       function compareNodeMinX$1(a, b) {
-         return a.minX - b.minX;
-       }
+             if (textDirection === 'rtl') {
+               bbox = {
+                 minX: p.x - width - textPadding,
+                 minY: p.y - height / 2 - textPadding,
+                 maxX: p.x + textPadding,
+                 maxY: p.y + height / 2 + textPadding
+               };
+             } else {
+               bbox = {
+                 minX: p.x - textPadding,
+                 minY: p.y - height / 2 - textPadding,
+                 maxX: p.x + width + textPadding,
+                 maxY: p.y + height / 2 + textPadding
+               };
+             }
 
-       function compareNodeMinY$1(a, b) {
-         return a.minY - b.minY;
-       }
+             if (tryInsert([bbox], entity.id, true)) {
+               return p;
+             }
+           }
 
-       function bboxArea$1(a) {
-         return (a.maxX - a.minX) * (a.maxY - a.minY);
-       }
+           function getLineLabel(entity, width, height) {
+             var viewport = geoExtent(context.projection.clipExtent()).polygon();
+             var points = graph.childNodes(entity).map(function (node) {
+               return projection(node.loc);
+             });
+             var length = geoPathLength(points);
+             if (length < width + 20) return; // % along the line to attempt to place the label
 
-       function bboxMargin$1(a) {
-         return a.maxX - a.minX + (a.maxY - a.minY);
-       }
+             var lineOffsets = [50, 45, 55, 40, 60, 35, 65, 30, 70, 25, 75, 20, 80, 15, 95, 10, 90, 5, 95];
+             var padding = 3;
 
-       function enlargedArea$1(a, b) {
-         return (Math.max(b.maxX, a.maxX) - Math.min(b.minX, a.minX)) * (Math.max(b.maxY, a.maxY) - Math.min(b.minY, a.minY));
-       }
+             for (var i = 0; i < lineOffsets.length; i++) {
+               var offset = lineOffsets[i];
+               var middle = offset / 100 * length;
+               var start = middle - width / 2;
+               if (start < 0 || start + width > length) continue; // generate subpath and ignore paths that are invalid or don't cross viewport.
 
-       function intersectionArea$1(a, b) {
-         var minX = Math.max(a.minX, b.minX),
-             minY = Math.max(a.minY, b.minY),
-             maxX = Math.min(a.maxX, b.maxX),
-             maxY = Math.min(a.maxY, b.maxY);
-         return Math.max(0, maxX - minX) * Math.max(0, maxY - minY);
-       }
+               var sub = subpath(points, start, start + width);
 
-       function contains$1(a, b) {
-         return a.minX <= b.minX && a.minY <= b.minY && b.maxX <= a.maxX && b.maxY <= a.maxY;
-       }
+               if (!sub || !geoPolygonIntersectsPolygon(viewport, sub, true)) {
+                 continue;
+               }
 
-       function intersects$1(a, b) {
-         return b.minX <= a.maxX && b.minY <= a.maxY && b.maxX >= a.minX && b.maxY >= a.minY;
-       }
+               var isReverse = reverse(sub);
 
-       function createNode$1(children) {
-         return {
-           children: children,
-           height: 1,
-           leaf: true,
-           minX: Infinity,
-           minY: Infinity,
-           maxX: -Infinity,
-           maxY: -Infinity
-         };
-       } // sort an array so that items come in groups of n unsorted items, with groups sorted between each other;
-       // combines selection algorithm with binary divide & conquer approach
+               if (isReverse) {
+                 sub = sub.reverse();
+               }
 
+               var bboxes = [];
+               var boxsize = (height + 2) / 2;
 
-       function multiSelect$1(arr, left, right, n, compare) {
-         var stack = [left, right],
-             mid;
+               for (var j = 0; j < sub.length - 1; j++) {
+                 var a = sub[j];
+                 var b = sub[j + 1]; // split up the text into small collision boxes
 
-         while (stack.length) {
-           right = stack.pop();
-           left = stack.pop();
-           if (right - left <= n) continue;
-           mid = left + Math.ceil((right - left) / n / 2) * n;
-           quickselect$2(arr, mid, left, right, compare);
-           stack.push(left, mid, mid, right);
-         }
-       }
-       rbush_1["default"] = _default$1;
+                 var num = Math.max(1, Math.floor(geoVecLength(a, b) / boxsize / 2));
+
+                 for (var box = 0; box < num; box++) {
+                   var p = geoVecInterp(a, b, box / num);
+                   var x0 = p[0] - boxsize - padding;
+                   var y0 = p[1] - boxsize - padding;
+                   var x1 = p[0] + boxsize + padding;
+                   var y1 = p[1] + boxsize + padding;
+                   bboxes.push({
+                     minX: Math.min(x0, x1),
+                     minY: Math.min(y0, y1),
+                     maxX: Math.max(x0, x1),
+                     maxY: Math.max(y0, y1)
+                   });
+                 }
+               }
+
+               if (tryInsert(bboxes, entity.id, false)) {
+                 // accept this one
+                 return {
+                   'font-size': height + 2,
+                   lineString: lineString(sub),
+                   startOffset: offset + '%'
+                 };
+               }
+             }
 
-       var lineclip_1 = lineclip$1;
-       lineclip$1.polyline = lineclip$1;
-       lineclip$1.polygon = polygonclip$1; // Cohen-Sutherland line clippign algorithm, adapted to efficiently
-       // handle polylines rather than just segments
+             function reverse(p) {
+               var angle = Math.atan2(p[1][1] - p[0][1], p[1][0] - p[0][0]);
+               return !(p[0][0] < p[p.length - 1][0] && angle < Math.PI / 2 && angle > -Math.PI / 2);
+             }
 
-       function lineclip$1(points, bbox, result) {
-         var len = points.length,
-             codeA = bitCode$1(points[0], bbox),
-             part = [],
-             i,
-             a,
-             b,
-             codeB,
-             lastCode;
-         if (!result) result = [];
+             function lineString(points) {
+               return 'M' + points.join('L');
+             }
 
-         for (i = 1; i < len; i++) {
-           a = points[i - 1];
-           b = points[i];
-           codeB = lastCode = bitCode$1(b, bbox);
+             function subpath(points, from, to) {
+               var sofar = 0;
+               var start, end, i0, i1;
 
-           while (true) {
-             if (!(codeA | codeB)) {
-               // accept
-               part.push(a);
+               for (var i = 0; i < points.length - 1; i++) {
+                 var a = points[i];
+                 var b = points[i + 1];
+                 var current = geoVecLength(a, b);
+                 var portion;
 
-               if (codeB !== lastCode) {
-                 // segment went outside
-                 part.push(b);
+                 if (!start && sofar + current >= from) {
+                   portion = (from - sofar) / current;
+                   start = [a[0] + portion * (b[0] - a[0]), a[1] + portion * (b[1] - a[1])];
+                   i0 = i + 1;
+                 }
 
-                 if (i < len - 1) {
-                   // start a new line
-                   result.push(part);
-                   part = [];
+                 if (!end && sofar + current >= to) {
+                   portion = (to - sofar) / current;
+                   end = [a[0] + portion * (b[0] - a[0]), a[1] + portion * (b[1] - a[1])];
+                   i1 = i + 1;
                  }
-               } else if (i === len - 1) {
-                 part.push(b);
+
+                 sofar += current;
                }
 
-               break;
-             } else if (codeA & codeB) {
-               // trivial reject
-               break;
-             } else if (codeA) {
-               // a outside, intersect with clip edge
-               a = intersect$1(a, b, codeA, bbox);
-               codeA = bitCode$1(a, bbox);
-             } else {
-               // b outside
-               b = intersect$1(a, b, codeB, bbox);
-               codeB = bitCode$1(b, bbox);
+               var result = points.slice(i0, i1);
+               result.unshift(start);
+               result.push(end);
+               return result;
              }
            }
 
-           codeA = lastCode;
-         }
-
-         if (part.length) result.push(part);
-         return result;
-       } // Sutherland-Hodgeman polygon clipping algorithm
+           function getAreaLabel(entity, width, height) {
+             var centroid = path.centroid(entity.asGeoJSON(graph));
+             var extent = entity.extent(graph);
+             var areaWidth = projection(extent[1])[0] - projection(extent[0])[0];
+             if (isNaN(centroid[0]) || areaWidth < 20) return;
+             var preset = _mainPresetIndex.match(entity, context.graph());
+             var picon = preset && preset.icon;
+             var iconSize = 17;
+             var padding = 2;
+             var p = {};
 
+             if (picon) {
+               // icon and label..
+               if (addIcon()) {
+                 addLabel(iconSize + padding);
+                 return p;
+               }
+             } else {
+               // label only..
+               if (addLabel(0)) {
+                 return p;
+               }
+             }
 
-       function polygonclip$1(points, bbox) {
-         var result, edge, prev, prevInside, i, p, inside; // clip against each side of the clip rectangle
+             function addIcon() {
+               var iconX = centroid[0] - iconSize / 2;
+               var iconY = centroid[1] - iconSize / 2;
+               var bbox = {
+                 minX: iconX,
+                 minY: iconY,
+                 maxX: iconX + iconSize,
+                 maxY: iconY + iconSize
+               };
 
-         for (edge = 1; edge <= 8; edge *= 2) {
-           result = [];
-           prev = points[points.length - 1];
-           prevInside = !(bitCode$1(prev, bbox) & edge);
+               if (tryInsert([bbox], entity.id + 'I', true)) {
+                 p.transform = 'translate(' + iconX + ',' + iconY + ')';
+                 return true;
+               }
 
-           for (i = 0; i < points.length; i++) {
-             p = points[i];
-             inside = !(bitCode$1(p, bbox) & edge); // if segment goes through the clip window, add an intersection
+               return false;
+             }
 
-             if (inside !== prevInside) result.push(intersect$1(prev, p, edge, bbox));
-             if (inside) result.push(p); // add a point if it's inside
+             function addLabel(yOffset) {
+               if (width && areaWidth >= width + 20) {
+                 var labelX = centroid[0];
+                 var labelY = centroid[1] + yOffset;
+                 var bbox = {
+                   minX: labelX - width / 2 - padding,
+                   minY: labelY - height / 2 - padding,
+                   maxX: labelX + width / 2 + padding,
+                   maxY: labelY + height / 2 + padding
+                 };
 
-             prev = p;
-             prevInside = inside;
-           }
+                 if (tryInsert([bbox], entity.id, true)) {
+                   p.x = labelX;
+                   p.y = labelY;
+                   p.textAnchor = 'middle';
+                   p.height = height;
+                   return true;
+                 }
+               }
 
-           points = result;
-           if (!points.length) break;
-         }
+               return false;
+             }
+           } // force insert a singular bounding box
+           // singular box only, no array, id better be unique
 
-         return result;
-       } // intersect a segment against one of the 4 lines that make up the bbox
 
+           function doInsert(bbox, id) {
+             bbox.id = id;
+             var oldbox = _entitybboxes[id];
 
-       function intersect$1(a, b, edge, bbox) {
-         return edge & 8 ? [a[0] + (b[0] - a[0]) * (bbox[3] - a[1]) / (b[1] - a[1]), bbox[3]] : // top
-         edge & 4 ? [a[0] + (b[0] - a[0]) * (bbox[1] - a[1]) / (b[1] - a[1]), bbox[1]] : // bottom
-         edge & 2 ? [bbox[2], a[1] + (b[1] - a[1]) * (bbox[2] - a[0]) / (b[0] - a[0])] : // right
-         edge & 1 ? [bbox[0], a[1] + (b[1] - a[1]) * (bbox[0] - a[0]) / (b[0] - a[0])] : // left
-         null;
-       } // bit code reflects the point position relative to the bbox:
-       //         left  mid  right
-       //    top  1001  1000  1010
-       //    mid  0001  0000  0010
-       // bottom  0101  0100  0110
+             if (oldbox) {
+               _rdrawn.remove(oldbox);
+             }
 
+             _entitybboxes[id] = bbox;
 
-       function bitCode$1(p, bbox) {
-         var code = 0;
-         if (p[0] < bbox[0]) code |= 1; // left
-         else if (p[0] > bbox[2]) code |= 2; // right
+             _rdrawn.insert(bbox);
+           }
 
-         if (p[1] < bbox[1]) code |= 4; // bottom
-         else if (p[1] > bbox[3]) code |= 8; // top
+           function tryInsert(bboxes, id, saveSkipped) {
+             var skipped = false;
 
-         return code;
-       }
+             for (var i = 0; i < bboxes.length; i++) {
+               var bbox = bboxes[i];
+               bbox.id = id; // Check that label is visible
 
-       var whichPolygon_1 = whichPolygon;
+               if (bbox.minX < 0 || bbox.minY < 0 || bbox.maxX > dimensions[0] || bbox.maxY > dimensions[1]) {
+                 skipped = true;
+                 break;
+               }
 
-       function whichPolygon(data) {
-         var bboxes = [];
+               if (_rdrawn.collides(bbox)) {
+                 skipped = true;
+                 break;
+               }
+             }
 
-         for (var i = 0; i < data.features.length; i++) {
-           var feature = data.features[i];
-           var coords = feature.geometry.coordinates;
+             _entitybboxes[id] = bboxes;
 
-           if (feature.geometry.type === 'Polygon') {
-             bboxes.push(treeItem(coords, feature.properties));
-           } else if (feature.geometry.type === 'MultiPolygon') {
-             for (var j = 0; j < coords.length; j++) {
-               bboxes.push(treeItem(coords[j], feature.properties));
+             if (skipped) {
+               if (saveSkipped) {
+                 _rskipped.load(bboxes);
+               }
+             } else {
+               _rdrawn.load(bboxes);
              }
-           }
-         }
 
-         var tree = rbush_1().load(bboxes);
+             return !skipped;
+           }
 
-         function query(p, multi) {
-           var output = [],
-               result = tree.search({
-             minX: p[0],
-             minY: p[1],
-             maxX: p[0],
-             maxY: p[1]
+           var layer = selection.selectAll('.layer-osm.labels');
+           layer.selectAll('.labels-group').data(['halo', 'label', 'debug']).enter().append('g').attr('class', function (d) {
+             return 'labels-group ' + d;
            });
+           var halo = layer.selectAll('.labels-group.halo');
+           var label = layer.selectAll('.labels-group.label');
+           var debug = layer.selectAll('.labels-group.debug'); // points
 
-           for (var i = 0; i < result.length; i++) {
-             if (insidePolygon(result[i].coords, p)) {
-               if (multi) output.push(result[i].props);else return result[i].props;
-             }
-           }
+           drawPointLabels(label, labelled.point, filter, 'pointlabel', positions.point);
+           drawPointLabels(halo, labelled.point, filter, 'pointlabel-halo', positions.point); // lines
 
-           return multi && output.length ? output : null;
-         }
+           drawLinePaths(layer, labelled.line, filter, '', positions.line);
+           drawLineLabels(label, labelled.line, filter, 'linelabel', positions.line);
+           drawLineLabels(halo, labelled.line, filter, 'linelabel-halo', positions.line); // areas
 
-         query.tree = tree;
+           drawAreaLabels(label, labelled.area, filter, 'arealabel', positions.area);
+           drawAreaLabels(halo, labelled.area, filter, 'arealabel-halo', positions.area);
+           drawAreaIcons(label, labelled.area, filter, 'areaicon', positions.area);
+           drawAreaIcons(halo, labelled.area, filter, 'areaicon-halo', positions.area); // debug
 
-         query.bbox = function queryBBox(bbox) {
-           var output = [];
-           var result = tree.search({
-             minX: bbox[0],
-             minY: bbox[1],
-             maxX: bbox[2],
-             maxY: bbox[3]
-           });
+           drawCollisionBoxes(debug, _rskipped, 'debug-skipped');
+           drawCollisionBoxes(debug, _rdrawn, 'debug-drawn');
+           layer.call(filterLabels);
+         }
 
-           for (var i = 0; i < result.length; i++) {
-             if (polygonIntersectsBBox(result[i].coords, bbox)) {
-               output.push(result[i].props);
-             }
-           }
+         function filterLabels(selection) {
+           var drawLayer = selection.selectAll('.layer-osm.labels');
+           var layers = drawLayer.selectAll('.labels-group.halo, .labels-group.label');
+           layers.selectAll('.nolabel').classed('nolabel', false);
+           var mouse = context.map().mouse();
+           var graph = context.graph();
+           var selectedIDs = context.selectedIDs();
+           var ids = [];
+           var pad, bbox; // hide labels near the mouse
 
-           return output;
-         };
+           if (mouse) {
+             pad = 20;
+             bbox = {
+               minX: mouse[0] - pad,
+               minY: mouse[1] - pad,
+               maxX: mouse[0] + pad,
+               maxY: mouse[1] + pad
+             };
 
-         return query;
-       }
+             var nearMouse = _rdrawn.search(bbox).map(function (entity) {
+               return entity.id;
+             });
 
-       function polygonIntersectsBBox(polygon, bbox) {
-         var bboxCenter = [(bbox[0] + bbox[2]) / 2, (bbox[1] + bbox[3]) / 2];
-         if (insidePolygon(polygon, bboxCenter)) return true;
+             ids.push.apply(ids, nearMouse);
+           } // hide labels on selected nodes (they look weird when dragging / haloed)
 
-         for (var i = 0; i < polygon.length; i++) {
-           if (lineclip_1(polygon[i], bbox).length > 0) return true;
-         }
 
-         return false;
-       } // ray casting algorithm for detecting if point is in polygon
+           for (var i = 0; i < selectedIDs.length; i++) {
+             var entity = graph.hasEntity(selectedIDs[i]);
 
+             if (entity && entity.type === 'node') {
+               ids.push(selectedIDs[i]);
+             }
+           }
 
-       function insidePolygon(rings, p) {
-         var inside = false;
+           layers.selectAll(utilEntitySelector(ids)).classed('nolabel', true); // draw the mouse bbox if debugging is on..
 
-         for (var i = 0, len = rings.length; i < len; i++) {
-           var ring = rings[i];
+           var debug = selection.selectAll('.labels-group.debug');
+           var gj = [];
 
-           for (var j = 0, len2 = ring.length, k = len2 - 1; j < len2; k = j++) {
-             if (rayIntersect(p, ring[j], ring[k])) inside = !inside;
+           if (context.getDebug('collision')) {
+             gj = bbox ? [{
+               type: 'Polygon',
+               coordinates: [[[bbox.minX, bbox.minY], [bbox.maxX, bbox.minY], [bbox.maxX, bbox.maxY], [bbox.minX, bbox.maxY], [bbox.minX, bbox.minY]]]
+             }] : [];
            }
+
+           var box = debug.selectAll('.debug-mouse').data(gj); // exit
+
+           box.exit().remove(); // enter/update
+
+           box.enter().append('path').attr('class', 'debug debug-mouse yellow').merge(box).attr('d', d3_geoPath());
          }
 
-         return inside;
-       }
+         var throttleFilterLabels = throttle(filterLabels, 100);
 
-       function rayIntersect(p, p1, p2) {
-         return p1[1] > p[1] !== p2[1] > p[1] && p[0] < (p2[0] - p1[0]) * (p[1] - p1[1]) / (p2[1] - p1[1]) + p1[0];
-       }
+         drawLabels.observe = function (selection) {
+           var listener = function listener() {
+             throttleFilterLabels(selection);
+           };
 
-       function treeItem(coords, props) {
-         var item = {
-           minX: Infinity,
-           minY: Infinity,
-           maxX: -Infinity,
-           maxY: -Infinity,
-           coords: coords,
-           props: props
+           selection.on('mousemove.hidelabels', listener);
+           context.on('enter.hidelabels', listener);
          };
 
-         for (var i = 0; i < coords[0].length; i++) {
-           var p = coords[0][i];
-           item.minX = Math.min(item.minX, p[0]);
-           item.minY = Math.min(item.minY, p[1]);
-           item.maxX = Math.max(item.maxX, p[0]);
-           item.maxY = Math.max(item.maxY, p[1]);
-         }
+         drawLabels.off = function (selection) {
+           throttleFilterLabels.cancel();
+           selection.on('mousemove.hidelabels', null);
+           context.on('enter.hidelabels', null);
+         };
 
-         return item;
+         return drawLabels;
        }
 
-       var type = "FeatureCollection";
-       var features = [{type:"Feature",properties:{m49:"680",wikidata:"Q3405693",nameEn:"Sark",country:"GB",groups:["GG","830","154","150"],level:"subterritory",driveSide:"left",roadSpeedUnit:"mph",callingCodes:["44 01481"]},geometry:{type:"MultiPolygon",coordinates:[[[[-2.36485,49.48223],[-2.65349,49.15373],[-2.09454,49.46288],[-2.36485,49.48223]]]]}},{type:"Feature",properties:{m49:"001",wikidata:"Q2",nameEn:"World",aliases:["Earth","Planet"],level:"world"},geometry:null},{type:"Feature",properties:{m49:"142",wikidata:"Q48",nameEn:"Asia",level:"region"},geometry:null},{type:"Feature",properties:{m49:"143",wikidata:"Q27275",nameEn:"Central Asia",groups:["142"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"145",wikidata:"Q27293",nameEn:"Western Asia",groups:["142"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"150",wikidata:"Q46",nameEn:"Europe",level:"region"},geometry:null},{type:"Feature",properties:{m49:"151",wikidata:"Q27468",nameEn:"Eastern Europe",groups:["150"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"154",wikidata:"Q27479",nameEn:"Northern Europe",groups:["150"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"155",wikidata:"Q27496",nameEn:"Western Europe",groups:["150"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"202",wikidata:"Q132959",nameEn:"Sub-Saharan Africa",groups:["002"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"419",wikidata:"Q72829598",nameEn:"Latin America and the Caribbean",groups:["019"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"830",wikidata:"Q42314",nameEn:"Channel Islands",groups:["150","154"],level:"intermediateRegion"},geometry:null},{type:"Feature",properties:{m49:"019",wikidata:"Q828",nameEn:"Americas",level:"region"},geometry:null},{type:"Feature",properties:{m49:"029",wikidata:"Q664609",nameEn:"Caribbean",groups:["419","019","003"],level:"intermediateRegion"},geometry:null},{type:"Feature",properties:{m49:"034",wikidata:"Q771405",nameEn:"Southern Asia",groups:["142"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"002",wikidata:"Q15",nameEn:"Africa",level:"region"},geometry:null},{type:"Feature",properties:{m49:"003",wikidata:"Q49",nameEn:"North America",groups:["019"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"017",wikidata:"Q27433",nameEn:"Middle Africa",groups:["202","002"],level:"intermediateRegion"},geometry:null},{type:"Feature",properties:{m49:"039",wikidata:"Q27449",nameEn:"Southern Europe",groups:["150"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"005",wikidata:"Q18",nameEn:"South America",groups:["419","019"],level:"intermediateRegion"},geometry:null},{type:"Feature",properties:{m49:"009",wikidata:"Q538",nameEn:"Oceania",level:"region"},geometry:null},{type:"Feature",properties:{m49:"061",wikidata:"Q35942",nameEn:"Polynesia",groups:["009"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"014",wikidata:"Q27407",nameEn:"Eastern Africa",groups:["202","002"],level:"intermediateRegion"},geometry:null},{type:"Feature",properties:{m49:"053",wikidata:"Q45256",nameEn:"Australia and New Zealand",aliases:["Australasia"],groups:["009"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"011",wikidata:"Q4412",nameEn:"Western Africa",groups:["202","002"],level:"intermediateRegion"},geometry:null},{type:"Feature",properties:{m49:"013",wikidata:"Q27611",nameEn:"Central America",groups:["419","019","003"],level:"intermediateRegion"},geometry:null},{type:"Feature",properties:{m49:"021",wikidata:"Q2017699",nameEn:"Northern America",groups:["019","003"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"035",wikidata:"Q11708",nameEn:"South-eastern Asia",groups:["142"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"018",wikidata:"Q27394",nameEn:"Southern Africa",groups:["202","002"],level:"intermediateRegion"},geometry:null},{type:"Feature",properties:{m49:"030",wikidata:"Q27231",nameEn:"Eastern Asia",groups:["142"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"015",wikidata:"Q27381",nameEn:"Northern Africa",groups:["002"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"054",wikidata:"Q37394",nameEn:"Melanesia",groups:["009"],level:"subregion"},geometry:null},{type:"Feature",properties:{m49:"057",wikidata:"Q3359409",nameEn:"Micronesia",groups:["009"],level:"subregion"},geometry:null},{type:"Feature",properties:{iso1A2:"AC",iso1A3:"ASC",wikidata:"Q46197",nameEn:"Ascension Island",country:"GB",groups:["SH","011","202","002"],isoStatus:"excRes",driveSide:"left",roadSpeedUnit:"mph",callingCodes:["247"]},geometry:{type:"MultiPolygon",coordinates:[[[[-14.82771,-8.70814],[-13.33271,-8.07391],[-14.91926,-6.63386],[-14.82771,-8.70814]]]]}},{type:"Feature",properties:{iso1A2:"AD",iso1A3:"AND",iso1N3:"020",wikidata:"Q228",nameEn:"Andorra",groups:["039","150"],callingCodes:["376"]},geometry:{type:"MultiPolygon",coordinates:[[[[1.72515,42.50338],[1.73683,42.55492],[1.7858,42.57698],[1.72588,42.59098],[1.73452,42.61515],[1.68267,42.62533],[1.6625,42.61982],[1.63485,42.62957],[1.60085,42.62703],[1.55418,42.65669],[1.50867,42.64483],[1.48043,42.65203],[1.46718,42.63296],[1.47986,42.61346],[1.44197,42.60217],[1.42512,42.58292],[1.44529,42.56722],[1.4234,42.55959],[1.41245,42.53539],[1.44759,42.54431],[1.46661,42.50949],[1.41648,42.48315],[1.43838,42.47848],[1.44529,42.43724],[1.5127,42.42959],[1.55073,42.43299],[1.55937,42.45808],[1.57953,42.44957],[1.58933,42.46275],[1.65674,42.47125],[1.66826,42.50779],[1.70571,42.48867],[1.72515,42.50338]]]]}},{type:"Feature",properties:{iso1A2:"AE",iso1A3:"ARE",iso1N3:"784",wikidata:"Q878",nameEn:"United Arab Emirates",groups:["145","142"],callingCodes:["971"]},geometry:{type:"MultiPolygon",coordinates:[[[[56.26534,25.62825],[56.25341,25.61443],[56.26636,25.60643],[56.25365,25.60211],[56.20473,25.61119],[56.18363,25.65508],[56.14826,25.66351],[56.13579,25.73524],[56.17416,25.77239],[56.13963,25.82765],[56.19334,25.9795],[56.15498,26.06828],[56.08666,26.05038],[55.81777,26.18798],[55.14145,25.62624],[53.97892,24.64436],[52.82259,25.51697],[52.35509,25.00368],[52.02277,24.75635],[51.83108,24.71675],[51.58834,24.66608],[51.41644,24.39615],[51.58871,24.27256],[51.59617,24.12041],[52.56622,22.94341],[55.13599,22.63334],[55.2137,22.71065],[55.22634,23.10378],[55.57358,23.669],[55.48677,23.94946],[55.73301,24.05994],[55.8308,24.01633],[56.01799,24.07426],[55.95472,24.2172],[55.83367,24.20193],[55.77658,24.23476],[55.76558,24.23227],[55.75257,24.23466],[55.75382,24.2466],[55.75939,24.26114],[55.76781,24.26209],[55.79145,24.27914],[55.80747,24.31069],[55.83395,24.32776],[55.83271,24.41521],[55.76461,24.5287],[55.83271,24.68567],[55.83408,24.77858],[55.81348,24.80102],[55.81116,24.9116],[55.85094,24.96858],[55.90849,24.96771],[55.96316,25.00857],[56.05715,24.95727],[56.05106,24.87461],[55.97467,24.89639],[55.97836,24.87673],[56.03535,24.81161],[56.06128,24.74457],[56.13684,24.73699],[56.20062,24.78565],[56.20568,24.85063],[56.30269,24.88334],[56.34873,24.93205],[56.3227,24.97284],[56.86325,25.03856],[56.82555,25.7713],[56.26534,25.62825]],[[56.26062,25.33108],[56.3005,25.31815],[56.3111,25.30107],[56.35172,25.30681],[56.34438,25.26653],[56.27628,25.23404],[56.24341,25.22867],[56.20872,25.24104],[56.20838,25.25668],[56.24465,25.27505],[56.25008,25.28843],[56.23362,25.31253],[56.26062,25.33108]]],[[[56.28423,25.26344],[56.29379,25.2754],[56.28102,25.28486],[56.2716,25.27916],[56.27086,25.26128],[56.28423,25.26344]]]]}},{type:"Feature",properties:{iso1A2:"AF",iso1A3:"AFG",iso1N3:"004",wikidata:"Q889",nameEn:"Afghanistan",groups:["034","142"],callingCodes:["93"]},geometry:{type:"MultiPolygon",coordinates:[[[[70.61526,38.34774],[70.60407,38.28046],[70.54673,38.24541],[70.4898,38.12546],[70.17206,37.93276],[70.1863,37.84296],[70.27694,37.81258],[70.28243,37.66706],[70.15015,37.52519],[69.95971,37.5659],[69.93362,37.61378],[69.84435,37.60616],[69.80041,37.5746],[69.51888,37.5844],[69.44954,37.4869],[69.36645,37.40462],[69.45022,37.23315],[69.39529,37.16752],[69.25152,37.09426],[69.03274,37.25174],[68.96407,37.32603],[68.88168,37.33368],[68.91189,37.26704],[68.80889,37.32494],[68.81438,37.23862],[68.6798,37.27906],[68.61851,37.19815],[68.41888,37.13906],[68.41201,37.10402],[68.29253,37.10621],[68.27605,37.00977],[68.18542,37.02074],[68.02194,36.91923],[67.87917,37.0591],[67.7803,37.08978],[67.78329,37.1834],[67.51868,37.26102],[67.2581,37.17216],[67.2224,37.24545],[67.13039,37.27168],[67.08232,37.35469],[66.95598,37.40162],[66.64699,37.32958],[66.55743,37.35409],[66.30993,37.32409],[65.72274,37.55438],[65.64137,37.45061],[65.64263,37.34388],[65.51778,37.23881],[64.97945,37.21913],[64.61141,36.6351],[64.62514,36.44311],[64.57295,36.34362],[64.43288,36.24401],[64.05385,36.10433],[63.98519,36.03773],[63.56496,35.95106],[63.53475,35.90881],[63.29579,35.85985],[63.12276,35.86208],[63.10318,35.81782],[63.23262,35.67487],[63.10079,35.63024],[63.12276,35.53196],[63.0898,35.43131],[62.90853,35.37086],[62.74098,35.25432],[62.62288,35.22067],[62.48006,35.28796],[62.29878,35.13312],[62.29191,35.25964],[62.15871,35.33278],[62.05709,35.43803],[61.97743,35.4604],[61.77693,35.41341],[61.58742,35.43803],[61.27371,35.61482],[61.18187,35.30249],[61.0991,35.27845],[61.12831,35.09938],[61.06926,34.82139],[61.00197,34.70631],[60.99922,34.63064],[60.72316,34.52857],[60.91321,34.30411],[60.66502,34.31539],[60.50209,34.13992],[60.5838,33.80793],[60.5485,33.73422],[60.57762,33.59772],[60.69573,33.56054],[60.91133,33.55596],[60.88908,33.50219],[60.56485,33.12944],[60.86191,32.22565],[60.84541,31.49561],[61.70929,31.37391],[61.80569,31.16167],[61.80957,31.12576],[61.83257,31.0452],[61.8335,30.97669],[61.78268,30.92724],[61.80829,30.84224],[60.87231,29.86514],[62.47751,29.40782],[63.5876,29.50456],[64.12966,29.39157],[64.19796,29.50407],[64.62116,29.58903],[65.04005,29.53957],[66.24175,29.85181],[66.36042,29.9583],[66.23609,30.06321],[66.34869,30.404],[66.28413,30.57001],[66.39194,30.9408],[66.42645,30.95309],[66.58175,30.97532],[66.68166,31.07597],[66.72561,31.20526],[66.83273,31.26867],[67.04147,31.31561],[67.03323,31.24519],[67.29964,31.19586],[67.78854,31.33203],[67.7748,31.4188],[67.62374,31.40473],[67.58323,31.52772],[67.72056,31.52304],[67.86887,31.63536],[68.00071,31.6564],[68.1655,31.82691],[68.25614,31.80357],[68.27605,31.75863],[68.44222,31.76446],[68.57475,31.83158],[68.6956,31.75687],[68.79997,31.61665],[68.91078,31.59687],[68.95995,31.64822],[69.00939,31.62249],[69.11514,31.70782],[69.20577,31.85957],[69.3225,31.93186],[69.27032,32.14141],[69.27932,32.29119],[69.23599,32.45946],[69.2868,32.53938],[69.38155,32.56601],[69.44747,32.6678],[69.43649,32.7302],[69.38018,32.76601],[69.47082,32.85834],[69.5436,32.8768],[69.49854,32.88843],[69.49004,33.01509],[69.57656,33.09911],[69.71526,33.09911],[69.79766,33.13247],[69.85259,33.09451],[70.02563,33.14282],[70.07369,33.22557],[70.13686,33.21064],[70.32775,33.34496],[70.17062,33.53535],[70.20141,33.64387],[70.14785,33.6553],[70.14236,33.71701],[70.00503,33.73528],[69.85671,33.93719],[69.87307,33.9689],[69.90203,34.04194],[70.54336,33.9463],[70.88119,33.97933],[71.07345,34.06242],[71.06933,34.10564],[71.09307,34.11961],[71.09453,34.13524],[71.13078,34.16503],[71.12815,34.26619],[71.17662,34.36769],[71.02401,34.44835],[71.0089,34.54568],[71.11602,34.63047],[71.08718,34.69034],[71.28356,34.80882],[71.29472,34.87728],[71.50329,34.97328],[71.49917,35.00478],[71.55273,35.02615],[71.52938,35.09023],[71.67495,35.21262],[71.5541,35.28776],[71.54294,35.31037],[71.65435,35.4479],[71.49917,35.6267],[71.55273,35.71483],[71.37969,35.95865],[71.19505,36.04134],[71.60491,36.39429],[71.80267,36.49924],[72.18135,36.71838],[72.6323,36.84601],[73.82685,36.91421],[74.04856,36.82648],[74.43389,37.00977],[74.53739,36.96224],[74.56453,37.03023],[74.49981,37.24518],[74.80605,37.21565],[74.88887,37.23275],[74.8294,37.3435],[74.68383,37.3948],[74.56161,37.37734],[74.41055,37.3948],[74.23339,37.41116],[74.20308,37.34208],[73.8564,37.26158],[73.82552,37.22659],[73.64974,37.23643],[73.61129,37.27469],[73.76647,37.33913],[73.77197,37.4417],[73.29633,37.46495],[73.06884,37.31729],[72.79693,37.22222],[72.66381,37.02014],[72.54095,37.00007],[72.31676,36.98115],[71.83229,36.68084],[71.67083,36.67346],[71.57195,36.74943],[71.51502,36.89128],[71.48481,36.93218],[71.46923,36.99925],[71.45578,37.03094],[71.43097,37.05855],[71.44127,37.11856],[71.4494,37.18137],[71.4555,37.21418],[71.47386,37.2269],[71.48339,37.23937],[71.4824,37.24921],[71.48536,37.26017],[71.50674,37.31502],[71.49821,37.31975],[71.4862,37.33405],[71.47685,37.40281],[71.49612,37.4279],[71.5256,37.47971],[71.50616,37.50733],[71.49693,37.53527],[71.5065,37.60912],[71.51972,37.61945],[71.54186,37.69691],[71.55234,37.73209],[71.53053,37.76534],[71.54324,37.77104],[71.55752,37.78677],[71.59255,37.79956],[71.58843,37.92425],[71.51565,37.95349],[71.32871,37.88564],[71.296,37.93403],[71.2809,37.91995],[71.24969,37.93031],[71.27278,37.96496],[71.27622,37.99946],[71.28922,38.01272],[71.29878,38.04429],[71.36444,38.15358],[71.37803,38.25641],[71.33869,38.27335],[71.33114,38.30339],[71.21291,38.32797],[71.1451,38.40106],[71.10957,38.40671],[71.10592,38.42077],[71.09542,38.42517],[71.0556,38.40176],[71.03545,38.44779],[70.98693,38.48862],[70.92728,38.43021],[70.88719,38.46826],[70.84376,38.44688],[70.82538,38.45394],[70.81697,38.44507],[70.80521,38.44447],[70.79766,38.44944],[70.78702,38.45031],[70.78581,38.45502],[70.77132,38.45548],[70.75455,38.4252],[70.72485,38.4131],[70.69807,38.41861],[70.67438,38.40597],[70.6761,38.39144],[70.69189,38.37031],[70.64966,38.34999],[70.61526,38.34774]]]]}},{type:"Feature",properties:{iso1A2:"AG",iso1A3:"ATG",iso1N3:"028",wikidata:"Q781",nameEn:"Antigua and Barbuda",groups:["029","003","419","019"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["1 268"]},geometry:{type:"MultiPolygon",coordinates:[[[[-62.12601,17.9235],[-62.27053,17.22145],[-62.62949,16.82364],[-62.52079,16.69392],[-62.14123,17.02632],[-61.83929,16.66647],[-61.44461,16.81958],[-61.45764,17.9187],[-62.12601,17.9235]]]]}},{type:"Feature",properties:{iso1A2:"AI",iso1A3:"AIA",iso1N3:"660",wikidata:"Q25228",nameEn:"Anguilla",country:"GB",groups:["029","003","419","019"],driveSide:"left",callingCodes:["1 264"]},geometry:{type:"MultiPolygon",coordinates:[[[[-63.83866,18.82518],[-63.35989,18.06012],[-62.86666,18.19278],[-62.75637,18.13489],[-62.46233,19.00569],[-63.83866,18.82518]]]]}},{type:"Feature",properties:{iso1A2:"AL",iso1A3:"ALB",iso1N3:"008",wikidata:"Q222",nameEn:"Albania",groups:["039","150"],callingCodes:["355"]},geometry:{type:"MultiPolygon",coordinates:[[[[20.07761,42.55582],[20.01834,42.54622],[20.00842,42.5109],[19.9324,42.51699],[19.82333,42.46581],[19.76549,42.50237],[19.74731,42.57422],[19.77375,42.58517],[19.73244,42.66299],[19.65972,42.62774],[19.4836,42.40831],[19.42352,42.36546],[19.42,42.33019],[19.28623,42.17745],[19.40687,42.10024],[19.37548,42.06835],[19.36867,42.02564],[19.37691,41.96977],[19.34601,41.95675],[19.33812,41.90669],[19.37451,41.8842],[19.37597,41.84849],[19.26406,41.74971],[19.0384,40.35325],[19.95905,39.82857],[19.97622,39.78684],[19.92466,39.69533],[19.98042,39.6504],[20.00957,39.69227],[20.05189,39.69112],[20.12956,39.65805],[20.15988,39.652],[20.22376,39.64532],[20.22707,39.67459],[20.27412,39.69884],[20.31961,39.72799],[20.29152,39.80421],[20.30804,39.81563],[20.38572,39.78516],[20.41475,39.81437],[20.41546,39.82832],[20.31135,39.99438],[20.37911,39.99058],[20.42373,40.06777],[20.48487,40.06271],[20.51297,40.08168],[20.55593,40.06524],[20.61081,40.07866],[20.62566,40.0897],[20.67162,40.09433],[20.71789,40.27739],[20.78234,40.35803],[20.7906,40.42726],[20.83688,40.47882],[20.94925,40.46625],[20.96908,40.51526],[21.03932,40.56299],[21.05833,40.66586],[20.98134,40.76046],[20.95752,40.76982],[20.98396,40.79109],[20.97887,40.85475],[20.97693,40.90103],[20.94305,40.92399],[20.83671,40.92752],[20.81567,40.89662],[20.73504,40.9081],[20.71634,40.91781],[20.65558,41.08009],[20.63454,41.0889],[20.59832,41.09066],[20.58546,41.11179],[20.59715,41.13644],[20.51068,41.2323],[20.49432,41.33679],[20.52119,41.34381],[20.55976,41.4087],[20.51301,41.442],[20.49039,41.49277],[20.45331,41.51436],[20.45809,41.5549],[20.52103,41.56473],[20.55508,41.58113],[20.51769,41.65975],[20.52937,41.69292],[20.51301,41.72433],[20.53405,41.78099],[20.57144,41.7897],[20.55976,41.87068],[20.59524,41.8818],[20.57946,41.91593],[20.63069,41.94913],[20.59434,42.03879],[20.55633,42.08173],[20.56955,42.12097],[20.48857,42.25444],[20.3819,42.3029],[20.34479,42.32656],[20.24399,42.32168],[20.21797,42.41237],[20.17127,42.50469],[20.07761,42.55582]]]]}},{type:"Feature",properties:{iso1A2:"AM",iso1A3:"ARM",iso1N3:"051",wikidata:"Q399",nameEn:"Armenia",groups:["145","142"],callingCodes:["374"]},geometry:{type:"MultiPolygon",coordinates:[[[[45.0133,41.29747],[44.93493,41.25685],[44.81437,41.30371],[44.80053,41.25949],[44.81749,41.23488],[44.84358,41.23088],[44.89911,41.21366],[44.87887,41.20195],[44.82084,41.21513],[44.72814,41.20338],[44.61462,41.24018],[44.59322,41.1933],[44.46791,41.18204],[44.34417,41.2382],[44.34337,41.20312],[44.32139,41.2079],[44.18148,41.24644],[44.16591,41.19141],[43.84835,41.16329],[43.74717,41.1117],[43.67712,41.13398],[43.4717,41.12611],[43.44984,41.0988],[43.47319,41.02251],[43.58683,40.98961],[43.67712,40.93084],[43.67712,40.84846],[43.74872,40.7365],[43.7425,40.66805],[43.63664,40.54159],[43.54791,40.47413],[43.60862,40.43267],[43.59928,40.34019],[43.71136,40.16673],[43.65221,40.14889],[43.65688,40.11199],[43.92307,40.01787],[44.1057,40.03555],[44.1778,40.02845],[44.26973,40.04866],[44.46635,39.97733],[44.61845,39.8281],[44.75779,39.7148],[44.88354,39.74432],[44.92869,39.72157],[45.06604,39.79277],[45.18554,39.67846],[45.17464,39.58614],[45.21784,39.58074],[45.23535,39.61373],[45.30385,39.61373],[45.29606,39.57654],[45.46992,39.49888],[45.70547,39.60174],[45.80804,39.56716],[45.83,39.46487],[45.79225,39.3695],[45.99774,39.28931],[46.02303,39.09978],[46.06973,39.0744],[46.14785,38.84206],[46.20601,38.85262],[46.34059,38.92076],[46.53497,38.86548],[46.51805,38.94982],[46.54296,39.07078],[46.44022,39.19636],[46.52584,39.18912],[46.54141,39.15895],[46.58032,39.21204],[46.63481,39.23013],[46.56476,39.24942],[46.50093,39.33736],[46.43244,39.35181],[46.37795,39.42039],[46.4013,39.45405],[46.53051,39.47809],[46.51027,39.52373],[46.57721,39.54414],[46.57098,39.56694],[46.52117,39.58734],[46.42465,39.57534],[46.40286,39.63651],[46.18493,39.60533],[45.96543,39.78859],[45.82533,39.82925],[45.7833,39.9475],[45.60895,39.97733],[45.59806,40.0131],[45.78642,40.03218],[45.83779,39.98925],[45.97944,40.181],[45.95609,40.27846],[45.65098,40.37696],[45.42994,40.53804],[45.45484,40.57707],[45.35366,40.65979],[45.4206,40.7424],[45.55914,40.78366],[45.60584,40.87436],[45.40814,40.97904],[45.44083,41.01663],[45.39725,41.02603],[45.35677,40.99784],[45.28859,41.03757],[45.26162,41.0228],[45.25897,41.0027],[45.1994,41.04518],[45.16493,41.05068],[45.1634,41.08082],[45.1313,41.09369],[45.12923,41.06059],[45.06784,41.05379],[45.08028,41.10917],[45.19942,41.13299],[45.1969,41.168],[45.11811,41.19967],[45.05201,41.19211],[45.02932,41.2101],[45.05497,41.2464],[45.0133,41.29747]],[[45.21324,40.9817],[45.21219,40.99001],[45.20518,40.99348],[45.19312,40.98998],[45.18382,41.0066],[45.20625,41.01484],[45.23487,41.00226],[45.23095,40.97828],[45.21324,40.9817]],[[45.00864,41.03411],[44.9903,41.05657],[44.96031,41.06345],[44.95383,41.07553],[44.97169,41.09176],[45.00864,41.09407],[45.03406,41.07931],[45.04517,41.06653],[45.03792,41.03938],[45.00864,41.03411]]],[[[45.50279,40.58424],[45.56071,40.64765],[45.51825,40.67382],[45.47927,40.65023],[45.50279,40.58424]]]]}},{type:"Feature",properties:{iso1A2:"AO",iso1A3:"AGO",iso1N3:"024",wikidata:"Q916",nameEn:"Angola",groups:["017","202","002"],callingCodes:["244"]},geometry:{type:"MultiPolygon",coordinates:[[[[16.55507,-5.85631],[13.04371,-5.87078],[12.42245,-6.07585],[11.95767,-5.94705],[12.20376,-5.76338],[12.26557,-5.74031],[12.52318,-5.74353],[12.52301,-5.17481],[12.53599,-5.1618],[12.53586,-5.14658],[12.51589,-5.1332],[12.49815,-5.14058],[12.46297,-5.09408],[12.60251,-5.01715],[12.63465,-4.94632],[12.70868,-4.95505],[12.8733,-4.74346],[13.11195,-4.67745],[13.09648,-4.63739],[12.91489,-4.47907],[12.87096,-4.40315],[12.76844,-4.38709],[12.64835,-4.55937],[12.40964,-4.60609],[12.32324,-4.78415],[12.25587,-4.79437],[12.20901,-4.75642],[12.16068,-4.90089],[12.00924,-5.02627],[11.50888,-5.33417],[10.5065,-17.25284],[11.75063,-17.25013],[12.07076,-17.15165],[12.52111,-17.24495],[12.97145,-16.98567],[13.36212,-16.98048],[13.95896,-17.43141],[14.28743,-17.38814],[18.39229,-17.38927],[18.84226,-17.80375],[21.14283,-17.94318],[21.42741,-18.02787],[23.47474,-17.62877],[23.20038,-17.47563],[22.17217,-16.50269],[22.00323,-16.18028],[21.97988,-13.00148],[24.03339,-12.99091],[23.90937,-12.844],[24.06672,-12.29058],[23.98804,-12.13149],[24.02603,-11.15368],[24.00027,-10.89356],[23.86868,-11.02856],[23.45631,-10.946],[23.16602,-11.10577],[22.54205,-11.05784],[22.25951,-11.24911],[22.17954,-10.85884],[22.32604,-10.76291],[22.19039,-9.94628],[21.84856,-9.59871],[21.79824,-7.29628],[20.56263,-7.28566],[20.61689,-6.90876],[20.31846,-6.91953],[20.30218,-6.98955],[19.5469,-7.00195],[19.33698,-7.99743],[18.33635,-8.00126],[17.5828,-8.13784],[16.96282,-7.21787],[16.55507,-5.85631]]]]}},{type:"Feature",properties:{iso1A2:"AQ",iso1A3:"ATA",iso1N3:"010",wikidata:"Q51",nameEn:"Antarctica",level:"region",callingCodes:["672"]},geometry:{type:"MultiPolygon",coordinates:[[[[180,-60],[-180,-60],[-180,-90],[180,-90],[180,-60]]]]}},{type:"Feature",properties:{iso1A2:"AR",iso1A3:"ARG",iso1N3:"032",wikidata:"Q414",nameEn:"Argentina",aliases:["RA"],groups:["005","419","019"],callingCodes:["54"]},geometry:{type:"MultiPolygon",coordinates:[[[[-72.31343,-50.58411],[-72.33873,-51.59954],[-71.99889,-51.98018],[-69.97824,-52.00845],[-68.41683,-52.33516],[-68.60702,-52.65781],[-68.60733,-54.9125],[-68.01394,-54.8753],[-67.46182,-54.92205],[-67.11046,-54.94199],[-66.07313,-55.19618],[-63.67376,-55.11859],[-54.78916,-36.21945],[-57.83001,-34.69099],[-58.34425,-34.15035],[-58.44442,-33.84033],[-58.40475,-33.11777],[-58.1224,-32.98842],[-58.22362,-32.52416],[-58.10036,-32.25338],[-58.20252,-31.86966],[-58.00076,-31.65016],[-58.0023,-31.53084],[-58.07569,-31.44916],[-57.98127,-31.3872],[-57.9908,-31.34924],[-57.86729,-31.06352],[-57.89476,-30.95994],[-57.8024,-30.77193],[-57.89115,-30.49572],[-57.64859,-30.35095],[-57.61478,-30.25165],[-57.65132,-30.19229],[-57.09386,-29.74211],[-56.81251,-29.48154],[-56.62789,-29.18073],[-56.57295,-29.11357],[-56.54171,-29.11447],[-56.05265,-28.62651],[-56.00458,-28.60421],[-56.01729,-28.51223],[-55.65418,-28.18304],[-55.6262,-28.17124],[-55.33303,-27.94661],[-55.16872,-27.86224],[-55.1349,-27.89759],[-54.90805,-27.73149],[-54.90159,-27.63132],[-54.67657,-27.57214],[-54.50416,-27.48232],[-54.41888,-27.40882],[-54.19268,-27.30751],[-54.19062,-27.27639],[-54.15978,-27.2889],[-53.80144,-27.09844],[-53.73372,-26.6131],[-53.68269,-26.33359],[-53.64505,-26.28089],[-53.64186,-26.25976],[-53.64632,-26.24798],[-53.63881,-26.25075],[-53.63739,-26.2496],[-53.65237,-26.23289],[-53.65018,-26.19501],[-53.73968,-26.10012],[-53.73391,-26.07006],[-53.7264,-26.0664],[-53.73086,-26.05842],[-53.73511,-26.04211],[-53.83691,-25.94849],[-53.90831,-25.55513],[-54.52926,-25.62846],[-54.5502,-25.58915],[-54.59398,-25.59224],[-54.62063,-25.91213],[-54.60664,-25.9691],[-54.67359,-25.98607],[-54.69333,-26.37705],[-54.70732,-26.45099],[-54.80868,-26.55669],[-55.00584,-26.78754],[-55.06351,-26.80195],[-55.16948,-26.96068],[-55.25243,-26.93808],[-55.39611,-26.97679],[-55.62322,-27.1941],[-55.59094,-27.32444],[-55.74475,-27.44485],[-55.89195,-27.3467],[-56.18313,-27.29851],[-56.85337,-27.5165],[-58.04205,-27.2387],[-58.59549,-27.29973],[-58.65321,-27.14028],[-58.3198,-26.83443],[-58.1188,-26.16704],[-57.87176,-25.93604],[-57.57431,-25.47269],[-57.80821,-25.13863],[-58.25492,-24.92528],[-58.33055,-24.97099],[-59.33886,-24.49935],[-59.45482,-24.34787],[-60.03367,-24.00701],[-60.28163,-24.04436],[-60.99754,-23.80934],[-61.0782,-23.62932],[-61.9756,-23.0507],[-62.22768,-22.55807],[-62.51761,-22.37684],[-62.64455,-22.25091],[-62.8078,-22.12534],[-62.81124,-21.9987],[-63.66482,-21.99918],[-63.68113,-22.0544],[-63.70963,-21.99934],[-63.93287,-21.99934],[-64.22918,-22.55807],[-64.31489,-22.88824],[-64.35108,-22.73282],[-64.4176,-22.67692],[-64.58888,-22.25035],[-64.67174,-22.18957],[-64.90014,-22.12136],[-64.99524,-22.08255],[-65.47435,-22.08908],[-65.57743,-22.07675],[-65.58694,-22.09794],[-65.61166,-22.09504],[-65.7467,-22.10105],[-65.9261,-21.93335],[-66.04832,-21.9187],[-66.03836,-21.84829],[-66.24077,-21.77837],[-66.29714,-22.08741],[-66.7298,-22.23644],[-67.18382,-22.81525],[-66.99632,-22.99839],[-67.33563,-24.04237],[-68.24825,-24.42596],[-68.56909,-24.69831],[-68.38372,-25.08636],[-68.57622,-25.32505],[-68.38372,-26.15353],[-68.56909,-26.28146],[-68.59048,-26.49861],[-68.27677,-26.90626],[-68.43363,-27.08414],[-68.77586,-27.16029],[-69.22504,-27.95042],[-69.66709,-28.44055],[-69.80969,-29.07185],[-69.99507,-29.28351],[-69.8596,-30.26131],[-70.14479,-30.36595],[-70.55832,-31.51559],[-69.88099,-33.34489],[-69.87386,-34.13344],[-70.49416,-35.24145],[-70.38008,-36.02375],[-70.95047,-36.4321],[-71.24279,-37.20264],[-70.89532,-38.6923],[-71.37826,-38.91474],[-71.92726,-40.72714],[-71.74901,-42.11711],[-72.15541,-42.15941],[-72.14828,-42.85321],[-71.64206,-43.64774],[-71.81318,-44.38097],[-71.16436,-44.46244],[-71.26418,-44.75684],[-72.06985,-44.81756],[-71.35687,-45.22075],[-71.75614,-45.61611],[-71.68577,-46.55385],[-71.94152,-47.13595],[-72.50478,-47.80586],[-72.27662,-48.28727],[-72.54042,-48.52392],[-72.56894,-48.81116],[-73.09655,-49.14342],[-73.45156,-49.79461],[-73.55259,-49.92488],[-73.15765,-50.78337],[-72.31343,-50.58411]]]]}},{type:"Feature",properties:{iso1A2:"AS",iso1A3:"ASM",iso1N3:"016",wikidata:"Q16641",nameEn:"American Samoa",country:"US",groups:["061","009"],roadSpeedUnit:"mph",callingCodes:["1 684"]},geometry:{type:"MultiPolygon",coordinates:[[[[-174.18596,-12.48057],[-171.14953,-12.4725],[-171.14262,-14.93704],[-167.73854,-14.92809],[-167.75195,-10.12005],[-174.17993,-10.13616],[-174.18596,-12.48057]]]]}},{type:"Feature",properties:{iso1A2:"AT",iso1A3:"AUT",iso1N3:"040",wikidata:"Q40",nameEn:"Austria",groups:["EU","155","150"],callingCodes:["43"]},geometry:{type:"MultiPolygon",coordinates:[[[[15.34823,48.98444],[15.28305,48.98831],[15.26177,48.95766],[15.16358,48.94278],[15.15534,48.99056],[14.99878,49.01444],[14.97612,48.96983],[14.98917,48.90082],[14.95072,48.79101],[14.98032,48.77959],[14.9782,48.7766],[14.98112,48.77524],[14.9758,48.76857],[14.95641,48.75915],[14.94773,48.76268],[14.81545,48.7874],[14.80821,48.77711],[14.80584,48.73489],[14.72756,48.69502],[14.71794,48.59794],[14.66762,48.58215],[14.60808,48.62881],[14.56139,48.60429],[14.4587,48.64695],[14.43076,48.58855],[14.33909,48.55852],[14.20691,48.5898],[14.09104,48.5943],[14.01482,48.63788],[14.06151,48.66873],[13.84023,48.76988],[13.82266,48.75544],[13.81863,48.73257],[13.79337,48.71375],[13.81791,48.69832],[13.81283,48.68426],[13.81901,48.6761],[13.82609,48.62345],[13.80038,48.59487],[13.80519,48.58026],[13.76921,48.55324],[13.7513,48.5624],[13.74816,48.53058],[13.72802,48.51208],[13.66113,48.53558],[13.65186,48.55092],[13.62508,48.55501],[13.59705,48.57013],[13.57535,48.55912],[13.51291,48.59023],[13.50131,48.58091],[13.50663,48.57506],[13.46967,48.55157],[13.45214,48.56472],[13.43695,48.55776],[13.45727,48.51092],[13.42527,48.45711],[13.43929,48.43386],[13.40709,48.37292],[13.30897,48.31575],[13.26039,48.29422],[13.18093,48.29577],[13.126,48.27867],[13.0851,48.27711],[13.02083,48.25689],[12.95306,48.20629],[12.87126,48.20318],[12.84475,48.16556],[12.836,48.1647],[12.8362,48.15876],[12.82673,48.15245],[12.80676,48.14979],[12.78595,48.12445],[12.7617,48.12796],[12.74973,48.10885],[12.76141,48.07373],[12.8549,48.01122],[12.87476,47.96195],[12.91683,47.95647],[12.9211,47.95135],[12.91985,47.94069],[12.92668,47.93879],[12.93419,47.94063],[12.93642,47.94436],[12.93886,47.94046],[12.94163,47.92927],[13.00588,47.84374],[12.98543,47.82896],[12.96311,47.79957],[12.93202,47.77302],[12.94371,47.76281],[12.9353,47.74788],[12.91711,47.74026],[12.90274,47.72513],[12.91333,47.7178],[12.92969,47.71094],[12.98578,47.7078],[13.01382,47.72116],[13.07692,47.68814],[13.09562,47.63304],[13.06407,47.60075],[13.06641,47.58577],[13.04537,47.58183],[13.05355,47.56291],[13.03252,47.53373],[13.04537,47.49426],[12.9998,47.46267],[12.98344,47.48716],[12.9624,47.47452],[12.85256,47.52741],[12.84672,47.54556],[12.80699,47.54477],[12.77427,47.58025],[12.82101,47.61493],[12.76492,47.64485],[12.77777,47.66689],[12.7357,47.6787],[12.6071,47.6741],[12.57438,47.63238],[12.53816,47.63553],[12.50076,47.62293],[12.44117,47.6741],[12.43883,47.6977],[12.37222,47.68433],[12.336,47.69534],[12.27991,47.68827],[12.26004,47.67725],[12.24017,47.69534],[12.26238,47.73544],[12.2542,47.7433],[12.22571,47.71776],[12.18303,47.70065],[12.16217,47.70105],[12.16769,47.68167],[12.18347,47.66663],[12.18507,47.65984],[12.19895,47.64085],[12.20801,47.61082],[12.20398,47.60667],[12.18568,47.6049],[12.17737,47.60121],[12.18145,47.61019],[12.17824,47.61506],[12.13734,47.60639],[12.05788,47.61742],[12.02282,47.61033],[12.0088,47.62451],[11.85572,47.60166],[11.84052,47.58354],[11.63934,47.59202],[11.60681,47.57881],[11.58811,47.55515],[11.58578,47.52281],[11.52618,47.50939],[11.4362,47.51413],[11.38128,47.47465],[11.4175,47.44621],[11.33804,47.44937],[11.29597,47.42566],[11.27844,47.39956],[11.22002,47.3964],[11.25157,47.43277],[11.20482,47.43198],[11.12536,47.41222],[11.11835,47.39719],[10.97111,47.39561],[10.97111,47.41617],[10.98513,47.42882],[10.92437,47.46991],[10.93839,47.48018],[10.90918,47.48571],[10.87061,47.4786],[10.86945,47.5015],[10.91268,47.51334],[10.88814,47.53701],[10.77596,47.51729],[10.7596,47.53228],[10.6965,47.54253],[10.68832,47.55752],[10.63456,47.5591],[10.60337,47.56755],[10.56912,47.53584],[10.48849,47.54057],[10.47329,47.58552],[10.43473,47.58394],[10.44992,47.5524],[10.4324,47.50111],[10.44291,47.48453],[10.46278,47.47901],[10.47446,47.43318],[10.4359,47.41183],[10.4324,47.38494],[10.39851,47.37623],[10.33424,47.30813],[10.23257,47.27088],[10.17531,47.27167],[10.17648,47.29149],[10.2147,47.31014],[10.19998,47.32832],[10.23757,47.37609],[10.22774,47.38904],[10.2127,47.38019],[10.17648,47.38889],[10.16362,47.36674],[10.11805,47.37228],[10.09819,47.35724],[10.06897,47.40709],[10.1052,47.4316],[10.09001,47.46005],[10.07131,47.45531],[10.03859,47.48927],[10.00003,47.48216],[9.96029,47.53899],[9.92407,47.53111],[9.87733,47.54688],[9.87499,47.52953],[9.8189,47.54688],[9.82591,47.58158],[9.80254,47.59419],[9.76748,47.5934],[9.72736,47.53457],[9.55125,47.53629],[9.56312,47.49495],[9.58208,47.48344],[9.59482,47.46305],[9.60205,47.46165],[9.60484,47.46358],[9.60841,47.47178],[9.62158,47.45858],[9.62475,47.45685],[9.6423,47.45599],[9.65728,47.45383],[9.65863,47.44847],[9.64483,47.43842],[9.6446,47.43233],[9.65043,47.41937],[9.65136,47.40504],[9.6629,47.39591],[9.67334,47.39191],[9.67445,47.38429],[9.6711,47.37824],[9.66243,47.37136],[9.65427,47.36824],[9.62476,47.36639],[9.59978,47.34671],[9.58513,47.31334],[9.55857,47.29919],[9.54773,47.2809],[9.53116,47.27029],[9.56766,47.24281],[9.55176,47.22585],[9.56981,47.21926],[9.58264,47.20673],[9.56539,47.17124],[9.62623,47.14685],[9.63395,47.08443],[9.61216,47.07732],[9.60717,47.06091],[9.87935,47.01337],[9.88266,46.93343],[9.98058,46.91434],[10.10715,46.84296],[10.22675,46.86942],[10.24128,46.93147],[10.30031,46.92093],[10.36933,47.00212],[10.48376,46.93891],[10.47197,46.85698],[10.54783,46.84505],[10.66405,46.87614],[10.75753,46.82258],[10.72974,46.78972],[11.00764,46.76896],[11.10618,46.92966],[11.33355,46.99862],[11.50739,47.00644],[11.74789,46.98484],[12.19254,47.09331],[12.21781,47.03996],[12.11675,47.01241],[12.2006,46.88854],[12.27591,46.88651],[12.38708,46.71529],[12.59992,46.6595],[12.94445,46.60401],[13.27627,46.56059],[13.64088,46.53438],[13.7148,46.5222],[13.89837,46.52331],[14.00422,46.48474],[14.04002,46.49117],[14.12097,46.47724],[14.15989,46.43327],[14.28326,46.44315],[14.314,46.43327],[14.42608,46.44614],[14.45877,46.41717],[14.52176,46.42617],[14.56463,46.37208],[14.5942,46.43434],[14.66892,46.44936],[14.72185,46.49974],[14.81836,46.51046],[14.83549,46.56614],[14.86419,46.59411],[14.87129,46.61],[14.92283,46.60848],[14.96002,46.63459],[14.98024,46.6009],[15.01451,46.641],[15.14215,46.66131],[15.23711,46.63994],[15.41235,46.65556],[15.45514,46.63697],[15.46906,46.61321],[15.54431,46.6312],[15.55333,46.64988],[15.54533,46.66985],[15.59826,46.68908],[15.62317,46.67947],[15.63255,46.68069],[15.6365,46.6894],[15.6543,46.69228],[15.6543,46.70616],[15.67411,46.70735],[15.69523,46.69823],[15.72279,46.69548],[15.73823,46.70011],[15.76771,46.69863],[15.78518,46.70712],[15.8162,46.71897],[15.87691,46.7211],[15.94864,46.68769],[15.98512,46.68463],[15.99988,46.67947],[16.04036,46.6549],[16.04347,46.68694],[16.02808,46.71094],[15.99769,46.7266],[15.98432,46.74991],[15.99126,46.78199],[15.99054,46.82772],[16.05786,46.83927],[16.10983,46.867],[16.19904,46.94134],[16.22403,46.939],[16.27594,46.9643],[16.28202,47.00159],[16.51369,47.00084],[16.43936,47.03548],[16.52176,47.05747],[16.46134,47.09395],[16.52863,47.13974],[16.44932,47.14418],[16.46442,47.16845],[16.4523,47.18812],[16.42801,47.18422],[16.41739,47.20649],[16.43663,47.21127],[16.44142,47.25079],[16.47782,47.25918],[16.45104,47.41181],[16.49908,47.39416],[16.52414,47.41007],[16.57152,47.40868],[16.6718,47.46139],[16.64821,47.50155],[16.71059,47.52692],[16.64193,47.63114],[16.58699,47.61772],[16.4222,47.66537],[16.55129,47.72268],[16.53514,47.73837],[16.54779,47.75074],[16.61183,47.76171],[16.65679,47.74197],[16.72089,47.73469],[16.7511,47.67878],[16.82938,47.68432],[16.86509,47.72268],[16.87538,47.68895],[17.08893,47.70928],[17.05048,47.79377],[17.07039,47.81129],[17.00997,47.86245],[17.08275,47.87719],[17.11022,47.92461],[17.09786,47.97336],[17.16001,48.00636],[17.07039,48.0317],[17.09168,48.09366],[17.05735,48.14179],[17.02919,48.13996],[16.97701,48.17385],[16.89461,48.31332],[16.90903,48.32519],[16.84243,48.35258],[16.83317,48.38138],[16.83588,48.3844],[16.8497,48.38321],[16.85204,48.44968],[16.94611,48.53614],[16.93955,48.60371],[16.90354,48.71541],[16.79779,48.70998],[16.71883,48.73806],[16.68518,48.7281],[16.67008,48.77699],[16.46134,48.80865],[16.40915,48.74576],[16.37345,48.729],[16.06034,48.75436],[15.84404,48.86921],[15.78087,48.87644],[15.75341,48.8516],[15.6921,48.85973],[15.61622,48.89541],[15.51357,48.91549],[15.48027,48.94481],[15.34823,48.98444]]]]}},{type:"Feature",properties:{iso1A2:"AU",iso1A3:"AUS",iso1N3:"036",wikidata:"Q408",nameEn:"Australia",groups:["053","009"],driveSide:"left",callingCodes:["61"]},geometry:{type:"MultiPolygon",coordinates:[[[[156.55918,-21.85134],[158.60851,-15.7108],[144.30183,-9.48146],[142.81927,-9.31709],[142.5723,-9.35994],[142.31447,-9.24611],[142.23304,-9.19253],[142.1462,-9.19923],[142.0953,-9.23534],[142.0601,-9.56571],[140.88922,-9.34945],[127.55165,-9.05052],[96.7091,-25.20343],[159.69067,-56.28945],[165.46901,-28.32101],[156.55918,-21.85134]]]]}},{type:"Feature",properties:{iso1A2:"AW",iso1A3:"ABW",iso1N3:"533",wikidata:"Q21203",nameEn:"Aruba",country:"NL",groups:["029","003","419","019"],callingCodes:["297"]},geometry:{type:"MultiPolygon",coordinates:[[[[-70.00823,12.98375],[-70.35625,12.58277],[-69.60231,12.17],[-70.00823,12.98375]]]]}},{type:"Feature",properties:{iso1A2:"AX",iso1A3:"ALA",iso1N3:"248",wikidata:"Q5689",nameEn:"Åland Islands",country:"FI",groups:["EU","154","150"],callingCodes:["358 18","358 457"]},geometry:{type:"MultiPolygon",coordinates:[[[[19.08191,60.19152],[20.5104,59.15546],[21.35468,59.67511],[21.02509,60.12142],[21.08159,60.20167],[21.15143,60.54555],[20.96741,60.71528],[19.23413,60.61414],[19.08191,60.19152]]]]}},{type:"Feature",properties:{iso1A2:"AZ",iso1A3:"AZE",iso1N3:"031",wikidata:"Q227",nameEn:"Azerbaijan",groups:["145","142"],callingCodes:["994"]},geometry:{type:"MultiPolygon",coordinates:[[[[46.42738,41.91323],[46.3984,41.84399],[46.30863,41.79133],[46.23962,41.75811],[46.20538,41.77205],[46.17891,41.72094],[46.19759,41.62327],[46.24429,41.59883],[46.26531,41.63339],[46.28182,41.60089],[46.3253,41.60912],[46.34039,41.5947],[46.34126,41.57454],[46.29794,41.5724],[46.33925,41.4963],[46.40307,41.48464],[46.4669,41.43331],[46.63658,41.37727],[46.72375,41.28609],[46.66148,41.20533],[46.63969,41.09515],[46.55096,41.1104],[46.48558,41.0576],[46.456,41.09984],[46.37661,41.10805],[46.27698,41.19011],[46.13221,41.19479],[45.95786,41.17956],[45.80842,41.2229],[45.69946,41.29545],[45.75705,41.35157],[45.71035,41.36208],[45.68389,41.3539],[45.45973,41.45898],[45.4006,41.42402],[45.31352,41.47168],[45.26285,41.46433],[45.1797,41.42231],[45.09867,41.34065],[45.0133,41.29747],[45.05497,41.2464],[45.02932,41.2101],[45.05201,41.19211],[45.11811,41.19967],[45.1969,41.168],[45.19942,41.13299],[45.08028,41.10917],[45.06784,41.05379],[45.12923,41.06059],[45.1313,41.09369],[45.1634,41.08082],[45.16493,41.05068],[45.1994,41.04518],[45.25897,41.0027],[45.26162,41.0228],[45.28859,41.03757],[45.35677,40.99784],[45.39725,41.02603],[45.44083,41.01663],[45.40814,40.97904],[45.60584,40.87436],[45.55914,40.78366],[45.4206,40.7424],[45.35366,40.65979],[45.45484,40.57707],[45.42994,40.53804],[45.65098,40.37696],[45.95609,40.27846],[45.97944,40.181],[45.83779,39.98925],[45.78642,40.03218],[45.59806,40.0131],[45.60895,39.97733],[45.7833,39.9475],[45.82533,39.82925],[45.96543,39.78859],[46.18493,39.60533],[46.40286,39.63651],[46.42465,39.57534],[46.52117,39.58734],[46.57098,39.56694],[46.57721,39.54414],[46.51027,39.52373],[46.53051,39.47809],[46.4013,39.45405],[46.37795,39.42039],[46.43244,39.35181],[46.50093,39.33736],[46.56476,39.24942],[46.63481,39.23013],[46.58032,39.21204],[46.54141,39.15895],[46.52584,39.18912],[46.44022,39.19636],[46.54296,39.07078],[46.51805,38.94982],[46.53497,38.86548],[46.75752,39.03231],[46.83822,39.13143],[46.92539,39.16644],[46.95341,39.13505],[47.05771,39.20143],[47.05927,39.24846],[47.31301,39.37492],[47.38978,39.45999],[47.50099,39.49615],[47.84774,39.66285],[47.98977,39.70999],[48.34264,39.42935],[48.37385,39.37584],[48.15984,39.30028],[48.12404,39.25208],[48.15361,39.19419],[48.31239,39.09278],[48.33884,39.03022],[48.28437,38.97186],[48.08627,38.94434],[48.07734,38.91616],[48.01409,38.90333],[48.02581,38.82705],[48.24773,38.71883],[48.3146,38.59958],[48.45084,38.61013],[48.58793,38.45076],[48.62217,38.40198],[48.70001,38.40564],[48.78979,38.45026],[48.81072,38.44853],[48.84969,38.45015],[48.88288,38.43975],[52.39847,39.43556],[48.80971,41.95365],[48.5867,41.84306],[48.55078,41.77917],[48.42301,41.65444],[48.40277,41.60441],[48.2878,41.56221],[48.22064,41.51472],[48.07587,41.49957],[47.87973,41.21798],[47.75831,41.19455],[47.62288,41.22969],[47.54504,41.20275],[47.49004,41.26366],[47.34579,41.27884],[47.10762,41.59044],[47.03757,41.55434],[46.99554,41.59743],[47.00955,41.63583],[46.8134,41.76252],[46.75269,41.8623],[46.58924,41.80547],[46.5332,41.87389],[46.42738,41.91323]],[[45.50279,40.58424],[45.47927,40.65023],[45.51825,40.67382],[45.56071,40.64765],[45.50279,40.58424]]],[[[45.00864,41.03411],[45.03792,41.03938],[45.04517,41.06653],[45.03406,41.07931],[45.00864,41.09407],[44.97169,41.09176],[44.95383,41.07553],[44.96031,41.06345],[44.9903,41.05657],[45.00864,41.03411]]],[[[45.21324,40.9817],[45.23095,40.97828],[45.23487,41.00226],[45.20625,41.01484],[45.18382,41.0066],[45.19312,40.98998],[45.20518,40.99348],[45.21219,40.99001],[45.21324,40.9817]]],[[[45.46992,39.49888],[45.29606,39.57654],[45.30385,39.61373],[45.23535,39.61373],[45.21784,39.58074],[45.17464,39.58614],[45.18554,39.67846],[45.06604,39.79277],[44.92869,39.72157],[44.88354,39.74432],[44.75779,39.7148],[44.80977,39.65768],[44.81043,39.62677],[44.88916,39.59653],[44.96746,39.42998],[45.05932,39.36435],[45.08751,39.35052],[45.16168,39.21952],[45.30489,39.18333],[45.40148,39.09007],[45.40452,39.07224],[45.44811,39.04927],[45.44966,38.99243],[45.6131,38.964],[45.6155,38.94304],[45.65172,38.95199],[45.83883,38.90768],[45.90266,38.87739],[45.94624,38.89072],[46.00228,38.87376],[46.06766,38.87861],[46.14785,38.84206],[46.06973,39.0744],[46.02303,39.09978],[45.99774,39.28931],[45.79225,39.3695],[45.83,39.46487],[45.80804,39.56716],[45.70547,39.60174],[45.46992,39.49888]]]]}},{type:"Feature",properties:{iso1A2:"BA",iso1A3:"BIH",iso1N3:"070",wikidata:"Q225",nameEn:"Bosnia and Herzegovina",groups:["039","150"],callingCodes:["387"]},geometry:{type:"MultiPolygon",coordinates:[[[[17.84826,45.04489],[17.66571,45.13408],[17.59104,45.10816],[17.51469,45.10791],[17.47589,45.12656],[17.45615,45.12523],[17.4498,45.16119],[17.41229,45.13335],[17.33573,45.14521],[17.32092,45.16246],[17.26815,45.18444],[17.25131,45.14957],[17.24325,45.146],[17.18438,45.14764],[17.0415,45.20759],[16.9385,45.22742],[16.92405,45.27607],[16.83804,45.18951],[16.81137,45.18434],[16.78219,45.19002],[16.74845,45.20393],[16.64962,45.20714],[16.60194,45.23042],[16.56559,45.22307],[16.5501,45.2212],[16.52982,45.22713],[16.49155,45.21153],[16.4634,45.14522],[16.40023,45.1147],[16.38309,45.05955],[16.38219,45.05139],[16.3749,45.05206],[16.35863,45.03529],[16.35404,45.00241],[16.29036,44.99732],[16.12153,45.09616],[15.98412,45.23088],[15.83512,45.22459],[15.76371,45.16508],[15.78842,45.11519],[15.74585,45.0638],[15.78568,44.97401],[15.74723,44.96818],[15.76096,44.87045],[15.79472,44.8455],[15.72584,44.82334],[15.8255,44.71501],[15.89348,44.74964],[16.05828,44.61538],[16.00884,44.58605],[16.03012,44.55572],[16.10566,44.52586],[16.16814,44.40679],[16.12969,44.38275],[16.21346,44.35231],[16.18688,44.27012],[16.36864,44.08263],[16.43662,44.07523],[16.43629,44.02826],[16.50528,44.0244],[16.55472,43.95326],[16.70922,43.84887],[16.75316,43.77157],[16.80736,43.76011],[17.00585,43.58037],[17.15828,43.49376],[17.24411,43.49376],[17.29699,43.44542],[17.25579,43.40353],[17.286,43.33065],[17.46986,43.16559],[17.64268,43.08595],[17.70879,42.97223],[17.5392,42.92787],[17.6444,42.88641],[17.68151,42.92725],[17.7948,42.89556],[17.80854,42.9182],[17.88201,42.83668],[18.24318,42.6112],[18.36197,42.61423],[18.43735,42.55921],[18.49778,42.58409],[18.53751,42.57376],[18.55504,42.58409],[18.52232,42.62279],[18.57373,42.64429],[18.54841,42.68328],[18.54603,42.69171],[18.55221,42.69045],[18.56789,42.72074],[18.47324,42.74992],[18.45921,42.81682],[18.47633,42.85829],[18.4935,42.86433],[18.49661,42.89306],[18.49076,42.95553],[18.52232,43.01451],[18.66254,43.03928],[18.64735,43.14766],[18.66605,43.2056],[18.71747,43.2286],[18.6976,43.25243],[18.76538,43.29838],[18.85342,43.32426],[18.84794,43.33735],[18.83912,43.34795],[18.90911,43.36383],[18.95819,43.32899],[18.95001,43.29327],[19.00844,43.24988],[19.04233,43.30008],[19.08206,43.29668],[19.08673,43.31453],[19.04071,43.397],[19.01078,43.43854],[18.96053,43.45042],[18.95469,43.49367],[18.91379,43.50299],[19.01078,43.55806],[19.04934,43.50384],[19.13933,43.5282],[19.15685,43.53943],[19.22807,43.5264],[19.24774,43.53061],[19.2553,43.5938],[19.33426,43.58833],[19.36653,43.60921],[19.41941,43.54056],[19.42696,43.57987],[19.50455,43.58385],[19.5176,43.71403],[19.3986,43.79668],[19.23465,43.98764],[19.24363,44.01502],[19.38439,43.96611],[19.52515,43.95573],[19.56498,43.99922],[19.61836,44.01464],[19.61991,44.05254],[19.57467,44.04716],[19.55999,44.06894],[19.51167,44.08158],[19.47321,44.1193],[19.48386,44.14332],[19.47338,44.15034],[19.43905,44.13088],[19.40927,44.16722],[19.3588,44.18353],[19.34773,44.23244],[19.32464,44.27185],[19.26945,44.26957],[19.23306,44.26097],[19.20508,44.2917],[19.18328,44.28383],[19.16741,44.28648],[19.13332,44.31492],[19.13556,44.338],[19.11547,44.34218],[19.1083,44.3558],[19.11865,44.36712],[19.10298,44.36924],[19.10365,44.37795],[19.10704,44.38249],[19.10749,44.39421],[19.11785,44.40313],[19.14681,44.41463],[19.14837,44.45253],[19.12278,44.50132],[19.13369,44.52521],[19.16699,44.52197],[19.26388,44.65412],[19.32543,44.74058],[19.36722,44.88164],[19.18183,44.92055],[19.01994,44.85493],[18.8704,44.85097],[18.76347,44.90669],[18.76369,44.93707],[18.80661,44.93561],[18.78357,44.97741],[18.65723,45.07544],[18.47939,45.05871],[18.41896,45.11083],[18.32077,45.1021],[18.24387,45.13699],[18.1624,45.07654],[18.03121,45.12632],[18.01594,45.15163],[17.99479,45.14958],[17.97834,45.13831],[17.97336,45.12245],[17.93706,45.08016],[17.87148,45.04645],[17.84826,45.04489]]]]}},{type:"Feature",properties:{iso1A2:"BB",iso1A3:"BRB",iso1N3:"052",wikidata:"Q244",nameEn:"Barbados",groups:["029","003","419","019"],driveSide:"left",callingCodes:["1 246"]},geometry:{type:"MultiPolygon",coordinates:[[[[-58.56442,13.24471],[-59.80731,13.87556],[-60.19227,12.37597],[-58.56442,13.24471]]]]}},{type:"Feature",properties:{iso1A2:"BD",iso1A3:"BGD",iso1N3:"050",wikidata:"Q902",nameEn:"Bangladesh",groups:["034","142"],driveSide:"left",callingCodes:["880"]},geometry:{type:"MultiPolygon",coordinates:[[[[89.15869,26.13708],[89.08899,26.38845],[88.95612,26.4564],[88.92357,26.40711],[88.91321,26.37984],[89.05328,26.2469],[88.85004,26.23211],[88.78961,26.31093],[88.67837,26.26291],[88.69485,26.38353],[88.62144,26.46783],[88.4298,26.54489],[88.41196,26.63837],[88.33093,26.48929],[88.35153,26.45241],[88.36938,26.48683],[88.48749,26.45855],[88.51649,26.35923],[88.35153,26.29123],[88.34757,26.22216],[88.1844,26.14417],[88.16581,26.0238],[88.08804,25.91334],[88.13138,25.78773],[88.242,25.80811],[88.45103,25.66245],[88.4559,25.59227],[88.677,25.46959],[88.81296,25.51546],[88.85278,25.34679],[89.01105,25.30303],[89.00463,25.26583],[88.94067,25.18534],[88.44766,25.20149],[88.46277,25.07468],[88.33917,24.86803],[88.27325,24.88796],[88.21832,24.96642],[88.14004,24.93529],[88.15515,24.85806],[88.00683,24.66477],[88.08786,24.63232],[88.12296,24.51301],[88.50934,24.32474],[88.68801,24.31464],[88.74841,24.1959],[88.6976,24.14703],[88.73743,23.91751],[88.66189,23.87607],[88.58087,23.87105],[88.56507,23.64044],[88.74841,23.47361],[88.79351,23.50535],[88.79254,23.46028],[88.71133,23.2492],[88.99148,23.21134],[88.86377,23.08759],[88.88327,23.03885],[88.87063,22.95235],[88.96713,22.83346],[88.9151,22.75228],[88.94614,22.66941],[88.9367,22.58527],[89.07114,22.15335],[89.03553,21.77397],[89.13927,21.60785],[89.13606,21.42955],[92.39837,20.38919],[92.4302,20.5688],[92.31348,20.57137],[92.28464,20.63179],[92.37665,20.72172],[92.26071,21.05697],[92.17752,21.17445],[92.20087,21.337],[92.37939,21.47764],[92.43158,21.37025],[92.55105,21.3856],[92.60187,21.24615],[92.68152,21.28454],[92.59775,21.6092],[92.62187,21.87037],[92.60949,21.97638],[92.56616,22.13554],[92.60029,22.1522],[92.5181,22.71441],[92.37665,22.9435],[92.38214,23.28705],[92.26541,23.70392],[92.15417,23.73409],[92.04706,23.64229],[91.95093,23.73284],[91.95642,23.47361],[91.84789,23.42235],[91.76417,23.26619],[91.81634,23.08001],[91.7324,23.00043],[91.61571,22.93929],[91.54993,23.01051],[91.46615,23.2328],[91.4035,23.27522],[91.40848,23.07117],[91.36453,23.06612],[91.28293,23.37538],[91.15579,23.6599],[91.25192,23.83463],[91.22308,23.89616],[91.29587,24.0041],[91.35741,23.99072],[91.37414,24.10693],[91.55542,24.08687],[91.63782,24.1132],[91.65292,24.22095],[91.73257,24.14703],[91.76004,24.23848],[91.82596,24.22345],[91.89258,24.14674],[91.96603,24.3799],[92.11662,24.38997],[92.15796,24.54435],[92.25854,24.9191],[92.38626,24.86055],[92.49887,24.88796],[92.39147,25.01471],[92.33957,25.07593],[92.0316,25.1834],[91.63648,25.12846],[91.25517,25.20677],[90.87427,25.15799],[90.65042,25.17788],[90.40034,25.1534],[90.1155,25.22686],[89.90478,25.31038],[89.87629,25.28337],[89.83371,25.29548],[89.84086,25.31854],[89.81208,25.37244],[89.86129,25.61714],[89.84388,25.70042],[89.80585,25.82489],[89.86592,25.93115],[89.77728,26.04254],[89.77865,26.08387],[89.73581,26.15818],[89.70201,26.15138],[89.63968,26.22595],[89.57101,25.9682],[89.53515,26.00382],[89.35953,26.0077],[89.15869,26.13708]]]]}},{type:"Feature",properties:{iso1A2:"BE",iso1A3:"BEL",iso1N3:"056",wikidata:"Q31",nameEn:"Belgium",groups:["EU","155","150"],callingCodes:["32"]},geometry:{type:"MultiPolygon",coordinates:[[[[4.93295,51.44945],[4.93909,51.44632],[4.9524,51.45014],[4.95244,51.45207],[4.93295,51.44945]]],[[[4.91493,51.4353],[4.92652,51.43329],[4.92952,51.42984],[4.93986,51.43064],[4.94265,51.44003],[4.93471,51.43861],[4.93416,51.44185],[4.94025,51.44193],[4.93544,51.44634],[4.92879,51.44161],[4.92815,51.43856],[4.92566,51.44273],[4.92811,51.4437],[4.92287,51.44741],[4.91811,51.44621],[4.92227,51.44252],[4.91935,51.43634],[4.91493,51.4353]]],[[[4.82946,51.4213],[4.82409,51.44736],[4.84139,51.4799],[4.78803,51.50284],[4.77321,51.50529],[4.74578,51.48937],[4.72935,51.48424],[4.65442,51.42352],[4.57489,51.4324],[4.53521,51.4243],[4.52846,51.45002],[4.54675,51.47265],[4.5388,51.48184],[4.47736,51.4778],[4.38122,51.44905],[4.39747,51.43316],[4.38064,51.41965],[4.43777,51.36989],[4.39292,51.35547],[4.34086,51.35738],[4.33265,51.37687],[4.21923,51.37443],[4.24024,51.35371],[4.16721,51.29348],[4.05165,51.24171],[4.01957,51.24504],[3.97889,51.22537],[3.90125,51.20371],[3.78783,51.2151],[3.78999,51.25766],[3.58939,51.30064],[3.51502,51.28697],[3.52698,51.2458],[3.43488,51.24135],[3.41704,51.25933],[3.38289,51.27331],[3.35847,51.31572],[3.38696,51.33436],[3.36263,51.37112],[2.56575,51.85301],[2.18458,51.52087],[2.55904,51.07014],[2.57551,51.00326],[2.63074,50.94746],[2.59093,50.91751],[2.63331,50.81457],[2.71165,50.81295],[2.81056,50.71773],[2.8483,50.72276],[2.86985,50.7033],[2.87937,50.70298],[2.88504,50.70656],[2.90069,50.69263],[2.91036,50.6939],[2.90873,50.702],[2.95019,50.75138],[2.96778,50.75242],[3.00537,50.76588],[3.04314,50.77674],[3.09163,50.77717],[3.10614,50.78303],[3.11206,50.79416],[3.11987,50.79188],[3.1257,50.78603],[3.15017,50.79031],[3.16476,50.76843],[3.18339,50.74981],[3.18811,50.74025],[3.20064,50.73547],[3.19017,50.72569],[3.20845,50.71662],[3.22042,50.71019],[3.24593,50.71389],[3.26063,50.70086],[3.26141,50.69151],[3.2536,50.68977],[3.264,50.67668],[3.23951,50.6585],[3.2729,50.60718],[3.28575,50.52724],[3.37693,50.49538],[3.44629,50.51009],[3.47385,50.53397],[3.51564,50.5256],[3.49509,50.48885],[3.5683,50.50192],[3.58361,50.49049],[3.61014,50.49568],[3.64426,50.46275],[3.66153,50.45165],[3.67494,50.40239],[3.67262,50.38663],[3.65709,50.36873],[3.66976,50.34563],[3.71009,50.30305],[3.70987,50.3191],[3.73911,50.34809],[3.84314,50.35219],[3.90781,50.32814],[3.96771,50.34989],[4.0268,50.35793],[4.0689,50.3254],[4.10237,50.31247],[4.10957,50.30234],[4.11954,50.30425],[4.13665,50.25609],[4.16808,50.25786],[4.15524,50.2833],[4.17347,50.28838],[4.17861,50.27443],[4.20651,50.27333],[4.21945,50.25539],[4.15524,50.21103],[4.16014,50.19239],[4.13561,50.13078],[4.20147,50.13535],[4.23101,50.06945],[4.16294,50.04719],[4.13508,50.01976],[4.14239,49.98034],[4.20532,49.95803],[4.31963,49.97043],[4.35051,49.95315],[4.43488,49.94122],[4.51098,49.94659],[4.5414,49.96911],[4.68695,49.99685],[4.70064,50.09384],[4.75237,50.11314],[4.82438,50.16878],[4.83279,50.15331],[4.88602,50.15182],[4.8382,50.06738],[4.78827,49.95609],[4.88529,49.9236],[4.85134,49.86457],[4.86965,49.82271],[4.85464,49.78995],[4.96714,49.79872],[5.09249,49.76193],[5.14545,49.70287],[5.26232,49.69456],[5.31465,49.66846],[5.33039,49.6555],[5.30214,49.63055],[5.3137,49.61225],[5.33851,49.61599],[5.34837,49.62889],[5.3974,49.61596],[5.43713,49.5707],[5.46734,49.52648],[5.46541,49.49825],[5.55001,49.52729],[5.60909,49.51228],[5.64505,49.55146],[5.75649,49.54321],[5.7577,49.55915],[5.77435,49.56298],[5.79195,49.55228],[5.81838,49.54777],[5.84143,49.5533],[5.84692,49.55663],[5.8424,49.56082],[5.87256,49.57539],[5.86986,49.58756],[5.84971,49.58674],[5.84826,49.5969],[5.8762,49.60898],[5.87609,49.62047],[5.88393,49.62802],[5.88552,49.63507],[5.90599,49.63853],[5.90164,49.6511],[5.9069,49.66377],[5.86175,49.67862],[5.86527,49.69291],[5.88677,49.70951],[5.86503,49.72739],[5.84193,49.72161],[5.82562,49.72395],[5.83149,49.74729],[5.82245,49.75048],[5.78871,49.7962],[5.75409,49.79239],[5.74953,49.81428],[5.74364,49.82058],[5.74844,49.82435],[5.7404,49.83452],[5.74076,49.83823],[5.74975,49.83933],[5.74953,49.84709],[5.75884,49.84811],[5.74567,49.85368],[5.75861,49.85631],[5.75269,49.8711],[5.78415,49.87922],[5.73621,49.89796],[5.77314,49.93646],[5.77291,49.96056],[5.80833,49.96451],[5.81163,49.97142],[5.83467,49.97823],[5.83968,49.9892],[5.82331,49.99662],[5.81866,50.01286],[5.8551,50.02683],[5.86904,50.04614],[5.85474,50.06342],[5.8857,50.07824],[5.89488,50.11476],[5.95929,50.13295],[5.96453,50.17259],[6.02488,50.18283],[6.03093,50.16362],[6.06406,50.15344],[6.08577,50.17246],[6.12028,50.16374],[6.1137,50.13668],[6.1379,50.12964],[6.15298,50.14126],[6.14132,50.14971],[6.14588,50.17106],[6.18739,50.1822],[6.18364,50.20815],[6.16853,50.2234],[6.208,50.25179],[6.28797,50.27458],[6.29949,50.30887],[6.32488,50.32333],[6.35701,50.31139],[6.40641,50.32425],[6.40785,50.33557],[6.3688,50.35898],[6.34406,50.37994],[6.36852,50.40776],[6.37219,50.45397],[6.34005,50.46083],[6.3465,50.48833],[6.30809,50.50058],[6.26637,50.50272],[6.22335,50.49578],[6.20599,50.52089],[6.19193,50.5212],[6.18716,50.52653],[6.19579,50.5313],[6.19735,50.53576],[6.17802,50.54179],[6.17739,50.55875],[6.20281,50.56952],[6.22581,50.5907],[6.24005,50.58732],[6.24888,50.59869],[6.2476,50.60392],[6.26957,50.62444],[6.17852,50.6245],[6.11707,50.72231],[6.04428,50.72861],[6.0406,50.71848],[6.0326,50.72647],[6.03889,50.74618],[6.01976,50.75398],[5.97545,50.75441],[5.95942,50.7622],[5.89132,50.75124],[5.89129,50.75125],[5.88734,50.77092],[5.84888,50.75448],[5.84548,50.76542],[5.80673,50.7558],[5.77513,50.78308],[5.76533,50.78159],[5.74356,50.7691],[5.73904,50.75674],[5.72216,50.76398],[5.69469,50.75529],[5.68091,50.75804],[5.70107,50.7827],[5.68995,50.79641],[5.70118,50.80764],[5.65259,50.82309],[5.64009,50.84742],[5.64504,50.87107],[5.67886,50.88142],[5.69858,50.91046],[5.71626,50.90796],[5.72644,50.91167],[5.72545,50.92312],[5.74644,50.94723],[5.75927,50.95601],[5.74752,50.96202],[5.72875,50.95428],[5.71864,50.96092],[5.76242,50.99703],[5.77688,51.02483],[5.75961,51.03113],[5.77258,51.06196],[5.79835,51.05834],[5.79903,51.09371],[5.82921,51.09328],[5.83226,51.10585],[5.8109,51.10861],[5.80798,51.11661],[5.85508,51.14445],[5.82564,51.16753],[5.77697,51.1522],[5.77735,51.17845],[5.74617,51.18928],[5.70344,51.1829],[5.65528,51.18736],[5.65145,51.19788],[5.5603,51.22249],[5.5569,51.26544],[5.515,51.29462],[5.48476,51.30053],[5.46519,51.2849],[5.4407,51.28169],[5.41672,51.26248],[5.347,51.27502],[5.33886,51.26314],[5.29716,51.26104],[5.26461,51.26693],[5.23814,51.26064],[5.22542,51.26888],[5.24244,51.30495],[5.2002,51.32243],[5.16222,51.31035],[5.13377,51.31592],[5.13105,51.34791],[5.07102,51.39469],[5.10456,51.43163],[5.07891,51.4715],[5.04774,51.47022],[5.03281,51.48679],[5.0106,51.47167],[5.00393,51.44406],[4.92152,51.39487],[4.90016,51.41404],[4.84988,51.41502],[4.78941,51.41102],[4.77229,51.41337],[4.76577,51.43046],[4.78314,51.43319],[4.82946,51.4213]]]]}},{type:"Feature",properties:{iso1A2:"BF",iso1A3:"BFA",iso1N3:"854",wikidata:"Q965",nameEn:"Burkina Faso",groups:["011","202","002"],callingCodes:["226"]},geometry:{type:"MultiPolygon",coordinates:[[[[0.23859,15.00135],[0.06588,14.96961],[-0.24673,15.07805],[-0.72004,15.08655],[-1.05875,14.7921],[-1.32166,14.72774],[-1.68083,14.50023],[-1.97945,14.47709],[-1.9992,14.19011],[-2.10223,14.14878],[-2.47587,14.29671],[-2.66175,14.14713],[-2.84667,14.05532],[-2.90831,13.81174],[-2.88189,13.64921],[-3.26407,13.70699],[-3.28396,13.5422],[-3.23599,13.29035],[-3.43507,13.27272],[-3.4313,13.1588],[-3.54454,13.1781],[-3.7911,13.36665],[-3.96282,13.38164],[-3.90558,13.44375],[-3.96501,13.49778],[-4.34477,13.12927],[-4.21819,12.95722],[-4.238,12.71467],[-4.47356,12.71252],[-4.41412,12.31922],[-4.57703,12.19875],[-4.54841,12.1385],[-4.62546,12.13204],[-4.62987,12.06531],[-4.70692,12.06746],[-4.72893,12.01579],[-5.07897,11.97918],[-5.26389,11.84778],[-5.40258,11.8327],[-5.26389,11.75728],[-5.29251,11.61715],[-5.22867,11.60421],[-5.20665,11.43811],[-5.25509,11.36905],[-5.25949,11.24816],[-5.32553,11.21578],[-5.32994,11.13371],[-5.49284,11.07538],[-5.41579,10.84628],[-5.47083,10.75329],[-5.46643,10.56074],[-5.51058,10.43177],[-5.39602,10.2929],[-5.12465,10.29788],[-4.96453,9.99923],[-4.96621,9.89132],[-4.6426,9.70696],[-4.31392,9.60062],[-4.25999,9.76012],[-3.69703,9.94279],[-3.31779,9.91125],[-3.27228,9.84981],[-3.19306,9.93781],[-3.16609,9.85147],[-3.00765,9.74019],[-2.93012,9.57403],[-2.76494,9.40778],[-2.68802,9.49343],[-2.76534,9.56589],[-2.74174,9.83172],[-2.83108,10.40252],[-2.94232,10.64281],[-2.83373,11.0067],[-0.67143,10.99811],[-0.61937,10.91305],[-0.44298,11.04292],[-0.42391,11.11661],[-0.38219,11.12596],[-0.35955,11.07801],[-0.28566,11.12713],[-0.27374,11.17157],[-0.13493,11.14075],[0.50388,11.01011],[0.48852,10.98561],[0.50521,10.98035],[0.4958,10.93269],[0.66104,10.99964],[0.91245,10.99597],[0.9813,11.08876],[1.03409,11.04719],[1.42823,11.46822],[2.00988,11.42227],[2.29983,11.68254],[2.39723,11.89473],[2.05785,12.35539],[2.26349,12.41915],[0.99167,13.10727],[0.99253,13.37515],[1.18873,13.31771],[1.21217,13.37853],[1.24516,13.33968],[1.28509,13.35488],[1.24429,13.39373],[1.20088,13.38951],[1.02813,13.46635],[0.99514,13.5668],[0.77637,13.64442],[0.77377,13.6866],[0.61924,13.68491],[0.38051,14.05575],[0.16936,14.51654],[0.23859,15.00135]]]]}},{type:"Feature",properties:{iso1A2:"BG",iso1A3:"BGR",iso1N3:"100",wikidata:"Q219",nameEn:"Bulgaria",groups:["EU","151","150"],callingCodes:["359"]},geometry:{type:"MultiPolygon",coordinates:[[[[23.05288,43.79494],[22.85314,43.84452],[22.83753,43.88055],[22.87873,43.9844],[23.01674,44.01946],[23.04988,44.07694],[22.67173,44.21564],[22.61711,44.16938],[22.61688,44.06534],[22.41449,44.00514],[22.35558,43.81281],[22.41043,43.69566],[22.47582,43.6558],[22.53397,43.47225],[22.82036,43.33665],[22.89727,43.22417],[23.00806,43.19279],[22.98104,43.11199],[22.89521,43.03625],[22.78397,42.98253],[22.74826,42.88701],[22.54302,42.87774],[22.43309,42.82057],[22.4997,42.74144],[22.43983,42.56851],[22.55669,42.50144],[22.51961,42.3991],[22.47498,42.3915],[22.45919,42.33822],[22.34773,42.31725],[22.38136,42.30339],[22.47251,42.20393],[22.50289,42.19527],[22.51224,42.15457],[22.67701,42.06614],[22.86749,42.02275],[22.90254,41.87587],[22.96682,41.77137],[23.01239,41.76527],[23.03342,41.71034],[22.95513,41.63265],[22.96331,41.35782],[22.93334,41.34104],[23.1833,41.31755],[23.21953,41.33773],[23.22771,41.37106],[23.31301,41.40525],[23.33639,41.36317],[23.40416,41.39999],[23.52453,41.40262],[23.63203,41.37632],[23.67644,41.41139],[23.76525,41.40175],[23.80148,41.43943],[23.89613,41.45257],[23.91483,41.47971],[23.96975,41.44118],[24.06908,41.46132],[24.06323,41.53222],[24.10063,41.54796],[24.18126,41.51735],[24.27124,41.57682],[24.30513,41.51297],[24.52599,41.56808],[24.61129,41.42278],[24.71529,41.41928],[24.8041,41.34913],[24.82514,41.4035],[24.86136,41.39298],[24.90928,41.40876],[24.942,41.38685],[25.11611,41.34212],[25.28322,41.23411],[25.48187,41.28506],[25.52394,41.2798],[25.55082,41.31667],[25.61042,41.30614],[25.66183,41.31316],[25.70507,41.29209],[25.8266,41.34563],[25.87919,41.30526],[26.12926,41.35878],[26.16548,41.42278],[26.20288,41.43943],[26.14796,41.47533],[26.176,41.50072],[26.17951,41.55409],[26.14328,41.55496],[26.15146,41.60828],[26.07083,41.64584],[26.06148,41.70345],[26.16841,41.74858],[26.21325,41.73223],[26.22888,41.74139],[26.2654,41.71544],[26.30255,41.70925],[26.35957,41.71149],[26.32952,41.73637],[26.33589,41.76802],[26.36952,41.82265],[26.53968,41.82653],[26.57961,41.90024],[26.56051,41.92995],[26.62996,41.97644],[26.79143,41.97386],[26.95638,42.00741],[27.03277,42.0809],[27.08486,42.08735],[27.19251,42.06028],[27.22376,42.10152],[27.27411,42.10409],[27.45478,41.96591],[27.52379,41.93756],[27.55191,41.90928],[27.69949,41.97515],[27.81235,41.94803],[27.83492,41.99709],[27.91479,41.97902],[28.02971,41.98066],[28.32297,41.98371],[29.24336,43.70874],[28.23293,43.76],[27.99558,43.84193],[27.92008,44.00761],[27.73468,43.95326],[27.64542,44.04958],[27.60834,44.01206],[27.39757,44.0141],[27.26845,44.12602],[26.95141,44.13555],[26.62712,44.05698],[26.38764,44.04356],[26.10115,43.96908],[26.05584,43.90925],[25.94911,43.85745],[25.72792,43.69263],[25.39528,43.61866],[25.17144,43.70261],[25.10718,43.6831],[24.96682,43.72693],[24.73542,43.68523],[24.62281,43.74082],[24.50264,43.76314],[24.35364,43.70211],[24.18149,43.68218],[23.73978,43.80627],[23.61687,43.79289],[23.4507,43.84936],[23.26772,43.84843],[23.05288,43.79494]]]]}},{type:"Feature",properties:{iso1A2:"BH",iso1A3:"BHR",iso1N3:"048",wikidata:"Q398",nameEn:"Bahrain",groups:["145","142"],callingCodes:["973"]},geometry:{type:"MultiPolygon",coordinates:[[[[50.93865,26.30758],[50.71771,26.73086],[50.38162,26.53976],[50.26923,26.08243],[50.302,25.87592],[50.57069,25.57887],[50.80824,25.54641],[50.7801,25.595],[50.86149,25.6965],[50.81266,25.88946],[50.93865,26.30758]]]]}},{type:"Feature",properties:{iso1A2:"BI",iso1A3:"BDI",iso1N3:"108",wikidata:"Q967",nameEn:"Burundi",groups:["014","202","002"],callingCodes:["257"]},geometry:{type:"MultiPolygon",coordinates:[[[[30.54501,-2.41404],[30.42933,-2.31064],[30.14034,-2.43626],[29.95911,-2.33348],[29.88237,-2.75105],[29.36805,-2.82933],[29.32234,-2.6483],[29.0562,-2.58632],[29.04081,-2.7416],[29.00167,-2.78523],[29.00404,-2.81978],[29.0505,-2.81774],[29.09119,-2.87871],[29.09797,-2.91935],[29.16037,-2.95457],[29.17258,-2.99385],[29.25633,-3.05471],[29.21463,-3.3514],[29.23708,-3.75856],[29.43673,-4.44845],[29.63827,-4.44681],[29.75109,-4.45836],[29.77289,-4.41733],[29.82885,-4.36153],[29.88172,-4.35743],[30.03323,-4.26631],[30.22042,-4.01738],[30.45915,-3.56532],[30.84165,-3.25152],[30.83823,-2.97837],[30.6675,-2.98987],[30.57926,-2.89791],[30.4987,-2.9573],[30.40662,-2.86151],[30.52747,-2.65841],[30.41789,-2.66266],[30.54501,-2.41404]]]]}},{type:"Feature",properties:{iso1A2:"BJ",iso1A3:"BEN",iso1N3:"204",wikidata:"Q962",nameEn:"Benin",aliases:["DY"],groups:["011","202","002"],callingCodes:["229"]},geometry:{type:"MultiPolygon",coordinates:[[[[3.59375,11.70269],[3.48187,11.86092],[3.31613,11.88495],[3.25352,12.01467],[2.83978,12.40585],[2.6593,12.30631],[2.37783,12.24804],[2.39657,12.10952],[2.45824,11.98672],[2.39723,11.89473],[2.29983,11.68254],[2.00988,11.42227],[1.42823,11.46822],[1.03409,11.04719],[0.9813,11.08876],[0.91245,10.99597],[0.8804,10.803],[0.80358,10.71459],[0.77666,10.37665],[1.35507,9.99525],[1.36624,9.5951],[1.33675,9.54765],[1.41746,9.3226],[1.5649,9.16941],[1.61838,9.0527],[1.64249,6.99562],[1.55877,6.99737],[1.61812,6.74843],[1.58105,6.68619],[1.76906,6.43189],[1.79826,6.28221],[1.62913,6.24075],[1.67336,6.02702],[2.74181,6.13349],[2.70566,6.38038],[2.70464,6.50831],[2.74334,6.57291],[2.7325,6.64057],[2.78204,6.70514],[2.78823,6.76356],[2.73405,6.78508],[2.74024,6.92802],[2.71702,6.95722],[2.76965,7.13543],[2.74489,7.42565],[2.79442,7.43486],[2.78668,7.5116],[2.73405,7.5423],[2.73095,7.7755],[2.67523,7.87825],[2.77907,9.06924],[3.08017,9.10006],[3.14147,9.28375],[3.13928,9.47167],[3.25093,9.61632],[3.34726,9.70696],[3.32099,9.78032],[3.35383,9.83641],[3.54429,9.87739],[3.66908,10.18136],[3.57275,10.27185],[3.6844,10.46351],[3.78292,10.40538],[3.84243,10.59316],[3.71505,11.13015],[3.49175,11.29765],[3.59375,11.70269]]]]}},{type:"Feature",properties:{iso1A2:"BL",iso1A3:"BLM",iso1N3:"652",wikidata:"Q25362",nameEn:"Saint-Barthélemy",country:"FR",groups:["029","003","419","019"],callingCodes:["590"]},geometry:{type:"MultiPolygon",coordinates:[[[[-62.75637,18.13489],[-62.93924,18.02904],[-63.07669,17.79659],[-62.76692,17.64353],[-62.54836,17.8636],[-62.75637,18.13489]]]]}},{type:"Feature",properties:{iso1A2:"BM",iso1A3:"BMU",iso1N3:"060",wikidata:"Q23635",nameEn:"Bermuda",country:"GB",groups:["021","003","019"],driveSide:"left",callingCodes:["1 441"]},geometry:{type:"MultiPolygon",coordinates:[[[[-63.20987,32.6953],[-65.31453,32.68437],[-65.63955,31.43417],[-63.20987,32.6953]]]]}},{type:"Feature",properties:{iso1A2:"BN",iso1A3:"BRN",iso1N3:"096",wikidata:"Q921",nameEn:"Brunei",groups:["035","142"],driveSide:"left",callingCodes:["673"]},geometry:{type:"MultiPolygon",coordinates:[[[[115.16236,5.01011],[115.02521,5.35005],[114.08532,4.64632],[114.07448,4.58441],[114.15813,4.57],[114.26876,4.49878],[114.32176,4.34942],[114.32176,4.2552],[114.4416,4.27588],[114.49922,4.13108],[114.64211,4.00694],[114.78539,4.12205],[114.88039,4.4257],[114.83189,4.42387],[114.77303,4.72871],[114.8266,4.75062],[114.88841,4.81905],[114.96982,4.81146],[114.99417,4.88201],[115.05038,4.90275],[115.02955,4.82087],[115.02278,4.74137],[115.04064,4.63706],[115.07737,4.53418],[115.09978,4.39123],[115.31275,4.30806],[115.36346,4.33563],[115.2851,4.42295],[115.27819,4.63661],[115.20737,4.8256],[115.15092,4.87604],[115.16236,5.01011]]]]}},{type:"Feature",properties:{iso1A2:"BO",iso1A3:"BOL",iso1N3:"068",wikidata:"Q750",nameEn:"Bolivia",groups:["005","419","019"],callingCodes:["591"]},geometry:{type:"MultiPolygon",coordinates:[[[[-63.90248,-12.52544],[-64.22539,-12.45267],[-64.30708,-12.46398],[-64.99778,-11.98604],[-65.30027,-11.48749],[-65.28141,-10.86289],[-65.35402,-10.78685],[-65.37923,-10.35141],[-65.29019,-9.86253],[-65.40615,-9.63894],[-65.56244,-9.84266],[-65.68343,-9.75323],[-67.17784,-10.34016],[-68.71533,-11.14749],[-68.7651,-11.0496],[-68.75179,-11.03688],[-68.75265,-11.02383],[-68.74802,-11.00891],[-69.42792,-10.93451],[-69.47839,-10.95254],[-69.57156,-10.94555],[-68.98115,-11.8979],[-68.65044,-12.50689],[-68.85615,-12.87769],[-68.8864,-13.40792],[-69.05265,-13.68546],[-68.88135,-14.18639],[-69.36254,-14.94634],[-69.14856,-15.23478],[-69.40336,-15.61358],[-69.20291,-16.16668],[-69.09986,-16.22693],[-68.96238,-16.194],[-68.79464,-16.33272],[-68.98358,-16.42165],[-69.04027,-16.57214],[-69.00853,-16.66769],[-69.16896,-16.72233],[-69.62883,-17.28142],[-69.46863,-17.37466],[-69.46897,-17.4988],[-69.46623,-17.60518],[-69.34126,-17.72753],[-69.28671,-17.94844],[-69.07496,-18.03715],[-69.14807,-18.16893],[-69.07432,-18.28259],[-68.94987,-18.93302],[-68.87082,-19.06003],[-68.80602,-19.08355],[-68.61989,-19.27584],[-68.41218,-19.40499],[-68.66761,-19.72118],[-68.54611,-19.84651],[-68.57132,-20.03134],[-68.74273,-20.08817],[-68.7276,-20.46178],[-68.44023,-20.62701],[-68.55383,-20.7355],[-68.53957,-20.91542],[-68.40403,-20.94562],[-68.18816,-21.28614],[-67.85114,-22.87076],[-67.54284,-22.89771],[-67.18382,-22.81525],[-66.7298,-22.23644],[-66.29714,-22.08741],[-66.24077,-21.77837],[-66.03836,-21.84829],[-66.04832,-21.9187],[-65.9261,-21.93335],[-65.7467,-22.10105],[-65.61166,-22.09504],[-65.58694,-22.09794],[-65.57743,-22.07675],[-65.47435,-22.08908],[-64.99524,-22.08255],[-64.90014,-22.12136],[-64.67174,-22.18957],[-64.58888,-22.25035],[-64.4176,-22.67692],[-64.35108,-22.73282],[-64.31489,-22.88824],[-64.22918,-22.55807],[-63.93287,-21.99934],[-63.70963,-21.99934],[-63.68113,-22.0544],[-63.66482,-21.99918],[-62.81124,-21.9987],[-62.8078,-22.12534],[-62.64455,-22.25091],[-62.2757,-21.06657],[-62.26883,-20.55311],[-61.93912,-20.10053],[-61.73723,-19.63958],[-60.00638,-19.2981],[-59.06965,-19.29148],[-58.23216,-19.80058],[-58.16225,-20.16193],[-57.8496,-19.98346],[-58.14215,-19.76276],[-57.78463,-19.03259],[-57.71113,-19.03161],[-57.69134,-19.00544],[-57.71995,-18.97546],[-57.71995,-18.89573],[-57.76764,-18.90087],[-57.56807,-18.25655],[-57.48237,-18.24219],[-57.69877,-17.8431],[-57.73949,-17.56095],[-57.90082,-17.44555],[-57.99661,-17.5273],[-58.32935,-17.28195],[-58.5058,-16.80958],[-58.30918,-16.3699],[-58.32431,-16.25861],[-58.41506,-16.32636],[-60.16069,-16.26479],[-60.23797,-15.50267],[-60.58224,-15.09887],[-60.23968,-15.09515],[-60.27887,-14.63021],[-60.46037,-14.22496],[-60.48053,-13.77981],[-61.05527,-13.50054],[-61.81151,-13.49564],[-63.76259,-12.42952],[-63.90248,-12.52544]]]]}},{type:"Feature",properties:{iso1A2:"BQ",iso1A3:"BES",iso1N3:"535",wikidata:"Q27561",nameEn:"Caribbean Netherlands",country:"NL",groups:["029","003","419","019"],callingCodes:["599 3","599 4","599 7"]},geometry:{type:"MultiPolygon",coordinates:[[[[-63.07669,17.79659],[-63.22932,17.32592],[-63.11114,17.23125],[-62.76692,17.64353],[-63.07669,17.79659]]],[[[-63.29212,17.90532],[-63.58819,17.61311],[-63.22932,17.32592],[-63.07669,17.79659],[-63.29212,17.90532]]],[[[-67.89186,12.4116],[-68.90012,12.62309],[-68.33524,11.78151],[-68.01417,11.77722],[-67.89186,12.4116]]]]}},{type:"Feature",properties:{iso1A2:"BR",iso1A3:"BRA",iso1N3:"076",wikidata:"Q155",nameEn:"Brazil",groups:["005","419","019"],callingCodes:["55"]},geometry:{type:"MultiPolygon",coordinates:[[[[-59.69361,4.34069],[-59.78878,4.45637],[-60.15953,4.53456],[-60.04189,4.69801],[-59.98129,5.07097],[-60.20944,5.28754],[-60.32352,5.21299],[-60.73204,5.20931],[-60.5802,4.94312],[-60.86539,4.70512],[-60.98303,4.54167],[-61.15703,4.49839],[-61.31457,4.54167],[-61.29675,4.44216],[-61.48569,4.43149],[-61.54629,4.2822],[-62.13094,4.08309],[-62.44822,4.18621],[-62.57656,4.04754],[-62.74411,4.03331],[-62.7655,3.73099],[-62.98296,3.59935],[-63.21111,3.96219],[-63.4464,3.9693],[-63.42233,3.89995],[-63.50611,3.83592],[-63.67099,4.01731],[-63.70218,3.91417],[-63.86082,3.94796],[-63.99183,3.90172],[-64.14512,4.12932],[-64.57648,4.12576],[-64.72977,4.28931],[-64.84028,4.24665],[-64.48379,3.7879],[-64.02908,2.79797],[-64.0257,2.48156],[-63.39114,2.4317],[-63.39827,2.16098],[-64.06135,1.94722],[-64.08274,1.64792],[-64.34654,1.35569],[-64.38932,1.5125],[-65.11657,1.12046],[-65.57288,0.62856],[-65.50158,0.92086],[-65.6727,1.01353],[-66.28507,0.74585],[-66.85795,1.22998],[-67.08222,1.17441],[-67.15784,1.80439],[-67.299,1.87494],[-67.40488,2.22258],[-67.9292,1.82455],[-68.18632,2.00091],[-68.26699,1.83463],[-68.18128,1.72881],[-69.38621,1.70865],[-69.53746,1.76408],[-69.83491,1.69353],[-69.82987,1.07864],[-69.26017,1.06856],[-69.14422,0.84172],[-69.20976,0.57958],[-69.47696,0.71065],[-70.04162,0.55437],[-70.03658,-0.19681],[-69.603,-0.51947],[-69.59796,-0.75136],[-69.4215,-1.01853],[-69.43395,-1.42219],[-69.94708,-4.2431],[-70.00888,-4.37833],[-70.11305,-4.27281],[-70.19582,-4.3607],[-70.33236,-4.15214],[-70.77601,-4.15717],[-70.96814,-4.36915],[-71.87003,-4.51661],[-72.64391,-5.0391],[-72.83973,-5.14765],[-73.24579,-6.05764],[-73.12983,-6.43852],[-73.73986,-6.87919],[-73.77011,-7.28944],[-73.96938,-7.58465],[-73.65485,-7.77897],[-73.76576,-7.89884],[-72.92886,-9.04074],[-73.21498,-9.40904],[-72.72216,-9.41397],[-72.31883,-9.5184],[-72.14742,-9.98049],[-71.23394,-9.9668],[-70.53373,-9.42628],[-70.58453,-9.58303],[-70.55429,-9.76692],[-70.62487,-9.80666],[-70.64134,-11.0108],[-70.51395,-10.92249],[-70.38791,-11.07096],[-69.90896,-10.92744],[-69.57835,-10.94051],[-69.57156,-10.94555],[-69.47839,-10.95254],[-69.42792,-10.93451],[-68.74802,-11.00891],[-68.75265,-11.02383],[-68.75179,-11.03688],[-68.7651,-11.0496],[-68.71533,-11.14749],[-67.17784,-10.34016],[-65.68343,-9.75323],[-65.56244,-9.84266],[-65.40615,-9.63894],[-65.29019,-9.86253],[-65.37923,-10.35141],[-65.35402,-10.78685],[-65.28141,-10.86289],[-65.30027,-11.48749],[-64.99778,-11.98604],[-64.30708,-12.46398],[-64.22539,-12.45267],[-63.90248,-12.52544],[-63.76259,-12.42952],[-61.81151,-13.49564],[-61.05527,-13.50054],[-60.48053,-13.77981],[-60.46037,-14.22496],[-60.27887,-14.63021],[-60.23968,-15.09515],[-60.58224,-15.09887],[-60.23797,-15.50267],[-60.16069,-16.26479],[-58.41506,-16.32636],[-58.32431,-16.25861],[-58.30918,-16.3699],[-58.5058,-16.80958],[-58.32935,-17.28195],[-57.99661,-17.5273],[-57.90082,-17.44555],[-57.73949,-17.56095],[-57.69877,-17.8431],[-57.48237,-18.24219],[-57.56807,-18.25655],[-57.76764,-18.90087],[-57.71995,-18.89573],[-57.71995,-18.97546],[-57.69134,-19.00544],[-57.71113,-19.03161],[-57.78463,-19.03259],[-58.14215,-19.76276],[-57.8496,-19.98346],[-58.16225,-20.16193],[-57.84536,-20.93155],[-57.93492,-21.65505],[-57.88239,-21.6868],[-57.94642,-21.73799],[-57.98625,-22.09157],[-56.6508,-22.28387],[-56.5212,-22.11556],[-56.45893,-22.08072],[-56.23206,-22.25347],[-55.8331,-22.29008],[-55.74941,-22.46436],[-55.741,-22.52018],[-55.72366,-22.5519],[-55.6986,-22.56268],[-55.68742,-22.58407],[-55.62493,-22.62765],[-55.63849,-22.95122],[-55.5446,-23.22811],[-55.52288,-23.2595],[-55.5555,-23.28237],[-55.43585,-23.87157],[-55.44117,-23.9185],[-55.41784,-23.9657],[-55.12292,-23.99669],[-55.0518,-23.98666],[-55.02691,-23.97317],[-54.6238,-23.83078],[-54.32807,-24.01865],[-54.28207,-24.07305],[-54.4423,-25.13381],[-54.62033,-25.46026],[-54.60196,-25.48397],[-54.59509,-25.53696],[-54.59398,-25.59224],[-54.5502,-25.58915],[-54.52926,-25.62846],[-53.90831,-25.55513],[-53.83691,-25.94849],[-53.73511,-26.04211],[-53.73086,-26.05842],[-53.7264,-26.0664],[-53.73391,-26.07006],[-53.73968,-26.10012],[-53.65018,-26.19501],[-53.65237,-26.23289],[-53.63739,-26.2496],[-53.63881,-26.25075],[-53.64632,-26.24798],[-53.64186,-26.25976],[-53.64505,-26.28089],[-53.68269,-26.33359],[-53.73372,-26.6131],[-53.80144,-27.09844],[-54.15978,-27.2889],[-54.19062,-27.27639],[-54.19268,-27.30751],[-54.41888,-27.40882],[-54.50416,-27.48232],[-54.67657,-27.57214],[-54.90159,-27.63132],[-54.90805,-27.73149],[-55.1349,-27.89759],[-55.16872,-27.86224],[-55.33303,-27.94661],[-55.6262,-28.17124],[-55.65418,-28.18304],[-56.01729,-28.51223],[-56.00458,-28.60421],[-56.05265,-28.62651],[-56.54171,-29.11447],[-56.57295,-29.11357],[-56.62789,-29.18073],[-56.81251,-29.48154],[-57.09386,-29.74211],[-57.65132,-30.19229],[-57.22502,-30.26121],[-56.90236,-30.02578],[-56.49267,-30.39471],[-56.4795,-30.3899],[-56.4619,-30.38457],[-55.87388,-31.05053],[-55.58866,-30.84117],[-55.5634,-30.8686],[-55.55373,-30.8732],[-55.55218,-30.88193],[-55.54572,-30.89051],[-55.53431,-30.89714],[-55.53276,-30.90218],[-55.52712,-30.89997],[-55.51862,-30.89828],[-55.50841,-30.9027],[-55.50821,-30.91349],[-54.17384,-31.86168],[-53.76024,-32.0751],[-53.39572,-32.58596],[-53.37671,-32.57005],[-53.1111,-32.71147],[-53.53459,-33.16843],[-53.52794,-33.68908],[-53.44031,-33.69344],[-53.39593,-33.75169],[-53.37138,-33.74313],[-52.83257,-34.01481],[-28.34015,-20.99094],[-28.99601,1.86593],[-51.35485,4.8383],[-51.63798,4.51394],[-51.61983,4.14596],[-51.79599,3.89336],[-51.82312,3.85825],[-51.85573,3.83427],[-52.31787,3.17896],[-52.6906,2.37298],[-52.96539,2.1881],[-53.78743,2.34412],[-54.16286,2.10779],[-54.6084,2.32856],[-55.01919,2.564],[-55.71493,2.40342],[-55.96292,2.53188],[-56.13054,2.27723],[-55.92159,2.05236],[-55.89863,1.89861],[-55.99278,1.83137],[-56.47045,1.95135],[-56.7659,1.89509],[-57.07092,1.95304],[-57.09109,2.01854],[-57.23981,1.95808],[-57.35073,1.98327],[-57.55743,1.69605],[-57.77281,1.73344],[-57.97336,1.64566],[-58.01873,1.51966],[-58.33887,1.58014],[-58.4858,1.48399],[-58.53571,1.29154],[-58.84229,1.17749],[-58.92072,1.31293],[-59.25583,1.40559],[-59.74066,1.87596],[-59.7264,2.27497],[-59.91177,2.36759],[-59.99733,2.92312],[-59.79769,3.37162],[-59.86899,3.57089],[-59.51963,3.91951],[-59.73353,4.20399],[-59.69361,4.34069]]]]}},{type:"Feature",properties:{iso1A2:"BS",iso1A3:"BHS",iso1N3:"044",wikidata:"Q778",nameEn:"The Bahamas",groups:["029","003","419","019"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["1 242"]},geometry:{type:"MultiPolygon",coordinates:[[[[-72.98446,20.4801],[-71.70065,25.7637],[-79.14818,27.83105],[-79.89631,24.6597],[-80.88924,23.80416],[-72.98446,20.4801]]]]}},{type:"Feature",properties:{iso1A2:"BT",iso1A3:"BTN",iso1N3:"064",wikidata:"Q917",nameEn:"Bhutan",groups:["034","142"],driveSide:"left",callingCodes:["975"]},geometry:{type:"MultiPolygon",coordinates:[[[[91.6469,27.76358],[91.5629,27.84823],[91.48973,27.93903],[91.46327,28.0064],[91.25779,28.07509],[91.20019,27.98715],[90.69894,28.07784],[90.58842,28.02838],[90.13387,28.19178],[89.79762,28.23979],[89.59525,28.16433],[89.12825,27.62502],[89.0582,27.60985],[88.97213,27.51671],[88.95355,27.4106],[89.00216,27.32532],[88.96947,27.30319],[88.93678,27.33777],[88.91901,27.32483],[88.74219,27.144],[88.86984,27.10937],[88.8714,26.97488],[88.92301,26.99286],[88.95807,26.92668],[89.09554,26.89089],[89.12825,26.81661],[89.1926,26.81329],[89.37913,26.86224],[89.38319,26.85963],[89.3901,26.84225],[89.42349,26.83727],[89.63369,26.74402],[89.86124,26.73307],[90.04535,26.72422],[90.30402,26.85098],[90.39271,26.90704],[90.48504,26.8594],[90.67715,26.77215],[91.50067,26.79223],[91.83181,26.87318],[92.05523,26.8692],[92.11863,26.893],[92.03457,27.07334],[92.04702,27.26861],[92.12019,27.27829],[92.01132,27.47352],[91.65007,27.48287],[91.55819,27.6144],[91.6469,27.76358]]]]}},{type:"Feature",properties:{iso1A2:"BV",iso1A3:"BVT",iso1N3:"074",wikidata:"Q23408",nameEn:"Bouvet Island",country:"NO",groups:["005","419","019"]},geometry:{type:"MultiPolygon",coordinates:[[[[4.54042,-54.0949],[2.28941,-54.13089],[3.35353,-55.17558],[4.54042,-54.0949]]]]}},{type:"Feature",properties:{iso1A2:"BW",iso1A3:"BWA",iso1N3:"072",wikidata:"Q963",nameEn:"Botswana",groups:["018","202","002"],driveSide:"left",callingCodes:["267"]},geometry:{type:"MultiPolygon",coordinates:[[[[25.26433,-17.79571],[25.16882,-17.78253],[25.05895,-17.84452],[24.95586,-17.79674],[24.73364,-17.89338],[24.71887,-17.9218],[24.6303,-17.9863],[24.57485,-18.07151],[24.40577,-17.95726],[24.19416,-18.01919],[23.61088,-18.4881],[23.29618,-17.99855],[23.0996,-18.00075],[21.45556,-18.31795],[20.99904,-18.31743],[20.99751,-22.00026],[19.99912,-21.99991],[19.99817,-24.76768],[20.02809,-24.78725],[20.03678,-24.81004],[20.29826,-24.94869],[20.64795,-25.47827],[20.86081,-26.14892],[20.61754,-26.4692],[20.63275,-26.78181],[20.68596,-26.9039],[20.87031,-26.80047],[21.13353,-26.86661],[21.37869,-26.82083],[21.69322,-26.86152],[21.7854,-26.79199],[21.77114,-26.69015],[21.83291,-26.65959],[21.90703,-26.66808],[22.06192,-26.61882],[22.21206,-26.3773],[22.41921,-26.23078],[22.56365,-26.19668],[22.70808,-25.99186],[22.86012,-25.50572],[23.03497,-25.29971],[23.47588,-25.29971],[23.9244,-25.64286],[24.18287,-25.62916],[24.36531,-25.773],[24.44703,-25.73021],[24.67319,-25.81749],[24.8946,-25.80723],[25.01718,-25.72507],[25.12266,-25.75931],[25.33076,-25.76616],[25.58543,-25.6343],[25.6643,-25.4491],[25.69661,-25.29284],[25.72702,-25.25503],[25.88571,-24.87802],[25.84295,-24.78661],[25.8515,-24.75727],[26.39409,-24.63468],[26.46346,-24.60358],[26.51667,-24.47219],[26.84165,-24.24885],[26.99749,-23.65486],[27.33768,-23.40917],[27.52393,-23.37952],[27.6066,-23.21894],[27.74154,-23.2137],[27.93539,-23.04941],[27.93729,-22.96194],[28.04752,-22.90243],[28.04562,-22.8394],[28.34874,-22.5694],[28.63287,-22.55887],[28.91889,-22.44299],[29.0151,-22.22907],[29.10881,-22.21202],[29.15268,-22.21399],[29.18974,-22.18599],[29.21955,-22.17771],[29.37703,-22.19581],[29.3533,-22.18363],[29.24648,-22.05967],[29.1974,-22.07472],[29.14501,-22.07275],[29.08495,-22.04867],[29.04108,-22.00563],[29.02191,-21.95665],[29.02191,-21.90647],[29.04023,-21.85864],[29.07763,-21.81877],[28.58114,-21.63455],[28.49942,-21.66634],[28.29416,-21.59037],[28.01669,-21.57624],[27.91407,-21.31621],[27.69171,-21.08409],[27.72972,-20.51735],[27.69361,-20.48531],[27.28865,-20.49873],[27.29831,-20.28935],[27.21278,-20.08244],[26.72246,-19.92707],[26.17227,-19.53709],[25.96226,-19.08152],[25.99837,-19.02943],[25.94326,-18.90362],[25.82353,-18.82808],[25.79217,-18.6355],[25.68859,-18.56165],[25.53465,-18.39041],[25.39972,-18.12691],[25.31799,-18.07091],[25.23909,-17.90832],[25.26433,-17.79571]]]]}},{type:"Feature",properties:{iso1A2:"BY",iso1A3:"BLR",iso1N3:"112",wikidata:"Q184",nameEn:"Belarus",groups:["151","150"],callingCodes:["375"]},geometry:{type:"MultiPolygon",coordinates:[[[[28.15217,56.16964],[27.97865,56.11849],[27.63065,55.89687],[27.61683,55.78558],[27.3541,55.8089],[27.27804,55.78299],[27.1559,55.85032],[26.97153,55.8102],[26.87448,55.7172],[26.76872,55.67658],[26.71802,55.70645],[26.64888,55.70515],[26.63231,55.67968],[26.63167,55.57887],[26.55094,55.5093],[26.5522,55.40277],[26.44937,55.34832],[26.5709,55.32572],[26.6714,55.33902],[26.80929,55.31642],[26.83266,55.30444],[26.835,55.28182],[26.73017,55.24226],[26.72983,55.21788],[26.68075,55.19787],[26.69243,55.16718],[26.54753,55.14181],[26.51481,55.16051],[26.46249,55.12814],[26.35121,55.1525],[26.30628,55.12536],[26.23202,55.10439],[26.26941,55.08032],[26.20397,54.99729],[26.13386,54.98924],[26.05907,54.94631],[25.99129,54.95705],[25.89462,54.93438],[25.74122,54.80108],[25.75977,54.57252],[25.68045,54.5321],[25.64813,54.48704],[25.62203,54.4656],[25.63371,54.42075],[25.5376,54.33158],[25.55425,54.31591],[25.68513,54.31727],[25.78553,54.23327],[25.78563,54.15747],[25.71084,54.16704],[25.64875,54.1259],[25.54724,54.14925],[25.51452,54.17799],[25.56823,54.25212],[25.509,54.30267],[25.35559,54.26544],[25.22705,54.26271],[25.19199,54.219],[25.0728,54.13419],[24.991,54.14241],[24.96894,54.17589],[24.77131,54.11091],[24.85311,54.02862],[24.74279,53.96663],[24.69185,53.96543],[24.69652,54.01901],[24.62275,54.00217],[24.44411,53.90076],[24.34128,53.90076],[24.19638,53.96405],[23.98837,53.92554],[23.95098,53.9613],[23.81309,53.94205],[23.80543,53.89558],[23.71726,53.93379],[23.61677,53.92691],[23.51284,53.95052],[23.62004,53.60942],[23.81995,53.24131],[23.85657,53.22923],[23.91393,53.16469],[23.87548,53.0831],[23.92184,53.02079],[23.94689,52.95919],[23.91805,52.94016],[23.93763,52.71332],[23.73615,52.6149],[23.58296,52.59868],[23.45112,52.53774],[23.34141,52.44845],[23.18196,52.28812],[23.20071,52.22848],[23.47859,52.18215],[23.54314,52.12148],[23.61,52.11264],[23.64066,52.07626],[23.68733,51.9906],[23.61523,51.92066],[23.62691,51.78208],[23.53198,51.74298],[23.57053,51.55938],[23.56236,51.53673],[23.62751,51.50512],[23.6736,51.50255],[23.60906,51.62122],[23.7766,51.66809],[23.91118,51.63316],[23.8741,51.59734],[23.99907,51.58369],[24.13075,51.66979],[24.3163,51.75063],[24.29021,51.80841],[24.37123,51.88222],[24.98784,51.91273],[25.20228,51.97143],[25.46163,51.92205],[25.73673,51.91973],[25.80574,51.94556],[25.83217,51.92587],[26.00408,51.92967],[26.19084,51.86781],[26.39367,51.87315],[26.46962,51.80501],[26.69759,51.82284],[26.80043,51.75777],[26.9489,51.73788],[26.99422,51.76933],[27.20602,51.77291],[27.20948,51.66713],[27.26613,51.65957],[27.24828,51.60161],[27.47212,51.61184],[27.51058,51.5854],[27.55727,51.63486],[27.71932,51.60672],[27.67125,51.50854],[27.76052,51.47604],[27.85253,51.62293],[27.91844,51.61952],[27.95827,51.56065],[28.10658,51.57857],[28.23452,51.66988],[28.37592,51.54505],[28.47051,51.59734],[28.64429,51.5664],[28.69161,51.44695],[28.73143,51.46236],[28.75615,51.41442],[28.78224,51.45294],[28.76027,51.48802],[28.81795,51.55552],[28.95528,51.59222],[28.99098,51.56833],[29.1187,51.65872],[29.16402,51.64679],[29.20659,51.56918],[29.25603,51.57089],[29.25191,51.49828],[29.32881,51.37843],[29.42357,51.4187],[29.49773,51.39814],[29.54372,51.48372],[29.7408,51.53417],[29.77376,51.4461],[30.17888,51.51025],[30.34642,51.42555],[30.36153,51.33984],[30.56203,51.25655],[30.64992,51.35014],[30.51946,51.59649],[30.68804,51.82806],[30.76443,51.89739],[30.90897,52.00699],[30.95589,52.07775],[31.13332,52.1004],[31.25142,52.04131],[31.38326,52.12991],[31.7822,52.11406],[31.77877,52.18636],[31.6895,52.1973],[31.70735,52.26711],[31.57971,52.32146],[31.62084,52.33849],[31.61397,52.48843],[31.56316,52.51518],[31.63869,52.55361],[31.50406,52.69707],[31.57277,52.71613],[31.592,52.79011],[31.35667,52.97854],[31.24147,53.031],[31.32283,53.04101],[31.33519,53.08805],[31.3915,53.09712],[31.36403,53.13504],[31.40523,53.21406],[31.56316,53.19432],[31.62496,53.22886],[31.787,53.18033],[31.82373,53.10042],[32.15368,53.07594],[32.40773,53.18856],[32.51725,53.28431],[32.73257,53.33494],[32.74968,53.45597],[32.47777,53.5548],[32.40499,53.6656],[32.50112,53.68594],[32.45717,53.74039],[32.36663,53.7166],[32.12621,53.81586],[31.89137,53.78099],[31.77028,53.80015],[31.85019,53.91801],[31.88744,54.03653],[31.89599,54.0837],[31.57002,54.14535],[31.30791,54.25315],[31.3177,54.34067],[31.22945,54.46585],[31.08543,54.50361],[31.21399,54.63113],[31.19339,54.66947],[30.99187,54.67046],[30.98226,54.68872],[31.0262,54.70698],[30.97127,54.71967],[30.95479,54.74346],[30.75165,54.80699],[30.8264,54.90062],[30.81759,54.94064],[30.93144,54.9585],[30.95754,54.98609],[30.9081,55.02232],[30.94243,55.03964],[31.00972,55.02783],[31.02071,55.06167],[30.97369,55.17134],[30.87944,55.28223],[30.81946,55.27931],[30.8257,55.3313],[30.93144,55.3914],[30.90123,55.46621],[30.95204,55.50667],[30.93419,55.6185],[30.86003,55.63169],[30.7845,55.58514],[30.72957,55.66268],[30.67464,55.64176],[30.63344,55.73079],[30.51037,55.76568],[30.51346,55.78982],[30.48257,55.81066],[30.30987,55.83592],[30.27776,55.86819],[30.12136,55.8358],[29.97975,55.87281],[29.80672,55.79569],[29.61446,55.77716],[29.51283,55.70294],[29.3604,55.75862],[29.44692,55.95978],[29.21717,55.98971],[29.08299,56.03427],[28.73418,55.97131],[28.63668,56.07262],[28.68337,56.10173],[28.5529,56.11705],[28.43068,56.09407],[28.37987,56.11399],[28.36888,56.05805],[28.30571,56.06035],[28.15217,56.16964]]]]}},{type:"Feature",properties:{iso1A2:"BZ",iso1A3:"BLZ",iso1N3:"084",wikidata:"Q242",nameEn:"Belize",groups:["013","003","419","019"],roadSpeedUnit:"mph",callingCodes:["501"]},geometry:{type:"MultiPolygon",coordinates:[[[[-88.3268,18.49048],[-88.48242,18.49164],[-88.71505,18.0707],[-88.8716,17.89535],[-89.03839,18.0067],[-89.15105,17.95104],[-89.14985,17.81563],[-89.15025,17.04813],[-89.22683,15.88619],[-89.17418,15.90898],[-89.02415,15.9063],[-88.95358,15.88698],[-88.40779,16.09624],[-86.92368,17.61462],[-87.84815,18.18511],[-87.85693,18.18266],[-87.86657,18.19971],[-87.87604,18.18313],[-87.90671,18.15213],[-88.03165,18.16657],[-88.03238,18.41778],[-88.26593,18.47617],[-88.29909,18.47591],[-88.3268,18.49048]]]]}},{type:"Feature",properties:{iso1A2:"CA",iso1A3:"CAN",iso1N3:"124",wikidata:"Q16",nameEn:"Canada",groups:["021","003","019"],callingCodes:["1"]},geometry:{type:"MultiPolygon",coordinates:[[[[-67.20349,45.1722],[-67.19603,45.16771],[-67.15965,45.16179],[-67.11316,45.11176],[-67.0216,44.95333],[-66.96824,44.90965],[-66.98249,44.87071],[-66.96824,44.83078],[-66.93432,44.82597],[-67.16117,44.20069],[-61.98255,37.34815],[-56.27503,47.39728],[-53.12387,41.40385],[-46.37635,57.3249],[-76.75614,76.72014],[-68.21821,80.48551],[-45.47832,84.58738],[-140.97446,84.39275],[-141.00116,60.30648],[-140.5227,60.22077],[-140.45648,60.30919],[-139.98024,60.18027],[-139.68991,60.33693],[-139.05831,60.35205],[-139.20603,60.08896],[-139.05365,59.99655],[-138.71149,59.90728],[-138.62145,59.76431],[-137.60623,59.24465],[-137.4925,58.89415],[-136.82619,59.16198],[-136.52365,59.16752],[-136.47323,59.46617],[-136.33727,59.44466],[-136.22381,59.55526],[-136.31566,59.59083],[-135.48007,59.79937],[-135.03069,59.56208],[-135.00267,59.28745],[-134.7047,59.2458],[-134.55699,59.1297],[-134.48059,59.13231],[-134.27175,58.8634],[-133.84645,58.73543],[-133.38523,58.42773],[-131.8271,56.62247],[-130.77769,56.36185],[-130.33965,56.10849],[-130.10173,56.12178],[-130.00093,56.00325],[-130.00857,55.91344],[-130.15373,55.74895],[-129.97513,55.28029],[-130.08035,55.21556],[-130.18765,55.07744],[-130.27203,54.97174],[-130.44184,54.85377],[-130.64499,54.76912],[-130.61931,54.70835],[-133.92876,54.62289],[-133.36909,48.51151],[-125.03842,48.53282],[-123.50039,48.21223],[-123.15614,48.35395],[-123.26565,48.6959],[-123.0093,48.76586],[-123.0093,48.83186],[-123.32163,49.00419],[-117.03266,49.00056],[-116.04938,48.99999],[-114.0683,48.99885],[-110.0051,48.99901],[-104.05004,48.99925],[-101.36198,48.99935],[-97.24024,48.99952],[-95.15355,48.9996],[-95.15357,49.384],[-95.12903,49.37056],[-95.05825,49.35311],[-95.01419,49.35647],[-94.99532,49.36579],[-94.95681,49.37035],[-94.85381,49.32492],[-94.8159,49.32299],[-94.82487,49.29483],[-94.77355,49.11998],[-94.75017,49.09931],[-94.687,48.84077],[-94.70087,48.8339],[-94.70486,48.82365],[-94.69669,48.80918],[-94.69335,48.77883],[-94.58903,48.71803],[-94.54885,48.71543],[-94.53826,48.70216],[-94.44258,48.69223],[-94.4174,48.71049],[-94.27153,48.70232],[-94.25172,48.68404],[-94.25104,48.65729],[-94.23215,48.65202],[-93.85769,48.63284],[-93.83288,48.62745],[-93.80676,48.58232],[-93.80939,48.52439],[-93.79267,48.51631],[-93.66382,48.51845],[-93.47022,48.54357],[-93.44472,48.59147],[-93.40693,48.60948],[-93.39758,48.60364],[-93.3712,48.60599],[-93.33946,48.62787],[-93.25391,48.64266],[-92.94973,48.60866],[-92.7287,48.54005],[-92.6342,48.54133],[-92.62747,48.50278],[-92.69927,48.49573],[-92.71323,48.46081],[-92.65606,48.43471],[-92.50712,48.44921],[-92.45588,48.40624],[-92.48147,48.36609],[-92.37185,48.22259],[-92.27167,48.25046],[-92.30939,48.31251],[-92.26662,48.35651],[-92.202,48.35252],[-92.14732,48.36578],[-92.05339,48.35958],[-91.98929,48.25409],[-91.86125,48.21278],[-91.71231,48.19875],[-91.70451,48.11805],[-91.55649,48.10611],[-91.58025,48.04339],[-91.45829,48.07454],[-91.43248,48.04912],[-91.25025,48.08522],[-91.08016,48.18096],[-90.87588,48.2484],[-90.75045,48.09143],[-90.56444,48.12184],[-90.56312,48.09488],[-90.07418,48.11043],[-89.89974,47.98109],[-89.77248,48.02607],[-89.57972,48.00023],[-89.48837,48.01412],[-88.37033,48.30586],[-84.85871,46.88881],[-84.55635,46.45974],[-84.47607,46.45225],[-84.4481,46.48972],[-84.42101,46.49853],[-84.34174,46.50683],[-84.29893,46.49127],[-84.26351,46.49508],[-84.2264,46.53337],[-84.1945,46.54061],[-84.17723,46.52753],[-84.12885,46.53068],[-84.11196,46.50248],[-84.13451,46.39218],[-84.11254,46.32329],[-84.11615,46.2681],[-84.09756,46.25512],[-84.1096,46.23987],[-83.95399,46.05634],[-83.90453,46.05922],[-83.83329,46.12169],[-83.57017,46.105],[-83.43746,45.99749],[-83.59589,45.82131],[-82.48419,45.30225],[-82.42469,42.992],[-82.4146,42.97626],[-82.4253,42.95423],[-82.45331,42.93139],[-82.4826,42.8068],[-82.46613,42.76615],[-82.51063,42.66025],[-82.51858,42.611],[-82.57583,42.5718],[-82.58873,42.54984],[-82.64242,42.55594],[-82.82964,42.37355],[-83.02253,42.33045],[-83.07837,42.30978],[-83.09837,42.28877],[-83.12724,42.2376],[-83.14962,42.04089],[-83.11184,41.95671],[-82.67862,41.67615],[-78.93684,42.82887],[-78.90712,42.89733],[-78.90905,42.93022],[-78.93224,42.95229],[-78.96312,42.95509],[-78.98126,42.97],[-79.02074,42.98444],[-79.02424,43.01983],[-78.99941,43.05612],[-79.01055,43.06659],[-79.07486,43.07845],[-79.05671,43.10937],[-79.06881,43.12029],[-79.0427,43.13934],[-79.04652,43.16396],[-79.05384,43.17418],[-79.05002,43.20133],[-79.05544,43.21224],[-79.05512,43.25375],[-79.06921,43.26183],[-79.25796,43.54052],[-76.79706,43.63099],[-76.43859,44.09393],[-76.35324,44.13493],[-76.31222,44.19894],[-76.244,44.19643],[-76.1664,44.23051],[-76.16285,44.28262],[-76.00018,44.34896],[-75.95947,44.34463],[-75.8217,44.43176],[-75.76813,44.51537],[-75.41441,44.76614],[-75.2193,44.87821],[-75.01363,44.95608],[-74.99101,44.98051],[-74.8447,45.00606],[-74.66689,45.00646],[-74.32699,44.99029],[-73.35025,45.00942],[-71.50067,45.01357],[-71.48735,45.07784],[-71.42778,45.12624],[-71.40364,45.21382],[-71.44252,45.2361],[-71.37133,45.24624],[-71.29371,45.29996],[-71.22338,45.25184],[-71.19723,45.25438],[-71.14568,45.24128],[-71.08364,45.30623],[-71.01866,45.31573],[-71.0107,45.34819],[-70.95193,45.33895],[-70.91169,45.29849],[-70.89864,45.2398],[-70.84816,45.22698],[-70.80236,45.37444],[-70.82638,45.39828],[-70.78372,45.43269],[-70.65383,45.37592],[-70.62518,45.42286],[-70.72651,45.49771],[-70.68516,45.56964],[-70.54019,45.67291],[-70.38934,45.73215],[-70.41523,45.79497],[-70.25976,45.89675],[-70.24694,45.95138],[-70.31025,45.96424],[-70.23855,46.1453],[-70.29078,46.18832],[-70.18547,46.35357],[-70.05812,46.41768],[-69.99966,46.69543],[-69.22119,47.46461],[-69.05148,47.42012],[-69.05073,47.30076],[-69.05039,47.2456],[-68.89222,47.1807],[-68.70125,47.24399],[-68.60575,47.24659],[-68.57914,47.28431],[-68.38332,47.28723],[-68.37458,47.35851],[-68.23244,47.35712],[-67.94843,47.1925],[-67.87993,47.10377],[-67.78578,47.06473],[-67.78111,45.9392],[-67.75196,45.91814],[-67.80961,45.87531],[-67.75654,45.82324],[-67.80653,45.80022],[-67.80705,45.69528],[-67.6049,45.60725],[-67.43815,45.59162],[-67.42144,45.50584],[-67.50578,45.48971],[-67.42394,45.37969],[-67.48201,45.27351],[-67.34927,45.122],[-67.29754,45.14865],[-67.29748,45.18173],[-67.27039,45.1934],[-67.22751,45.16344],[-67.20349,45.1722]]]]}},{type:"Feature",properties:{iso1A2:"CC",iso1A3:"CCK",iso1N3:"166",wikidata:"Q36004",nameEn:"Cocos (Keeling) Islands",country:"AU",groups:["053","009"],driveSide:"left",callingCodes:["61"]},geometry:{type:"MultiPolygon",coordinates:[[[[96.61846,-10.82438],[96.02343,-12.68334],[97.93979,-12.33309],[96.61846,-10.82438]]]]}},{type:"Feature",properties:{iso1A2:"CD",iso1A3:"COD",iso1N3:"180",wikidata:"Q974",nameEn:"Democratic Republic of the Congo",aliases:["ZR"],groups:["017","202","002"],callingCodes:["243"]},geometry:{type:"MultiPolygon",coordinates:[[[[27.44012,5.07349],[27.09575,5.22305],[26.93064,5.13535],[26.85579,5.03887],[26.74572,5.10685],[26.48595,5.04984],[26.13371,5.25594],[25.86073,5.19455],[25.53271,5.37431],[25.34558,5.29101],[25.31256,5.03668],[24.71816,4.90509],[24.46719,5.0915],[23.38847,4.60013],[22.94817,4.82392],[22.89094,4.79321],[22.84691,4.69887],[22.78526,4.71423],[22.6928,4.47285],[22.60915,4.48821],[22.5431,4.22041],[22.45504,4.13039],[22.27682,4.11347],[22.10721,4.20723],[21.6405,4.317],[21.55904,4.25553],[21.25744,4.33676],[21.21341,4.29285],[21.11214,4.33895],[21.08793,4.39603],[20.90383,4.44877],[20.60184,4.42394],[18.62755,3.47564],[18.63857,3.19342],[18.10683,2.26876],[18.08034,1.58553],[17.85887,1.04327],[17.86989,0.58873],[17.95255,0.48128],[17.93877,0.32424],[17.81204,0.23884],[17.66051,-0.26535],[17.72112,-0.52707],[17.32438,-0.99265],[16.97999,-1.12762],[16.70724,-1.45815],[16.50336,-1.8795],[16.16173,-2.16586],[16.22785,-2.59528],[16.1755,-3.25014],[16.21407,-3.2969],[15.89448,-3.9513],[15.53081,-4.042],[15.48121,-4.22062],[15.41785,-4.28381],[15.32693,-4.27282],[15.25411,-4.31121],[15.1978,-4.32388],[14.83101,-4.80838],[14.67948,-4.92093],[14.5059,-4.84956],[14.41499,-4.8825],[14.37366,-4.56125],[14.47284,-4.42941],[14.3957,-4.36623],[14.40672,-4.28381],[13.9108,-4.50906],[13.81162,-4.41842],[13.71794,-4.44864],[13.70417,-4.72601],[13.50305,-4.77818],[13.41764,-4.89897],[13.11182,-4.5942],[13.09648,-4.63739],[13.11195,-4.67745],[12.8733,-4.74346],[12.70868,-4.95505],[12.63465,-4.94632],[12.60251,-5.01715],[12.46297,-5.09408],[12.49815,-5.14058],[12.51589,-5.1332],[12.53586,-5.14658],[12.53599,-5.1618],[12.52301,-5.17481],[12.52318,-5.74353],[12.26557,-5.74031],[12.20376,-5.76338],[11.95767,-5.94705],[12.42245,-6.07585],[13.04371,-5.87078],[16.55507,-5.85631],[16.96282,-7.21787],[17.5828,-8.13784],[18.33635,-8.00126],[19.33698,-7.99743],[19.5469,-7.00195],[20.30218,-6.98955],[20.31846,-6.91953],[20.61689,-6.90876],[20.56263,-7.28566],[21.79824,-7.29628],[21.84856,-9.59871],[22.19039,-9.94628],[22.32604,-10.76291],[22.17954,-10.85884],[22.25951,-11.24911],[22.54205,-11.05784],[23.16602,-11.10577],[23.45631,-10.946],[23.86868,-11.02856],[24.00027,-10.89356],[24.34528,-11.06816],[24.42612,-11.44975],[25.34069,-11.19707],[25.33058,-11.65767],[26.01777,-11.91488],[26.88687,-12.01868],[27.04351,-11.61312],[27.22541,-11.60323],[27.21025,-11.76157],[27.59932,-12.22123],[28.33199,-12.41375],[29.01918,-13.41353],[29.60531,-13.21685],[29.65078,-13.41844],[29.81551,-13.44683],[29.8139,-12.14898],[29.48404,-12.23604],[29.4992,-12.43843],[29.18592,-12.37921],[28.48357,-11.87532],[28.37241,-11.57848],[28.65032,-10.65133],[28.62795,-9.92942],[28.68532,-9.78],[28.56208,-9.49122],[28.51627,-9.44726],[28.52636,-9.35379],[28.36562,-9.30091],[28.38526,-9.23393],[28.9711,-8.66935],[28.88917,-8.4831],[30.79243,-8.27382],[30.2567,-7.14121],[29.52552,-6.2731],[29.43673,-4.44845],[29.23708,-3.75856],[29.21463,-3.3514],[29.25633,-3.05471],[29.17258,-2.99385],[29.16037,-2.95457],[29.09797,-2.91935],[29.09119,-2.87871],[29.0505,-2.81774],[29.00404,-2.81978],[29.00167,-2.78523],[29.04081,-2.7416],[29.00357,-2.70596],[28.94346,-2.69124],[28.89793,-2.66111],[28.90226,-2.62385],[28.89288,-2.55848],[28.87943,-2.55165],[28.86193,-2.53185],[28.86209,-2.5231],[28.87497,-2.50887],[28.88846,-2.50493],[28.89342,-2.49017],[28.89132,-2.47557],[28.86846,-2.44866],[28.86826,-2.41888],[28.89601,-2.37321],[28.95642,-2.37321],[29.00051,-2.29001],[29.105,-2.27043],[29.17562,-2.12278],[29.11847,-1.90576],[29.24458,-1.69663],[29.24323,-1.66826],[29.36322,-1.50887],[29.45038,-1.5054],[29.53062,-1.40499],[29.59061,-1.39016],[29.58388,-0.89821],[29.63006,-0.8997],[29.62708,-0.71055],[29.67176,-0.55714],[29.67474,-0.47969],[29.65091,-0.46777],[29.72687,-0.08051],[29.7224,0.07291],[29.77454,0.16675],[29.81922,0.16824],[29.87284,0.39166],[29.97413,0.52124],[29.95477,0.64486],[29.98307,0.84295],[30.1484,0.89805],[30.22139,0.99635],[30.24671,1.14974],[30.48503,1.21675],[31.30127,2.11006],[31.28042,2.17853],[31.20148,2.2217],[31.1985,2.29462],[31.12104,2.27676],[31.07934,2.30207],[31.06593,2.35862],[30.96911,2.41071],[30.91102,2.33332],[30.83059,2.42559],[30.74271,2.43601],[30.75612,2.5863],[30.8857,2.83923],[30.8574,2.9508],[30.77101,3.04897],[30.84251,3.26908],[30.93486,3.40737],[30.94081,3.50847],[30.85153,3.48867],[30.85997,3.5743],[30.80713,3.60506],[30.78512,3.67097],[30.56277,3.62703],[30.57378,3.74567],[30.55396,3.84451],[30.47691,3.83353],[30.27658,3.95653],[30.22374,3.93896],[30.1621,4.10586],[30.06964,4.13221],[29.79666,4.37809],[29.82087,4.56246],[29.49726,4.7007],[29.43341,4.50101],[29.22207,4.34297],[29.03054,4.48784],[28.8126,4.48784],[28.6651,4.42638],[28.20719,4.35614],[27.79551,4.59976],[27.76469,4.79284],[27.65462,4.89375],[27.56656,4.89375],[27.44012,5.07349]]]]}},{type:"Feature",properties:{iso1A2:"CF",iso1A3:"CAF",iso1N3:"140",wikidata:"Q929",nameEn:"Central African Republic",groups:["017","202","002"],callingCodes:["236"]},geometry:{type:"MultiPolygon",coordinates:[[[[22.87758,10.91915],[22.45889,11.00246],[21.72139,10.64136],[21.71479,10.29932],[21.63553,10.217],[21.52766,10.2105],[21.34934,9.95907],[21.26348,9.97642],[20.82979,9.44696],[20.36748,9.11019],[19.06421,9.00367],[18.86388,8.87971],[19.11044,8.68172],[18.79783,8.25929],[18.67455,8.22226],[18.62612,8.14163],[18.64153,8.08714],[18.6085,8.05009],[18.02731,8.01085],[17.93926,7.95853],[17.67288,7.98905],[16.8143,7.53971],[16.6668,7.67281],[16.658,7.75353],[16.59415,7.76444],[16.58315,7.88657],[16.41583,7.77971],[16.40703,7.68809],[15.79942,7.44149],[15.73118,7.52006],[15.49743,7.52179],[15.23397,7.25135],[15.04717,6.77085],[14.96311,6.75693],[14.79966,6.39043],[14.80122,6.34866],[14.74206,6.26356],[14.56149,6.18928],[14.43073,6.08867],[14.42917,6.00508],[14.49455,5.91683],[14.60974,5.91838],[14.62375,5.70466],[14.58951,5.59777],[14.62531,5.51411],[14.52724,5.28319],[14.57083,5.23979],[14.65489,5.21343],[14.73383,4.6135],[15.00825,4.41458],[15.08609,4.30282],[15.10644,4.1362],[15.17482,4.05131],[15.07686,4.01805],[15.73522,3.24348],[15.77725,3.26835],[16.05449,3.02306],[16.08252,2.45708],[16.19357,2.21537],[16.50126,2.84739],[16.46701,2.92512],[16.57598,3.47999],[16.68283,3.54257],[17.01746,3.55136],[17.35649,3.63045],[17.46876,3.70515],[17.60966,3.63705],[17.83421,3.61068],[17.85842,3.53378],[18.05656,3.56893],[18.14902,3.54476],[18.17323,3.47665],[18.24148,3.50302],[18.2723,3.57992],[18.39558,3.58212],[18.49245,3.63924],[18.58711,3.49423],[18.62755,3.47564],[20.60184,4.42394],[20.90383,4.44877],[21.08793,4.39603],[21.11214,4.33895],[21.21341,4.29285],[21.25744,4.33676],[21.55904,4.25553],[21.6405,4.317],[22.10721,4.20723],[22.27682,4.11347],[22.45504,4.13039],[22.5431,4.22041],[22.60915,4.48821],[22.6928,4.47285],[22.78526,4.71423],[22.84691,4.69887],[22.89094,4.79321],[22.94817,4.82392],[23.38847,4.60013],[24.46719,5.0915],[24.71816,4.90509],[25.31256,5.03668],[25.34558,5.29101],[25.53271,5.37431],[25.86073,5.19455],[26.13371,5.25594],[26.48595,5.04984],[26.74572,5.10685],[26.85579,5.03887],[26.93064,5.13535],[27.09575,5.22305],[27.44012,5.07349],[27.26886,5.25876],[27.23017,5.37167],[27.28621,5.56382],[27.22705,5.62889],[27.22705,5.71254],[26.51721,6.09655],[26.58259,6.1987],[26.32729,6.36272],[26.38022,6.63493],[25.90076,7.09549],[25.37461,7.33024],[25.35281,7.42595],[25.20337,7.50312],[25.20649,7.61115],[25.29214,7.66675],[25.25319,7.8487],[24.98855,7.96588],[24.85156,8.16933],[24.35965,8.26177],[24.13238,8.36959],[24.25691,8.69288],[23.51905,8.71749],[23.59065,8.99743],[23.44744,8.99128],[23.4848,9.16959],[23.56263,9.19418],[23.64358,9.28637],[23.64981,9.44303],[23.62179,9.53823],[23.69155,9.67566],[23.67164,9.86923],[23.3128,10.45214],[23.02221,10.69235],[22.87758,10.91915]]]]}},{type:"Feature",properties:{iso1A2:"CG",iso1A3:"COG",iso1N3:"178",wikidata:"Q971",nameEn:"Republic of the Congo",groups:["017","202","002"],callingCodes:["242"]},geometry:{type:"MultiPolygon",coordinates:[[[[18.62755,3.47564],[18.58711,3.49423],[18.49245,3.63924],[18.39558,3.58212],[18.2723,3.57992],[18.24148,3.50302],[18.17323,3.47665],[18.14902,3.54476],[18.05656,3.56893],[17.85842,3.53378],[17.83421,3.61068],[17.60966,3.63705],[17.46876,3.70515],[17.35649,3.63045],[17.01746,3.55136],[16.68283,3.54257],[16.57598,3.47999],[16.46701,2.92512],[16.50126,2.84739],[16.19357,2.21537],[16.15568,2.18955],[16.08563,2.19733],[16.05294,1.9811],[16.14634,1.70259],[16.02647,1.65591],[16.02959,1.76483],[15.48942,1.98265],[15.34776,1.91264],[15.22634,2.03243],[15.00996,1.98887],[14.61145,2.17866],[13.29457,2.16106],[13.13461,1.57238],[13.25447,1.32339],[13.15519,1.23368],[13.89582,1.4261],[14.25186,1.39842],[14.48179,0.9152],[14.26066,0.57255],[14.10909,0.58563],[13.88648,0.26652],[13.90632,-0.2287],[14.06862,-0.20826],[14.2165,-0.38261],[14.41887,-0.44799],[14.52569,-0.57818],[14.41838,-1.89412],[14.25932,-1.97624],[14.23518,-2.15671],[14.16202,-2.23916],[14.23829,-2.33715],[14.10442,-2.49268],[13.85846,-2.46935],[13.92073,-2.35581],[13.75884,-2.09293],[13.47977,-2.43224],[13.02759,-2.33098],[12.82172,-1.91091],[12.61312,-1.8129],[12.44656,-1.92025],[12.47925,-2.32626],[12.04895,-2.41704],[11.96866,-2.33559],[11.74605,-2.39936],[11.57637,-2.33379],[11.64487,-2.61865],[11.5359,-2.85654],[11.64798,-2.81146],[11.80365,-3.00424],[11.70558,-3.0773],[11.70227,-3.17465],[11.96554,-3.30267],[11.8318,-3.5812],[11.92719,-3.62768],[11.87083,-3.71571],[11.68608,-3.68942],[11.57949,-3.52798],[11.48764,-3.51089],[11.22301,-3.69888],[11.12647,-3.94169],[10.75913,-4.39519],[11.50888,-5.33417],[12.00924,-5.02627],[12.16068,-4.90089],[12.20901,-4.75642],[12.25587,-4.79437],[12.32324,-4.78415],[12.40964,-4.60609],[12.64835,-4.55937],[12.76844,-4.38709],[12.87096,-4.40315],[12.91489,-4.47907],[13.09648,-4.63739],[13.11182,-4.5942],[13.41764,-4.89897],[13.50305,-4.77818],[13.70417,-4.72601],[13.71794,-4.44864],[13.81162,-4.41842],[13.9108,-4.50906],[14.40672,-4.28381],[14.3957,-4.36623],[14.47284,-4.42941],[14.37366,-4.56125],[14.41499,-4.8825],[14.5059,-4.84956],[14.67948,-4.92093],[14.83101,-4.80838],[15.1978,-4.32388],[15.25411,-4.31121],[15.32693,-4.27282],[15.41785,-4.28381],[15.48121,-4.22062],[15.53081,-4.042],[15.89448,-3.9513],[16.21407,-3.2969],[16.1755,-3.25014],[16.22785,-2.59528],[16.16173,-2.16586],[16.50336,-1.8795],[16.70724,-1.45815],[16.97999,-1.12762],[17.32438,-0.99265],[17.72112,-0.52707],[17.66051,-0.26535],[17.81204,0.23884],[17.93877,0.32424],[17.95255,0.48128],[17.86989,0.58873],[17.85887,1.04327],[18.08034,1.58553],[18.10683,2.26876],[18.63857,3.19342],[18.62755,3.47564]]]]}},{type:"Feature",properties:{iso1A2:"CH",iso1A3:"CHE",iso1N3:"756",wikidata:"Q39",nameEn:"Switzerland",groups:["155","150"],callingCodes:["41"]},geometry:{type:"MultiPolygon",coordinates:[[[[8.72809,47.69282],[8.72617,47.69651],[8.73671,47.7169],[8.70543,47.73121],[8.74251,47.75168],[8.71778,47.76571],[8.68985,47.75686],[8.68022,47.78599],[8.65292,47.80066],[8.64425,47.76398],[8.62408,47.7626],[8.61657,47.79998],[8.56415,47.80633],[8.56814,47.78001],[8.48868,47.77215],[8.45771,47.7493],[8.44807,47.72426],[8.40569,47.69855],[8.4211,47.68407],[8.40473,47.67499],[8.41346,47.66676],[8.42264,47.66667],[8.44711,47.65379],[8.4667,47.65747],[8.46605,47.64103],[8.49656,47.64709],[8.5322,47.64687],[8.52801,47.66059],[8.56141,47.67088],[8.57683,47.66158],[8.6052,47.67258],[8.61113,47.66332],[8.62884,47.65098],[8.62049,47.63757],[8.60412,47.63735],[8.61471,47.64514],[8.60701,47.65271],[8.59545,47.64298],[8.60348,47.61204],[8.57586,47.59537],[8.55756,47.62394],[8.51686,47.63476],[8.50747,47.61897],[8.45578,47.60121],[8.46637,47.58389],[8.48949,47.588],[8.49431,47.58107],[8.43235,47.56617],[8.39477,47.57826],[8.38273,47.56608],[8.32735,47.57133],[8.30277,47.58607],[8.29524,47.5919],[8.29722,47.60603],[8.2824,47.61225],[8.26313,47.6103],[8.25863,47.61571],[8.23809,47.61204],[8.22577,47.60385],[8.22011,47.6181],[8.20617,47.62141],[8.19378,47.61636],[8.1652,47.5945],[8.14947,47.59558],[8.13823,47.59147],[8.13662,47.58432],[8.11543,47.5841],[8.10395,47.57918],[8.10002,47.56504],[8.08557,47.55768],[8.06663,47.56374],[8.04383,47.55443],[8.02136,47.55096],[8.00113,47.55616],[7.97581,47.55493],[7.95682,47.55789],[7.94494,47.54511],[7.91251,47.55031],[7.90673,47.57674],[7.88664,47.58854],[7.84412,47.5841],[7.81901,47.58798],[7.79486,47.55691],[7.75261,47.54599],[7.71961,47.54219],[7.69642,47.53297],[7.68101,47.53232],[7.6656,47.53752],[7.66174,47.54554],[7.65083,47.54662],[7.63338,47.56256],[7.67655,47.56435],[7.68904,47.57133],[7.67115,47.5871],[7.68486,47.59601],[7.69385,47.60099],[7.68229,47.59905],[7.67395,47.59212],[7.64599,47.59695],[7.64213,47.5944],[7.64309,47.59151],[7.61929,47.57683],[7.60459,47.57869],[7.60523,47.58519],[7.58945,47.59017],[7.58386,47.57536],[7.56684,47.57785],[7.56548,47.57617],[7.55689,47.57232],[7.55652,47.56779],[7.53634,47.55553],[7.52831,47.55347],[7.51723,47.54578],[7.50873,47.54546],[7.49691,47.53821],[7.50588,47.52856],[7.51904,47.53515],[7.53199,47.5284],[7.5229,47.51644],[7.49804,47.51798],[7.51076,47.49651],[7.47534,47.47932],[7.43356,47.49712],[7.42923,47.48628],[7.4583,47.47216],[7.4462,47.46264],[7.43088,47.45846],[7.40308,47.43638],[7.35603,47.43432],[7.33526,47.44186],[7.24669,47.4205],[7.17026,47.44312],[7.19583,47.49455],[7.16249,47.49025],[7.12781,47.50371],[7.07425,47.48863],[7.0231,47.50522],[6.98425,47.49432],[7.0024,47.45264],[6.93953,47.43388],[6.93744,47.40714],[6.88542,47.37262],[6.87959,47.35335],[7.03125,47.36996],[7.0564,47.35134],[7.05305,47.33304],[6.94316,47.28747],[6.95108,47.26428],[6.9508,47.24338],[6.8489,47.15933],[6.76788,47.1208],[6.68823,47.06616],[6.71531,47.0494],[6.43341,46.92703],[6.46456,46.88865],[6.43216,46.80336],[6.45209,46.77502],[6.38351,46.73171],[6.27135,46.68251],[6.11084,46.57649],[6.1567,46.54402],[6.07269,46.46244],[6.08427,46.44305],[6.06407,46.41676],[6.09926,46.40768],[6.15016,46.3778],[6.15985,46.37721],[6.16987,46.36759],[6.15738,46.3491],[6.13876,46.33844],[6.1198,46.31157],[6.11697,46.29547],[6.1013,46.28512],[6.11926,46.2634],[6.12446,46.25059],[6.10071,46.23772],[6.08563,46.24651],[6.07072,46.24085],[6.0633,46.24583],[6.05029,46.23518],[6.04602,46.23127],[6.03342,46.2383],[6.02461,46.23313],[5.97542,46.21525],[5.96515,46.19638],[5.99573,46.18587],[5.98846,46.17046],[5.98188,46.17392],[5.97508,46.15863],[5.9641,46.14412],[5.95781,46.12925],[5.97893,46.13303],[5.9871,46.14499],[6.01791,46.14228],[6.03614,46.13712],[6.04564,46.14031],[6.05203,46.15191],[6.07491,46.14879],[6.09199,46.15191],[6.09926,46.14373],[6.13397,46.1406],[6.15305,46.15194],[6.18116,46.16187],[6.18871,46.16644],[6.18707,46.17999],[6.19552,46.18401],[6.19807,46.18369],[6.20539,46.19163],[6.21114,46.1927],[6.21273,46.19409],[6.21603,46.19507],[6.21844,46.19837],[6.22222,46.19888],[6.22175,46.20045],[6.23544,46.20714],[6.23913,46.20511],[6.24821,46.20531],[6.26007,46.21165],[6.27694,46.21566],[6.29663,46.22688],[6.31041,46.24417],[6.29474,46.26221],[6.26749,46.24745],[6.24952,46.26255],[6.23775,46.27822],[6.25137,46.29014],[6.24826,46.30175],[6.21981,46.31304],[6.25432,46.3632],[6.53358,46.45431],[6.82312,46.42661],[6.8024,46.39171],[6.77152,46.34784],[6.86052,46.28512],[6.78968,46.14058],[6.89321,46.12548],[6.87868,46.03855],[6.93862,46.06502],[7.00946,45.9944],[7.04151,45.92435],[7.10685,45.85653],[7.56343,45.97421],[7.85949,45.91485],[7.9049,45.99945],[7.98881,45.99867],[8.02906,46.10331],[8.11383,46.11577],[8.16866,46.17817],[8.08814,46.26692],[8.31162,46.38044],[8.30648,46.41587],[8.42464,46.46367],[8.46317,46.43712],[8.45032,46.26869],[8.62242,46.12112],[8.75697,46.10395],[8.80778,46.10085],[8.85617,46.0748],[8.79414,46.00913],[8.78585,45.98973],[8.79362,45.99207],[8.8319,45.9879],[8.85121,45.97239],[8.86688,45.96135],[8.88904,45.95465],[8.93649,45.86775],[8.94372,45.86587],[8.93504,45.86245],[8.91129,45.8388],[8.94737,45.84285],[8.9621,45.83707],[8.99663,45.83466],[9.00324,45.82055],[9.0298,45.82127],[9.03279,45.82865],[9.03793,45.83548],[9.03505,45.83976],[9.04059,45.8464],[9.04546,45.84968],[9.06642,45.8761],[9.09065,45.89906],[8.99257,45.9698],[9.01618,46.04928],[9.24503,46.23616],[9.29226,46.32717],[9.25502,46.43743],[9.28136,46.49685],[9.36128,46.5081],[9.40487,46.46621],[9.45936,46.50873],[9.46117,46.37481],[9.57015,46.2958],[9.71273,46.29266],[9.73086,46.35071],[9.95249,46.38045],[10.07055,46.21668],[10.14439,46.22992],[10.17862,46.25626],[10.10506,46.3372],[10.165,46.41051],[10.03715,46.44479],[10.10307,46.61003],[10.23674,46.63484],[10.25309,46.57432],[10.46136,46.53164],[10.49375,46.62049],[10.44686,46.64162],[10.40475,46.63671],[10.38659,46.67847],[10.47197,46.85698],[10.48376,46.93891],[10.36933,47.00212],[10.30031,46.92093],[10.24128,46.93147],[10.22675,46.86942],[10.10715,46.84296],[9.98058,46.91434],[9.88266,46.93343],[9.87935,47.01337],[9.60717,47.06091],[9.55721,47.04762],[9.54041,47.06495],[9.47548,47.05257],[9.47139,47.06402],[9.51362,47.08505],[9.52089,47.10019],[9.51044,47.13727],[9.48774,47.17402],[9.4891,47.19346],[9.50318,47.22153],[9.52406,47.24959],[9.53116,47.27029],[9.54773,47.2809],[9.55857,47.29919],[9.58513,47.31334],[9.59978,47.34671],[9.62476,47.36639],[9.65427,47.36824],[9.66243,47.37136],[9.6711,47.37824],[9.67445,47.38429],[9.67334,47.39191],[9.6629,47.39591],[9.65136,47.40504],[9.65043,47.41937],[9.6446,47.43233],[9.64483,47.43842],[9.65863,47.44847],[9.65728,47.45383],[9.6423,47.45599],[9.62475,47.45685],[9.62158,47.45858],[9.60841,47.47178],[9.60484,47.46358],[9.60205,47.46165],[9.59482,47.46305],[9.58208,47.48344],[9.56312,47.49495],[9.55125,47.53629],[9.25619,47.65939],[9.18203,47.65598],[9.17593,47.65399],[9.1755,47.65584],[9.1705,47.65513],[9.15181,47.66904],[9.13845,47.66389],[9.09891,47.67801],[9.02093,47.6868],[8.94093,47.65596],[8.89946,47.64769],[8.87625,47.65441],[8.87383,47.67045],[8.85065,47.68209],[8.86989,47.70504],[8.82002,47.71458],[8.80663,47.73821],[8.77309,47.72059],[8.76965,47.7075],[8.79966,47.70222],[8.79511,47.67462],[8.75856,47.68969],[8.72809,47.69282]],[[8.95861,45.96485],[8.96668,45.98436],[8.97741,45.98317],[8.97604,45.96151],[8.95861,45.96485]],[[8.70847,47.68904],[8.68985,47.69552],[8.66837,47.68437],[8.65769,47.68928],[8.67508,47.6979],[8.66416,47.71367],[8.70237,47.71453],[8.71773,47.69088],[8.70847,47.68904]]]]}},{type:"Feature",properties:{iso1A2:"CI",iso1A3:"CIV",iso1N3:"384",wikidata:"Q1008",nameEn:"Côte d'Ivoire",groups:["011","202","002"],callingCodes:["225"]},geometry:{type:"MultiPolygon",coordinates:[[[[-7.52774,3.7105],[-3.34019,4.17519],[-3.10675,5.08515],[-3.11073,5.12675],[-3.063,5.13665],[-2.96554,5.10397],[-2.95261,5.12477],[-2.75502,5.10657],[-2.73074,5.1364],[-2.77625,5.34621],[-2.72737,5.34789],[-2.76614,5.60963],[-2.85378,5.65156],[-2.93132,5.62137],[-2.96671,5.6415],[-2.95323,5.71865],[-3.01896,5.71697],[-3.25999,6.62521],[-3.21954,6.74407],[-3.23327,6.81744],[-2.95438,7.23737],[-2.97822,7.27165],[-2.92339,7.60847],[-2.79467,7.86002],[-2.78395,7.94974],[-2.74819,7.92613],[-2.67787,8.02055],[-2.61232,8.02645],[-2.62901,8.11495],[-2.49037,8.20872],[-2.58243,8.7789],[-2.66357,9.01771],[-2.77799,9.04949],[-2.69814,9.22717],[-2.68802,9.49343],[-2.76494,9.40778],[-2.93012,9.57403],[-3.00765,9.74019],[-3.16609,9.85147],[-3.19306,9.93781],[-3.27228,9.84981],[-3.31779,9.91125],[-3.69703,9.94279],[-4.25999,9.76012],[-4.31392,9.60062],[-4.6426,9.70696],[-4.96621,9.89132],[-4.96453,9.99923],[-5.12465,10.29788],[-5.39602,10.2929],[-5.51058,10.43177],[-5.65135,10.46767],[-5.78124,10.43952],[-5.99478,10.19694],[-6.18851,10.24244],[-6.1731,10.46983],[-6.24795,10.74248],[-6.325,10.68624],[-6.40646,10.69922],[-6.42847,10.5694],[-6.52974,10.59104],[-6.63541,10.66893],[-6.68164,10.35074],[-6.93921,10.35291],[-7.01186,10.25111],[-6.97444,10.21644],[-7.00966,10.15794],[-7.0603,10.14711],[-7.13331,10.24877],[-7.3707,10.24677],[-7.44555,10.44602],[-7.52261,10.4655],[-7.54462,10.40921],[-7.63048,10.46334],[-7.92107,10.15577],[-7.97971,10.17117],[-8.01225,10.1021],[-8.11921,10.04577],[-8.15652,9.94288],[-8.09434,9.86936],[-8.14657,9.55062],[-8.03463,9.39604],[-7.85056,9.41812],[-7.90777,9.20456],[-7.73862,9.08422],[-7.92518,8.99332],[-7.95503,8.81146],[-7.69882,8.66148],[-7.65653,8.36873],[-7.92518,8.50652],[-8.22991,8.48438],[-8.2411,8.24196],[-8.062,8.16071],[-7.98675,8.20134],[-7.99919,8.11023],[-7.94695,8.00925],[-8.06449,8.04989],[-8.13414,7.87991],[-8.09931,7.78626],[-8.21374,7.54466],[-8.4003,7.6285],[-8.47114,7.55676],[-8.41935,7.51203],[-8.37458,7.25794],[-8.29249,7.1691],[-8.31736,6.82837],[-8.59456,6.50612],[-8.48652,6.43797],[-8.45666,6.49977],[-8.38453,6.35887],[-8.3298,6.36381],[-8.17557,6.28222],[-8.00642,6.31684],[-7.90692,6.27728],[-7.83478,6.20309],[-7.8497,6.08932],[-7.79747,6.07696],[-7.78254,5.99037],[-7.70294,5.90625],[-7.67309,5.94337],[-7.48155,5.80974],[-7.46165,5.84934],[-7.43677,5.84687],[-7.43926,5.74787],[-7.37209,5.61173],[-7.43428,5.42355],[-7.36463,5.32944],[-7.46165,5.26256],[-7.48901,5.14118],[-7.55369,5.08667],[-7.53876,4.94294],[-7.59349,4.8909],[-7.53259,4.35145],[-7.52774,3.7105]]]]}},{type:"Feature",properties:{iso1A2:"CK",iso1A3:"COK",iso1N3:"184",wikidata:"Q26988",nameEn:"Cook Islands",country:"NZ",groups:["061","009"],driveSide:"left",callingCodes:["682"]},geometry:{type:"MultiPolygon",coordinates:[[[[-167.73854,-14.92809],[-167.73129,-23.22266],[-156.46451,-23.21255],[-156.4957,-12.32002],[-156.50903,-7.4975],[-167.75329,-7.52784],[-167.75195,-10.12005],[-167.73854,-14.92809]]]]}},{type:"Feature",properties:{iso1A2:"CL",iso1A3:"CHL",iso1N3:"152",wikidata:"Q298",nameEn:"Chile",groups:["005","419","019"],callingCodes:["56"]},geometry:{type:"MultiPolygon",coordinates:[[[[-68.60702,-52.65781],[-68.41683,-52.33516],[-69.97824,-52.00845],[-71.99889,-51.98018],[-72.33873,-51.59954],[-72.31343,-50.58411],[-73.15765,-50.78337],[-73.55259,-49.92488],[-73.45156,-49.79461],[-73.09655,-49.14342],[-72.56894,-48.81116],[-72.54042,-48.52392],[-72.27662,-48.28727],[-72.50478,-47.80586],[-71.94152,-47.13595],[-71.68577,-46.55385],[-71.75614,-45.61611],[-71.35687,-45.22075],[-72.06985,-44.81756],[-71.26418,-44.75684],[-71.16436,-44.46244],[-71.81318,-44.38097],[-71.64206,-43.64774],[-72.14828,-42.85321],[-72.15541,-42.15941],[-71.74901,-42.11711],[-71.92726,-40.72714],[-71.37826,-38.91474],[-70.89532,-38.6923],[-71.24279,-37.20264],[-70.95047,-36.4321],[-70.38008,-36.02375],[-70.49416,-35.24145],[-69.87386,-34.13344],[-69.88099,-33.34489],[-70.55832,-31.51559],[-70.14479,-30.36595],[-69.8596,-30.26131],[-69.99507,-29.28351],[-69.80969,-29.07185],[-69.66709,-28.44055],[-69.22504,-27.95042],[-68.77586,-27.16029],[-68.43363,-27.08414],[-68.27677,-26.90626],[-68.59048,-26.49861],[-68.56909,-26.28146],[-68.38372,-26.15353],[-68.57622,-25.32505],[-68.38372,-25.08636],[-68.56909,-24.69831],[-68.24825,-24.42596],[-67.33563,-24.04237],[-66.99632,-22.99839],[-67.18382,-22.81525],[-67.54284,-22.89771],[-67.85114,-22.87076],[-68.18816,-21.28614],[-68.40403,-20.94562],[-68.53957,-20.91542],[-68.55383,-20.7355],[-68.44023,-20.62701],[-68.7276,-20.46178],[-68.74273,-20.08817],[-68.57132,-20.03134],[-68.54611,-19.84651],[-68.66761,-19.72118],[-68.41218,-19.40499],[-68.61989,-19.27584],[-68.80602,-19.08355],[-68.87082,-19.06003],[-68.94987,-18.93302],[-69.07432,-18.28259],[-69.14807,-18.16893],[-69.07496,-18.03715],[-69.28671,-17.94844],[-69.34126,-17.72753],[-69.46623,-17.60518],[-69.46897,-17.4988],[-69.66483,-17.65083],[-69.79087,-17.65563],[-69.82868,-17.72048],[-69.75305,-17.94605],[-69.81607,-18.12582],[-69.96732,-18.25992],[-70.16394,-18.31737],[-70.31267,-18.31258],[-70.378,-18.3495],[-70.59118,-18.35072],[-113.52687,-26.52828],[-68.11646,-58.14883],[-66.07313,-55.19618],[-67.11046,-54.94199],[-67.46182,-54.92205],[-68.01394,-54.8753],[-68.60733,-54.9125],[-68.60702,-52.65781]]]]}},{type:"Feature",properties:{iso1A2:"CM",iso1A3:"CMR",iso1N3:"120",wikidata:"Q1009",nameEn:"Cameroon",groups:["017","202","002"],callingCodes:["237"]},geometry:{type:"MultiPolygon",coordinates:[[[[14.83314,12.62963],[14.55058,12.78256],[14.56101,12.91036],[14.46881,13.08259],[14.08251,13.0797],[14.20204,12.53405],[14.17523,12.41916],[14.22215,12.36533],[14.4843,12.35223],[14.6474,12.17466],[14.61612,11.7798],[14.55207,11.72001],[14.64591,11.66166],[14.6124,11.51283],[14.17821,11.23831],[13.97489,11.30258],[13.78945,11.00154],[13.7403,11.00593],[13.70753,10.94451],[13.73434,10.9255],[13.54964,10.61236],[13.5705,10.53183],[13.43644,10.13326],[13.34111,10.12299],[13.25025,10.03647],[13.25323,10.00127],[13.286,9.9822],[13.27409,9.93232],[13.24132,9.91031],[13.25025,9.86042],[13.29941,9.8296],[13.25472,9.76795],[13.22642,9.57266],[13.02385,9.49334],[12.85628,9.36698],[12.91958,9.33905],[12.90022,9.11411],[12.81085,8.91992],[12.79,8.75361],[12.71701,8.7595],[12.68722,8.65938],[12.44146,8.6152],[12.4489,8.52536],[12.26123,8.43696],[12.24782,8.17904],[12.19271,8.10826],[12.20909,7.97553],[11.99908,7.67302],[12.01844,7.52981],[11.93205,7.47812],[11.84864,7.26098],[11.87396,7.09398],[11.63117,6.9905],[11.55818,6.86186],[11.57755,6.74059],[11.51499,6.60892],[11.42264,6.5882],[11.42041,6.53789],[11.09495,6.51717],[11.09644,6.68437],[10.94302,6.69325],[10.8179,6.83377],[10.83727,6.9358],[10.60789,7.06885],[10.59746,7.14719],[10.57214,7.16345],[10.53639,6.93432],[10.21466,6.88996],[10.15135,7.03781],[9.86314,6.77756],[9.77824,6.79088],[9.70674,6.51717],[9.51757,6.43874],[8.84209,5.82562],[8.88156,5.78857],[8.83687,5.68483],[8.92029,5.58403],[8.78027,5.1243],[8.60302,4.87353],[8.34397,4.30689],[9.22018,3.72052],[9.81162,2.33797],[9.82123,2.35097],[9.83754,2.32428],[9.83238,2.29079],[9.84716,2.24676],[9.89012,2.20457],[9.90749,2.20049],[9.991,2.16561],[11.3561,2.17217],[11.37116,2.29975],[13.28534,2.25716],[13.29457,2.16106],[14.61145,2.17866],[15.00996,1.98887],[15.22634,2.03243],[15.34776,1.91264],[15.48942,1.98265],[16.02959,1.76483],[16.02647,1.65591],[16.14634,1.70259],[16.05294,1.9811],[16.08563,2.19733],[16.15568,2.18955],[16.19357,2.21537],[16.08252,2.45708],[16.05449,3.02306],[15.77725,3.26835],[15.73522,3.24348],[15.07686,4.01805],[15.17482,4.05131],[15.10644,4.1362],[15.08609,4.30282],[15.00825,4.41458],[14.73383,4.6135],[14.65489,5.21343],[14.57083,5.23979],[14.52724,5.28319],[14.62531,5.51411],[14.58951,5.59777],[14.62375,5.70466],[14.60974,5.91838],[14.49455,5.91683],[14.42917,6.00508],[14.43073,6.08867],[14.56149,6.18928],[14.74206,6.26356],[14.80122,6.34866],[14.79966,6.39043],[14.96311,6.75693],[15.04717,6.77085],[15.23397,7.25135],[15.49743,7.52179],[15.56964,7.58936],[15.59272,7.7696],[15.50743,7.79302],[15.20426,8.50892],[15.09484,8.65982],[14.83566,8.80557],[14.35707,9.19611],[14.37094,9.2954],[13.97544,9.6365],[14.01793,9.73169],[14.1317,9.82413],[14.20411,10.00055],[14.4673,10.00264],[14.80082,9.93818],[14.95722,9.97926],[15.05999,9.94845],[15.14043,9.99246],[15.24618,9.99246],[15.41408,9.92876],[15.68761,9.99344],[15.50535,10.1098],[15.30874,10.31063],[15.23724,10.47764],[15.14936,10.53915],[15.15532,10.62846],[15.06737,10.80921],[15.09127,10.87431],[15.04957,11.02347],[15.10021,11.04101],[15.0585,11.40481],[15.13149,11.5537],[15.06595,11.71126],[15.11579,11.79313],[15.04808,11.8731],[15.05786,12.0608],[15.0349,12.10698],[15.00146,12.1223],[14.96952,12.0925],[14.89019,12.16593],[14.90827,12.3269],[14.83314,12.62963]]]]}},{type:"Feature",properties:{iso1A2:"CN",iso1A3:"CHN",iso1N3:"156",wikidata:"Q148",nameEn:"China",aliases:["RC"],groups:["030","142"],callingCodes:["86"]},geometry:{type:"MultiPolygon",coordinates:[[[[125.6131,53.07229],[125.17522,53.20225],[124.46078,53.21881],[123.86158,53.49391],[123.26989,53.54843],[122.85966,53.47395],[122.35063,53.49565],[121.39213,53.31888],[120.85633,53.28499],[120.0451,52.7359],[120.04049,52.58773],[120.46454,52.63811],[120.71673,52.54099],[120.61346,52.32447],[120.77337,52.20805],[120.65907,51.93544],[120.10963,51.671],[119.13553,50.37412],[119.38598,50.35162],[119.27996,50.13348],[119.11003,50.00276],[118.61623,49.93809],[117.82343,49.52696],[117.48208,49.62324],[117.27597,49.62544],[117.07142,49.68482],[116.71193,49.83813],[116.03781,48.87014],[116.06565,48.81716],[115.78876,48.51781],[115.811,48.25699],[115.52082,48.15367],[115.57128,47.91988],[115.94296,47.67741],[116.08431,47.80693],[116.2527,47.87766],[116.4465,47.83662],[116.67405,47.89039],[116.87527,47.88836],[117.08918,47.82242],[117.37875,47.63627],[117.50181,47.77216],[117.80196,48.01661],[118.03676,48.00982],[118.11009,48.04],[118.22677,48.03853],[118.29654,48.00246],[118.55766,47.99277],[118.7564,47.76947],[119.12343,47.66458],[119.13995,47.53997],[119.35892,47.48104],[119.31964,47.42617],[119.54918,47.29505],[119.56019,47.24874],[119.62403,47.24575],[119.71209,47.19192],[119.85518,46.92196],[119.91242,46.90091],[119.89261,46.66423],[119.80455,46.67631],[119.77373,46.62947],[119.68127,46.59015],[119.65265,46.62342],[119.42827,46.63783],[119.37306,46.61132],[119.30261,46.6083],[119.24978,46.64761],[119.10448,46.65516],[119.00541,46.74273],[118.92616,46.72765],[118.89974,46.77139],[118.8337,46.77742],[118.78747,46.68689],[118.30534,46.73519],[117.69554,46.50991],[117.60748,46.59771],[117.41782,46.57862],[117.36609,46.36335],[117.07252,46.35818],[116.83166,46.38637],[116.75551,46.33083],[116.58612,46.30211],[116.26678,45.96479],[116.24012,45.8778],[116.27366,45.78637],[116.16989,45.68603],[115.91898,45.6227],[115.69688,45.45761],[115.35757,45.39106],[114.94546,45.37377],[114.74612,45.43585],[114.54801,45.38337],[114.5166,45.27189],[114.08071,44.92847],[113.909,44.91444],[113.63821,44.74326],[112.74662,44.86297],[112.4164,45.06858],[111.98695,45.09074],[111.76275,44.98032],[111.40498,44.3461],[111.96289,43.81596],[111.93776,43.68709],[111.79758,43.6637],[111.59087,43.51207],[111.0149,43.3289],[110.4327,42.78293],[110.08401,42.6411],[109.89402,42.63111],[109.452,42.44842],[109.00679,42.45302],[108.84489,42.40246],[108.23156,42.45532],[107.57258,42.40898],[107.49681,42.46221],[107.29755,42.41395],[107.24774,42.36107],[106.76517,42.28741],[105.24708,41.7442],[105.01119,41.58382],[104.91272,41.64619],[104.51667,41.66113],[104.52258,41.8706],[103.92804,41.78246],[103.3685,41.89696],[102.72403,42.14675],[102.42826,42.15137],[102.07645,42.22519],[101.80515,42.50074],[101.28833,42.58524],[100.84979,42.67087],[100.33297,42.68231],[99.50671,42.56535],[97.1777,42.7964],[96.37926,42.72055],[96.35658,42.90363],[95.89543,43.2528],[95.52594,43.99353],[95.32891,44.02407],[95.39772,44.2805],[95.01191,44.25274],[94.71959,44.35284],[94.10003,44.71016],[93.51161,44.95964],[91.64048,45.07408],[90.89169,45.19667],[90.65114,45.49314],[90.70907,45.73437],[91.03026,46.04194],[90.99672,46.14207],[90.89639,46.30711],[91.07696,46.57315],[91.0147,46.58171],[91.03649,46.72916],[90.84035,46.99525],[90.76108,46.99399],[90.48542,47.30438],[90.48854,47.41826],[90.33598,47.68303],[90.10871,47.7375],[90.06512,47.88177],[89.76624,47.82745],[89.55453,48.0423],[89.0711,47.98528],[88.93186,48.10263],[88.8011,48.11302],[88.58316,48.21893],[88.58939,48.34531],[87.96361,48.58478],[88.0788,48.71436],[87.73822,48.89582],[87.88171,48.95853],[87.81333,49.17354],[87.48983,49.13794],[87.478,49.07403],[87.28386,49.11626],[86.87238,49.12432],[86.73568,48.99918],[86.75343,48.70331],[86.38069,48.46064],[85.73581,48.3939],[85.5169,48.05493],[85.61067,47.49753],[85.69696,47.2898],[85.54294,47.06171],[85.22443,47.04816],[84.93995,46.87399],[84.73077,47.01394],[83.92184,46.98912],[83.04622,47.19053],[82.21792,45.56619],[82.58474,45.40027],[82.51374,45.1755],[81.73278,45.3504],[80.11169,45.03352],[79.8987,44.89957],[80.38384,44.63073],[80.40229,44.23319],[80.40031,44.10986],[80.75156,43.44948],[80.69718,43.32589],[80.77771,43.30065],[80.78817,43.14235],[80.62913,43.141],[80.3735,43.01557],[80.58999,42.9011],[80.38169,42.83142],[80.26886,42.8366],[80.16892,42.61137],[80.26841,42.23797],[80.17807,42.21166],[80.17842,42.03211],[79.92977,42.04113],[78.3732,41.39603],[78.15757,41.38565],[78.12873,41.23091],[77.81287,41.14307],[77.76206,41.01574],[77.52723,41.00227],[77.3693,41.0375],[77.28004,41.0033],[76.99302,41.0696],[76.75681,40.95354],[76.5261,40.46114],[76.33659,40.3482],[75.96168,40.38064],[75.91361,40.2948],[75.69663,40.28642],[75.5854,40.66874],[75.22834,40.45382],[75.08243,40.43945],[74.82013,40.52197],[74.78168,40.44886],[74.85996,40.32857],[74.69875,40.34668],[74.35063,40.09742],[74.25533,40.13191],[73.97049,40.04378],[73.83006,39.76136],[73.9051,39.75073],[73.92354,39.69565],[73.94683,39.60733],[73.87018,39.47879],[73.59831,39.46425],[73.59241,39.40843],[73.5004,39.38402],[73.55396,39.3543],[73.54572,39.27567],[73.60638,39.24534],[73.75823,39.023],[73.81728,39.04007],[73.82964,38.91517],[73.7445,38.93867],[73.7033,38.84782],[73.80656,38.66449],[73.79806,38.61106],[73.97933,38.52945],[74.17022,38.65504],[74.51217,38.47034],[74.69619,38.42947],[74.69894,38.22155],[74.80331,38.19889],[74.82665,38.07359],[74.9063,38.03033],[74.92416,37.83428],[75.00935,37.77486],[74.8912,37.67576],[74.94338,37.55501],[75.06011,37.52779],[75.15899,37.41443],[75.09719,37.37297],[75.12328,37.31839],[74.88887,37.23275],[74.80605,37.21565],[74.49981,37.24518],[74.56453,37.03023],[75.13839,37.02622],[75.40481,36.95382],[75.45562,36.71971],[75.72737,36.7529],[75.92391,36.56986],[76.0324,36.41198],[76.00906,36.17511],[75.93028,36.13136],[76.15325,35.9264],[76.14913,35.82848],[76.33453,35.84296],[76.50961,35.8908],[76.77323,35.66062],[76.84539,35.67356],[76.96624,35.5932],[77.44277,35.46132],[77.70232,35.46244],[77.80532,35.52058],[78.11664,35.48022],[78.03466,35.3785],[78.00033,35.23954],[78.22692,34.88771],[78.18435,34.7998],[78.27781,34.61484],[78.54964,34.57283],[78.56475,34.50835],[78.74465,34.45174],[79.05364,34.32482],[78.99802,34.3027],[78.91769,34.15452],[78.66225,34.08858],[78.65657,34.03195],[78.73367,34.01121],[78.77349,33.73871],[78.67599,33.66445],[78.73636,33.56521],[79.15252,33.17156],[79.14016,33.02545],[79.46562,32.69668],[79.26768,32.53277],[79.13174,32.47766],[79.0979,32.38051],[78.99322,32.37948],[78.96713,32.33655],[78.7831,32.46873],[78.73916,32.69438],[78.38897,32.53938],[78.4645,32.45367],[78.49609,32.2762],[78.68754,32.10256],[78.74404,32.00384],[78.78036,31.99478],[78.69933,31.78723],[78.84516,31.60631],[78.71032,31.50197],[78.77898,31.31209],[79.01931,31.42817],[79.14016,31.43403],[79.22805,31.34963],[79.59884,30.93943],[79.93255,30.88288],[80.20721,30.58541],[80.54504,30.44936],[80.83343,30.32023],[81.03953,30.20059],[81.12842,30.01395],[81.24362,30.0126],[81.29032,30.08806],[81.2623,30.14596],[81.33355,30.15303],[81.39928,30.21862],[81.41018,30.42153],[81.62033,30.44703],[81.99082,30.33423],[82.10135,30.35439],[82.10757,30.23745],[82.19475,30.16884],[82.16984,30.0692],[82.38622,30.02608],[82.5341,29.9735],[82.73024,29.81695],[83.07116,29.61957],[83.28131,29.56813],[83.44787,29.30513],[83.63156,29.16249],[83.82303,29.30513],[83.97559,29.33091],[84.18107,29.23451],[84.24801,29.02783],[84.2231,28.89571],[84.47528,28.74023],[84.62317,28.73887],[84.85511,28.58041],[85.06059,28.68562],[85.19135,28.62825],[85.18668,28.54076],[85.10729,28.34092],[85.38127,28.28336],[85.4233,28.32996],[85.59765,28.30529],[85.60854,28.25045],[85.69105,28.38475],[85.71907,28.38064],[85.74864,28.23126],[85.84672,28.18187],[85.90743,28.05144],[85.97813,27.99023],[85.94946,27.9401],[86.06309,27.90021],[86.12069,27.93047],[86.08333,28.02121],[86.088,28.09264],[86.18607,28.17364],[86.22966,27.9786],[86.42736,27.91122],[86.51609,27.96623],[86.56265,28.09569],[86.74181,28.10638],[86.75582,28.04182],[87.03757,27.94835],[87.11696,27.84104],[87.56996,27.84517],[87.72718,27.80938],[87.82681,27.95248],[88.13378,27.88015],[88.1278,27.95417],[88.25332,27.9478],[88.54858,28.06057],[88.63235,28.12356],[88.83559,28.01936],[88.88091,27.85192],[88.77517,27.45415],[88.82981,27.38814],[88.91901,27.32483],[88.93678,27.33777],[88.96947,27.30319],[89.00216,27.32532],[88.95355,27.4106],[88.97213,27.51671],[89.0582,27.60985],[89.12825,27.62502],[89.59525,28.16433],[89.79762,28.23979],[90.13387,28.19178],[90.58842,28.02838],[90.69894,28.07784],[91.20019,27.98715],[91.25779,28.07509],[91.46327,28.0064],[91.48973,27.93903],[91.5629,27.84823],[91.6469,27.76358],[91.84722,27.76325],[91.87057,27.7195],[92.27432,27.89077],[92.32101,27.79363],[92.42538,27.80092],[92.7275,27.98662],[92.73025,28.05814],[92.65472,28.07632],[92.67486,28.15018],[92.93075,28.25671],[93.14635,28.37035],[93.18069,28.50319],[93.44621,28.67189],[93.72797,28.68821],[94.35897,29.01965],[94.2752,29.11687],[94.69318,29.31739],[94.81353,29.17804],[95.0978,29.14446],[95.11291,29.09527],[95.2214,29.10727],[95.26122,29.07727],[95.3038,29.13847],[95.41091,29.13007],[95.50842,29.13487],[95.72086,29.20797],[95.75149,29.32063],[95.84899,29.31464],[96.05361,29.38167],[96.31316,29.18643],[96.18682,29.11087],[96.20467,29.02325],[96.3626,29.10607],[96.61391,28.72742],[96.40929,28.51526],[96.48895,28.42955],[96.6455,28.61657],[96.85561,28.4875],[96.88445,28.39452],[96.98882,28.32564],[97.1289,28.3619],[97.34547,28.21385],[97.41729,28.29783],[97.47085,28.2688],[97.50518,28.49716],[97.56835,28.55628],[97.70705,28.5056],[97.79632,28.33168],[97.90069,28.3776],[98.15337,28.12114],[98.13964,27.9478],[98.32641,27.51385],[98.42529,27.55404],[98.43353,27.67086],[98.69582,27.56499],[98.7333,26.85615],[98.77547,26.61994],[98.72741,26.36183],[98.67797,26.24487],[98.7329,26.17218],[98.66884,26.09165],[98.63128,26.15492],[98.57085,26.11547],[98.60763,26.01512],[98.70818,25.86241],[98.63128,25.79937],[98.54064,25.85129],[98.40606,25.61129],[98.31268,25.55307],[98.25774,25.6051],[98.16848,25.62739],[98.18084,25.56298],[98.12591,25.50722],[98.14925,25.41547],[97.92541,25.20815],[97.83614,25.2715],[97.77023,25.11492],[97.72216,25.08508],[97.72903,24.91332],[97.79949,24.85655],[97.76481,24.8289],[97.73127,24.83015],[97.70181,24.84557],[97.64354,24.79171],[97.56648,24.76475],[97.56383,24.75535],[97.5542,24.74943],[97.54675,24.74202],[97.56525,24.72838],[97.56286,24.54535],[97.52757,24.43748],[97.60029,24.4401],[97.66998,24.45288],[97.7098,24.35658],[97.65624,24.33781],[97.66723,24.30027],[97.71941,24.29652],[97.76799,24.26365],[97.72998,24.2302],[97.72799,24.18883],[97.75305,24.16902],[97.72903,24.12606],[97.62363,24.00506],[97.5247,23.94032],[97.64667,23.84574],[97.72302,23.89288],[97.79456,23.94836],[97.79416,23.95663],[97.84328,23.97603],[97.86545,23.97723],[97.88811,23.97446],[97.8955,23.97758],[97.89676,23.97931],[97.89683,23.98389],[97.88814,23.98605],[97.88414,23.99405],[97.88616,24.00463],[97.90998,24.02094],[97.93951,24.01953],[97.98691,24.03897],[97.99583,24.04932],[98.04709,24.07616],[98.05302,24.07408],[98.05671,24.07961],[98.0607,24.07812],[98.06703,24.08028],[98.07806,24.07988],[98.20666,24.11406],[98.54476,24.13119],[98.59256,24.08371],[98.85319,24.13042],[98.87998,24.15624],[98.89632,24.10612],[98.67797,23.9644],[98.68209,23.80492],[98.79607,23.77947],[98.82933,23.72921],[98.81775,23.694],[98.88396,23.59555],[98.80294,23.5345],[98.82877,23.47908],[98.87683,23.48995],[98.92104,23.36946],[98.87573,23.33038],[98.93958,23.31414],[98.92515,23.29535],[98.88597,23.18656],[99.05975,23.16382],[99.04601,23.12215],[99.25741,23.09025],[99.34127,23.13099],[99.52214,23.08218],[99.54218,22.90014],[99.43537,22.94086],[99.45654,22.85726],[99.31243,22.73893],[99.38247,22.57544],[99.37972,22.50188],[99.28771,22.4105],[99.17318,22.18025],[99.19176,22.16983],[99.1552,22.15874],[99.33166,22.09656],[99.47585,22.13345],[99.85351,22.04183],[99.96612,22.05965],[99.99084,21.97053],[99.94003,21.82782],[99.98654,21.71064],[100.04956,21.66843],[100.12679,21.70539],[100.17486,21.65306],[100.10757,21.59945],[100.12542,21.50365],[100.1625,21.48704],[100.18447,21.51898],[100.25863,21.47043],[100.35201,21.53176],[100.42892,21.54325],[100.4811,21.46148],[100.57861,21.45637],[100.72143,21.51898],[100.87265,21.67396],[101.11744,21.77659],[101.15156,21.56129],[101.2124,21.56422],[101.19349,21.41959],[101.26912,21.36482],[101.2229,21.23271],[101.29326,21.17254],[101.54563,21.25668],[101.6068,21.23329],[101.59491,21.18621],[101.60886,21.17947],[101.66977,21.20004],[101.70548,21.14911],[101.7622,21.14813],[101.79266,21.19025],[101.76745,21.21571],[101.83887,21.20983],[101.84412,21.25291],[101.74014,21.30967],[101.74224,21.48276],[101.7727,21.51794],[101.7475,21.5873],[101.80001,21.57461],[101.83257,21.61562],[101.74555,21.72852],[101.7791,21.83019],[101.62566,21.96574],[101.57525,22.13026],[101.60675,22.13513],[101.53638,22.24794],[101.56789,22.28876],[101.61306,22.27515],[101.68973,22.46843],[101.7685,22.50337],[101.86828,22.38397],[101.90714,22.38688],[101.91344,22.44417],[101.98487,22.42766],[102.03633,22.46164],[102.1245,22.43372],[102.14099,22.40092],[102.16621,22.43336],[102.26428,22.41321],[102.25339,22.4607],[102.41061,22.64184],[102.38415,22.67919],[102.42618,22.69212],[102.46665,22.77108],[102.51802,22.77969],[102.57095,22.7036],[102.60675,22.73376],[102.8636,22.60735],[102.9321,22.48659],[103.0722,22.44775],[103.07843,22.50097],[103.17961,22.55705],[103.15782,22.59873],[103.18895,22.64471],[103.28079,22.68063],[103.32282,22.8127],[103.43179,22.75816],[103.43646,22.70648],[103.52675,22.59155],[103.57812,22.65764],[103.56255,22.69499],[103.64506,22.79979],[103.87904,22.56683],[103.93286,22.52703],[103.94513,22.52553],[103.95191,22.5134],[103.96352,22.50584],[103.96783,22.51173],[103.97384,22.50634],[103.99247,22.51958],[104.01088,22.51823],[104.03734,22.72945],[104.11384,22.80363],[104.27084,22.8457],[104.25683,22.76534],[104.35593,22.69353],[104.47225,22.75813],[104.58122,22.85571],[104.60457,22.81841],[104.65283,22.83419],[104.72755,22.81984],[104.77114,22.90017],[104.84942,22.93631],[104.86765,22.95178],[104.8334,23.01484],[104.79478,23.12934],[104.87382,23.12854],[104.87992,23.17141],[104.91435,23.18666],[104.9486,23.17235],[104.96532,23.20463],[104.98712,23.19176],[105.07002,23.26248],[105.11672,23.25247],[105.17276,23.28679],[105.22569,23.27249],[105.32376,23.39684],[105.40782,23.28107],[105.42805,23.30824],[105.49966,23.20669],[105.56037,23.16806],[105.57594,23.075],[105.72382,23.06641],[105.8726,22.92756],[105.90119,22.94168],[105.99568,22.94178],[106.00179,22.99049],[106.19705,22.98475],[106.27022,22.87722],[106.34961,22.86718],[106.49749,22.91164],[106.51306,22.94891],[106.55976,22.92311],[106.60179,22.92884],[106.6516,22.86862],[106.6734,22.89587],[106.71387,22.88296],[106.71128,22.85982],[106.78422,22.81532],[106.81271,22.8226],[106.83685,22.8098],[106.82404,22.7881],[106.76293,22.73491],[106.72321,22.63606],[106.71698,22.58432],[106.65316,22.5757],[106.61269,22.60301],[106.58395,22.474],[106.55665,22.46498],[106.57221,22.37],[106.55976,22.34841],[106.6516,22.33977],[106.69986,22.22309],[106.67495,22.1885],[106.6983,22.15102],[106.70142,22.02409],[106.68274,21.99811],[106.69276,21.96013],[106.72551,21.97923],[106.74345,22.00965],[106.81038,21.97934],[106.9178,21.97357],[106.92714,21.93459],[106.97228,21.92592],[106.99252,21.95191],[107.05634,21.92303],[107.06101,21.88982],[107.00964,21.85948],[107.02615,21.81981],[107.10771,21.79879],[107.20734,21.71493],[107.24625,21.7077],[107.29296,21.74674],[107.35834,21.6672],[107.35989,21.60063],[107.38636,21.59774],[107.41593,21.64839],[107.47197,21.6672],[107.49532,21.62958],[107.49065,21.59774],[107.54047,21.5934],[107.56537,21.61945],[107.66967,21.60787],[107.80355,21.66141],[107.86114,21.65128],[107.90006,21.5905],[107.92652,21.58906],[107.95232,21.5388],[107.96774,21.53601],[107.97074,21.54072],[107.97383,21.53961],[107.97932,21.54503],[108.02926,21.54997],[108.0569,21.53604],[108.10003,21.47338],[108.00365,17.98159],[111.60491,13.57105],[118.41371,24.06775],[118.11703,24.39734],[118.28244,24.51231],[118.35291,24.51645],[118.42453,24.54644],[118.56434,24.49266],[120.49232,25.22863],[121.03532,26.8787],[123.5458,31.01942],[122.29378,31.76513],[122.80525,33.30571],[123.85601,37.49093],[123.90497,38.79949],[124.17532,39.8232],[124.23201,39.9248],[124.35029,39.95639],[124.37089,40.03004],[124.3322,40.05573],[124.38556,40.11047],[124.40719,40.13655],[124.86913,40.45387],[125.71172,40.85223],[125.76869,40.87908],[126.00335,40.92835],[126.242,41.15454],[126.53189,41.35206],[126.60631,41.65565],[126.90729,41.79955],[127.17841,41.59714],[127.29712,41.49473],[127.92943,41.44291],[128.02633,41.42103],[128.03311,41.39232],[128.12967,41.37931],[128.18546,41.41279],[128.20061,41.40895],[128.30716,41.60322],[128.15119,41.74568],[128.04487,42.01769],[128.94007,42.03537],[128.96068,42.06657],[129.15178,42.17224],[129.22285,42.26491],[129.22423,42.3553],[129.28541,42.41574],[129.42882,42.44702],[129.54701,42.37254],[129.60482,42.44461],[129.72541,42.43739],[129.75294,42.59409],[129.77183,42.69435],[129.7835,42.76521],[129.80719,42.79218],[129.83277,42.86746],[129.85261,42.96494],[129.8865,43.00395],[129.95082,43.01051],[129.96409,42.97306],[130.12957,42.98361],[130.09764,42.91425],[130.26095,42.9027],[130.23068,42.80125],[130.2385,42.71127],[130.41826,42.6011],[130.44361,42.54849],[130.50123,42.61636],[130.55143,42.52158],[130.62107,42.58413],[130.56576,42.68925],[130.40213,42.70788],[130.44361,42.76205],[130.66524,42.84753],[131.02438,42.86518],[131.02668,42.91246],[131.135,42.94114],[131.10274,43.04734],[131.20414,43.13654],[131.19031,43.21385],[131.30324,43.39498],[131.29402,43.46695],[131.19492,43.53047],[131.21105,43.82383],[131.26176,43.94011],[131.23583,43.96085],[131.25484,44.03131],[131.30365,44.04262],[131.1108,44.70266],[130.95639,44.85154],[131.48415,44.99513],[131.68466,45.12374],[131.66852,45.2196],[131.76532,45.22609],[131.86903,45.33636],[131.99417,45.2567],[132.83978,45.05916],[132.96373,45.0212],[133.12293,45.1332],[133.09279,45.25693],[133.19419,45.51913],[133.41083,45.57723],[133.48457,45.86203],[133.60442,45.90053],[133.67569,45.9759],[133.72695,46.05576],[133.68047,46.14697],[133.88097,46.25066],[133.91496,46.4274],[133.84104,46.46681],[134.03538,46.75668],[134.20016,47.33458],[134.50898,47.4812],[134.7671,47.72051],[134.55508,47.98651],[134.67098,48.1564],[134.75328,48.36763],[134.49516,48.42884],[132.66989,47.96491],[132.57309,47.71741],[131.90448,47.68011],[131.2635,47.73325],[131.09871,47.6852],[130.95985,47.6957],[130.90915,47.90623],[130.65103,48.10052],[130.84462,48.30942],[130.52147,48.61745],[130.66946,48.88251],[130.43232,48.90844],[130.2355,48.86741],[129.85416,49.11067],[129.67598,49.29596],[129.50685,49.42398],[129.40398,49.44194],[129.35317,49.3481],[129.23232,49.40353],[129.11153,49.36813],[128.72896,49.58676],[127.83476,49.5748],[127.53516,49.84306],[127.49299,50.01251],[127.60515,50.23503],[127.37384,50.28393],[127.36009,50.43787],[127.28765,50.46585],[127.36335,50.58306],[127.28165,50.72075],[127.14586,50.91152],[126.93135,51.0841],[126.90369,51.3238],[126.68349,51.70607],[126.44606,51.98254],[126.558,52.13738],[125.6131,53.07229]],[[113.56865,22.20973],[113.57123,22.20416],[113.60504,22.20464],[113.63011,22.10782],[113.57191,22.07696],[113.54839,22.10909],[113.54942,22.14519],[113.54093,22.15497],[113.52659,22.18271],[113.53552,22.20607],[113.53301,22.21235],[113.53591,22.21369],[113.54093,22.21314],[113.54333,22.21688],[113.5508,22.21672],[113.56865,22.20973]],[[114.50148,22.15017],[113.92195,22.13873],[113.83338,22.1826],[113.81621,22.2163],[113.86771,22.42972],[114.03113,22.5065],[114.05438,22.5026],[114.05729,22.51104],[114.06272,22.51617],[114.07267,22.51855],[114.07817,22.52997],[114.08606,22.53276],[114.09048,22.53716],[114.09692,22.53435],[114.1034,22.5352],[114.11181,22.52878],[114.11656,22.53415],[114.12665,22.54003],[114.13823,22.54319],[114.1482,22.54091],[114.15123,22.55163],[114.1597,22.56041],[114.17247,22.55944],[114.18338,22.55444],[114.20655,22.55706],[114.22185,22.55343],[114.22888,22.5436],[114.25154,22.55977],[114.44998,22.55977],[114.50148,22.15017]]]]}},{type:"Feature",properties:{iso1A2:"CO",iso1A3:"COL",iso1N3:"170",wikidata:"Q739",nameEn:"Colombia",groups:["005","419","019"],callingCodes:["57"]},geometry:{type:"MultiPolygon",coordinates:[[[[-71.19849,12.65801],[-81.58685,18.0025],[-82.06974,14.49418],[-82.56142,11.91792],[-78.79327,9.93766],[-77.58292,9.22278],[-77.32389,8.81247],[-77.45064,8.49991],[-77.17257,7.97422],[-77.57185,7.51147],[-77.72514,7.72348],[-77.72157,7.47612],[-77.81426,7.48319],[-77.89178,7.22681],[-78.06168,7.07793],[-82.12561,4.00341],[-78.87137,1.47457],[-78.42749,1.15389],[-77.85677,0.80197],[-77.7148,0.85003],[-77.68613,0.83029],[-77.66416,0.81604],[-77.67815,0.73863],[-77.49984,0.64476],[-77.52001,0.40782],[-76.89177,0.24736],[-76.4094,0.24015],[-76.41215,0.38228],[-76.23441,0.42294],[-75.82927,0.09578],[-75.25764,-0.11943],[-75.18513,-0.0308],[-74.42701,-0.50218],[-74.26675,-0.97229],[-73.65312,-1.26222],[-72.92587,-2.44514],[-71.75223,-2.15058],[-70.94377,-2.23142],[-70.04609,-2.73906],[-70.71396,-3.7921],[-70.52393,-3.87553],[-70.3374,-3.79505],[-69.94708,-4.2431],[-69.43395,-1.42219],[-69.4215,-1.01853],[-69.59796,-0.75136],[-69.603,-0.51947],[-70.03658,-0.19681],[-70.04162,0.55437],[-69.47696,0.71065],[-69.20976,0.57958],[-69.14422,0.84172],[-69.26017,1.06856],[-69.82987,1.07864],[-69.83491,1.69353],[-69.53746,1.76408],[-69.38621,1.70865],[-68.18128,1.72881],[-68.26699,1.83463],[-68.18632,2.00091],[-67.9292,1.82455],[-67.40488,2.22258],[-67.299,1.87494],[-67.15784,1.80439],[-67.08222,1.17441],[-66.85795,1.22998],[-67.21967,2.35778],[-67.65696,2.81691],[-67.85862,2.79173],[-67.85862,2.86727],[-67.30945,3.38393],[-67.50067,3.75812],[-67.62671,3.74303],[-67.85358,4.53249],[-67.83341,5.31104],[-67.59141,5.5369],[-67.63914,5.64963],[-67.58558,5.84537],[-67.43513,5.98835],[-67.4625,6.20625],[-67.60654,6.2891],[-69.41843,6.1072],[-70.10716,6.96516],[-70.7596,7.09799],[-71.03941,6.98163],[-71.37234,7.01588],[-71.42212,7.03854],[-71.44118,7.02116],[-71.82441,7.04314],[-72.04895,7.03837],[-72.19437,7.37034],[-72.43132,7.40034],[-72.47415,7.48928],[-72.45321,7.57232],[-72.47827,7.65604],[-72.46763,7.79518],[-72.44454,7.86031],[-72.46183,7.90682],[-72.45806,7.91141],[-72.47042,7.92306],[-72.48183,7.92909],[-72.48801,7.94329],[-72.47213,7.96106],[-72.39137,8.03534],[-72.35163,8.01163],[-72.36987,8.19976],[-72.4042,8.36513],[-72.65474,8.61428],[-72.77415,9.10165],[-72.94052,9.10663],[-73.02119,9.27584],[-73.36905,9.16636],[-72.98085,9.85253],[-72.88002,10.44309],[-72.4767,11.1117],[-72.24983,11.14138],[-71.9675,11.65536],[-71.3275,11.85],[-70.92579,11.96275],[-71.19849,12.65801]]]]}},{type:"Feature",properties:{iso1A2:"CP",iso1A3:"CPT",wikidata:"Q161258",nameEn:"Clipperton Island",country:"FR",isoStatus:"excRes"},geometry:{type:"MultiPolygon",coordinates:[[[[-110.36279,9.79626],[-108.755,9.84085],[-109.04145,11.13245],[-110.36279,9.79626]]]]}},{type:"Feature",properties:{iso1A2:"CR",iso1A3:"CRI",iso1N3:"188",wikidata:"Q800",nameEn:"Costa Rica",groups:["013","003","419","019"],callingCodes:["506"]},geometry:{type:"MultiPolygon",coordinates:[[[[-83.68276,11.01562],[-83.66597,10.79916],[-83.90838,10.71161],[-84.68197,11.07568],[-84.92439,10.9497],[-85.60529,11.22607],[-85.71223,11.06868],[-86.14524,11.09059],[-87.41779,5.02401],[-82.94503,7.93865],[-82.89978,8.04083],[-82.89137,8.05755],[-82.88641,8.10219],[-82.9388,8.26634],[-83.05209,8.33394],[-82.93056,8.43465],[-82.8679,8.44042],[-82.8382,8.48117],[-82.83322,8.52464],[-82.83975,8.54755],[-82.82739,8.60153],[-82.8794,8.6981],[-82.92068,8.74832],[-82.91377,8.774],[-82.88253,8.83331],[-82.72126,8.97125],[-82.93516,9.07687],[-82.93516,9.46741],[-82.84871,9.4973],[-82.87919,9.62645],[-82.77206,9.59573],[-82.66667,9.49746],[-82.61345,9.49881],[-82.56507,9.57279],[-82.51044,9.65379],[-83.54024,10.96805],[-83.68276,11.01562]]]]}},{type:"Feature",properties:{iso1A2:"CU",iso1A3:"CUB",iso1N3:"192",wikidata:"Q241",nameEn:"Cuba",groups:["029","003","419","019"],callingCodes:["53"]},geometry:{type:"MultiPolygon",coordinates:[[[[-73.62304,20.6935],[-82.02215,24.23074],[-85.77883,21.92705],[-74.81171,18.82201],[-73.62304,20.6935]]]]}},{type:"Feature",properties:{iso1A2:"CV",iso1A3:"CPV",iso1N3:"132",wikidata:"Q1011",nameEn:"Cape Verde",groups:["011","202","002"],callingCodes:["238"]},geometry:{type:"MultiPolygon",coordinates:[[[[-28.81604,14.57305],[-20.39702,14.12816],[-23.37101,19.134],[-28.81604,14.57305]]]]}},{type:"Feature",properties:{iso1A2:"CW",iso1A3:"CUW",iso1N3:"531",wikidata:"Q25279",nameEn:"Curaçao",country:"NL",groups:["029","003","419","019"],callingCodes:["599"]},geometry:{type:"MultiPolygon",coordinates:[[[[-68.90012,12.62309],[-69.59009,12.46019],[-68.99639,11.79035],[-68.33524,11.78151],[-68.90012,12.62309]]]]}},{type:"Feature",properties:{iso1A2:"CX",iso1A3:"CXR",iso1N3:"162",wikidata:"Q31063",nameEn:"Christmas Island",country:"AU",groups:["053","009"],driveSide:"left",callingCodes:["61"]},geometry:{type:"MultiPolygon",coordinates:[[[[105.66835,-9.31927],[104.67494,-11.2566],[106.66176,-11.14349],[105.66835,-9.31927]]]]}},{type:"Feature",properties:{iso1A2:"CY",iso1A3:"CYP",iso1N3:"196",wikidata:"Q229",nameEn:"Cyprus",groups:["EU","145","142"],driveSide:"left",callingCodes:["357"]},geometry:{type:"MultiPolygon",coordinates:[[[[33.70639,34.99303],[33.71514,35.00294],[33.69731,35.01754],[33.69938,35.03123],[33.67678,35.03866],[33.67742,35.05963],[33.68474,35.06602],[33.69095,35.06237],[33.70861,35.07644],[33.7161,35.07279],[33.70209,35.04882],[33.71482,35.03722],[33.73824,35.05321],[33.76106,35.04253],[33.78581,35.05104],[33.82067,35.07826],[33.84168,35.06823],[33.8541,35.07201],[33.87479,35.08881],[33.87097,35.09389],[33.87622,35.10457],[33.87224,35.12293],[33.88561,35.12449],[33.88943,35.12007],[33.88737,35.11408],[33.89853,35.11377],[33.91789,35.08688],[33.91299,35.07579],[33.90247,35.07686],[33.89485,35.06873],[33.88367,35.07877],[33.85261,35.0574],[33.8355,35.05777],[33.82051,35.0667],[33.8012,35.04786],[33.81524,35.04192],[33.83055,35.02865],[33.82875,35.01685],[33.84045,35.00616],[33.85216,35.00579],[33.85891,35.001],[33.85621,34.98956],[33.83505,34.98108],[33.84811,34.97075],[33.86432,34.97592],[33.90075,34.96623],[33.98684,34.76642],[35.48515,34.70851],[35.51152,36.10954],[32.82353,35.70297],[30.15137,34.08517],[32.74412,34.43926],[32.75515,34.64985],[32.76136,34.68318],[32.79433,34.67883],[32.82717,34.70622],[32.86014,34.70585],[32.86167,34.68734],[32.9068,34.66102],[32.91398,34.67343],[32.93043,34.67091],[32.92807,34.66736],[32.93449,34.66241],[32.93693,34.67027],[32.94379,34.67111],[32.94683,34.67907],[32.95539,34.68471],[32.99135,34.68061],[32.98668,34.67268],[32.99014,34.65518],[32.97736,34.65277],[32.97079,34.66112],[32.95325,34.66462],[32.94796,34.6587],[32.94976,34.65204],[32.95471,34.64528],[32.95323,34.64075],[32.95891,34.62919],[32.96718,34.63446],[32.96968,34.64046],[33.0138,34.64424],[33.26744,34.49942],[33.83531,34.73974],[33.70575,34.97947],[33.70639,34.99303]]],[[[33.74144,35.01053],[33.7492,35.01319],[33.74983,35.02274],[33.74265,35.02329],[33.73781,35.02181],[33.7343,35.01178],[33.74144,35.01053]]],[[[33.77312,34.9976],[33.75994,35.00113],[33.75682,34.99916],[33.76605,34.99543],[33.76738,34.99188],[33.7778,34.98981],[33.77843,34.988],[33.78149,34.98854],[33.78318,34.98699],[33.78571,34.98951],[33.78917,34.98854],[33.79191,34.98914],[33.78516,34.99582],[33.77553,34.99518],[33.77312,34.9976]]]]}},{type:"Feature",properties:{iso1A2:"CZ",iso1A3:"CZE",iso1N3:"203",wikidata:"Q213",nameEn:"Czechia",groups:["EU","151","150"],callingCodes:["420"]},geometry:{type:"MultiPolygon",coordinates:[[[[14.82803,50.86966],[14.79139,50.81438],[14.70661,50.84096],[14.61993,50.86049],[14.63434,50.8883],[14.65259,50.90513],[14.64802,50.93241],[14.58024,50.91443],[14.56374,50.922],[14.59702,50.96148],[14.59908,50.98685],[14.58215,50.99306],[14.56432,51.01008],[14.53438,51.00374],[14.53321,51.01679],[14.49873,51.02242],[14.50809,51.0427],[14.49991,51.04692],[14.49154,51.04382],[14.49202,51.02286],[14.45827,51.03712],[14.41335,51.02086],[14.30098,51.05515],[14.25665,50.98935],[14.28776,50.97718],[14.32353,50.98556],[14.32793,50.97379],[14.30251,50.96606],[14.31422,50.95243],[14.39848,50.93866],[14.38691,50.89907],[14.30098,50.88448],[14.27123,50.89386],[14.24314,50.88761],[14.22331,50.86049],[14.02982,50.80662],[13.98864,50.8177],[13.89113,50.78533],[13.89444,50.74142],[13.82942,50.7251],[13.76316,50.73487],[13.70204,50.71771],[13.65977,50.73096],[13.52474,50.70394],[13.53748,50.67654],[13.5226,50.64721],[13.49742,50.63133],[13.46413,50.60102],[13.42189,50.61243],[13.37485,50.64931],[13.37805,50.627],[13.32264,50.60317],[13.32594,50.58009],[13.29454,50.57904],[13.25158,50.59268],[13.19043,50.50237],[13.13424,50.51709],[13.08301,50.50132],[13.0312,50.50944],[13.02038,50.4734],[13.02147,50.44763],[12.98433,50.42016],[12.94058,50.40944],[12.82465,50.45738],[12.73476,50.43237],[12.73044,50.42268],[12.70731,50.39948],[12.67261,50.41949],[12.51356,50.39694],[12.48747,50.37278],[12.49214,50.35228],[12.48256,50.34784],[12.46643,50.35527],[12.43722,50.33774],[12.43371,50.32506],[12.39924,50.32302],[12.40158,50.29521],[12.36594,50.28289],[12.35425,50.23993],[12.33263,50.24367],[12.32445,50.20442],[12.33847,50.19432],[12.32596,50.17146],[12.29232,50.17524],[12.28063,50.19544],[12.28755,50.22429],[12.23943,50.24594],[12.24791,50.25525],[12.26953,50.25189],[12.25119,50.27079],[12.20823,50.2729],[12.18013,50.32146],[12.10907,50.32041],[12.13716,50.27396],[12.09287,50.25032],[12.19335,50.19997],[12.21484,50.16399],[12.1917,50.13434],[12.2073,50.10315],[12.23709,50.10213],[12.27433,50.0771],[12.26111,50.06331],[12.30798,50.05719],[12.49908,49.97305],[12.47264,49.94222],[12.55197,49.92094],[12.48256,49.83575],[12.46603,49.78882],[12.40489,49.76321],[12.4462,49.70233],[12.52553,49.68415],[12.53544,49.61888],[12.56188,49.6146],[12.60155,49.52887],[12.64782,49.52565],[12.64121,49.47628],[12.669,49.42935],[12.71227,49.42363],[12.75854,49.3989],[12.78168,49.34618],[12.88414,49.33541],[12.88249,49.35479],[12.94859,49.34079],[13.03618,49.30417],[13.02957,49.27399],[13.05883,49.26259],[13.17665,49.16713],[13.17019,49.14339],[13.20405,49.12303],[13.23689,49.11412],[13.28242,49.1228],[13.39479,49.04812],[13.40802,48.98851],[13.50221,48.93752],[13.50552,48.97441],[13.58319,48.96899],[13.61624,48.9462],[13.67739,48.87886],[13.73854,48.88538],[13.76994,48.83537],[13.78977,48.83319],[13.8096,48.77877],[13.84023,48.76988],[14.06151,48.66873],[14.01482,48.63788],[14.09104,48.5943],[14.20691,48.5898],[14.33909,48.55852],[14.43076,48.58855],[14.4587,48.64695],[14.56139,48.60429],[14.60808,48.62881],[14.66762,48.58215],[14.71794,48.59794],[14.72756,48.69502],[14.80584,48.73489],[14.80821,48.77711],[14.81545,48.7874],[14.94773,48.76268],[14.95641,48.75915],[14.9758,48.76857],[14.98112,48.77524],[14.9782,48.7766],[14.98032,48.77959],[14.95072,48.79101],[14.98917,48.90082],[14.97612,48.96983],[14.99878,49.01444],[15.15534,48.99056],[15.16358,48.94278],[15.26177,48.95766],[15.28305,48.98831],[15.34823,48.98444],[15.48027,48.94481],[15.51357,48.91549],[15.61622,48.89541],[15.6921,48.85973],[15.75341,48.8516],[15.78087,48.87644],[15.84404,48.86921],[16.06034,48.75436],[16.37345,48.729],[16.40915,48.74576],[16.46134,48.80865],[16.67008,48.77699],[16.68518,48.7281],[16.71883,48.73806],[16.79779,48.70998],[16.90354,48.71541],[16.93955,48.60371],[17.00215,48.70887],[17.11202,48.82925],[17.19355,48.87602],[17.29054,48.85546],[17.3853,48.80936],[17.45671,48.85004],[17.5295,48.81117],[17.7094,48.86721],[17.73126,48.87885],[17.77944,48.92318],[17.87831,48.92679],[17.91814,49.01784],[18.06885,49.03157],[18.1104,49.08624],[18.15022,49.24518],[18.18456,49.28909],[18.36446,49.3267],[18.4139,49.36517],[18.4084,49.40003],[18.44686,49.39467],[18.54848,49.47059],[18.53063,49.49022],[18.57183,49.51162],[18.6144,49.49824],[18.67757,49.50895],[18.74761,49.492],[18.84521,49.51672],[18.84786,49.5446],[18.80479,49.6815],[18.72838,49.68163],[18.69817,49.70473],[18.62676,49.71983],[18.62943,49.74603],[18.62645,49.75002],[18.61368,49.75426],[18.61278,49.7618],[18.57183,49.83334],[18.60341,49.86256],[18.57045,49.87849],[18.57697,49.91565],[18.54299,49.92537],[18.54495,49.9079],[18.53423,49.89906],[18.41604,49.93498],[18.33562,49.94747],[18.33278,49.92415],[18.31914,49.91565],[18.27794,49.93863],[18.27107,49.96779],[18.21752,49.97309],[18.20241,49.99958],[18.10628,50.00223],[18.07898,50.04535],[18.03212,50.06574],[18.00396,50.04954],[18.04585,50.03311],[18.04585,50.01194],[18.00191,50.01723],[17.86886,49.97452],[17.77669,50.02253],[17.7506,50.07896],[17.6888,50.12037],[17.66683,50.10275],[17.59404,50.16437],[17.70528,50.18812],[17.76296,50.23382],[17.72176,50.25665],[17.74648,50.29966],[17.69292,50.32859],[17.67764,50.28977],[17.58889,50.27837],[17.3702,50.28123],[17.34548,50.2628],[17.34273,50.32947],[17.27681,50.32246],[17.19991,50.3654],[17.19579,50.38817],[17.14498,50.38117],[17.1224,50.39494],[16.89229,50.45117],[16.85933,50.41093],[16.90877,50.38642],[16.94448,50.31281],[16.99803,50.30316],[17.02138,50.27772],[16.99803,50.25753],[17.02825,50.23118],[17.00353,50.21449],[16.98018,50.24172],[16.8456,50.20834],[16.7014,50.09659],[16.63137,50.1142],[16.55446,50.16613],[16.56407,50.21009],[16.42674,50.32509],[16.39379,50.3207],[16.3622,50.34875],[16.36495,50.37679],[16.30289,50.38292],[16.28118,50.36891],[16.22821,50.41054],[16.21585,50.40627],[16.19526,50.43291],[16.31413,50.50274],[16.34572,50.49575],[16.44597,50.58041],[16.33611,50.66579],[16.23174,50.67101],[16.20839,50.63096],[16.10265,50.66405],[16.02437,50.60046],[15.98317,50.61528],[16.0175,50.63009],[15.97219,50.69799],[15.87331,50.67188],[15.81683,50.75666],[15.73186,50.73885],[15.43798,50.80833],[15.3803,50.77187],[15.36656,50.83956],[15.2773,50.8907],[15.27043,50.97724],[15.2361,50.99886],[15.1743,50.9833],[15.16744,51.01959],[15.11937,50.99021],[15.10152,51.01095],[15.06218,51.02269],[15.03895,51.0123],[15.02433,51.0242],[14.96419,50.99108],[15.01088,50.97984],[14.99852,50.86817],[14.82803,50.86966]]]]}},{type:"Feature",properties:{iso1A2:"DE",iso1A3:"DEU",iso1N3:"276",wikidata:"Q183",nameEn:"Germany",groups:["EU","155","150"],callingCodes:["49"]},geometry:{type:"MultiPolygon",coordinates:[[[[8.70847,47.68904],[8.71773,47.69088],[8.70237,47.71453],[8.66416,47.71367],[8.67508,47.6979],[8.65769,47.68928],[8.66837,47.68437],[8.68985,47.69552],[8.70847,47.68904]]],[[[8.72617,47.69651],[8.72809,47.69282],[8.75856,47.68969],[8.79511,47.67462],[8.79966,47.70222],[8.76965,47.7075],[8.77309,47.72059],[8.80663,47.73821],[8.82002,47.71458],[8.86989,47.70504],[8.85065,47.68209],[8.87383,47.67045],[8.87625,47.65441],[8.89946,47.64769],[8.94093,47.65596],[9.02093,47.6868],[9.09891,47.67801],[9.13845,47.66389],[9.15181,47.66904],[9.1705,47.65513],[9.1755,47.65584],[9.17593,47.65399],[9.18203,47.65598],[9.25619,47.65939],[9.55125,47.53629],[9.72736,47.53457],[9.76748,47.5934],[9.80254,47.59419],[9.82591,47.58158],[9.8189,47.54688],[9.87499,47.52953],[9.87733,47.54688],[9.92407,47.53111],[9.96029,47.53899],[10.00003,47.48216],[10.03859,47.48927],[10.07131,47.45531],[10.09001,47.46005],[10.1052,47.4316],[10.06897,47.40709],[10.09819,47.35724],[10.11805,47.37228],[10.16362,47.36674],[10.17648,47.38889],[10.2127,47.38019],[10.22774,47.38904],[10.23757,47.37609],[10.19998,47.32832],[10.2147,47.31014],[10.17648,47.29149],[10.17531,47.27167],[10.23257,47.27088],[10.33424,47.30813],[10.39851,47.37623],[10.4324,47.38494],[10.4359,47.41183],[10.47446,47.43318],[10.46278,47.47901],[10.44291,47.48453],[10.4324,47.50111],[10.44992,47.5524],[10.43473,47.58394],[10.47329,47.58552],[10.48849,47.54057],[10.56912,47.53584],[10.60337,47.56755],[10.63456,47.5591],[10.68832,47.55752],[10.6965,47.54253],[10.7596,47.53228],[10.77596,47.51729],[10.88814,47.53701],[10.91268,47.51334],[10.86945,47.5015],[10.87061,47.4786],[10.90918,47.48571],[10.93839,47.48018],[10.92437,47.46991],[10.98513,47.42882],[10.97111,47.41617],[10.97111,47.39561],[11.11835,47.39719],[11.12536,47.41222],[11.20482,47.43198],[11.25157,47.43277],[11.22002,47.3964],[11.27844,47.39956],[11.29597,47.42566],[11.33804,47.44937],[11.4175,47.44621],[11.38128,47.47465],[11.4362,47.51413],[11.52618,47.50939],[11.58578,47.52281],[11.58811,47.55515],[11.60681,47.57881],[11.63934,47.59202],[11.84052,47.58354],[11.85572,47.60166],[12.0088,47.62451],[12.02282,47.61033],[12.05788,47.61742],[12.13734,47.60639],[12.17824,47.61506],[12.18145,47.61019],[12.17737,47.60121],[12.18568,47.6049],[12.20398,47.60667],[12.20801,47.61082],[12.19895,47.64085],[12.18507,47.65984],[12.18347,47.66663],[12.16769,47.68167],[12.16217,47.70105],[12.18303,47.70065],[12.22571,47.71776],[12.2542,47.7433],[12.26238,47.73544],[12.24017,47.69534],[12.26004,47.67725],[12.27991,47.68827],[12.336,47.69534],[12.37222,47.68433],[12.43883,47.6977],[12.44117,47.6741],[12.50076,47.62293],[12.53816,47.63553],[12.57438,47.63238],[12.6071,47.6741],[12.7357,47.6787],[12.77777,47.66689],[12.76492,47.64485],[12.82101,47.61493],[12.77427,47.58025],[12.80699,47.54477],[12.84672,47.54556],[12.85256,47.52741],[12.9624,47.47452],[12.98344,47.48716],[12.9998,47.46267],[13.04537,47.49426],[13.03252,47.53373],[13.05355,47.56291],[13.04537,47.58183],[13.06641,47.58577],[13.06407,47.60075],[13.09562,47.63304],[13.07692,47.68814],[13.01382,47.72116],[12.98578,47.7078],[12.92969,47.71094],[12.91333,47.7178],[12.90274,47.72513],[12.91711,47.74026],[12.9353,47.74788],[12.94371,47.76281],[12.93202,47.77302],[12.96311,47.79957],[12.98543,47.82896],[13.00588,47.84374],[12.94163,47.92927],[12.93886,47.94046],[12.93642,47.94436],[12.93419,47.94063],[12.92668,47.93879],[12.91985,47.94069],[12.9211,47.95135],[12.91683,47.95647],[12.87476,47.96195],[12.8549,48.01122],[12.76141,48.07373],[12.74973,48.10885],[12.7617,48.12796],[12.78595,48.12445],[12.80676,48.14979],[12.82673,48.15245],[12.8362,48.15876],[12.836,48.1647],[12.84475,48.16556],[12.87126,48.20318],[12.95306,48.20629],[13.02083,48.25689],[13.0851,48.27711],[13.126,48.27867],[13.18093,48.29577],[13.26039,48.29422],[13.30897,48.31575],[13.40709,48.37292],[13.43929,48.43386],[13.42527,48.45711],[13.45727,48.51092],[13.43695,48.55776],[13.45214,48.56472],[13.46967,48.55157],[13.50663,48.57506],[13.50131,48.58091],[13.51291,48.59023],[13.57535,48.55912],[13.59705,48.57013],[13.62508,48.55501],[13.65186,48.55092],[13.66113,48.53558],[13.72802,48.51208],[13.74816,48.53058],[13.7513,48.5624],[13.76921,48.55324],[13.80519,48.58026],[13.80038,48.59487],[13.82609,48.62345],[13.81901,48.6761],[13.81283,48.68426],[13.81791,48.69832],[13.79337,48.71375],[13.81863,48.73257],[13.82266,48.75544],[13.84023,48.76988],[13.8096,48.77877],[13.78977,48.83319],[13.76994,48.83537],[13.73854,48.88538],[13.67739,48.87886],[13.61624,48.9462],[13.58319,48.96899],[13.50552,48.97441],[13.50221,48.93752],[13.40802,48.98851],[13.39479,49.04812],[13.28242,49.1228],[13.23689,49.11412],[13.20405,49.12303],[13.17019,49.14339],[13.17665,49.16713],[13.05883,49.26259],[13.02957,49.27399],[13.03618,49.30417],[12.94859,49.34079],[12.88249,49.35479],[12.88414,49.33541],[12.78168,49.34618],[12.75854,49.3989],[12.71227,49.42363],[12.669,49.42935],[12.64121,49.47628],[12.64782,49.52565],[12.60155,49.52887],[12.56188,49.6146],[12.53544,49.61888],[12.52553,49.68415],[12.4462,49.70233],[12.40489,49.76321],[12.46603,49.78882],[12.48256,49.83575],[12.55197,49.92094],[12.47264,49.94222],[12.49908,49.97305],[12.30798,50.05719],[12.26111,50.06331],[12.27433,50.0771],[12.23709,50.10213],[12.2073,50.10315],[12.1917,50.13434],[12.21484,50.16399],[12.19335,50.19997],[12.09287,50.25032],[12.13716,50.27396],[12.10907,50.32041],[12.18013,50.32146],[12.20823,50.2729],[12.25119,50.27079],[12.26953,50.25189],[12.24791,50.25525],[12.23943,50.24594],[12.28755,50.22429],[12.28063,50.19544],[12.29232,50.17524],[12.32596,50.17146],[12.33847,50.19432],[12.32445,50.20442],[12.33263,50.24367],[12.35425,50.23993],[12.36594,50.28289],[12.40158,50.29521],[12.39924,50.32302],[12.43371,50.32506],[12.43722,50.33774],[12.46643,50.35527],[12.48256,50.34784],[12.49214,50.35228],[12.48747,50.37278],[12.51356,50.39694],[12.67261,50.41949],[12.70731,50.39948],[12.73044,50.42268],[12.73476,50.43237],[12.82465,50.45738],[12.94058,50.40944],[12.98433,50.42016],[13.02147,50.44763],[13.02038,50.4734],[13.0312,50.50944],[13.08301,50.50132],[13.13424,50.51709],[13.19043,50.50237],[13.25158,50.59268],[13.29454,50.57904],[13.32594,50.58009],[13.32264,50.60317],[13.37805,50.627],[13.37485,50.64931],[13.42189,50.61243],[13.46413,50.60102],[13.49742,50.63133],[13.5226,50.64721],[13.53748,50.67654],[13.52474,50.70394],[13.65977,50.73096],[13.70204,50.71771],[13.76316,50.73487],[13.82942,50.7251],[13.89444,50.74142],[13.89113,50.78533],[13.98864,50.8177],[14.02982,50.80662],[14.22331,50.86049],[14.24314,50.88761],[14.27123,50.89386],[14.30098,50.88448],[14.38691,50.89907],[14.39848,50.93866],[14.31422,50.95243],[14.30251,50.96606],[14.32793,50.97379],[14.32353,50.98556],[14.28776,50.97718],[14.25665,50.98935],[14.30098,51.05515],[14.41335,51.02086],[14.45827,51.03712],[14.49202,51.02286],[14.49154,51.04382],[14.49991,51.04692],[14.50809,51.0427],[14.49873,51.02242],[14.53321,51.01679],[14.53438,51.00374],[14.56432,51.01008],[14.58215,50.99306],[14.59908,50.98685],[14.59702,50.96148],[14.56374,50.922],[14.58024,50.91443],[14.64802,50.93241],[14.65259,50.90513],[14.63434,50.8883],[14.61993,50.86049],[14.70661,50.84096],[14.79139,50.81438],[14.82803,50.86966],[14.81664,50.88148],[14.89681,50.9422],[14.89252,50.94999],[14.92942,50.99744],[14.95529,51.04552],[14.97938,51.07742],[14.98229,51.11354],[14.99689,51.12205],[14.99079,51.14284],[14.99646,51.14365],[15.00083,51.14974],[14.99414,51.15813],[14.99311,51.16249],[15.0047,51.16874],[15.01242,51.21285],[15.04288,51.28387],[14.98008,51.33449],[14.96899,51.38367],[14.9652,51.44793],[14.94749,51.47155],[14.73219,51.52922],[14.72652,51.53902],[14.73047,51.54606],[14.71125,51.56209],[14.7727,51.61263],[14.75759,51.62318],[14.75392,51.67445],[14.69065,51.70842],[14.66386,51.73282],[14.64625,51.79472],[14.60493,51.80473],[14.59089,51.83302],[14.6588,51.88359],[14.6933,51.9044],[14.70601,51.92944],[14.7177,51.94048],[14.72163,51.95188],[14.71836,51.95606],[14.7139,51.95643],[14.70488,51.97679],[14.71339,52.00337],[14.76026,52.06624],[14.72971,52.09167],[14.6917,52.10283],[14.67683,52.13936],[14.70616,52.16927],[14.68344,52.19612],[14.71319,52.22144],[14.70139,52.25038],[14.58149,52.28007],[14.56378,52.33838],[14.55228,52.35264],[14.54423,52.42568],[14.63056,52.48993],[14.60081,52.53116],[14.6289,52.57136],[14.61073,52.59847],[14.22071,52.81175],[14.13806,52.82392],[14.12256,52.84311],[14.15873,52.87715],[14.14056,52.95786],[14.25954,53.00264],[14.35044,53.05829],[14.38679,53.13669],[14.36696,53.16444],[14.37853,53.20405],[14.40662,53.21098],[14.45125,53.26241],[14.44133,53.27427],[14.4215,53.27724],[14.35209,53.49506],[14.3273,53.50587],[14.30416,53.55499],[14.31904,53.61581],[14.2853,53.63392],[14.28477,53.65955],[14.27133,53.66613],[14.2836,53.67721],[14.26782,53.69866],[14.27249,53.74464],[14.21323,53.8664],[14.20823,53.90776],[14.18544,53.91258],[14.20647,53.91671],[14.22634,53.9291],[14.20811,54.12784],[13.93395,54.84044],[12.85844,54.82438],[11.90309,54.38543],[11.00303,54.63689],[10.31111,54.65968],[10.16755,54.73883],[9.89314,54.84171],[9.73563,54.8247],[9.61187,54.85548],[9.62734,54.88057],[9.58937,54.88785],[9.4659,54.83131],[9.43155,54.82586],[9.41213,54.84254],[9.38532,54.83968],[9.36496,54.81749],[9.33849,54.80233],[9.32771,54.80602],[9.2474,54.8112],[9.23445,54.83432],[9.24631,54.84726],[9.20571,54.85841],[9.14275,54.87421],[9.04629,54.87249],[8.92795,54.90452],[8.81178,54.90518],[8.76387,54.8948],[8.63979,54.91069],[8.55769,54.91837],[8.45719,55.06747],[8.02459,55.09613],[5.45168,54.20039],[6.91025,53.44221],[7.00198,53.32672],[7.19052,53.31866],[7.21679,53.20058],[7.22681,53.18165],[7.17898,53.13817],[7.21694,53.00742],[7.07253,52.81083],[7.04557,52.63318],[6.77307,52.65375],[6.71641,52.62905],[6.69507,52.488],[6.94293,52.43597],[6.99041,52.47235],[7.03417,52.40237],[7.07044,52.37805],[7.02703,52.27941],[7.06365,52.23789],[7.03729,52.22695],[6.9897,52.2271],[6.97189,52.20329],[6.83984,52.11728],[6.76117,52.11895],[6.68128,52.05052],[6.83035,51.9905],[6.82357,51.96711],[6.72319,51.89518],[6.68386,51.91861],[6.58556,51.89386],[6.50231,51.86313],[6.47179,51.85395],[6.38815,51.87257],[6.40704,51.82771],[6.30593,51.84998],[6.29872,51.86801],[6.21443,51.86801],[6.15349,51.90439],[6.11551,51.89769],[6.16902,51.84094],[6.10337,51.84829],[6.06705,51.86136],[5.99848,51.83195],[5.94568,51.82786],[5.98665,51.76944],[5.95003,51.7493],[6.04091,51.71821],[6.02767,51.6742],[6.11759,51.65609],[6.09055,51.60564],[6.18017,51.54096],[6.21724,51.48568],[6.20654,51.40049],[6.22641,51.39948],[6.22674,51.36135],[6.16977,51.33169],[6.07889,51.24432],[6.07889,51.17038],[6.17384,51.19589],[6.16706,51.15677],[5.98292,51.07469],[5.9541,51.03496],[5.9134,51.06736],[5.86735,51.05182],[5.87849,51.01969],[5.90493,51.00198],[5.90296,50.97356],[5.95282,50.98728],[6.02697,50.98303],[6.01615,50.93367],[6.09297,50.92066],[6.07486,50.89307],[6.08805,50.87223],[6.07693,50.86025],[6.07431,50.84674],[6.05702,50.85179],[6.05623,50.8572],[6.01921,50.84435],[6.02328,50.81694],[6.00462,50.80065],[5.98404,50.80988],[5.97497,50.79992],[6.02624,50.77453],[6.01976,50.75398],[6.03889,50.74618],[6.0326,50.72647],[6.0406,50.71848],[6.04428,50.72861],[6.11707,50.72231],[6.17852,50.6245],[6.26957,50.62444],[6.2476,50.60392],[6.24888,50.59869],[6.24005,50.58732],[6.22581,50.5907],[6.20281,50.56952],[6.17739,50.55875],[6.17802,50.54179],[6.19735,50.53576],[6.19579,50.5313],[6.18716,50.52653],[6.19193,50.5212],[6.20599,50.52089],[6.22335,50.49578],[6.26637,50.50272],[6.30809,50.50058],[6.3465,50.48833],[6.34005,50.46083],[6.37219,50.45397],[6.36852,50.40776],[6.34406,50.37994],[6.3688,50.35898],[6.40785,50.33557],[6.40641,50.32425],[6.35701,50.31139],[6.32488,50.32333],[6.29949,50.30887],[6.28797,50.27458],[6.208,50.25179],[6.16853,50.2234],[6.18364,50.20815],[6.18739,50.1822],[6.14588,50.17106],[6.14132,50.14971],[6.15298,50.14126],[6.1379,50.12964],[6.12055,50.09171],[6.11274,50.05916],[6.13458,50.04141],[6.13044,50.02929],[6.14666,50.02207],[6.13794,50.01466],[6.13273,50.02019],[6.1295,50.01849],[6.13806,50.01056],[6.14948,50.00908],[6.14147,49.99563],[6.1701,49.98518],[6.16466,49.97086],[6.17872,49.9537],[6.18554,49.95622],[6.18045,49.96611],[6.19089,49.96991],[6.19856,49.95053],[6.22094,49.94955],[6.22608,49.929],[6.21882,49.92403],[6.22926,49.92096],[6.23496,49.89972],[6.26146,49.88203],[6.28874,49.87592],[6.29692,49.86685],[6.30963,49.87021],[6.32303,49.85133],[6.32098,49.83728],[6.33585,49.83785],[6.34267,49.84974],[6.36576,49.85032],[6.40022,49.82029],[6.42521,49.81591],[6.42905,49.81091],[6.44131,49.81443],[6.45425,49.81164],[6.47111,49.82263],[6.48718,49.81267],[6.50647,49.80916],[6.51215,49.80124],[6.52121,49.81338],[6.53122,49.80666],[6.52169,49.79787],[6.50534,49.78952],[6.51669,49.78336],[6.51056,49.77515],[6.51828,49.76855],[6.51646,49.75961],[6.50174,49.75292],[6.50193,49.73291],[6.51805,49.72425],[6.51397,49.72058],[6.50261,49.72718],[6.49535,49.72645],[6.49694,49.72205],[6.5042,49.71808],[6.50647,49.71353],[6.49785,49.71118],[6.48014,49.69767],[6.46048,49.69092],[6.44654,49.67799],[6.42937,49.66857],[6.42726,49.66078],[6.43768,49.66021],[6.4413,49.65722],[6.41861,49.61723],[6.39822,49.60081],[6.385,49.59946],[6.37464,49.58886],[6.38342,49.5799],[6.38024,49.57593],[6.36676,49.57813],[6.35825,49.57053],[6.38228,49.55855],[6.38072,49.55171],[6.35666,49.52931],[6.36788,49.50377],[6.36907,49.48931],[6.36778,49.46937],[6.38352,49.46463],[6.39168,49.4667],[6.40274,49.46546],[6.42432,49.47683],[6.55404,49.42464],[6.533,49.40748],[6.60091,49.36864],[6.58807,49.35358],[6.572,49.35027],[6.60186,49.31055],[6.66583,49.28065],[6.69274,49.21661],[6.71843,49.2208],[6.73256,49.20486],[6.71137,49.18808],[6.73765,49.16375],[6.78265,49.16793],[6.83385,49.15162],[6.84703,49.15734],[6.86225,49.18185],[6.85016,49.19354],[6.85119,49.20038],[6.83555,49.21249],[6.85939,49.22376],[6.89298,49.20863],[6.91875,49.22261],[6.93831,49.2223],[6.94028,49.21641],[6.95963,49.203],[6.97273,49.2099],[7.01318,49.19018],[7.03459,49.19096],[7.0274,49.17042],[7.03178,49.15734],[7.04662,49.13724],[7.04409,49.12123],[7.04843,49.11422],[7.05548,49.11185],[7.06642,49.11415],[7.07162,49.1255],[7.09007,49.13094],[7.07859,49.15031],[7.10715,49.15631],[7.10384,49.13787],[7.12504,49.14253],[7.1358,49.1282],[7.1593,49.1204],[7.23473,49.12971],[7.29514,49.11426],[7.3195,49.14231],[7.35995,49.14399],[7.3662,49.17308],[7.44052,49.18354],[7.44455,49.16765],[7.49473,49.17],[7.49172,49.13915],[7.53012,49.09818],[7.56416,49.08136],[7.62575,49.07654],[7.63618,49.05428],[7.75948,49.04562],[7.79557,49.06583],[7.86386,49.03499],[7.93641,49.05544],[7.97783,49.03161],[8.14189,48.97833],[8.22604,48.97352],[8.20031,48.95856],[8.19989,48.95825],[8.12813,48.87985],[8.10253,48.81829],[8.06802,48.78957],[8.0326,48.79017],[8.01534,48.76085],[7.96994,48.75606],[7.96812,48.72491],[7.89002,48.66317],[7.84098,48.64217],[7.80057,48.5857],[7.80167,48.54758],[7.80647,48.51239],[7.76833,48.48945],[7.73109,48.39192],[7.74562,48.32736],[7.69022,48.30018],[7.6648,48.22219],[7.57137,48.12292],[7.56966,48.03265],[7.62302,47.97898],[7.55673,47.87371],[7.52921,47.77747],[7.54761,47.72912],[7.53722,47.71635],[7.51266,47.70197],[7.51915,47.68335],[7.52067,47.66437],[7.53384,47.65115],[7.5591,47.63849],[7.57423,47.61628],[7.58851,47.60794],[7.59301,47.60058],[7.58945,47.59017],[7.60523,47.58519],[7.60459,47.57869],[7.61929,47.57683],[7.64309,47.59151],[7.64213,47.5944],[7.64599,47.59695],[7.67395,47.59212],[7.68229,47.59905],[7.69385,47.60099],[7.68486,47.59601],[7.67115,47.5871],[7.68904,47.57133],[7.67655,47.56435],[7.63338,47.56256],[7.65083,47.54662],[7.66174,47.54554],[7.6656,47.53752],[7.68101,47.53232],[7.69642,47.53297],[7.71961,47.54219],[7.75261,47.54599],[7.79486,47.55691],[7.81901,47.58798],[7.84412,47.5841],[7.88664,47.58854],[7.90673,47.57674],[7.91251,47.55031],[7.94494,47.54511],[7.95682,47.55789],[7.97581,47.55493],[8.00113,47.55616],[8.02136,47.55096],[8.04383,47.55443],[8.06663,47.56374],[8.08557,47.55768],[8.10002,47.56504],[8.10395,47.57918],[8.11543,47.5841],[8.13662,47.58432],[8.13823,47.59147],[8.14947,47.59558],[8.1652,47.5945],[8.19378,47.61636],[8.20617,47.62141],[8.22011,47.6181],[8.22577,47.60385],[8.23809,47.61204],[8.25863,47.61571],[8.26313,47.6103],[8.2824,47.61225],[8.29722,47.60603],[8.29524,47.5919],[8.30277,47.58607],[8.32735,47.57133],[8.35512,47.57014],[8.38273,47.56608],[8.39477,47.57826],[8.43235,47.56617],[8.49431,47.58107],[8.48949,47.588],[8.46637,47.58389],[8.45578,47.60121],[8.50747,47.61897],[8.51686,47.63476],[8.55756,47.62394],[8.57586,47.59537],[8.60348,47.61204],[8.59545,47.64298],[8.60701,47.65271],[8.61471,47.64514],[8.60412,47.63735],[8.62049,47.63757],[8.62884,47.65098],[8.61113,47.66332],[8.6052,47.67258],[8.57683,47.66158],[8.56141,47.67088],[8.52801,47.66059],[8.5322,47.64687],[8.49656,47.64709],[8.46605,47.64103],[8.4667,47.65747],[8.44711,47.65379],[8.42264,47.66667],[8.41346,47.66676],[8.40473,47.67499],[8.4211,47.68407],[8.40569,47.69855],[8.44807,47.72426],[8.45771,47.7493],[8.48868,47.77215],[8.56814,47.78001],[8.56415,47.80633],[8.61657,47.79998],[8.62408,47.7626],[8.64425,47.76398],[8.65292,47.80066],[8.68022,47.78599],[8.68985,47.75686],[8.71778,47.76571],[8.74251,47.75168],[8.70543,47.73121],[8.73671,47.7169],[8.72617,47.69651]]]]}},{type:"Feature",properties:{iso1A2:"DG",iso1A3:"DGA",wikidata:"Q184851",nameEn:"Diego Garcia",country:"GB",groups:["IO","014","202","002"],isoStatus:"excRes",callingCodes:["246"]},geometry:{type:"MultiPolygon",coordinates:[[[[73.14823,-7.76302],[73.09982,-6.07324],[71.43792,-7.73904],[73.14823,-7.76302]]]]}},{type:"Feature",properties:{iso1A2:"DJ",iso1A3:"DJI",iso1N3:"262",wikidata:"Q977",nameEn:"Djibouti",groups:["014","202","002"],callingCodes:["253"]},geometry:{type:"MultiPolygon",coordinates:[[[[43.42425,11.70983],[43.90659,12.3823],[43.32909,12.59711],[43.29075,12.79154],[42.86195,12.58747],[42.7996,12.42629],[42.6957,12.36201],[42.46941,12.52661],[42.4037,12.46478],[41.95461,11.81157],[41.82878,11.72361],[41.77727,11.49902],[41.8096,11.33606],[41.80056,10.97127],[42.06302,10.92599],[42.13691,10.97586],[42.42669,10.98493],[42.62989,11.09711],[42.75111,11.06992],[42.79037,10.98493],[42.95776,10.98533],[43.42425,11.70983]]]]}},{type:"Feature",properties:{iso1A2:"DK",iso1A3:"DNK",iso1N3:"208",wikidata:"Q35",nameEn:"Denmark",groups:["EU","154","150"],callingCodes:["45"]},geometry:{type:"MultiPolygon",coordinates:[[[[12.16597,56.60205],[10.40861,58.38489],[7.28637,57.35913],[8.02459,55.09613],[8.45719,55.06747],[8.55769,54.91837],[8.63979,54.91069],[8.76387,54.8948],[8.81178,54.90518],[8.92795,54.90452],[9.04629,54.87249],[9.14275,54.87421],[9.20571,54.85841],[9.24631,54.84726],[9.23445,54.83432],[9.2474,54.8112],[9.32771,54.80602],[9.33849,54.80233],[9.36496,54.81749],[9.38532,54.83968],[9.41213,54.84254],[9.43155,54.82586],[9.4659,54.83131],[9.58937,54.88785],[9.62734,54.88057],[9.61187,54.85548],[9.73563,54.8247],[9.89314,54.84171],[10.16755,54.73883],[10.31111,54.65968],[11.00303,54.63689],[11.90309,54.38543],[12.85844,54.82438],[13.93395,54.84044],[15.36991,54.73263],[15.79951,55.54655],[14.89259,55.5623],[14.28399,55.1553],[12.84405,55.13257],[12.60345,55.42675],[12.88472,55.63369],[12.6372,55.91371],[12.65312,56.04345],[12.07466,56.29488],[12.16597,56.60205]]]]}},{type:"Feature",properties:{iso1A2:"DM",iso1A3:"DMA",iso1N3:"212",wikidata:"Q784",nameEn:"Dominica",groups:["029","003","419","019"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["1 767"]},geometry:{type:"MultiPolygon",coordinates:[[[[-61.51867,14.96709],[-60.69955,15.22234],[-60.95725,15.70997],[-61.44899,15.79571],[-61.81728,15.58058],[-61.51867,14.96709]]]]}},{type:"Feature",properties:{iso1A2:"DO",iso1A3:"DOM",iso1N3:"214",wikidata:"Q786",nameEn:"Dominican Republic",groups:["029","003","419","019"],callingCodes:["1 809","1 829","1 849"]},geometry:{type:"MultiPolygon",coordinates:[[[[-67.87844,21.7938],[-72.38946,20.27111],[-71.77419,19.73128],[-71.75865,19.70231],[-71.7429,19.58445],[-71.71449,19.55364],[-71.71268,19.53374],[-71.6802,19.45008],[-71.69448,19.37866],[-71.77766,19.33823],[-71.73229,19.26686],[-71.62642,19.21212],[-71.65337,19.11759],[-71.69938,19.10916],[-71.71088,19.08353],[-71.74088,19.0437],[-71.88102,18.95007],[-71.77766,18.95007],[-71.72624,18.87802],[-71.71885,18.78423],[-71.82556,18.62551],[-71.95412,18.64939],[-72.00201,18.62312],[-71.88102,18.50125],[-71.90875,18.45821],[-71.69952,18.34101],[-71.78271,18.18302],[-71.75465,18.14405],[-71.74994,18.11115],[-71.73783,18.07177],[-71.75671,18.03456],[-72.29523,17.48026],[-68.39466,16.14167],[-67.87844,21.7938]]]]}},{type:"Feature",properties:{iso1A2:"DZ",iso1A3:"DZA",iso1N3:"012",wikidata:"Q262",nameEn:"Algeria",groups:["015","002"],callingCodes:["213"]},geometry:{type:"MultiPolygon",coordinates:[[[[8.59123,37.14286],[2.46645,37.97429],[-2.27707,35.35051],[-2.21248,35.08532],[-2.21445,35.04378],[-2.04734,34.93218],[-1.97833,34.93218],[-1.97469,34.886],[-1.73707,34.74226],[-1.84569,34.61907],[-1.69788,34.48056],[-1.78042,34.39018],[-1.64666,34.10405],[-1.73494,33.71721],[-1.59508,33.59929],[-1.67067,33.27084],[-1.46249,33.0499],[-1.54244,32.95499],[-1.37794,32.73628],[-0.9912,32.52467],[-1.24998,32.32993],[-1.24453,32.1917],[-1.15735,32.12096],[-1.22829,32.07832],[-2.46166,32.16603],[-2.93873,32.06557],[-2.82784,31.79459],[-3.66314,31.6339],[-3.66386,31.39202],[-3.77647,31.31912],[-3.77103,31.14984],[-3.54944,31.0503],[-3.65418,30.85566],[-3.64735,30.67539],[-4.31774,30.53229],[-4.6058,30.28343],[-5.21671,29.95253],[-5.58831,29.48103],[-5.72121,29.52322],[-5.75616,29.61407],[-6.69965,29.51623],[-6.78351,29.44634],[-6.95824,29.50924],[-7.61585,29.36252],[-8.6715,28.71194],[-8.66879,27.6666],[-8.66674,27.31569],[-4.83423,24.99935],[1.15698,21.12843],[1.20992,20.73533],[3.24648,19.81703],[3.12501,19.1366],[3.36082,18.9745],[4.26651,19.14224],[5.8153,19.45101],[7.38361,20.79165],[7.48273,20.87258],[11.96886,23.51735],[11.62498,24.26669],[11.41061,24.21456],[10.85323,24.5595],[10.33159,24.5465],[10.02432,24.98124],[10.03146,25.35635],[9.38834,26.19288],[9.51696,26.39148],[9.89569,26.57696],[9.78136,29.40961],[9.3876,30.16738],[9.55544,30.23971],[9.07483,32.07865],[8.35999,32.50101],[8.31895,32.83483],[8.1179,33.05086],[8.11433,33.10175],[7.83028,33.18851],[7.73687,33.42114],[7.54088,33.7726],[7.52851,34.06493],[7.66174,34.20167],[7.74207,34.16492],[7.81242,34.21841],[7.86264,34.3987],[8.20482,34.57575],[8.29655,34.72798],[8.25189,34.92009],[8.30727,34.95378],[8.3555,35.10007],[8.47318,35.23376],[8.30329,35.29884],[8.36086,35.47774],[8.35371,35.66373],[8.26472,35.73669],[8.2626,35.91733],[8.40731,36.42208],[8.18936,36.44939],[8.16167,36.48817],[8.47609,36.66607],[8.46537,36.7706],[8.57613,36.78062],[8.67706,36.8364],[8.62972,36.86499],[8.64044,36.9401],[8.59123,37.14286]]]]}},{type:"Feature",properties:{iso1A2:"EA",wikidata:"Q28868874",nameEn:"Ceuta, Melilla",country:"ES",groups:["015","002"],isoStatus:"excRes",callingCodes:["34"]},geometry:{type:"MultiPolygon",coordinates:[[[[-5.38491,35.92591],[-5.37338,35.88417],[-5.35844,35.87375],[-5.34379,35.8711],[-5.27056,35.88794],[-5.27635,35.91222],[-5.38491,35.92591]]],[[[-2.92224,35.3401],[-2.96038,35.31609],[-2.96648,35.30475],[-2.96978,35.29459],[-2.97035,35.28852],[-2.96507,35.28801],[-2.96826,35.28296],[-2.96516,35.27967],[-2.95431,35.2728],[-2.95065,35.26576],[-2.93893,35.26737],[-2.92674,35.27313],[-2.92181,35.28599],[-2.92224,35.3401]]]]}},{type:"Feature",properties:{iso1A2:"EC",iso1A3:"ECU",iso1N3:"218",wikidata:"Q736",nameEn:"Ecuador",groups:["005","419","019"],callingCodes:["593"]},geometry:{type:"MultiPolygon",coordinates:[[[[-75.25764,-0.11943],[-75.82927,0.09578],[-76.23441,0.42294],[-76.41215,0.38228],[-76.4094,0.24015],[-76.89177,0.24736],[-77.52001,0.40782],[-77.49984,0.64476],[-77.67815,0.73863],[-77.66416,0.81604],[-77.68613,0.83029],[-77.7148,0.85003],[-77.85677,0.80197],[-78.42749,1.15389],[-78.87137,1.47457],[-93.12365,2.64343],[-92.46744,-2.52874],[-80.30602,-3.39149],[-80.20647,-3.431],[-80.24123,-3.46124],[-80.24475,-3.47846],[-80.24586,-3.48677],[-80.23651,-3.48652],[-80.22629,-3.501],[-80.20535,-3.51667],[-80.21642,-3.5888],[-80.19848,-3.59249],[-80.18741,-3.63994],[-80.19926,-3.68894],[-80.13232,-3.90317],[-80.46386,-4.01342],[-80.4822,-4.05477],[-80.45023,-4.20938],[-80.32114,-4.21323],[-80.46386,-4.41516],[-80.39256,-4.48269],[-80.13945,-4.29786],[-79.79722,-4.47558],[-79.59402,-4.46848],[-79.26248,-4.95167],[-79.1162,-4.97774],[-79.01659,-5.01481],[-78.85149,-4.66795],[-78.68394,-4.60754],[-78.34362,-3.38633],[-78.24589,-3.39907],[-78.22642,-3.51113],[-78.14324,-3.47653],[-78.19369,-3.36431],[-77.94147,-3.05454],[-76.6324,-2.58397],[-76.05203,-2.12179],[-75.57429,-1.55961],[-75.3872,-0.9374],[-75.22862,-0.95588],[-75.22862,-0.60048],[-75.53615,-0.19213],[-75.60169,-0.18708],[-75.61997,-0.10012],[-75.40192,-0.17196],[-75.25764,-0.11943]]]]}},{type:"Feature",properties:{iso1A2:"EE",iso1A3:"EST",iso1N3:"233",wikidata:"Q191",nameEn:"Estonia",aliases:["EW"],groups:["EU","154","150"],callingCodes:["372"]},geometry:{type:"MultiPolygon",coordinates:[[[[26.32936,60.00121],[20.5104,59.15546],[19.84909,57.57876],[22.80496,57.87798],[23.20055,57.56697],[24.26221,57.91787],[24.3579,57.87471],[25.19484,58.0831],[25.28237,57.98539],[25.29581,58.08288],[25.73499,57.90193],[26.05949,57.84744],[26.0324,57.79037],[26.02456,57.78342],[26.027,57.78158],[26.0266,57.77441],[26.02069,57.77169],[26.02415,57.76865],[26.03332,57.7718],[26.0543,57.76105],[26.08098,57.76619],[26.2029,57.7206],[26.1866,57.6849],[26.29253,57.59244],[26.46527,57.56885],[26.54675,57.51813],[26.90364,57.62823],[27.34698,57.52242],[27.31919,57.57672],[27.40393,57.62125],[27.3746,57.66834],[27.52615,57.72843],[27.50171,57.78842],[27.56689,57.83356],[27.78526,57.83963],[27.81841,57.89244],[27.67282,57.92627],[27.62393,58.09462],[27.48541,58.22615],[27.55489,58.39525],[27.36366,58.78381],[27.74429,58.98351],[27.80482,59.1116],[27.87978,59.18097],[27.90911,59.24353],[28.00689,59.28351],[28.14215,59.28934],[28.19284,59.35791],[28.20537,59.36491],[28.21137,59.38058],[28.19061,59.39962],[28.04187,59.47017],[27.85643,59.58538],[26.90044,59.63819],[26.32936,60.00121]]]]}},{type:"Feature",properties:{iso1A2:"EG",iso1A3:"EGY",iso1N3:"818",wikidata:"Q79",nameEn:"Egypt",groups:["015","002"],callingCodes:["20"]},geometry:{type:"MultiPolygon",coordinates:[[[[33.62659,31.82938],[25.63787,31.9359],[25.14001,31.67534],[25.06041,31.57937],[24.83101,31.31921],[25.01077,30.73861],[24.71117,30.17441],[24.99968,29.24574],[24.99885,21.99535],[33.17563,22.00405],[34.0765,22.00501],[37.8565,22.00903],[34.51305,27.70027],[34.46254,27.99552],[34.88293,29.37455],[34.92298,29.45305],[34.26742,31.21998],[34.24012,31.29591],[34.23572,31.2966],[34.21853,31.32363],[34.052,31.46619],[33.62659,31.82938]]]]}},{type:"Feature",properties:{iso1A2:"EH",iso1A3:"ESH",iso1N3:"732",wikidata:"Q6250",nameEn:"Western Sahara",groups:["015","002"],callingCodes:["212"]},geometry:{type:"MultiPolygon",coordinates:[[[[-8.66879,27.6666],[-8.77527,27.66663],[-8.71787,26.9898],[-9.08698,26.98639],[-9.56957,26.90042],[-9.81998,26.71379],[-10.68417,26.90984],[-11.35695,26.8505],[-11.23622,26.72023],[-11.38635,26.611],[-11.62052,26.05229],[-12.06001,26.04442],[-12.12281,25.13682],[-12.92147,24.39502],[-13.00628,24.01923],[-13.75627,23.77231],[-14.10361,22.75501],[-14.1291,22.41636],[-14.48112,22.00886],[-14.47329,21.63839],[-14.78487,21.36587],[-16.44269,21.39745],[-16.9978,21.36239],[-17.02707,21.34022],[-17.21511,21.34226],[-17.35589,20.80492],[-17.0471,20.76408],[-17.0695,20.85742],[-17.06781,20.92697],[-17.0396,20.9961],[-17.0357,21.05368],[-16.99806,21.12142],[-16.95474,21.33997],[-13.01525,21.33343],[-13.08438,22.53866],[-13.15313,22.75649],[-13.10753,22.89493],[-13.00412,23.02297],[-12.5741,23.28975],[-12.36213,23.3187],[-12.14969,23.41935],[-12.00251,23.4538],[-12.0002,25.9986],[-8.66721,25.99918],[-8.66674,27.31569],[-8.66879,27.6666]]]]}},{type:"Feature",properties:{iso1A2:"ER",iso1A3:"ERI",iso1N3:"232",wikidata:"Q986",nameEn:"Eritrea",groups:["014","202","002"],callingCodes:["291"]},geometry:{type:"MultiPolygon",coordinates:[[[[41.37609,16.19728],[39.63762,18.37348],[38.57727,17.98125],[38.45916,17.87167],[38.37133,17.66269],[38.13362,17.53906],[37.50967,17.32199],[37.42694,17.04041],[36.99777,17.07172],[36.92193,16.23451],[36.76371,15.80831],[36.69761,15.75323],[36.54276,15.23478],[36.44337,15.14963],[36.54376,14.25597],[36.56536,14.26177],[36.55659,14.28237],[36.63364,14.31172],[36.85787,14.32201],[37.01622,14.2561],[37.09486,14.27155],[37.13206,14.40746],[37.3106,14.44657],[37.47319,14.2149],[37.528,14.18413],[37.91287,14.89447],[38.0364,14.72745],[38.25562,14.67287],[38.3533,14.51323],[38.45748,14.41445],[38.78306,14.4754],[38.98058,14.54895],[39.02834,14.63717],[39.16074,14.65187],[39.14772,14.61827],[39.19547,14.56996],[39.23888,14.56365],[39.26927,14.48801],[39.2302,14.44598],[39.2519,14.40393],[39.37685,14.54402],[39.52756,14.49011],[39.50585,14.55735],[39.58182,14.60987],[39.76632,14.54264],[39.9443,14.41024],[40.07236,14.54264],[40.14649,14.53969],[40.21128,14.39342],[40.25686,14.41445],[40.9167,14.11152],[41.25097,13.60787],[41.62864,13.38626],[42.05841,12.80912],[42.21469,12.75832],[42.2798,12.6355],[42.4037,12.46478],[42.46941,12.52661],[42.6957,12.36201],[42.7996,12.42629],[42.86195,12.58747],[43.29075,12.79154],[42.63806,13.58268],[41.29956,15.565],[41.37609,16.19728]]]]}},{type:"Feature",properties:{iso1A2:"ES",iso1A3:"ESP",iso1N3:"724",wikidata:"Q29",nameEn:"Spain",groups:["EU","039","150"],callingCodes:["34"]},geometry:{type:"MultiPolygon",coordinates:[[[[-2.41312,35.17111],[-2.41265,35.1877],[-2.44896,35.18777],[-2.44887,35.17075],[-2.41312,35.17111]]],[[[-3.90602,35.21494],[-3.88926,35.20841],[-3.88617,35.21406],[-3.90288,35.22024],[-3.90602,35.21494]]],[[[-4.30191,35.17419],[-4.30112,35.17058],[-4.29436,35.17149],[-4.30191,35.17419]]],[[[-7.27694,35.93599],[-5.64962,35.93752],[-5.10878,36.05227],[-2.85819,35.63219],[-2.27707,35.35051],[2.46645,37.97429],[5.18061,39.43581],[3.4481,42.4358],[3.17156,42.43545],[3.11379,42.43646],[3.10027,42.42621],[3.08167,42.42748],[3.03734,42.47363],[2.96518,42.46692],[2.94283,42.48174],[2.92107,42.4573],[2.88413,42.45938],[2.86983,42.46843],[2.85675,42.45444],[2.84335,42.45724],[2.77464,42.41046],[2.75497,42.42578],[2.72056,42.42298],[2.65311,42.38771],[2.6747,42.33974],[2.57934,42.35808],[2.55516,42.35351],[2.54382,42.33406],[2.48457,42.33933],[2.43508,42.37568],[2.43299,42.39423],[2.38504,42.39977],[2.25551,42.43757],[2.20578,42.41633],[2.16599,42.42314],[2.12789,42.41291],[2.11621,42.38393],[2.06241,42.35906],[2.00488,42.35399],[1.96482,42.37787],[1.9574,42.42401],[1.94084,42.43039],[1.94061,42.43333],[1.94292,42.44316],[1.93663,42.45439],[1.88853,42.4501],[1.83037,42.48395],[1.76335,42.48863],[1.72515,42.50338],[1.70571,42.48867],[1.66826,42.50779],[1.65674,42.47125],[1.58933,42.46275],[1.57953,42.44957],[1.55937,42.45808],[1.55073,42.43299],[1.5127,42.42959],[1.44529,42.43724],[1.43838,42.47848],[1.41648,42.48315],[1.46661,42.50949],[1.44759,42.54431],[1.41245,42.53539],[1.4234,42.55959],[1.44529,42.56722],[1.42512,42.58292],[1.44197,42.60217],[1.35562,42.71944],[1.15928,42.71407],[1.0804,42.78569],[0.98292,42.78754],[0.96166,42.80629],[0.93089,42.79154],[0.711,42.86372],[0.66121,42.84021],[0.65421,42.75872],[0.67873,42.69458],[0.40214,42.69779],[0.36251,42.72282],[0.29407,42.67431],[0.25336,42.7174],[0.17569,42.73424],[-0.02468,42.68513],[-0.10519,42.72761],[-0.16141,42.79535],[-0.17939,42.78974],[-0.3122,42.84788],[-0.38833,42.80132],[-0.41319,42.80776],[-0.44334,42.79939],[-0.50863,42.82713],[-0.55497,42.77846],[-0.67637,42.88303],[-0.69837,42.87945],[-0.72608,42.89318],[-0.73422,42.91228],[-0.72037,42.92541],[-0.75478,42.96916],[-0.81652,42.95166],[-0.97133,42.96239],[-1.00963,42.99279],[-1.10333,43.0059],[-1.22881,43.05534],[-1.25244,43.04164],[-1.30531,43.06859],[-1.30052,43.09581],[-1.27118,43.11961],[-1.32209,43.1127],[-1.34419,43.09665],[-1.35272,43.02658],[-1.44067,43.047],[-1.47555,43.08372],[-1.41562,43.12815],[-1.3758,43.24511],[-1.40942,43.27272],[-1.45289,43.27049],[-1.50992,43.29481],[-1.55963,43.28828],[-1.57674,43.25269],[-1.61341,43.25269],[-1.63052,43.28591],[-1.62481,43.30726],[-1.69407,43.31378],[-1.73074,43.29481],[-1.7397,43.32979],[-1.75079,43.3317],[-1.75334,43.34107],[-1.77068,43.34396],[-1.78714,43.35476],[-1.78332,43.36399],[-1.79319,43.37497],[-1.77289,43.38957],[-1.81005,43.59738],[-10.14298,44.17365],[-9.14112,41.86623],[-8.87157,41.86488],[-8.81794,41.90375],[-8.75712,41.92833],[-8.74606,41.9469],[-8.7478,41.96282],[-8.69071,41.98862],[-8.6681,41.99703],[-8.65832,42.02972],[-8.64626,42.03668],[-8.63791,42.04691],[-8.59493,42.05708],[-8.58086,42.05147],[-8.54563,42.0537],[-8.5252,42.06264],[-8.52837,42.07658],[-8.48185,42.0811],[-8.44123,42.08218],[-8.42512,42.07199],[-8.40143,42.08052],[-8.38323,42.07683],[-8.36353,42.09065],[-8.33912,42.08358],[-8.32161,42.10218],[-8.29809,42.106],[-8.2732,42.12396],[-8.24681,42.13993],[-8.22406,42.1328],[-8.1986,42.15402],[-8.18947,42.13853],[-8.19406,42.12141],[-8.18178,42.06436],[-8.11729,42.08537],[-8.08847,42.05767],[-8.08796,42.01398],[-8.16232,41.9828],[-8.2185,41.91237],[-8.19551,41.87459],[-8.16944,41.87944],[-8.16455,41.81753],[-8.0961,41.81024],[-8.01136,41.83453],[-7.9804,41.87337],[-7.92336,41.8758],[-7.90707,41.92432],[-7.88751,41.92553],[-7.88055,41.84571],[-7.84188,41.88065],[-7.69848,41.90977],[-7.65774,41.88308],[-7.58603,41.87944],[-7.62188,41.83089],[-7.52737,41.83939],[-7.49803,41.87095],[-7.45566,41.86488],[-7.44759,41.84451],[-7.42854,41.83262],[-7.42864,41.80589],[-7.37092,41.85031],[-7.32366,41.8406],[-7.18677,41.88793],[-7.18549,41.97515],[-7.14115,41.98855],[-7.08574,41.97401],[-7.07596,41.94977],[-7.01078,41.94977],[-6.98144,41.9728],[-6.95537,41.96553],[-6.94396,41.94403],[-6.82174,41.94493],[-6.81196,41.99097],[-6.76959,41.98734],[-6.75004,41.94129],[-6.61967,41.94008],[-6.58544,41.96674],[-6.5447,41.94371],[-6.56752,41.88429],[-6.51374,41.8758],[-6.56426,41.74219],[-6.54633,41.68623],[-6.49907,41.65823],[-6.44204,41.68258],[-6.29863,41.66432],[-6.19128,41.57638],[-6.26777,41.48796],[-6.3306,41.37677],[-6.38553,41.38655],[-6.38551,41.35274],[-6.55937,41.24417],[-6.65046,41.24725],[-6.68286,41.21641],[-6.69711,41.1858],[-6.77319,41.13049],[-6.75655,41.10187],[-6.79241,41.05397],[-6.80942,41.03629],[-6.84781,41.02692],[-6.88843,41.03027],[-6.913,41.03922],[-6.9357,41.02888],[-6.8527,40.93958],[-6.84292,40.89771],[-6.80707,40.88047],[-6.79892,40.84842],[-6.82337,40.84472],[-6.82826,40.74603],[-6.79567,40.65955],[-6.84292,40.56801],[-6.80218,40.55067],[-6.7973,40.51723],[-6.84944,40.46394],[-6.84618,40.42177],[-6.78426,40.36468],[-6.80218,40.33239],[-6.86085,40.2976],[-6.86085,40.26776],[-7.00426,40.23169],[-7.02544,40.18564],[-7.00589,40.12087],[-6.94233,40.10716],[-6.86737,40.01986],[-6.91463,39.86618],[-6.97492,39.81488],[-7.01613,39.66877],[-7.24707,39.66576],[-7.33507,39.64569],[-7.54121,39.66717],[-7.49477,39.58794],[-7.2927,39.45847],[-7.3149,39.34857],[-7.23403,39.27579],[-7.23566,39.20132],[-7.12811,39.17101],[-7.14929,39.11287],[-7.10692,39.10275],[-7.04011,39.11919],[-6.97004,39.07619],[-6.95211,39.0243],[-7.051,38.907],[-7.03848,38.87221],[-7.26174,38.72107],[-7.265,38.61674],[-7.32529,38.44336],[-7.15581,38.27597],[-7.09389,38.17227],[-6.93418,38.21454],[-7.00375,38.01914],[-7.05966,38.01966],[-7.10366,38.04404],[-7.12648,38.00296],[-7.24544,37.98884],[-7.27314,37.90145],[-7.33441,37.81193],[-7.41981,37.75729],[-7.51759,37.56119],[-7.46878,37.47127],[-7.43974,37.38913],[-7.43227,37.25152],[-7.41854,37.23813],[-7.41133,37.20314],[-7.39769,37.16868],[-7.37282,36.96896],[-7.27694,35.93599]],[[-5.28217,36.09907],[-5.3004,36.07439],[-5.32837,36.05935],[-5.36503,36.06205],[-5.39074,36.10278],[-5.40134,36.14896],[-5.38545,36.15481],[-5.36494,36.15496],[-5.34536,36.15501],[-5.33822,36.15272],[-5.27801,36.14942],[-5.28217,36.09907]]],[[[1.99838,42.44682],[2.01564,42.45171],[1.99216,42.46208],[1.98579,42.47486],[1.99766,42.4858],[1.98916,42.49351],[1.98022,42.49569],[1.97697,42.48568],[1.97227,42.48487],[1.97003,42.48081],[1.96215,42.47854],[1.95606,42.45785],[1.96125,42.45364],[1.98378,42.44697],[1.99838,42.44682]]]]}},{type:"Feature",properties:{iso1A2:"ET",iso1A3:"ETH",iso1N3:"231",wikidata:"Q115",nameEn:"Ethiopia",groups:["014","202","002"],callingCodes:["251"]},geometry:{type:"MultiPolygon",coordinates:[[[[42.4037,12.46478],[42.2798,12.6355],[42.21469,12.75832],[42.05841,12.80912],[41.62864,13.38626],[41.25097,13.60787],[40.9167,14.11152],[40.25686,14.41445],[40.21128,14.39342],[40.14649,14.53969],[40.07236,14.54264],[39.9443,14.41024],[39.76632,14.54264],[39.58182,14.60987],[39.50585,14.55735],[39.52756,14.49011],[39.37685,14.54402],[39.2519,14.40393],[39.2302,14.44598],[39.26927,14.48801],[39.23888,14.56365],[39.19547,14.56996],[39.14772,14.61827],[39.16074,14.65187],[39.02834,14.63717],[38.98058,14.54895],[38.78306,14.4754],[38.45748,14.41445],[38.3533,14.51323],[38.25562,14.67287],[38.0364,14.72745],[37.91287,14.89447],[37.528,14.18413],[37.47319,14.2149],[37.3106,14.44657],[37.13206,14.40746],[37.09486,14.27155],[37.01622,14.2561],[36.85787,14.32201],[36.63364,14.31172],[36.55659,14.28237],[36.56536,14.26177],[36.54376,14.25597],[36.44653,13.95666],[36.48824,13.83954],[36.38993,13.56459],[36.24545,13.36759],[36.13374,12.92665],[36.16651,12.88019],[36.14268,12.70879],[36.01458,12.72478],[35.70476,12.67101],[35.24302,11.91132],[35.11492,11.85156],[35.05832,11.71158],[35.09556,11.56278],[34.95704,11.24448],[35.01215,11.19626],[34.93172,10.95946],[34.97789,10.91559],[34.97491,10.86147],[34.86916,10.78832],[34.86618,10.74588],[34.77532,10.69027],[34.77383,10.74588],[34.59062,10.89072],[34.4372,10.781],[34.2823,10.53508],[34.34783,10.23914],[34.32102,10.11599],[34.22718,10.02506],[34.20484,9.9033],[34.13186,9.7492],[34.08717,9.55243],[34.10229,9.50238],[34.14304,9.04654],[34.14453,8.60204],[34.01346,8.50041],[33.89579,8.4842],[33.87195,8.41938],[33.71407,8.3678],[33.66938,8.44442],[33.54575,8.47094],[33.3119,8.45474],[33.19721,8.40317],[33.1853,8.29264],[33.18083,8.13047],[33.08401,8.05822],[33.0006,7.90333],[33.04944,7.78989],[33.24637,7.77939],[33.32531,7.71297],[33.44745,7.7543],[33.71407,7.65983],[33.87642,7.5491],[34.02984,7.36449],[34.03878,7.27437],[34.01495,7.25664],[34.19369,7.12807],[34.19369,7.04382],[34.35753,6.91963],[34.47669,6.91076],[34.53925,6.82794],[34.53776,6.74808],[34.65096,6.72589],[34.77459,6.5957],[34.87736,6.60161],[35.01738,6.46991],[34.96227,6.26415],[35.00546,5.89387],[35.12611,5.68937],[35.13058,5.62118],[35.31188,5.50106],[35.29938,5.34042],[35.50792,5.42431],[35.8576,5.33413],[35.81968,5.10757],[35.82118,4.77382],[35.9419,4.61933],[35.95449,4.53244],[36.03924,4.44406],[36.84474,4.44518],[37.07724,4.33503],[38.14168,3.62487],[38.45812,3.60445],[38.52336,3.62551],[38.91938,3.51198],[39.07736,3.5267],[39.19954,3.47834],[39.49444,3.45521],[39.51551,3.40895],[39.55132,3.39634],[39.58339,3.47434],[39.76808,3.67058],[39.86043,3.86974],[40.77498,4.27683],[41.1754,3.94079],[41.89488,3.97375],[42.07619,4.17667],[42.55853,4.20518],[42.84526,4.28357],[42.97746,4.44032],[43.04177,4.57923],[43.40263,4.79289],[44.02436,4.9451],[44.98104,4.91821],[47.97917,8.00124],[47.92477,8.00111],[46.99339,7.9989],[44.19222,8.93028],[43.32613,9.59205],[43.23518,9.84605],[43.0937,9.90579],[42.87643,10.18441],[42.69452,10.62672],[42.95776,10.98533],[42.79037,10.98493],[42.75111,11.06992],[42.62989,11.09711],[42.42669,10.98493],[42.13691,10.97586],[42.06302,10.92599],[41.80056,10.97127],[41.8096,11.33606],[41.77727,11.49902],[41.82878,11.72361],[41.95461,11.81157],[42.4037,12.46478]]]]}},{type:"Feature",properties:{iso1A2:"EU",iso1A3:"EUE",wikidata:"Q458",nameEn:"European Union",level:"union",isoStatus:"excRes"},geometry:null},{type:"Feature",properties:{iso1A2:"FI",iso1A3:"FIN",iso1N3:"246",wikidata:"Q33",nameEn:"Finland",aliases:["SF"],groups:["EU","154","150"],callingCodes:["358"]},geometry:{type:"MultiPolygon",coordinates:[[[[29.12697,69.69193],[28.36883,69.81658],[28.32849,69.88605],[27.97558,69.99671],[27.95542,70.0965],[27.57226,70.06215],[27.05802,69.92069],[26.64461,69.96565],[26.40261,69.91377],[25.96904,69.68397],[25.69679,69.27039],[25.75729,68.99383],[25.61613,68.89602],[25.42455,68.90328],[25.12206,68.78684],[25.10189,68.63307],[24.93048,68.61102],[24.90023,68.55579],[24.74898,68.65143],[24.18432,68.73936],[24.02299,68.81601],[23.781,68.84514],[23.68017,68.70276],[23.13064,68.64684],[22.53321,68.74393],[22.38367,68.71561],[22.27276,68.89514],[21.63833,69.27485],[21.27827,69.31281],[21.00732,69.22755],[20.98641,69.18809],[21.11099,69.10291],[21.05775,69.0356],[20.72171,69.11874],[20.55258,69.06069],[20.78802,69.03087],[20.91658,68.96764],[20.85104,68.93142],[20.90649,68.89696],[21.03001,68.88969],[22.00429,68.50692],[22.73028,68.40881],[23.10336,68.26551],[23.15377,68.14759],[23.26469,68.15134],[23.40081,68.05545],[23.65793,67.9497],[23.45627,67.85297],[23.54701,67.59306],[23.39577,67.46974],[23.75372,67.43688],[23.75372,67.29914],[23.54701,67.25435],[23.58735,67.20752],[23.56214,67.17038],[23.98563,66.84149],[23.98059,66.79585],[23.89488,66.772],[23.85959,66.56434],[23.63776,66.43568],[23.67591,66.3862],[23.64982,66.30603],[23.71339,66.21299],[23.90497,66.15802],[24.15791,65.85385],[24.14798,65.83466],[24.15107,65.81427],[24.14112,65.39731],[20.15877,63.06556],[19.23413,60.61414],[20.96741,60.71528],[21.15143,60.54555],[21.08159,60.20167],[21.02509,60.12142],[21.35468,59.67511],[20.5104,59.15546],[26.32936,60.00121],[27.44953,60.22766],[27.71177,60.3893],[27.77352,60.52722],[28.47974,60.93365],[28.82816,61.1233],[29.01829,61.17448],[31.10136,62.43042],[31.38369,62.66284],[31.58535,62.91642],[31.29294,63.09035],[31.23244,63.22239],[30.49637,63.46666],[29.98213,63.75795],[30.25437,63.83364],[30.55687,64.09036],[30.4762,64.25728],[30.06279,64.35782],[30.01238,64.57513],[30.12329,64.64862],[30.05271,64.79072],[29.68972,64.80789],[29.61914,65.05993],[29.84096,65.1109],[29.8813,65.22101],[29.61914,65.23791],[29.68972,65.31803],[29.84096,65.56945],[29.74013,65.64025],[29.97205,65.70256],[30.16363,65.66935],[29.91155,66.13863],[28.9839,66.94139],[29.91155,67.51507],[30.02041,67.67523],[29.66955,67.79872],[29.34179,68.06655],[28.62982,68.19816],[28.43941,68.53366],[28.78224,68.86696],[28.45957,68.91417],[28.91738,69.04774],[28.81248,69.11997],[28.8629,69.22395],[29.31664,69.47994],[29.12697,69.69193]]]]}},{type:"Feature",properties:{iso1A2:"FJ",iso1A3:"FJI",iso1N3:"242",wikidata:"Q712",nameEn:"Fiji",groups:["054","009"],driveSide:"left",callingCodes:["679"]},geometry:{type:"MultiPolygon",coordinates:[[[[174,-22.5],[179.99999,-22.5],[179.99999,-11.5],[174,-11.5],[174,-22.5]]],[[[-178.60161,-14.95666],[-180,-14.96041],[-180,-22.90585],[-176.74538,-22.89767],[-176.76826,-14.95183],[-178.60161,-14.95666]]]]}},{type:"Feature",properties:{iso1A2:"FK",iso1A3:"FLK",iso1N3:"238",wikidata:"Q9648",nameEn:"Falkland Islands",country:"GB",groups:["005","419","019"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["500"]},geometry:{type:"MultiPolygon",coordinates:[[[[-63.67376,-55.11859],[-54.56126,-51.26248],[-61.26735,-50.63919],[-63.67376,-55.11859]]]]}},{type:"Feature",properties:{iso1A2:"FM",iso1A3:"FSM",iso1N3:"583",wikidata:"Q702",nameEn:"Federated States of Micronesia",groups:["057","009"],roadSpeedUnit:"mph",callingCodes:["691"]},geometry:{type:"MultiPolygon",coordinates:[[[[136.04605,12.45908],[136.27107,6.73747],[156.88247,-1.39237],[165.35175,6.367],[159.04653,10.59067],[136.04605,12.45908]]]]}},{type:"Feature",properties:{iso1A2:"FO",iso1A3:"FRO",iso1N3:"234",wikidata:"Q4628",nameEn:"Faroe Islands",country:"DK",groups:["154","150"],callingCodes:["298"]},geometry:{type:"MultiPolygon",coordinates:[[[[-8.51774,62.35338],[-6.51083,60.95272],[-5.70102,62.77194],[-8.51774,62.35338]]]]}},{type:"Feature",properties:{iso1A2:"FR",iso1A3:"FRA",iso1N3:"250",wikidata:"Q142",nameEn:"France",groups:["EU","155","150"],callingCodes:["33"]},geometry:null},{type:"Feature",properties:{iso1A2:"FX",iso1A3:"FXX",iso1N3:"249",wikidata:"Q212429",nameEn:"Metropolitan France",country:"FR",groups:["EU","155","150"],isoStatus:"excRes",callingCodes:["33"]},geometry:{type:"MultiPolygon",coordinates:[[[[2.55904,51.07014],[2.18458,51.52087],[1.17405,50.74239],[-2.02963,49.91866],[-2.09454,49.46288],[-1.83944,49.23037],[-2.00491,48.86706],[-2.65349,49.15373],[-6.13339,48.73907],[-1.81005,43.59738],[-1.77289,43.38957],[-1.79319,43.37497],[-1.78332,43.36399],[-1.78714,43.35476],[-1.77068,43.34396],[-1.75334,43.34107],[-1.75079,43.3317],[-1.7397,43.32979],[-1.73074,43.29481],[-1.69407,43.31378],[-1.62481,43.30726],[-1.63052,43.28591],[-1.61341,43.25269],[-1.57674,43.25269],[-1.55963,43.28828],[-1.50992,43.29481],[-1.45289,43.27049],[-1.40942,43.27272],[-1.3758,43.24511],[-1.41562,43.12815],[-1.47555,43.08372],[-1.44067,43.047],[-1.35272,43.02658],[-1.34419,43.09665],[-1.32209,43.1127],[-1.27118,43.11961],[-1.30052,43.09581],[-1.30531,43.06859],[-1.25244,43.04164],[-1.22881,43.05534],[-1.10333,43.0059],[-1.00963,42.99279],[-0.97133,42.96239],[-0.81652,42.95166],[-0.75478,42.96916],[-0.72037,42.92541],[-0.73422,42.91228],[-0.72608,42.89318],[-0.69837,42.87945],[-0.67637,42.88303],[-0.55497,42.77846],[-0.50863,42.82713],[-0.44334,42.79939],[-0.41319,42.80776],[-0.38833,42.80132],[-0.3122,42.84788],[-0.17939,42.78974],[-0.16141,42.79535],[-0.10519,42.72761],[-0.02468,42.68513],[0.17569,42.73424],[0.25336,42.7174],[0.29407,42.67431],[0.36251,42.72282],[0.40214,42.69779],[0.67873,42.69458],[0.65421,42.75872],[0.66121,42.84021],[0.711,42.86372],[0.93089,42.79154],[0.96166,42.80629],[0.98292,42.78754],[1.0804,42.78569],[1.15928,42.71407],[1.35562,42.71944],[1.44197,42.60217],[1.47986,42.61346],[1.46718,42.63296],[1.48043,42.65203],[1.50867,42.64483],[1.55418,42.65669],[1.60085,42.62703],[1.63485,42.62957],[1.6625,42.61982],[1.68267,42.62533],[1.73452,42.61515],[1.72588,42.59098],[1.7858,42.57698],[1.73683,42.55492],[1.72515,42.50338],[1.76335,42.48863],[1.83037,42.48395],[1.88853,42.4501],[1.93663,42.45439],[1.94292,42.44316],[1.94061,42.43333],[1.94084,42.43039],[1.9574,42.42401],[1.96482,42.37787],[2.00488,42.35399],[2.06241,42.35906],[2.11621,42.38393],[2.12789,42.41291],[2.16599,42.42314],[2.20578,42.41633],[2.25551,42.43757],[2.38504,42.39977],[2.43299,42.39423],[2.43508,42.37568],[2.48457,42.33933],[2.54382,42.33406],[2.55516,42.35351],[2.57934,42.35808],[2.6747,42.33974],[2.65311,42.38771],[2.72056,42.42298],[2.75497,42.42578],[2.77464,42.41046],[2.84335,42.45724],[2.85675,42.45444],[2.86983,42.46843],[2.88413,42.45938],[2.92107,42.4573],[2.94283,42.48174],[2.96518,42.46692],[3.03734,42.47363],[3.08167,42.42748],[3.10027,42.42621],[3.11379,42.43646],[3.17156,42.43545],[3.4481,42.4358],[7.60802,41.05927],[10.09675,41.44089],[9.56115,43.20816],[7.50102,43.51859],[7.42422,43.72209],[7.40903,43.7296],[7.41113,43.73156],[7.41291,43.73168],[7.41298,43.73311],[7.41233,43.73439],[7.42062,43.73977],[7.42299,43.74176],[7.42443,43.74087],[7.42809,43.74396],[7.43013,43.74895],[7.43624,43.75014],[7.43708,43.75197],[7.4389,43.75151],[7.4379,43.74963],[7.47823,43.73341],[7.53006,43.78405],[7.50423,43.84345],[7.49355,43.86551],[7.51162,43.88301],[7.56075,43.89932],[7.56858,43.94506],[7.60771,43.95772],[7.65266,43.9763],[7.66848,43.99943],[7.6597,44.03009],[7.72508,44.07578],[7.66878,44.12795],[7.68694,44.17487],[7.63245,44.17877],[7.62155,44.14881],[7.36364,44.11882],[7.34547,44.14359],[7.27827,44.1462],[7.16929,44.20352],[7.00764,44.23736],[6.98221,44.28289],[6.89171,44.36637],[6.88784,44.42043],[6.94504,44.43112],[6.86233,44.49834],[6.85507,44.53072],[6.96042,44.62129],[6.95133,44.66264],[7.00582,44.69364],[7.07484,44.68073],[7.00401,44.78782],[7.02217,44.82519],[6.93499,44.8664],[6.90774,44.84322],[6.75518,44.89915],[6.74519,44.93661],[6.74791,45.01939],[6.66981,45.02324],[6.62803,45.11175],[6.7697,45.16044],[6.85144,45.13226],[6.96706,45.20841],[7.07074,45.21228],[7.13115,45.25386],[7.10572,45.32924],[7.18019,45.40071],[7.00037,45.509],[6.98948,45.63869],[6.80785,45.71864],[6.80785,45.83265],[6.95315,45.85163],[7.04151,45.92435],[7.00946,45.9944],[6.93862,46.06502],[6.87868,46.03855],[6.89321,46.12548],[6.78968,46.14058],[6.86052,46.28512],[6.77152,46.34784],[6.8024,46.39171],[6.82312,46.42661],[6.53358,46.45431],[6.25432,46.3632],[6.21981,46.31304],[6.24826,46.30175],[6.25137,46.29014],[6.23775,46.27822],[6.24952,46.26255],[6.26749,46.24745],[6.29474,46.26221],[6.31041,46.24417],[6.29663,46.22688],[6.27694,46.21566],[6.26007,46.21165],[6.24821,46.20531],[6.23913,46.20511],[6.23544,46.20714],[6.22175,46.20045],[6.22222,46.19888],[6.21844,46.19837],[6.21603,46.19507],[6.21273,46.19409],[6.21114,46.1927],[6.20539,46.19163],[6.19807,46.18369],[6.19552,46.18401],[6.18707,46.17999],[6.18871,46.16644],[6.18116,46.16187],[6.15305,46.15194],[6.13397,46.1406],[6.09926,46.14373],[6.09199,46.15191],[6.07491,46.14879],[6.05203,46.15191],[6.04564,46.14031],[6.03614,46.13712],[6.01791,46.14228],[5.9871,46.14499],[5.97893,46.13303],[5.95781,46.12925],[5.9641,46.14412],[5.97508,46.15863],[5.98188,46.17392],[5.98846,46.17046],[5.99573,46.18587],[5.96515,46.19638],[5.97542,46.21525],[6.02461,46.23313],[6.03342,46.2383],[6.04602,46.23127],[6.05029,46.23518],[6.0633,46.24583],[6.07072,46.24085],[6.08563,46.24651],[6.10071,46.23772],[6.12446,46.25059],[6.11926,46.2634],[6.1013,46.28512],[6.11697,46.29547],[6.1198,46.31157],[6.13876,46.33844],[6.15738,46.3491],[6.16987,46.36759],[6.15985,46.37721],[6.15016,46.3778],[6.09926,46.40768],[6.06407,46.41676],[6.08427,46.44305],[6.07269,46.46244],[6.1567,46.54402],[6.11084,46.57649],[6.27135,46.68251],[6.38351,46.73171],[6.45209,46.77502],[6.43216,46.80336],[6.46456,46.88865],[6.43341,46.92703],[6.71531,47.0494],[6.68823,47.06616],[6.76788,47.1208],[6.8489,47.15933],[6.9508,47.24338],[6.95108,47.26428],[6.94316,47.28747],[7.05305,47.33304],[7.0564,47.35134],[7.03125,47.36996],[6.87959,47.35335],[6.88542,47.37262],[6.93744,47.40714],[6.93953,47.43388],[7.0024,47.45264],[6.98425,47.49432],[7.0231,47.50522],[7.07425,47.48863],[7.12781,47.50371],[7.16249,47.49025],[7.19583,47.49455],[7.17026,47.44312],[7.24669,47.4205],[7.33526,47.44186],[7.35603,47.43432],[7.40308,47.43638],[7.43088,47.45846],[7.4462,47.46264],[7.4583,47.47216],[7.42923,47.48628],[7.43356,47.49712],[7.47534,47.47932],[7.51076,47.49651],[7.49804,47.51798],[7.5229,47.51644],[7.53199,47.5284],[7.51904,47.53515],[7.50588,47.52856],[7.49691,47.53821],[7.50873,47.54546],[7.51723,47.54578],[7.52831,47.55347],[7.53634,47.55553],[7.55652,47.56779],[7.55689,47.57232],[7.56548,47.57617],[7.56684,47.57785],[7.58386,47.57536],[7.58945,47.59017],[7.59301,47.60058],[7.58851,47.60794],[7.57423,47.61628],[7.5591,47.63849],[7.53384,47.65115],[7.52067,47.66437],[7.51915,47.68335],[7.51266,47.70197],[7.53722,47.71635],[7.54761,47.72912],[7.52921,47.77747],[7.55673,47.87371],[7.62302,47.97898],[7.56966,48.03265],[7.57137,48.12292],[7.6648,48.22219],[7.69022,48.30018],[7.74562,48.32736],[7.73109,48.39192],[7.76833,48.48945],[7.80647,48.51239],[7.80167,48.54758],[7.80057,48.5857],[7.84098,48.64217],[7.89002,48.66317],[7.96812,48.72491],[7.96994,48.75606],[8.01534,48.76085],[8.0326,48.79017],[8.06802,48.78957],[8.10253,48.81829],[8.12813,48.87985],[8.19989,48.95825],[8.20031,48.95856],[8.22604,48.97352],[8.14189,48.97833],[7.97783,49.03161],[7.93641,49.05544],[7.86386,49.03499],[7.79557,49.06583],[7.75948,49.04562],[7.63618,49.05428],[7.62575,49.07654],[7.56416,49.08136],[7.53012,49.09818],[7.49172,49.13915],[7.49473,49.17],[7.44455,49.16765],[7.44052,49.18354],[7.3662,49.17308],[7.35995,49.14399],[7.3195,49.14231],[7.29514,49.11426],[7.23473,49.12971],[7.1593,49.1204],[7.1358,49.1282],[7.12504,49.14253],[7.10384,49.13787],[7.10715,49.15631],[7.07859,49.15031],[7.09007,49.13094],[7.07162,49.1255],[7.06642,49.11415],[7.05548,49.11185],[7.04843,49.11422],[7.04409,49.12123],[7.04662,49.13724],[7.03178,49.15734],[7.0274,49.17042],[7.03459,49.19096],[7.01318,49.19018],[6.97273,49.2099],[6.95963,49.203],[6.94028,49.21641],[6.93831,49.2223],[6.91875,49.22261],[6.89298,49.20863],[6.85939,49.22376],[6.83555,49.21249],[6.85119,49.20038],[6.85016,49.19354],[6.86225,49.18185],[6.84703,49.15734],[6.83385,49.15162],[6.78265,49.16793],[6.73765,49.16375],[6.71137,49.18808],[6.73256,49.20486],[6.71843,49.2208],[6.69274,49.21661],[6.66583,49.28065],[6.60186,49.31055],[6.572,49.35027],[6.58807,49.35358],[6.60091,49.36864],[6.533,49.40748],[6.55404,49.42464],[6.42432,49.47683],[6.40274,49.46546],[6.39168,49.4667],[6.38352,49.46463],[6.36778,49.46937],[6.3687,49.4593],[6.28818,49.48465],[6.27875,49.503],[6.25029,49.50609],[6.2409,49.51408],[6.19543,49.50536],[6.17386,49.50934],[6.15366,49.50226],[6.16115,49.49297],[6.14321,49.48796],[6.12814,49.49365],[6.12346,49.4735],[6.10325,49.4707],[6.09845,49.46351],[6.10072,49.45268],[6.08373,49.45594],[6.07887,49.46399],[6.05553,49.46663],[6.04176,49.44801],[6.02743,49.44845],[6.02648,49.45451],[5.97693,49.45513],[5.96876,49.49053],[5.94224,49.49608],[5.94128,49.50034],[5.86571,49.50015],[5.83389,49.52152],[5.83467,49.52717],[5.84466,49.53027],[5.83648,49.5425],[5.81664,49.53775],[5.80871,49.5425],[5.81838,49.54777],[5.79195,49.55228],[5.77435,49.56298],[5.7577,49.55915],[5.75649,49.54321],[5.64505,49.55146],[5.60909,49.51228],[5.55001,49.52729],[5.46541,49.49825],[5.46734,49.52648],[5.43713,49.5707],[5.3974,49.61596],[5.34837,49.62889],[5.33851,49.61599],[5.3137,49.61225],[5.30214,49.63055],[5.33039,49.6555],[5.31465,49.66846],[5.26232,49.69456],[5.14545,49.70287],[5.09249,49.76193],[4.96714,49.79872],[4.85464,49.78995],[4.86965,49.82271],[4.85134,49.86457],[4.88529,49.9236],[4.78827,49.95609],[4.8382,50.06738],[4.88602,50.15182],[4.83279,50.15331],[4.82438,50.16878],[4.75237,50.11314],[4.70064,50.09384],[4.68695,49.99685],[4.5414,49.96911],[4.51098,49.94659],[4.43488,49.94122],[4.35051,49.95315],[4.31963,49.97043],[4.20532,49.95803],[4.14239,49.98034],[4.13508,50.01976],[4.16294,50.04719],[4.23101,50.06945],[4.20147,50.13535],[4.13561,50.13078],[4.16014,50.19239],[4.15524,50.21103],[4.21945,50.25539],[4.20651,50.27333],[4.17861,50.27443],[4.17347,50.28838],[4.15524,50.2833],[4.16808,50.25786],[4.13665,50.25609],[4.11954,50.30425],[4.10957,50.30234],[4.10237,50.31247],[4.0689,50.3254],[4.0268,50.35793],[3.96771,50.34989],[3.90781,50.32814],[3.84314,50.35219],[3.73911,50.34809],[3.70987,50.3191],[3.71009,50.30305],[3.66976,50.34563],[3.65709,50.36873],[3.67262,50.38663],[3.67494,50.40239],[3.66153,50.45165],[3.64426,50.46275],[3.61014,50.49568],[3.58361,50.49049],[3.5683,50.50192],[3.49509,50.48885],[3.51564,50.5256],[3.47385,50.53397],[3.44629,50.51009],[3.37693,50.49538],[3.28575,50.52724],[3.2729,50.60718],[3.23951,50.6585],[3.264,50.67668],[3.2536,50.68977],[3.26141,50.69151],[3.26063,50.70086],[3.24593,50.71389],[3.22042,50.71019],[3.20845,50.71662],[3.19017,50.72569],[3.20064,50.73547],[3.18811,50.74025],[3.18339,50.74981],[3.16476,50.76843],[3.15017,50.79031],[3.1257,50.78603],[3.11987,50.79188],[3.11206,50.79416],[3.10614,50.78303],[3.09163,50.77717],[3.04314,50.77674],[3.00537,50.76588],[2.96778,50.75242],[2.95019,50.75138],[2.90873,50.702],[2.91036,50.6939],[2.90069,50.69263],[2.88504,50.70656],[2.87937,50.70298],[2.86985,50.7033],[2.8483,50.72276],[2.81056,50.71773],[2.71165,50.81295],[2.63331,50.81457],[2.59093,50.91751],[2.63074,50.94746],[2.57551,51.00326],[2.55904,51.07014]]]]}},{type:"Feature",properties:{iso1A2:"GA",iso1A3:"GAB",iso1N3:"266",wikidata:"Q1000",nameEn:"Gabon",groups:["017","202","002"],callingCodes:["241"]},geometry:{type:"MultiPolygon",coordinates:[[[[13.29457,2.16106],[13.28534,2.25716],[11.37116,2.29975],[11.3561,2.17217],[11.35307,1.00251],[9.79648,1.0019],[9.78058,1.03996],[9.76085,1.05949],[9.73014,1.06721],[9.68638,1.06836],[9.66092,1.05865],[9.62096,1.03039],[9.54793,1.0185],[9.51998,0.96418],[9.35563,0.84865],[7.24416,-0.64092],[10.75913,-4.39519],[11.12647,-3.94169],[11.22301,-3.69888],[11.48764,-3.51089],[11.57949,-3.52798],[11.68608,-3.68942],[11.87083,-3.71571],[11.92719,-3.62768],[11.8318,-3.5812],[11.96554,-3.30267],[11.70227,-3.17465],[11.70558,-3.0773],[11.80365,-3.00424],[11.64798,-2.81146],[11.5359,-2.85654],[11.64487,-2.61865],[11.57637,-2.33379],[11.74605,-2.39936],[11.96866,-2.33559],[12.04895,-2.41704],[12.47925,-2.32626],[12.44656,-1.92025],[12.61312,-1.8129],[12.82172,-1.91091],[13.02759,-2.33098],[13.47977,-2.43224],[13.75884,-2.09293],[13.92073,-2.35581],[13.85846,-2.46935],[14.10442,-2.49268],[14.23829,-2.33715],[14.16202,-2.23916],[14.23518,-2.15671],[14.25932,-1.97624],[14.41838,-1.89412],[14.52569,-0.57818],[14.41887,-0.44799],[14.2165,-0.38261],[14.06862,-0.20826],[13.90632,-0.2287],[13.88648,0.26652],[14.10909,0.58563],[14.26066,0.57255],[14.48179,0.9152],[14.25186,1.39842],[13.89582,1.4261],[13.15519,1.23368],[13.25447,1.32339],[13.13461,1.57238],[13.29457,2.16106]]]]}},{type:"Feature",properties:{iso1A2:"GB",iso1A3:"GBR",iso1N3:"826",wikidata:"Q145",nameEn:"United Kingdom",aliases:["UK","Britain","Great Britain"],groups:["154","150"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["44"]},geometry:{type:"MultiPolygon",coordinates:[[[[-5.83481,53.87749],[-4.1819,54.57861],[-3.64906,54.12723],[-5.37267,53.63269],[-5.79914,52.03902],[-7.74976,48.64773],[1.17405,50.74239],[2.18458,51.52087],[2.56575,51.85301],[-0.3751,61.32236],[-14.78497,57.60709],[-7.93366,55.84142],[-6.79943,55.54107],[-6.71944,55.27952],[-6.9734,55.19878],[-7.2471,55.06933],[-7.34464,55.04688],[-7.4033,55.00391],[-7.40004,54.94498],[-7.44404,54.9403],[-7.4473,54.87003],[-7.47626,54.83084],[-7.54508,54.79401],[-7.54671,54.74606],[-7.64449,54.75265],[-7.75041,54.7103],[-7.83352,54.73854],[-7.93293,54.66603],[-7.70315,54.62077],[-7.8596,54.53671],[-7.99812,54.54427],[-8.04538,54.48941],[-8.179,54.46763],[-8.04555,54.36292],[-7.87101,54.29299],[-7.8596,54.21779],[-7.81397,54.20159],[-7.69501,54.20731],[-7.55812,54.12239],[-7.4799,54.12239],[-7.44567,54.1539],[-7.32834,54.11475],[-7.30553,54.11869],[-7.34005,54.14698],[-7.29157,54.17191],[-7.28017,54.16714],[-7.29687,54.1354],[-7.29493,54.12013],[-7.26316,54.13863],[-7.25012,54.20063],[-7.14908,54.22732],[-7.19145,54.31296],[-7.02034,54.4212],[-6.87775,54.34682],[-6.85179,54.29176],[-6.81583,54.22791],[-6.74575,54.18788],[-6.70175,54.20218],[-6.6382,54.17071],[-6.66264,54.0666],[-6.62842,54.03503],[-6.47849,54.06947],[-6.36605,54.07234],[-6.36279,54.11248],[-6.32694,54.09337],[-6.29003,54.11278],[-6.26218,54.09785],[-5.83481,53.87749]]],[[[33.70575,34.97947],[33.83531,34.73974],[33.98684,34.76642],[33.90075,34.96623],[33.86432,34.97592],[33.84811,34.97075],[33.83505,34.98108],[33.85621,34.98956],[33.85891,35.001],[33.85216,35.00579],[33.84045,35.00616],[33.82875,35.01685],[33.83055,35.02865],[33.81524,35.04192],[33.8012,35.04786],[33.82051,35.0667],[33.8355,35.05777],[33.85261,35.0574],[33.88367,35.07877],[33.89485,35.06873],[33.90247,35.07686],[33.91299,35.07579],[33.91789,35.08688],[33.89853,35.11377],[33.88737,35.11408],[33.88943,35.12007],[33.88561,35.12449],[33.87224,35.12293],[33.87622,35.10457],[33.87097,35.09389],[33.87479,35.08881],[33.8541,35.07201],[33.84168,35.06823],[33.82067,35.07826],[33.78581,35.05104],[33.76106,35.04253],[33.73824,35.05321],[33.71482,35.03722],[33.70209,35.04882],[33.7161,35.07279],[33.70861,35.07644],[33.69095,35.06237],[33.68474,35.06602],[33.67742,35.05963],[33.67678,35.03866],[33.69938,35.03123],[33.69731,35.01754],[33.71514,35.00294],[33.70639,34.99303],[33.70575,34.97947]],[[33.77312,34.9976],[33.77553,34.99518],[33.78516,34.99582],[33.79191,34.98914],[33.78917,34.98854],[33.78571,34.98951],[33.78318,34.98699],[33.78149,34.98854],[33.77843,34.988],[33.7778,34.98981],[33.76738,34.99188],[33.76605,34.99543],[33.75682,34.99916],[33.75994,35.00113],[33.77312,34.9976]],[[33.74144,35.01053],[33.7343,35.01178],[33.73781,35.02181],[33.74265,35.02329],[33.74983,35.02274],[33.7492,35.01319],[33.74144,35.01053]]],[[[32.86014,34.70585],[32.82717,34.70622],[32.79433,34.67883],[32.76136,34.68318],[32.75515,34.64985],[32.74412,34.43926],[33.26744,34.49942],[33.0138,34.64424],[32.96968,34.64046],[32.96718,34.63446],[32.95891,34.62919],[32.95323,34.64075],[32.95471,34.64528],[32.94976,34.65204],[32.94796,34.6587],[32.95325,34.66462],[32.97079,34.66112],[32.97736,34.65277],[32.99014,34.65518],[32.98668,34.67268],[32.99135,34.68061],[32.95539,34.68471],[32.94683,34.67907],[32.94379,34.67111],[32.93693,34.67027],[32.93449,34.66241],[32.92807,34.66736],[32.93043,34.67091],[32.91398,34.67343],[32.9068,34.66102],[32.86167,34.68734],[32.86014,34.70585]]]]}},{type:"Feature",properties:{iso1A2:"GD",iso1A3:"GRD",iso1N3:"308",wikidata:"Q769",nameEn:"Grenada",aliases:["WG"],groups:["029","003","419","019"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["1 473"]},geometry:{type:"MultiPolygon",coordinates:[[[[-62.14806,11.87638],[-61.57265,11.65795],[-61.13395,12.51526],[-61.38256,12.52991],[-61.73897,12.61191],[-62.14806,11.87638]]]]}},{type:"Feature",properties:{iso1A2:"GE",iso1A3:"GEO",iso1N3:"268",wikidata:"Q230",nameEn:"Georgia",groups:["145","142"],callingCodes:["995"]},geometry:{type:"MultiPolygon",coordinates:[[[[46.42738,41.91323],[45.61676,42.20768],[45.78692,42.48358],[45.36501,42.55268],[45.15318,42.70598],[44.88754,42.74934],[44.80941,42.61277],[44.70002,42.74679],[44.54202,42.75699],[43.95517,42.55396],[43.73119,42.62043],[43.81453,42.74297],[43.0419,43.02413],[43.03322,43.08883],[42.75889,43.19651],[42.66667,43.13917],[42.40563,43.23226],[41.64935,43.22331],[40.65957,43.56212],[40.10657,43.57344],[40.04445,43.47776],[40.03312,43.44262],[40.01007,43.42411],[40.01552,43.42025],[40.00853,43.40578],[40.0078,43.38551],[39.81147,43.06294],[40.89217,41.72528],[41.54366,41.52185],[41.7148,41.4932],[41.7124,41.47417],[41.81939,41.43621],[41.95134,41.52466],[42.26387,41.49346],[42.51772,41.43606],[42.59202,41.58183],[42.72794,41.59714],[42.84471,41.58912],[42.78995,41.50126],[42.84899,41.47265],[42.8785,41.50516],[43.02956,41.37891],[43.21707,41.30331],[43.13373,41.25503],[43.1945,41.25242],[43.23096,41.17536],[43.36118,41.2028],[43.44973,41.17666],[43.4717,41.12611],[43.67712,41.13398],[43.74717,41.1117],[43.84835,41.16329],[44.16591,41.19141],[44.18148,41.24644],[44.32139,41.2079],[44.34337,41.20312],[44.34417,41.2382],[44.46791,41.18204],[44.59322,41.1933],[44.61462,41.24018],[44.72814,41.20338],[44.82084,41.21513],[44.87887,41.20195],[44.89911,41.21366],[44.84358,41.23088],[44.81749,41.23488],[44.80053,41.25949],[44.81437,41.30371],[44.93493,41.25685],[45.0133,41.29747],[45.09867,41.34065],[45.1797,41.42231],[45.26285,41.46433],[45.31352,41.47168],[45.4006,41.42402],[45.45973,41.45898],[45.68389,41.3539],[45.71035,41.36208],[45.75705,41.35157],[45.69946,41.29545],[45.80842,41.2229],[45.95786,41.17956],[46.13221,41.19479],[46.27698,41.19011],[46.37661,41.10805],[46.456,41.09984],[46.48558,41.0576],[46.55096,41.1104],[46.63969,41.09515],[46.66148,41.20533],[46.72375,41.28609],[46.63658,41.37727],[46.4669,41.43331],[46.40307,41.48464],[46.33925,41.4963],[46.29794,41.5724],[46.34126,41.57454],[46.34039,41.5947],[46.3253,41.60912],[46.28182,41.60089],[46.26531,41.63339],[46.24429,41.59883],[46.19759,41.62327],[46.17891,41.72094],[46.20538,41.77205],[46.23962,41.75811],[46.30863,41.79133],[46.3984,41.84399],[46.42738,41.91323]]]]}},{type:"Feature",properties:{iso1A2:"GF",iso1A3:"GUF",iso1N3:"254",wikidata:"Q3769",nameEn:"French Guiana",country:"FR",groups:["EU","005","419","019"],callingCodes:["594"]},geometry:{type:"MultiPolygon",coordinates:[[[[-51.35485,4.8383],[-53.7094,6.2264],[-54.01074,5.68785],[-54.01877,5.52789],[-54.26916,5.26909],[-54.4717,4.91964],[-54.38444,4.13222],[-54.19367,3.84387],[-54.05128,3.63557],[-53.98914,3.627],[-53.9849,3.58697],[-54.28534,2.67798],[-54.42864,2.42442],[-54.6084,2.32856],[-54.16286,2.10779],[-53.78743,2.34412],[-52.96539,2.1881],[-52.6906,2.37298],[-52.31787,3.17896],[-51.85573,3.83427],[-51.82312,3.85825],[-51.79599,3.89336],[-51.61983,4.14596],[-51.63798,4.51394],[-51.35485,4.8383]]]]}},{type:"Feature",properties:{iso1A2:"GG",iso1A3:"GGY",iso1N3:"831",wikidata:"Q25230",nameEn:"Guernsey",country:"GB",groups:["830","154","150"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["44 01481"]},geometry:{type:"MultiPolygon",coordinates:[[[[-2.65349,49.15373],[-2.36485,49.48223],[-2.09454,49.46288],[-2.02963,49.91866],[-3.28154,49.57329],[-2.65349,49.15373]]]]}},{type:"Feature",properties:{iso1A2:"GH",iso1A3:"GHA",iso1N3:"288",wikidata:"Q117",nameEn:"Ghana",groups:["011","202","002"],callingCodes:["233"]},geometry:{type:"MultiPolygon",coordinates:[[[[-0.13493,11.14075],[-0.27374,11.17157],[-0.28566,11.12713],[-0.35955,11.07801],[-0.38219,11.12596],[-0.42391,11.11661],[-0.44298,11.04292],[-0.61937,10.91305],[-0.67143,10.99811],[-2.83373,11.0067],[-2.94232,10.64281],[-2.83108,10.40252],[-2.74174,9.83172],[-2.76534,9.56589],[-2.68802,9.49343],[-2.69814,9.22717],[-2.77799,9.04949],[-2.66357,9.01771],[-2.58243,8.7789],[-2.49037,8.20872],[-2.62901,8.11495],[-2.61232,8.02645],[-2.67787,8.02055],[-2.74819,7.92613],[-2.78395,7.94974],[-2.79467,7.86002],[-2.92339,7.60847],[-2.97822,7.27165],[-2.95438,7.23737],[-3.23327,6.81744],[-3.21954,6.74407],[-3.25999,6.62521],[-3.01896,5.71697],[-2.95323,5.71865],[-2.96671,5.6415],[-2.93132,5.62137],[-2.85378,5.65156],[-2.76614,5.60963],[-2.72737,5.34789],[-2.77625,5.34621],[-2.73074,5.1364],[-2.75502,5.10657],[-2.95261,5.12477],[-2.96554,5.10397],[-3.063,5.13665],[-3.11073,5.12675],[-3.10675,5.08515],[-3.34019,4.17519],[1.07031,5.15655],[1.27574,5.93551],[1.19771,6.11522],[1.19966,6.17069],[1.09187,6.17074],[1.05969,6.22998],[1.03108,6.24064],[0.99652,6.33779],[0.89283,6.33779],[0.71048,6.53083],[0.74862,6.56517],[0.63659,6.63857],[0.6497,6.73682],[0.58176,6.76049],[0.57406,6.80348],[0.52853,6.82921],[0.56508,6.92971],[0.52098,6.94391],[0.52217,6.9723],[0.59606,7.01252],[0.65327,7.31643],[0.62943,7.41099],[0.57223,7.39326],[0.52455,7.45354],[0.51979,7.58706],[0.58295,7.62368],[0.62943,7.85751],[0.58891,8.12779],[0.6056,8.13959],[0.61156,8.18324],[0.5913,8.19622],[0.63897,8.25873],[0.73432,8.29529],[0.64731,8.48866],[0.47211,8.59945],[0.37319,8.75262],[0.52455,8.87746],[0.45424,9.04581],[0.56388,9.40697],[0.49118,9.48339],[0.36485,9.49749],[0.33148,9.44812],[0.25758,9.42696],[0.2254,9.47869],[0.31241,9.50337],[0.30406,9.521],[0.2409,9.52335],[0.23851,9.57389],[0.38153,9.58682],[0.36008,9.6256],[0.29334,9.59387],[0.26712,9.66437],[0.28261,9.69022],[0.32313,9.6491],[0.34816,9.66907],[0.34816,9.71607],[0.32075,9.72781],[0.36366,10.03309],[0.41252,10.02018],[0.41371,10.06361],[0.35293,10.09412],[0.39584,10.31112],[0.33028,10.30408],[0.29453,10.41546],[0.18846,10.4096],[0.12886,10.53149],[-0.05945,10.63458],[-0.09141,10.7147],[-0.07327,10.71845],[-0.07183,10.76794],[-0.0228,10.81916],[-0.02685,10.8783],[-0.00908,10.91644],[-0.0063,10.96417],[0.03355,10.9807],[0.02395,11.06229],[0.00342,11.08317],[-0.00514,11.10763],[-0.0275,11.11202],[-0.05733,11.08628],[-0.14462,11.10811],[-0.13493,11.14075]]]]}},{type:"Feature",properties:{iso1A2:"GI",iso1A3:"GIB",iso1N3:"292",wikidata:"Q1410",nameEn:"Gibraltar",country:"GB",groups:["039","150"],callingCodes:["350"]},geometry:{type:"MultiPolygon",coordinates:[[[[-5.28217,36.09907],[-5.27801,36.14942],[-5.33822,36.15272],[-5.34536,36.15501],[-5.36494,36.15496],[-5.38545,36.15481],[-5.40134,36.14896],[-5.39074,36.10278],[-5.36503,36.06205],[-5.32837,36.05935],[-5.3004,36.07439],[-5.28217,36.09907]]]]}},{type:"Feature",properties:{iso1A2:"GL",iso1A3:"GRL",iso1N3:"304",wikidata:"Q223",nameEn:"Greenland",country:"DK",groups:["021","003","019"],callingCodes:["299"]},geometry:{type:"MultiPolygon",coordinates:[[[[-45.47832,84.58738],[-68.21821,80.48551],[-76.75614,76.72014],[-46.37635,57.3249],[-9.68082,72.73731],[-5.7106,84.28058],[-45.47832,84.58738]]]]}},{type:"Feature",properties:{iso1A2:"GM",iso1A3:"GMB",iso1N3:"270",wikidata:"Q1005",nameEn:"The Gambia",groups:["011","202","002"],callingCodes:["220"]},geometry:{type:"MultiPolygon",coordinates:[[[[-15.14917,13.57989],[-14.36795,13.23033],[-13.79409,13.34472],[-13.8955,13.59126],[-14.34721,13.46578],[-14.93719,13.80173],[-15.36504,13.79313],[-15.47902,13.58758],[-17.43598,13.59273],[-17.43966,13.04579],[-16.74676,13.06025],[-16.69343,13.16791],[-15.80355,13.16729],[-15.80478,13.34832],[-15.26908,13.37768],[-15.14917,13.57989]]]]}},{type:"Feature",properties:{iso1A2:"GN",iso1A3:"GIN",iso1N3:"324",wikidata:"Q1006",nameEn:"Guinea",groups:["011","202","002"],callingCodes:["224"]},geometry:{type:"MultiPolygon",coordinates:[[[[-11.37536,12.40788],[-11.46267,12.44559],[-11.91331,12.42008],[-12.35415,12.32758],[-12.87336,12.51892],[-13.06603,12.49342],[-13.05296,12.64003],[-13.70523,12.68013],[-13.7039,12.60313],[-13.65089,12.49515],[-13.64168,12.42764],[-13.70851,12.24978],[-13.92745,12.24077],[-13.94589,12.16869],[-13.7039,12.00869],[-13.7039,11.70195],[-14.09799,11.63649],[-14.26623,11.67486],[-14.31513,11.60713],[-14.51173,11.49708],[-14.66677,11.51188],[-14.77786,11.36323],[-14.95993,10.99244],[-15.07174,10.89557],[-15.96748,10.162],[-14.36218,8.64107],[-13.29911,9.04245],[-13.18586,9.0925],[-13.08953,9.0409],[-12.94095,9.26335],[-12.76788,9.3133],[-12.47254,9.86834],[-12.24262,9.92386],[-12.12634,9.87203],[-11.91023,9.93927],[-11.89624,9.99763],[-11.2118,10.00098],[-10.6534,9.29919],[-10.74484,9.07998],[-10.5783,9.06386],[-10.56197,8.81225],[-10.47707,8.67669],[-10.61422,8.5314],[-10.70565,8.29235],[-10.63934,8.35326],[-10.54891,8.31174],[-10.37257,8.48941],[-10.27575,8.48711],[-10.203,8.47991],[-10.14579,8.52665],[-10.05375,8.50697],[-10.05873,8.42578],[-9.77763,8.54633],[-9.47415,8.35195],[-9.50898,8.18455],[-9.41445,8.02448],[-9.44928,7.9284],[-9.35724,7.74111],[-9.37465,7.62032],[-9.48161,7.37122],[-9.41943,7.41809],[-9.305,7.42056],[-9.20798,7.38109],[-9.18311,7.30461],[-9.09107,7.1985],[-8.93435,7.2824],[-8.85724,7.26019],[-8.8448,7.35149],[-8.72789,7.51429],[-8.67814,7.69428],[-8.55874,7.70167],[-8.55874,7.62525],[-8.47114,7.55676],[-8.4003,7.6285],[-8.21374,7.54466],[-8.09931,7.78626],[-8.13414,7.87991],[-8.06449,8.04989],[-7.94695,8.00925],[-7.99919,8.11023],[-7.98675,8.20134],[-8.062,8.16071],[-8.2411,8.24196],[-8.22991,8.48438],[-7.92518,8.50652],[-7.65653,8.36873],[-7.69882,8.66148],[-7.95503,8.81146],[-7.92518,8.99332],[-7.73862,9.08422],[-7.90777,9.20456],[-7.85056,9.41812],[-8.03463,9.39604],[-8.14657,9.55062],[-8.09434,9.86936],[-8.15652,9.94288],[-8.11921,10.04577],[-8.01225,10.1021],[-7.97971,10.17117],[-7.9578,10.2703],[-8.10207,10.44649],[-8.22711,10.41722],[-8.32614,10.69273],[-8.2667,10.91762],[-8.35083,11.06234],[-8.66923,10.99397],[-8.40058,11.37466],[-8.80854,11.66715],[-8.94784,12.34842],[-9.13689,12.50875],[-9.38067,12.48446],[-9.32097,12.29009],[-9.63938,12.18312],[-9.714,12.0226],[-10.30604,12.24634],[-10.71897,11.91552],[-10.80355,12.1053],[-10.99758,12.24634],[-11.24136,12.01286],[-11.50006,12.17826],[-11.37536,12.40788]]]]}},{type:"Feature",properties:{iso1A2:"GP",iso1A3:"GLP",iso1N3:"312",wikidata:"Q17012",nameEn:"Guadeloupe",country:"FR",groups:["EU","029","003","419","019"],callingCodes:["590"]},geometry:{type:"MultiPolygon",coordinates:[[[[-60.95725,15.70997],[-60.71337,16.48911],[-61.44461,16.81958],[-61.83929,16.66647],[-62.17275,16.35721],[-61.81728,15.58058],[-61.44899,15.79571],[-60.95725,15.70997]]]]}},{type:"Feature",properties:{iso1A2:"GQ",iso1A3:"GNQ",iso1N3:"226",wikidata:"Q983",nameEn:"Equatorial Guinea",groups:["017","202","002"],callingCodes:["240"]},geometry:{type:"MultiPolygon",coordinates:[[[[9.22018,3.72052],[8.34397,4.30689],[8.05799,3.48284],[8.0168,1.79377],[6.69416,-0.53945],[5.38965,-1.19244],[5.3459,-2.30107],[7.24416,-0.64092],[9.35563,0.84865],[9.51998,0.96418],[9.54793,1.0185],[9.62096,1.03039],[9.66092,1.05865],[9.68638,1.06836],[9.73014,1.06721],[9.76085,1.05949],[9.78058,1.03996],[9.79648,1.0019],[11.35307,1.00251],[11.3561,2.17217],[9.991,2.16561],[9.90749,2.20049],[9.89012,2.20457],[9.84716,2.24676],[9.83238,2.29079],[9.83754,2.32428],[9.82123,2.35097],[9.81162,2.33797],[9.22018,3.72052]]]]}},{type:"Feature",properties:{iso1A2:"GR",iso1A3:"GRC",iso1N3:"300",wikidata:"Q41",nameEn:"Greece",aliases:["EL"],groups:["EU","039","150"],callingCodes:["30"]},geometry:{type:"MultiPolygon",coordinates:[[[[26.03489,40.73051],[26.0754,40.72772],[26.08638,40.73214],[26.12495,40.74283],[26.12854,40.77339],[26.15685,40.80709],[26.21351,40.83298],[26.20856,40.86048],[26.26169,40.9168],[26.29441,40.89119],[26.28623,40.93005],[26.32259,40.94042],[26.35894,40.94292],[26.33297,40.98388],[26.3606,41.02027],[26.31928,41.07386],[26.32259,41.24929],[26.39861,41.25053],[26.5209,41.33993],[26.5837,41.32131],[26.62997,41.34613],[26.61767,41.42281],[26.59742,41.48058],[26.59196,41.60491],[26.5209,41.62592],[26.47958,41.67037],[26.35957,41.71149],[26.30255,41.70925],[26.2654,41.71544],[26.22888,41.74139],[26.21325,41.73223],[26.16841,41.74858],[26.06148,41.70345],[26.07083,41.64584],[26.15146,41.60828],[26.14328,41.55496],[26.17951,41.55409],[26.176,41.50072],[26.14796,41.47533],[26.20288,41.43943],[26.16548,41.42278],[26.12926,41.35878],[25.87919,41.30526],[25.8266,41.34563],[25.70507,41.29209],[25.66183,41.31316],[25.61042,41.30614],[25.55082,41.31667],[25.52394,41.2798],[25.48187,41.28506],[25.28322,41.23411],[25.11611,41.34212],[24.942,41.38685],[24.90928,41.40876],[24.86136,41.39298],[24.82514,41.4035],[24.8041,41.34913],[24.71529,41.41928],[24.61129,41.42278],[24.52599,41.56808],[24.30513,41.51297],[24.27124,41.57682],[24.18126,41.51735],[24.10063,41.54796],[24.06323,41.53222],[24.06908,41.46132],[23.96975,41.44118],[23.91483,41.47971],[23.89613,41.45257],[23.80148,41.43943],[23.76525,41.40175],[23.67644,41.41139],[23.63203,41.37632],[23.52453,41.40262],[23.40416,41.39999],[23.33639,41.36317],[23.31301,41.40525],[23.22771,41.37106],[23.21953,41.33773],[23.1833,41.31755],[22.93334,41.34104],[22.81199,41.3398],[22.76408,41.32225],[22.74538,41.16321],[22.71266,41.13945],[22.65306,41.18168],[22.62852,41.14385],[22.58295,41.11568],[22.5549,41.13065],[22.42285,41.11921],[22.26744,41.16409],[22.17629,41.15969],[22.1424,41.12449],[22.06527,41.15617],[21.90869,41.09191],[21.91102,41.04786],[21.7556,40.92525],[21.69601,40.9429],[21.57448,40.86076],[21.53007,40.90759],[21.41555,40.9173],[21.35595,40.87578],[21.25779,40.86165],[21.21105,40.8855],[21.15262,40.85546],[20.97887,40.85475],[20.98396,40.79109],[20.95752,40.76982],[20.98134,40.76046],[21.05833,40.66586],[21.03932,40.56299],[20.96908,40.51526],[20.94925,40.46625],[20.83688,40.47882],[20.7906,40.42726],[20.78234,40.35803],[20.71789,40.27739],[20.67162,40.09433],[20.62566,40.0897],[20.61081,40.07866],[20.55593,40.06524],[20.51297,40.08168],[20.48487,40.06271],[20.42373,40.06777],[20.37911,39.99058],[20.31135,39.99438],[20.41546,39.82832],[20.41475,39.81437],[20.38572,39.78516],[20.30804,39.81563],[20.29152,39.80421],[20.31961,39.72799],[20.27412,39.69884],[20.22707,39.67459],[20.22376,39.64532],[20.15988,39.652],[20.12956,39.65805],[20.05189,39.69112],[20.00957,39.69227],[19.98042,39.6504],[19.92466,39.69533],[19.97622,39.78684],[19.95905,39.82857],[19.0384,40.35325],[19.20409,39.7532],[22.5213,33.45682],[29.73302,35.92555],[29.69611,36.10365],[29.61805,36.14179],[29.61002,36.1731],[29.48192,36.18377],[29.30783,36.01033],[28.23708,36.56812],[27.95037,36.46155],[27.89482,36.69898],[27.46117,36.53789],[27.24613,36.71622],[27.45627,36.9008],[27.20312,36.94571],[27.14757,37.32],[26.95583,37.64989],[26.99377,37.69034],[27.16428,37.72343],[27.05537,37.9131],[26.21136,38.17558],[26.24183,38.44695],[26.32173,38.48731],[26.21136,38.65436],[26.61814,38.81372],[26.70773,39.0312],[26.43357,39.43096],[25.94257,39.39358],[25.61285,40.17161],[26.04292,40.3958],[25.94795,40.72797],[26.03489,40.73051]]]]}},{type:"Feature",properties:{iso1A2:"GS",iso1A3:"SGS",iso1N3:"239",wikidata:"Q35086",nameEn:"South Georgia and South Sandwich Islands",country:"GB",groups:["005","419","019"],driveSide:"left",callingCodes:["500"]},geometry:{type:"MultiPolygon",coordinates:[[[[-35.26394,-43.68272],[-53.39656,-59.87088],[-22.31757,-59.85974],[-35.26394,-43.68272]]]]}},{type:"Feature",properties:{iso1A2:"GT",iso1A3:"GTM",iso1N3:"320",wikidata:"Q774",nameEn:"Guatemala",groups:["013","003","419","019"],callingCodes:["502"]},geometry:{type:"MultiPolygon",coordinates:[[[[-89.14985,17.81563],[-90.98678,17.81655],[-90.99199,17.25192],[-91.43809,17.25373],[-91.04436,16.92175],[-90.69064,16.70697],[-90.61212,16.49832],[-90.40499,16.40524],[-90.44567,16.07573],[-91.73182,16.07371],[-92.20983,15.26077],[-92.0621,15.07406],[-92.1454,14.98143],[-92.1423,14.88647],[-92.18161,14.84147],[-92.1454,14.6804],[-92.2261,14.53423],[-92.37213,14.39277],[-90.55276,12.8866],[-90.11344,13.73679],[-90.10505,13.85104],[-89.88937,14.0396],[-89.81807,14.07073],[-89.76103,14.02923],[-89.73251,14.04133],[-89.75569,14.07073],[-89.70756,14.1537],[-89.61844,14.21937],[-89.52397,14.22628],[-89.50614,14.26084],[-89.58814,14.33165],[-89.57441,14.41637],[-89.39028,14.44561],[-89.34776,14.43013],[-89.35189,14.47553],[-89.23719,14.58046],[-89.15653,14.57802],[-89.13132,14.71582],[-89.23467,14.85596],[-89.15149,14.97775],[-89.18048,14.99967],[-89.15149,15.07392],[-88.97343,15.14039],[-88.32467,15.63665],[-88.31459,15.66942],[-88.24022,15.69247],[-88.22552,15.72294],[-88.20359,16.03858],[-88.40779,16.09624],[-88.95358,15.88698],[-89.02415,15.9063],[-89.17418,15.90898],[-89.22683,15.88619],[-89.15025,17.04813],[-89.14985,17.81563]]]]}},{type:"Feature",properties:{iso1A2:"GU",iso1A3:"GUM",iso1N3:"316",wikidata:"Q16635",nameEn:"Guam",country:"US",groups:["057","009"],roadSpeedUnit:"mph",callingCodes:["1 671"]},geometry:{type:"MultiPolygon",coordinates:[[[[146.25931,13.85876],[143.82485,13.92273],[144.61642,12.82462],[146.25931,13.85876]]]]}},{type:"Feature",properties:{iso1A2:"GW",iso1A3:"GNB",iso1N3:"624",wikidata:"Q1007",nameEn:"Guinea-Bissau",groups:["011","202","002"],callingCodes:["245"]},geometry:{type:"MultiPolygon",coordinates:[[[[-14.31513,11.60713],[-14.26623,11.67486],[-14.09799,11.63649],[-13.7039,11.70195],[-13.7039,12.00869],[-13.94589,12.16869],[-13.92745,12.24077],[-13.70851,12.24978],[-13.64168,12.42764],[-13.65089,12.49515],[-13.7039,12.60313],[-13.70523,12.68013],[-15.17582,12.6847],[-15.67302,12.42974],[-16.20591,12.46157],[-16.38191,12.36449],[-16.70562,12.34803],[-17.4623,11.92379],[-15.96748,10.162],[-15.07174,10.89557],[-14.95993,10.99244],[-14.77786,11.36323],[-14.66677,11.51188],[-14.51173,11.49708],[-14.31513,11.60713]]]]}},{type:"Feature",properties:{iso1A2:"GY",iso1A3:"GUY",iso1N3:"328",wikidata:"Q734",nameEn:"Guyana",groups:["005","419","019"],driveSide:"left",callingCodes:["592"]},geometry:{type:"MultiPolygon",coordinates:[[[[-56.84822,6.73257],[-59.54058,8.6862],[-59.98508,8.53046],[-59.85562,8.35213],[-59.80661,8.28906],[-59.83156,8.23261],[-59.97059,8.20791],[-60.02407,8.04557],[-60.38056,7.8302],[-60.51959,7.83373],[-60.64793,7.56877],[-60.71923,7.55817],[-60.59802,7.33194],[-60.63367,7.25061],[-60.54098,7.14804],[-60.44116,7.20817],[-60.28074,7.1162],[-60.39419,6.94847],[-60.54873,6.8631],[-61.13632,6.70922],[-61.20762,6.58174],[-61.15058,6.19558],[-61.4041,5.95304],[-60.73204,5.20931],[-60.32352,5.21299],[-60.20944,5.28754],[-59.98129,5.07097],[-60.04189,4.69801],[-60.15953,4.53456],[-59.78878,4.45637],[-59.69361,4.34069],[-59.73353,4.20399],[-59.51963,3.91951],[-59.86899,3.57089],[-59.79769,3.37162],[-59.99733,2.92312],[-59.91177,2.36759],[-59.7264,2.27497],[-59.74066,1.87596],[-59.25583,1.40559],[-58.92072,1.31293],[-58.84229,1.17749],[-58.53571,1.29154],[-58.4858,1.48399],[-58.33887,1.58014],[-58.01873,1.51966],[-57.97336,1.64566],[-57.77281,1.73344],[-57.55743,1.69605],[-57.35073,1.98327],[-57.23981,1.95808],[-57.09109,2.01854],[-57.07092,1.95304],[-56.7659,1.89509],[-56.47045,1.95135],[-56.55439,2.02003],[-56.70519,2.02964],[-57.35891,3.32121],[-58.0307,3.95513],[-57.8699,4.89394],[-57.37442,5.0208],[-57.22536,5.15605],[-57.31629,5.33714],[-56.84822,6.73257]]]]}},{type:"Feature",properties:{iso1A2:"HK",iso1A3:"HKG",iso1N3:"344",wikidata:"Q8646",nameEn:"Hong Kong",country:"CN",groups:["030","142"],driveSide:"left",callingCodes:["852"]},geometry:{type:"MultiPolygon",coordinates:[[[[113.92195,22.13873],[114.50148,22.15017],[114.44998,22.55977],[114.25154,22.55977],[114.22888,22.5436],[114.22185,22.55343],[114.20655,22.55706],[114.18338,22.55444],[114.17247,22.55944],[114.1597,22.56041],[114.15123,22.55163],[114.1482,22.54091],[114.13823,22.54319],[114.12665,22.54003],[114.11656,22.53415],[114.11181,22.52878],[114.1034,22.5352],[114.09692,22.53435],[114.09048,22.53716],[114.08606,22.53276],[114.07817,22.52997],[114.07267,22.51855],[114.06272,22.51617],[114.05729,22.51104],[114.05438,22.5026],[114.03113,22.5065],[113.86771,22.42972],[113.81621,22.2163],[113.83338,22.1826],[113.92195,22.13873]]]]}},{type:"Feature",properties:{iso1A2:"HM",iso1A3:"HMD",iso1N3:"334",wikidata:"Q131198",nameEn:"Heard Island and McDonald Islands",country:"AU",groups:["053","009"],driveSide:"left"},geometry:{type:"MultiPolygon",coordinates:[[[[71.08716,-53.87687],[75.44182,-53.99822],[72.87012,-51.48322],[71.08716,-53.87687]]]]}},{type:"Feature",properties:{iso1A2:"HN",iso1A3:"HND",iso1N3:"340",wikidata:"Q783",nameEn:"Honduras",groups:["013","003","419","019"],callingCodes:["504"]},geometry:{type:"MultiPolygon",coordinates:[[[[-83.86109,17.73736],[-88.20359,16.03858],[-88.22552,15.72294],[-88.24022,15.69247],[-88.31459,15.66942],[-88.32467,15.63665],[-88.97343,15.14039],[-89.15149,15.07392],[-89.18048,14.99967],[-89.15149,14.97775],[-89.23467,14.85596],[-89.13132,14.71582],[-89.15653,14.57802],[-89.23719,14.58046],[-89.35189,14.47553],[-89.34776,14.43013],[-89.04187,14.33644],[-88.94608,14.20207],[-88.85785,14.17763],[-88.815,14.11652],[-88.73182,14.10919],[-88.70661,14.04317],[-88.49738,13.97224],[-88.48982,13.86458],[-88.25791,13.91108],[-88.23018,13.99915],[-88.07641,13.98447],[-88.00331,13.86948],[-87.7966,13.91353],[-87.68821,13.80829],[-87.73106,13.75443],[-87.78148,13.52906],[-87.71657,13.50577],[-87.72115,13.46083],[-87.73841,13.44169],[-87.77354,13.45767],[-87.83467,13.44655],[-87.84675,13.41078],[-87.80177,13.35689],[-87.73714,13.32715],[-87.69751,13.25228],[-87.55124,13.12523],[-87.37107,12.98646],[-87.06306,13.00892],[-87.03785,12.98682],[-86.93197,13.05313],[-86.93383,13.18677],[-86.87066,13.30641],[-86.71267,13.30348],[-86.76812,13.79605],[-86.35219,13.77157],[-86.14801,14.04317],[-86.00685,14.08474],[-86.03458,13.99181],[-85.75477,13.8499],[-85.73964,13.9698],[-85.45762,14.11304],[-85.32149,14.2562],[-85.18602,14.24929],[-85.1575,14.53934],[-84.90082,14.80489],[-84.82596,14.82212],[-84.70119,14.68078],[-84.48373,14.63249],[-84.10584,14.76353],[-83.89551,14.76697],[-83.62101,14.89448],[-83.49268,15.01158],[-83.13724,15.00002],[-83.04763,15.03256],[-82.06974,14.49418],[-81.58685,18.0025],[-83.86109,17.73736]]]]}},{type:"Feature",properties:{iso1A2:"HR",iso1A3:"HRV",iso1N3:"191",wikidata:"Q224",nameEn:"Croatia",groups:["EU","039","150"],callingCodes:["385"]},geometry:{type:"MultiPolygon",coordinates:[[[[17.6444,42.88641],[17.5392,42.92787],[17.70879,42.97223],[17.64268,43.08595],[17.46986,43.16559],[17.286,43.33065],[17.25579,43.40353],[17.29699,43.44542],[17.24411,43.49376],[17.15828,43.49376],[17.00585,43.58037],[16.80736,43.76011],[16.75316,43.77157],[16.70922,43.84887],[16.55472,43.95326],[16.50528,44.0244],[16.43629,44.02826],[16.43662,44.07523],[16.36864,44.08263],[16.18688,44.27012],[16.21346,44.35231],[16.12969,44.38275],[16.16814,44.40679],[16.10566,44.52586],[16.03012,44.55572],[16.00884,44.58605],[16.05828,44.61538],[15.89348,44.74964],[15.8255,44.71501],[15.72584,44.82334],[15.79472,44.8455],[15.76096,44.87045],[15.74723,44.96818],[15.78568,44.97401],[15.74585,45.0638],[15.78842,45.11519],[15.76371,45.16508],[15.83512,45.22459],[15.98412,45.23088],[16.12153,45.09616],[16.29036,44.99732],[16.35404,45.00241],[16.35863,45.03529],[16.3749,45.05206],[16.38219,45.05139],[16.38309,45.05955],[16.40023,45.1147],[16.4634,45.14522],[16.49155,45.21153],[16.52982,45.22713],[16.5501,45.2212],[16.56559,45.22307],[16.60194,45.23042],[16.64962,45.20714],[16.74845,45.20393],[16.78219,45.19002],[16.81137,45.18434],[16.83804,45.18951],[16.92405,45.27607],[16.9385,45.22742],[17.0415,45.20759],[17.18438,45.14764],[17.24325,45.146],[17.25131,45.14957],[17.26815,45.18444],[17.32092,45.16246],[17.33573,45.14521],[17.41229,45.13335],[17.4498,45.16119],[17.45615,45.12523],[17.47589,45.12656],[17.51469,45.10791],[17.59104,45.10816],[17.66571,45.13408],[17.84826,45.04489],[17.87148,45.04645],[17.93706,45.08016],[17.97336,45.12245],[17.97834,45.13831],[17.99479,45.14958],[18.01594,45.15163],[18.03121,45.12632],[18.1624,45.07654],[18.24387,45.13699],[18.32077,45.1021],[18.41896,45.11083],[18.47939,45.05871],[18.65723,45.07544],[18.78357,44.97741],[18.80661,44.93561],[18.76369,44.93707],[18.76347,44.90669],[18.8704,44.85097],[19.01994,44.85493],[18.98957,44.90645],[19.02871,44.92541],[19.06853,44.89915],[19.15573,44.95409],[19.05205,44.97692],[19.1011,45.01191],[19.07952,45.14668],[19.14063,45.12972],[19.19144,45.17863],[19.43589,45.17137],[19.41941,45.23475],[19.28208,45.23813],[19.10774,45.29547],[18.97446,45.37528],[18.99918,45.49333],[19.08364,45.48804],[19.07471,45.53086],[18.94562,45.53712],[18.88776,45.57253],[18.96691,45.66731],[18.90305,45.71863],[18.85783,45.85493],[18.81394,45.91329],[18.80211,45.87995],[18.6792,45.92057],[18.57483,45.80772],[18.44368,45.73972],[18.12439,45.78905],[18.08869,45.76511],[17.99805,45.79671],[17.87377,45.78522],[17.66545,45.84207],[17.56821,45.93728],[17.35672,45.95209],[17.14592,46.16697],[16.8903,46.28122],[16.8541,46.36255],[16.7154,46.39523],[16.6639,46.45203],[16.59527,46.47524],[16.52604,46.47831],[16.5007,46.49644],[16.44036,46.5171],[16.38771,46.53608],[16.37193,46.55008],[16.29793,46.5121],[16.26733,46.51505],[16.26759,46.50566],[16.23961,46.49653],[16.25124,46.48067],[16.27398,46.42875],[16.27329,46.41467],[16.30162,46.40437],[16.30233,46.37837],[16.18824,46.38282],[16.14859,46.40547],[16.05281,46.39141],[16.05065,46.3833],[16.07314,46.36458],[16.07616,46.3463],[15.97965,46.30652],[15.79284,46.25811],[15.78817,46.21719],[15.75479,46.20336],[15.75436,46.21969],[15.67395,46.22478],[15.6434,46.21396],[15.64904,46.19229],[15.59909,46.14761],[15.6083,46.11992],[15.62317,46.09103],[15.72977,46.04682],[15.71246,46.01196],[15.70327,46.00015],[15.70636,45.92116],[15.67967,45.90455],[15.68383,45.88867],[15.68232,45.86819],[15.70411,45.8465],[15.66662,45.84085],[15.64185,45.82915],[15.57952,45.84953],[15.52234,45.82195],[15.47325,45.8253],[15.47531,45.79802],[15.40836,45.79491],[15.25423,45.72275],[15.30872,45.69014],[15.34919,45.71623],[15.4057,45.64727],[15.38952,45.63682],[15.34214,45.64702],[15.34695,45.63382],[15.31027,45.6303],[15.27747,45.60504],[15.29837,45.5841],[15.30249,45.53224],[15.38188,45.48752],[15.33051,45.45258],[15.27758,45.46678],[15.16862,45.42309],[15.05187,45.49079],[15.02385,45.48533],[14.92266,45.52788],[14.90554,45.47769],[14.81992,45.45913],[14.80124,45.49515],[14.71718,45.53442],[14.68605,45.53006],[14.69694,45.57366],[14.59576,45.62812],[14.60977,45.66403],[14.57397,45.67165],[14.53816,45.6205],[14.5008,45.60852],[14.49769,45.54424],[14.36693,45.48642],[14.32487,45.47142],[14.27681,45.4902],[14.26611,45.48239],[14.24239,45.50607],[14.22371,45.50388],[14.20348,45.46896],[14.07116,45.48752],[14.00578,45.52352],[13.96063,45.50825],[13.99488,45.47551],[13.97309,45.45258],[13.90771,45.45149],[13.88124,45.42637],[13.81742,45.43729],[13.7785,45.46787],[13.67398,45.4436],[13.62902,45.45898],[13.56979,45.4895],[13.45644,45.59464],[13.05142,45.33128],[13.12821,44.48877],[16.15283,42.18525],[18.45131,42.21682],[18.54128,42.39171],[18.52152,42.42302],[18.43588,42.48556],[18.44307,42.51077],[18.43735,42.55921],[18.36197,42.61423],[18.24318,42.6112],[17.88201,42.83668],[17.80854,42.9182],[17.7948,42.89556],[17.68151,42.92725],[17.6444,42.88641]]]]}},{type:"Feature",properties:{iso1A2:"HT",iso1A3:"HTI",iso1N3:"332",wikidata:"Q790",nameEn:"Haiti",aliases:["RH"],groups:["029","003","419","019"],callingCodes:["509"]},geometry:{type:"MultiPolygon",coordinates:[[[[-71.71885,18.78423],[-71.72624,18.87802],[-71.77766,18.95007],[-71.88102,18.95007],[-71.74088,19.0437],[-71.71088,19.08353],[-71.69938,19.10916],[-71.65337,19.11759],[-71.62642,19.21212],[-71.73229,19.26686],[-71.77766,19.33823],[-71.69448,19.37866],[-71.6802,19.45008],[-71.71268,19.53374],[-71.71449,19.55364],[-71.7429,19.58445],[-71.75865,19.70231],[-71.77419,19.73128],[-72.38946,20.27111],[-73.37289,20.43199],[-74.7289,18.71009],[-74.76465,18.06252],[-72.29523,17.48026],[-71.75671,18.03456],[-71.73783,18.07177],[-71.74994,18.11115],[-71.75465,18.14405],[-71.78271,18.18302],[-71.69952,18.34101],[-71.90875,18.45821],[-71.88102,18.50125],[-72.00201,18.62312],[-71.95412,18.64939],[-71.82556,18.62551],[-71.71885,18.78423]]]]}},{type:"Feature",properties:{iso1A2:"HU",iso1A3:"HUN",iso1N3:"348",wikidata:"Q28",nameEn:"Hungary",groups:["EU","151","150"],callingCodes:["36"]},geometry:{type:"MultiPolygon",coordinates:[[[[21.72525,48.34628],[21.67134,48.3989],[21.6068,48.50365],[21.44063,48.58456],[21.11516,48.49546],[20.83248,48.5824],[20.5215,48.53336],[20.29943,48.26104],[20.24312,48.2784],[19.92452,48.1283],[19.63338,48.25006],[19.52489,48.19791],[19.47957,48.09437],[19.28182,48.08336],[19.23924,48.0595],[19.01952,48.07052],[18.82176,48.04206],[18.76134,47.97499],[18.76821,47.87469],[18.8506,47.82308],[18.74074,47.8157],[18.66521,47.76772],[18.56496,47.76588],[18.29305,47.73541],[18.02938,47.75665],[17.71215,47.7548],[17.23699,48.02094],[17.16001,48.00636],[17.09786,47.97336],[17.11022,47.92461],[17.08275,47.87719],[17.00997,47.86245],[17.07039,47.81129],[17.05048,47.79377],[17.08893,47.70928],[16.87538,47.68895],[16.86509,47.72268],[16.82938,47.68432],[16.7511,47.67878],[16.72089,47.73469],[16.65679,47.74197],[16.61183,47.76171],[16.54779,47.75074],[16.53514,47.73837],[16.55129,47.72268],[16.4222,47.66537],[16.58699,47.61772],[16.64193,47.63114],[16.71059,47.52692],[16.64821,47.50155],[16.6718,47.46139],[16.57152,47.40868],[16.52414,47.41007],[16.49908,47.39416],[16.45104,47.41181],[16.47782,47.25918],[16.44142,47.25079],[16.43663,47.21127],[16.41739,47.20649],[16.42801,47.18422],[16.4523,47.18812],[16.46442,47.16845],[16.44932,47.14418],[16.52863,47.13974],[16.46134,47.09395],[16.52176,47.05747],[16.43936,47.03548],[16.51369,47.00084],[16.28202,47.00159],[16.27594,46.9643],[16.22403,46.939],[16.19904,46.94134],[16.10983,46.867],[16.14365,46.8547],[16.15711,46.85434],[16.21892,46.86961],[16.2365,46.87775],[16.2941,46.87137],[16.34547,46.83836],[16.3408,46.80641],[16.31303,46.79838],[16.30966,46.7787],[16.37816,46.69975],[16.42641,46.69228],[16.41863,46.66238],[16.38594,46.6549],[16.39217,46.63673],[16.50139,46.56684],[16.52885,46.53303],[16.52604,46.5051],[16.59527,46.47524],[16.6639,46.45203],[16.7154,46.39523],[16.8541,46.36255],[16.8903,46.28122],[17.14592,46.16697],[17.35672,45.95209],[17.56821,45.93728],[17.66545,45.84207],[17.87377,45.78522],[17.99805,45.79671],[18.08869,45.76511],[18.12439,45.78905],[18.44368,45.73972],[18.57483,45.80772],[18.6792,45.92057],[18.80211,45.87995],[18.81394,45.91329],[18.99712,45.93537],[19.01284,45.96529],[19.0791,45.96458],[19.10388,46.04015],[19.14543,45.9998],[19.28826,45.99694],[19.52473,46.1171],[19.56113,46.16824],[19.66007,46.19005],[19.81491,46.1313],[19.93508,46.17553],[20.01816,46.17696],[20.03533,46.14509],[20.09713,46.17315],[20.26068,46.12332],[20.28324,46.1438],[20.35573,46.16629],[20.45377,46.14405],[20.49718,46.18721],[20.63863,46.12728],[20.76085,46.21002],[20.74574,46.25467],[20.86797,46.28884],[21.06572,46.24897],[21.16872,46.30118],[21.28061,46.44941],[21.26929,46.4993],[21.33214,46.63035],[21.43926,46.65109],[21.5151,46.72147],[21.48935,46.7577],[21.52028,46.84118],[21.59307,46.86935],[21.59581,46.91628],[21.68645,46.99595],[21.648,47.03902],[21.78395,47.11104],[21.94463,47.38046],[22.01055,47.37767],[22.03389,47.42508],[22.00917,47.50492],[22.31816,47.76126],[22.41979,47.7391],[22.46559,47.76583],[22.67247,47.7871],[22.76617,47.8417],[22.77991,47.87211],[22.89849,47.95851],[22.84276,47.98602],[22.87847,48.04665],[22.81804,48.11363],[22.73427,48.12005],[22.66835,48.09162],[22.58733,48.10813],[22.59007,48.15121],[22.49806,48.25189],[22.38133,48.23726],[22.2083,48.42534],[22.14689,48.4005],[21.83339,48.36242],[21.8279,48.33321],[21.72525,48.34628]]]]}},{type:"Feature",properties:{iso1A2:"IC",wikidata:"Q5813",nameEn:"Canary Islands",country:"ES",groups:["EU","039","150"],isoStatus:"excRes",callingCodes:["34"]},geometry:{type:"MultiPolygon",coordinates:[[[[-15.92339,29.50503],[-25.3475,27.87574],[-14.43883,27.02969],[-9.94494,32.97138],[-15.92339,29.50503]]]]}},{type:"Feature",properties:{iso1A2:"ID",iso1A3:"IDN",iso1N3:"360",wikidata:"Q252",nameEn:"Indonesia",aliases:["RI"],groups:["035","142"],driveSide:"left",callingCodes:["62"]},geometry:{type:"MultiPolygon",coordinates:[[[[141.02352,0.08993],[128.97621,3.08804],[126.69413,6.02692],[124.97752,4.82064],[118.41402,3.99509],[118.07935,4.15511],[117.89538,4.16637],[117.67641,4.16535],[117.47313,4.18857],[117.25801,4.35108],[115.90217,4.37708],[115.58276,3.93499],[115.53713,3.14776],[115.11343,2.82879],[115.1721,2.49671],[114.80706,2.21665],[114.80706,1.92351],[114.57892,1.5],[114.03788,1.44787],[113.64677,1.23933],[113.01448,1.42832],[113.021,1.57819],[112.48648,1.56516],[112.2127,1.44135],[112.15679,1.17004],[111.94553,1.12016],[111.82846,0.99349],[111.55434,0.97864],[111.22979,1.08326],[110.62374,0.873],[110.49182,0.88088],[110.35354,0.98869],[109.66397,1.60425],[109.66397,1.79972],[109.57923,1.80624],[109.53794,1.91771],[109.62558,1.99182],[109.64506,2.08014],[109.71058,2.32059],[108.10426,5.42408],[105.01437,3.24936],[104.56723,1.44271],[104.34728,1.33529],[104.12282,1.27714],[104.03085,1.26954],[103.74084,1.12902],[103.66049,1.18825],[103.56591,1.19719],[103.03657,1.30383],[96.11174,6.69841],[74.28481,-3.17525],[122.14954,-11.52517],[125.68138,-9.85176],[125.09025,-9.46406],[124.97892,-9.19281],[125.04044,-9.17093],[125.09434,-9.19669],[125.18907,-9.16434],[125.18632,-9.03142],[125.11764,-8.96359],[124.97742,-9.08128],[124.94011,-8.85617],[124.46701,-9.13002],[124.45971,-9.30263],[124.38554,-9.3582],[124.35258,-9.43002],[124.3535,-9.48493],[124.28115,-9.50453],[124.28115,-9.42189],[124.21247,-9.36904],[124.14517,-9.42324],[124.10539,-9.41206],[124.04286,-9.34243],[124.04628,-9.22671],[124.33472,-9.11416],[124.92337,-8.75859],[125.31127,-8.22976],[125.65946,-8.06136],[125.87691,-8.31789],[127.42116,-8.22471],[127.55165,-9.05052],[140.88922,-9.34945],[141.00782,-9.1242],[141.01763,-6.90181],[140.85295,-6.72996],[140.99813,-6.3233],[141.02352,0.08993]]]]}},{type:"Feature",properties:{iso1A2:"IE",iso1A3:"IRL",iso1N3:"372",wikidata:"Q27",nameEn:"Ireland",groups:["EU","154","150"],driveSide:"left",callingCodes:["353"]},geometry:{type:"MultiPolygon",coordinates:[[[[-6.26218,54.09785],[-6.29003,54.11278],[-6.32694,54.09337],[-6.36279,54.11248],[-6.36605,54.07234],[-6.47849,54.06947],[-6.62842,54.03503],[-6.66264,54.0666],[-6.6382,54.17071],[-6.70175,54.20218],[-6.74575,54.18788],[-6.81583,54.22791],[-6.85179,54.29176],[-6.87775,54.34682],[-7.02034,54.4212],[-7.19145,54.31296],[-7.14908,54.22732],[-7.25012,54.20063],[-7.26316,54.13863],[-7.29493,54.12013],[-7.29687,54.1354],[-7.28017,54.16714],[-7.29157,54.17191],[-7.34005,54.14698],[-7.30553,54.11869],[-7.32834,54.11475],[-7.44567,54.1539],[-7.4799,54.12239],[-7.55812,54.12239],[-7.69501,54.20731],[-7.81397,54.20159],[-7.8596,54.21779],[-7.87101,54.29299],[-8.04555,54.36292],[-8.179,54.46763],[-8.04538,54.48941],[-7.99812,54.54427],[-7.8596,54.53671],[-7.70315,54.62077],[-7.93293,54.66603],[-7.83352,54.73854],[-7.75041,54.7103],[-7.64449,54.75265],[-7.54671,54.74606],[-7.54508,54.79401],[-7.47626,54.83084],[-7.4473,54.87003],[-7.44404,54.9403],[-7.40004,54.94498],[-7.4033,55.00391],[-7.34464,55.04688],[-7.2471,55.06933],[-6.9734,55.19878],[-6.71944,55.27952],[-6.79943,55.54107],[-7.93366,55.84142],[-22.01468,48.19557],[-5.79914,52.03902],[-5.37267,53.63269],[-5.83481,53.87749],[-6.26218,54.09785]]]]}},{type:"Feature",properties:{iso1A2:"IL",iso1A3:"ISR",iso1N3:"376",wikidata:"Q801",nameEn:"Israel",groups:["145","142"],callingCodes:["972"]},geometry:{type:"MultiPolygon",coordinates:[[[[34.052,31.46619],[34.29262,31.70393],[34.48681,31.59711],[34.56797,31.54197],[34.48892,31.48365],[34.40077,31.40926],[34.36505,31.36404],[34.37381,31.30598],[34.36523,31.28963],[34.29417,31.24194],[34.26742,31.21998],[34.92298,29.45305],[34.97718,29.54294],[34.98207,29.58147],[35.02147,29.66343],[35.14108,30.07374],[35.19183,30.34636],[35.16218,30.43535],[35.19595,30.50297],[35.21379,30.60401],[35.29311,30.71365],[35.33456,30.81224],[35.33984,30.8802],[35.41371,30.95565],[35.43658,31.12444],[35.40316,31.25535],[35.47672,31.49578],[35.39675,31.49572],[35.22921,31.37445],[35.13033,31.3551],[35.02459,31.35979],[34.92571,31.34337],[34.88932,31.37093],[34.87833,31.39321],[34.89756,31.43891],[34.93258,31.47816],[34.94356,31.50743],[34.9415,31.55601],[34.95249,31.59813],[35.00879,31.65426],[35.08226,31.69107],[35.10782,31.71594],[35.11895,31.71454],[35.12933,31.7325],[35.13931,31.73012],[35.15119,31.73634],[35.15474,31.73352],[35.16478,31.73242],[35.18023,31.72067],[35.20538,31.72388],[35.21937,31.71578],[35.22392,31.71899],[35.23972,31.70896],[35.24315,31.71244],[35.2438,31.7201],[35.24981,31.72543],[35.25182,31.73945],[35.26319,31.74846],[35.25225,31.7678],[35.26058,31.79064],[35.25573,31.81362],[35.26404,31.82567],[35.251,31.83085],[35.25753,31.8387],[35.24816,31.8458],[35.2304,31.84222],[35.2249,31.85433],[35.22817,31.8638],[35.22567,31.86745],[35.22294,31.87889],[35.22014,31.88264],[35.2136,31.88241],[35.21276,31.88153],[35.21016,31.88237],[35.20945,31.8815],[35.20791,31.8821],[35.20673,31.88151],[35.20381,31.86716],[35.21128,31.863],[35.216,31.83894],[35.21469,31.81835],[35.19461,31.82687],[35.18169,31.82542],[35.18603,31.80901],[35.14174,31.81325],[35.07677,31.85627],[35.05617,31.85685],[35.01978,31.82944],[34.9724,31.83352],[34.99712,31.85569],[35.03489,31.85919],[35.03978,31.89276],[35.03489,31.92448],[35.00124,31.93264],[34.98682,31.96935],[35.00261,32.027],[34.9863,32.09551],[34.99437,32.10962],[34.98507,32.12606],[34.99039,32.14626],[34.96009,32.17503],[34.95703,32.19522],[34.98885,32.20758],[35.01841,32.23981],[35.02939,32.2671],[35.01119,32.28684],[35.01772,32.33863],[35.04243,32.35008],[35.05142,32.3667],[35.0421,32.38242],[35.05311,32.4024],[35.05423,32.41754],[35.07059,32.4585],[35.08564,32.46948],[35.09236,32.47614],[35.10024,32.47856],[35.10882,32.4757],[35.15937,32.50466],[35.2244,32.55289],[35.25049,32.52453],[35.29306,32.50947],[35.30685,32.51024],[35.35212,32.52047],[35.40224,32.50136],[35.42034,32.46009],[35.41598,32.45593],[35.41048,32.43706],[35.42078,32.41562],[35.55807,32.38674],[35.55494,32.42687],[35.57485,32.48669],[35.56614,32.64393],[35.59813,32.65159],[35.61669,32.67999],[35.66527,32.681],[35.68467,32.70715],[35.75983,32.74803],[35.78745,32.77938],[35.83758,32.82817],[35.84021,32.8725],[35.87012,32.91976],[35.89298,32.9456],[35.87188,32.98028],[35.84802,33.1031],[35.81911,33.11077],[35.81911,33.1336],[35.84285,33.16673],[35.83846,33.19397],[35.81647,33.2028],[35.81295,33.24841],[35.77513,33.27342],[35.813,33.3172],[35.77477,33.33609],[35.62019,33.27278],[35.62283,33.24226],[35.58502,33.26653],[35.58326,33.28381],[35.56523,33.28969],[35.55555,33.25844],[35.54544,33.25513],[35.54808,33.236],[35.5362,33.23196],[35.54228,33.19865],[35.52573,33.11921],[35.50335,33.114],[35.50272,33.09056],[35.448,33.09264],[35.43059,33.06659],[35.35223,33.05617],[35.31429,33.10515],[35.1924,33.08743],[35.10645,33.09318],[34.78515,33.20368],[33.62659,31.82938],[34.052,31.46619]]]]}},{type:"Feature",properties:{iso1A2:"IM",iso1A3:"IMN",iso1N3:"833",wikidata:"Q9676",nameEn:"Isle of Man",country:"GB",groups:["154","150"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["44 01624","44 07624","44 07524","44 07924"]},geometry:{type:"MultiPolygon",coordinates:[[[[-3.64906,54.12723],[-4.1819,54.57861],[-5.83481,53.87749],[-5.37267,53.63269],[-3.64906,54.12723]]]]}},{type:"Feature",properties:{iso1A2:"IN",iso1A3:"IND",iso1N3:"356",wikidata:"Q668",nameEn:"India",groups:["034","142"],driveSide:"left",callingCodes:["91"]},geometry:{type:"MultiPolygon",coordinates:[[[[78.11664,35.48022],[77.80532,35.52058],[77.70232,35.46244],[77.44277,35.46132],[76.96624,35.5932],[76.84539,35.67356],[76.77323,35.66062],[76.75475,35.52617],[76.85088,35.39754],[76.93465,35.39866],[77.11796,35.05419],[76.99251,34.93349],[76.87193,34.96906],[76.74514,34.92488],[76.74377,34.84039],[76.67648,34.76371],[76.47186,34.78965],[76.15463,34.6429],[76.04614,34.67566],[75.75438,34.51827],[75.38009,34.55021],[75.01479,34.64629],[74.6663,34.703],[74.58083,34.77386],[74.31239,34.79626],[74.12897,34.70073],[73.96423,34.68244],[73.93401,34.63386],[73.93951,34.57169],[73.89419,34.54568],[73.88732,34.48911],[73.74999,34.3781],[73.74862,34.34183],[73.8475,34.32935],[73.90517,34.35317],[73.98208,34.2522],[73.90677,34.10504],[73.88732,34.05105],[73.91341,34.01235],[74.21554,34.03853],[74.25262,34.01577],[74.26086,33.92237],[74.14001,33.83002],[74.05898,33.82089],[74.00891,33.75437],[73.96423,33.73071],[73.98968,33.66155],[73.97367,33.64061],[74.03576,33.56718],[74.10115,33.56392],[74.18121,33.4745],[74.17983,33.3679],[74.08782,33.26232],[74.01366,33.25199],[74.02144,33.18908],[74.15374,33.13477],[74.17571,33.07495],[74.31854,33.02891],[74.34875,32.97823],[74.31227,32.92795],[74.41467,32.90563],[74.45312,32.77755],[74.6289,32.75561],[74.64675,32.82604],[74.7113,32.84219],[74.65345,32.71225],[74.69542,32.66792],[74.64424,32.60985],[74.65251,32.56416],[74.67431,32.56676],[74.68362,32.49298],[74.84725,32.49075],[74.97634,32.45367],[75.03265,32.49538],[75.28259,32.36556],[75.38046,32.26836],[75.25649,32.10187],[75.00793,32.03786],[74.9269,32.0658],[74.86236,32.04485],[74.79919,31.95983],[74.58907,31.87824],[74.47771,31.72227],[74.57498,31.60382],[74.61517,31.55698],[74.59319,31.50197],[74.64713,31.45605],[74.59773,31.4136],[74.53223,31.30321],[74.51629,31.13829],[74.56023,31.08303],[74.60281,31.10419],[74.60006,31.13711],[74.6852,31.12771],[74.67971,31.05479],[74.5616,31.04153],[73.88993,30.36305],[73.95736,30.28466],[73.97225,30.19829],[73.80299,30.06969],[73.58665,30.01848],[73.3962,29.94707],[73.28094,29.56646],[73.05886,29.1878],[73.01337,29.16422],[72.94272,29.02487],[72.40402,28.78283],[72.29495,28.66367],[72.20329,28.3869],[71.9244,28.11555],[71.89921,27.96035],[70.79054,27.68423],[70.60927,28.02178],[70.37307,28.01208],[70.12502,27.8057],[70.03136,27.56627],[69.58519,27.18109],[69.50904,26.74892],[69.88555,26.56836],[70.05584,26.60398],[70.17532,26.55362],[70.17532,26.24118],[70.08193,26.08094],[70.0985,25.93238],[70.2687,25.71156],[70.37444,25.67443],[70.53649,25.68928],[70.60378,25.71898],[70.67382,25.68186],[70.66695,25.39314],[70.89148,25.15064],[70.94002,24.92843],[71.09405,24.69017],[70.97594,24.60904],[71.00341,24.46038],[71.12838,24.42662],[71.04461,24.34657],[70.94985,24.3791],[70.85784,24.30903],[70.88393,24.27398],[70.71502,24.23517],[70.57906,24.27774],[70.5667,24.43787],[70.11712,24.30915],[70.03428,24.172],[69.73335,24.17007],[69.59579,24.29777],[69.29778,24.28712],[69.19341,24.25646],[69.07806,24.29777],[68.97781,24.26021],[68.90914,24.33156],[68.7416,24.31904],[68.74643,23.97027],[68.39339,23.96838],[68.20763,23.85849],[68.11329,23.53945],[72.15131,7.6285],[78.52781,7.63099],[79.50447,8.91876],[79.42124,9.80115],[80.48418,10.20786],[94.53911,5.99016],[94.6371,13.81803],[92.61042,13.76986],[89.13606,21.42955],[89.13927,21.60785],[89.03553,21.77397],[89.07114,22.15335],[88.9367,22.58527],[88.94614,22.66941],[88.9151,22.75228],[88.96713,22.83346],[88.87063,22.95235],[88.88327,23.03885],[88.86377,23.08759],[88.99148,23.21134],[88.71133,23.2492],[88.79254,23.46028],[88.79351,23.50535],[88.74841,23.47361],[88.56507,23.64044],[88.58087,23.87105],[88.66189,23.87607],[88.73743,23.91751],[88.6976,24.14703],[88.74841,24.1959],[88.68801,24.31464],[88.50934,24.32474],[88.12296,24.51301],[88.08786,24.63232],[88.00683,24.66477],[88.15515,24.85806],[88.14004,24.93529],[88.21832,24.96642],[88.27325,24.88796],[88.33917,24.86803],[88.46277,25.07468],[88.44766,25.20149],[88.94067,25.18534],[89.00463,25.26583],[89.01105,25.30303],[88.85278,25.34679],[88.81296,25.51546],[88.677,25.46959],[88.4559,25.59227],[88.45103,25.66245],[88.242,25.80811],[88.13138,25.78773],[88.08804,25.91334],[88.16581,26.0238],[88.1844,26.14417],[88.34757,26.22216],[88.35153,26.29123],[88.51649,26.35923],[88.48749,26.45855],[88.36938,26.48683],[88.35153,26.45241],[88.33093,26.48929],[88.41196,26.63837],[88.4298,26.54489],[88.62144,26.46783],[88.69485,26.38353],[88.67837,26.26291],[88.78961,26.31093],[88.85004,26.23211],[89.05328,26.2469],[88.91321,26.37984],[88.92357,26.40711],[88.95612,26.4564],[89.08899,26.38845],[89.15869,26.13708],[89.35953,26.0077],[89.53515,26.00382],[89.57101,25.9682],[89.63968,26.22595],[89.70201,26.15138],[89.73581,26.15818],[89.77865,26.08387],[89.77728,26.04254],[89.86592,25.93115],[89.80585,25.82489],[89.84388,25.70042],[89.86129,25.61714],[89.81208,25.37244],[89.84086,25.31854],[89.83371,25.29548],[89.87629,25.28337],[89.90478,25.31038],[90.1155,25.22686],[90.40034,25.1534],[90.65042,25.17788],[90.87427,25.15799],[91.25517,25.20677],[91.63648,25.12846],[92.0316,25.1834],[92.33957,25.07593],[92.39147,25.01471],[92.49887,24.88796],[92.38626,24.86055],[92.25854,24.9191],[92.15796,24.54435],[92.11662,24.38997],[91.96603,24.3799],[91.89258,24.14674],[91.82596,24.22345],[91.76004,24.23848],[91.73257,24.14703],[91.65292,24.22095],[91.63782,24.1132],[91.55542,24.08687],[91.37414,24.10693],[91.35741,23.99072],[91.29587,24.0041],[91.22308,23.89616],[91.25192,23.83463],[91.15579,23.6599],[91.28293,23.37538],[91.36453,23.06612],[91.40848,23.07117],[91.4035,23.27522],[91.46615,23.2328],[91.54993,23.01051],[91.61571,22.93929],[91.7324,23.00043],[91.81634,23.08001],[91.76417,23.26619],[91.84789,23.42235],[91.95642,23.47361],[91.95093,23.73284],[92.04706,23.64229],[92.15417,23.73409],[92.26541,23.70392],[92.38214,23.28705],[92.37665,22.9435],[92.5181,22.71441],[92.60029,22.1522],[92.56616,22.13554],[92.60949,21.97638],[92.67532,22.03547],[92.70416,22.16017],[92.86208,22.05456],[92.89504,21.95143],[92.93899,22.02656],[92.99804,21.98964],[92.99255,22.05965],[93.04885,22.20595],[93.15734,22.18687],[93.14224,22.24535],[93.19991,22.25425],[93.18206,22.43716],[93.13537,22.45873],[93.11477,22.54374],[93.134,22.59573],[93.09417,22.69459],[93.134,22.92498],[93.12988,23.05772],[93.2878,23.00464],[93.38478,23.13698],[93.36862,23.35426],[93.38781,23.36139],[93.39981,23.38828],[93.38805,23.4728],[93.43475,23.68299],[93.3908,23.7622],[93.3908,23.92925],[93.36059,23.93176],[93.32351,24.04468],[93.34735,24.10151],[93.41415,24.07854],[93.46633,23.97067],[93.50616,23.94432],[93.62871,24.00922],[93.75952,24.0003],[93.80279,23.92549],[93.92089,23.95812],[94.14081,23.83333],[94.30215,24.23752],[94.32362,24.27692],[94.45279,24.56656],[94.50729,24.59281],[94.5526,24.70764],[94.60204,24.70889],[94.73937,25.00545],[94.74212,25.13606],[94.57458,25.20318],[94.68032,25.47003],[94.80117,25.49359],[95.18556,26.07338],[95.11428,26.1019],[95.12801,26.38397],[95.05798,26.45408],[95.23513,26.68499],[95.30339,26.65372],[95.437,26.7083],[95.81603,27.01335],[95.93002,27.04149],[96.04949,27.19428],[96.15591,27.24572],[96.40779,27.29818],[96.55761,27.29928],[96.73888,27.36638],[96.88445,27.25046],[96.85287,27.2065],[96.89132,27.17474],[97.14675,27.09041],[97.17422,27.14052],[96.91431,27.45752],[96.90112,27.62149],[97.29919,27.92233],[97.35824,27.87256],[97.38845,28.01329],[97.35412,28.06663],[97.31292,28.06784],[97.34547,28.21385],[97.1289,28.3619],[96.98882,28.32564],[96.88445,28.39452],[96.85561,28.4875],[96.6455,28.61657],[96.48895,28.42955],[96.40929,28.51526],[96.61391,28.72742],[96.3626,29.10607],[96.20467,29.02325],[96.18682,29.11087],[96.31316,29.18643],[96.05361,29.38167],[95.84899,29.31464],[95.75149,29.32063],[95.72086,29.20797],[95.50842,29.13487],[95.41091,29.13007],[95.3038,29.13847],[95.26122,29.07727],[95.2214,29.10727],[95.11291,29.09527],[95.0978,29.14446],[94.81353,29.17804],[94.69318,29.31739],[94.2752,29.11687],[94.35897,29.01965],[93.72797,28.68821],[93.44621,28.67189],[93.18069,28.50319],[93.14635,28.37035],[92.93075,28.25671],[92.67486,28.15018],[92.65472,28.07632],[92.73025,28.05814],[92.7275,27.98662],[92.42538,27.80092],[92.32101,27.79363],[92.27432,27.89077],[91.87057,27.7195],[91.84722,27.76325],[91.6469,27.76358],[91.55819,27.6144],[91.65007,27.48287],[92.01132,27.47352],[92.12019,27.27829],[92.04702,27.26861],[92.03457,27.07334],[92.11863,26.893],[92.05523,26.8692],[91.83181,26.87318],[91.50067,26.79223],[90.67715,26.77215],[90.48504,26.8594],[90.39271,26.90704],[90.30402,26.85098],[90.04535,26.72422],[89.86124,26.73307],[89.63369,26.74402],[89.42349,26.83727],[89.3901,26.84225],[89.38319,26.85963],[89.37913,26.86224],[89.1926,26.81329],[89.12825,26.81661],[89.09554,26.89089],[88.95807,26.92668],[88.92301,26.99286],[88.8714,26.97488],[88.86984,27.10937],[88.74219,27.144],[88.91901,27.32483],[88.82981,27.38814],[88.77517,27.45415],[88.88091,27.85192],[88.83559,28.01936],[88.63235,28.12356],[88.54858,28.06057],[88.25332,27.9478],[88.1278,27.95417],[88.13378,27.88015],[88.1973,27.85067],[88.19107,27.79285],[88.04008,27.49223],[88.07277,27.43007],[88.01646,27.21612],[88.01587,27.21388],[87.9887,27.11045],[88.11719,26.98758],[88.13422,26.98705],[88.12302,26.95324],[88.19107,26.75516],[88.1659,26.68177],[88.16452,26.64111],[88.09963,26.54195],[88.09414,26.43732],[88.00895,26.36029],[87.90115,26.44923],[87.89085,26.48565],[87.84193,26.43663],[87.7918,26.46737],[87.76004,26.40711],[87.67893,26.43501],[87.66803,26.40294],[87.59175,26.38342],[87.55274,26.40596],[87.51571,26.43106],[87.46566,26.44058],[87.37314,26.40815],[87.34568,26.34787],[87.26568,26.37294],[87.26587,26.40592],[87.24682,26.4143],[87.18863,26.40558],[87.14751,26.40542],[87.09147,26.45039],[87.0707,26.58571],[87.04691,26.58685],[87.01559,26.53228],[86.95912,26.52076],[86.94543,26.52076],[86.82898,26.43919],[86.76797,26.45892],[86.74025,26.42386],[86.69124,26.45169],[86.62686,26.46891],[86.61313,26.48658],[86.57073,26.49825],[86.54258,26.53819],[86.49726,26.54218],[86.31564,26.61925],[86.26235,26.61886],[86.22513,26.58863],[86.13596,26.60651],[86.02729,26.66756],[85.8492,26.56667],[85.85126,26.60866],[85.83126,26.61134],[85.76907,26.63076],[85.72315,26.67471],[85.73483,26.79613],[85.66239,26.84822],[85.61621,26.86721],[85.59461,26.85161],[85.5757,26.85955],[85.56471,26.84133],[85.47752,26.79292],[85.34302,26.74954],[85.21159,26.75933],[85.18046,26.80519],[85.19291,26.86909],[85.15883,26.86966],[85.02635,26.85381],[85.05592,26.88991],[85.00536,26.89523],[84.97186,26.9149],[84.96687,26.95599],[84.85754,26.98984],[84.82913,27.01989],[84.793,26.9968],[84.64496,27.04669],[84.69166,27.21294],[84.62161,27.33885],[84.29315,27.39],[84.25735,27.44941],[84.21376,27.45218],[84.10791,27.52399],[84.02229,27.43836],[83.93306,27.44939],[83.86182,27.4241],[83.85595,27.35797],[83.61288,27.47013],[83.39495,27.4798],[83.38872,27.39276],[83.35136,27.33885],[83.29999,27.32778],[83.2673,27.36235],[83.27197,27.38309],[83.19413,27.45632],[82.94938,27.46036],[82.93261,27.50328],[82.74119,27.49838],[82.70378,27.72122],[82.46405,27.6716],[82.06554,27.92222],[81.97214,27.93322],[81.91223,27.84995],[81.47867,28.08303],[81.48179,28.12148],[81.38683,28.17638],[81.32923,28.13521],[81.19847,28.36284],[81.08507,28.38346],[80.89648,28.47237],[80.55142,28.69182],[80.50575,28.6706],[80.52443,28.54897],[80.44504,28.63098],[80.37188,28.63371],[80.12125,28.82346],[80.06957,28.82763],[80.05743,28.91479],[80.18085,29.13649],[80.23178,29.11626],[80.26602,29.13938],[80.24112,29.21414],[80.28626,29.20327],[80.31428,29.30784],[80.24322,29.44299],[80.37939,29.57098],[80.41858,29.63581],[80.38428,29.68513],[80.36803,29.73865],[80.41554,29.79451],[80.43458,29.80466],[80.48997,29.79566],[80.56247,29.86661],[80.56957,29.88176],[80.60226,29.95732],[80.67076,29.95732],[80.8778,30.13384],[80.93695,30.18229],[81.03953,30.20059],[80.83343,30.32023],[80.54504,30.44936],[80.20721,30.58541],[79.93255,30.88288],[79.59884,30.93943],[79.22805,31.34963],[79.14016,31.43403],[79.01931,31.42817],[78.77898,31.31209],[78.71032,31.50197],[78.84516,31.60631],[78.69933,31.78723],[78.78036,31.99478],[78.74404,32.00384],[78.68754,32.10256],[78.49609,32.2762],[78.4645,32.45367],[78.38897,32.53938],[78.73916,32.69438],[78.7831,32.46873],[78.96713,32.33655],[78.99322,32.37948],[79.0979,32.38051],[79.13174,32.47766],[79.26768,32.53277],[79.46562,32.69668],[79.14016,33.02545],[79.15252,33.17156],[78.73636,33.56521],[78.67599,33.66445],[78.77349,33.73871],[78.73367,34.01121],[78.65657,34.03195],[78.66225,34.08858],[78.91769,34.15452],[78.99802,34.3027],[79.05364,34.32482],[78.74465,34.45174],[78.56475,34.50835],[78.54964,34.57283],[78.27781,34.61484],[78.18435,34.7998],[78.22692,34.88771],[78.00033,35.23954],[78.03466,35.3785],[78.11664,35.48022]]]]}},{type:"Feature",properties:{iso1A2:"IO",iso1A3:"IOT",iso1N3:"086",wikidata:"Q43448",nameEn:"British Indian Ocean Territory",country:"GB",groups:["014","202","002"],callingCodes:["246"]},geometry:{type:"MultiPolygon",coordinates:[[[[70.64754,-4.95745],[70.67958,-8.2663],[73.70488,-4.92492],[70.64754,-4.95745]]]]}},{type:"Feature",properties:{iso1A2:"IQ",iso1A3:"IRQ",iso1N3:"368",wikidata:"Q796",nameEn:"Iraq",groups:["145","142"],callingCodes:["964"]},geometry:{type:"MultiPolygon",coordinates:[[[[42.78887,37.38615],[42.56725,37.14878],[42.35724,37.10998],[42.36697,37.0627],[41.81736,36.58782],[41.40058,36.52502],[41.28864,36.35368],[41.2564,36.06012],[41.37027,35.84095],[41.38184,35.62502],[41.26569,35.42708],[41.21654,35.1508],[41.2345,34.80049],[41.12388,34.65742],[40.97676,34.39788],[40.64314,34.31604],[38.79171,33.37328],[39.08202,32.50304],[38.98762,32.47694],[39.04251,32.30203],[39.26157,32.35555],[39.29903,32.23259],[40.01521,32.05667],[42.97601,30.72204],[42.97796,30.48295],[44.72255,29.19736],[46.42415,29.05947],[46.5527,29.10283],[46.89695,29.50584],[47.15166,30.01044],[47.37192,30.10421],[47.7095,30.10453],[48.01114,29.98906],[48.06782,30.02906],[48.17332,30.02448],[48.40479,29.85763],[48.59531,29.66815],[48.83867,29.78572],[48.61441,29.93675],[48.51011,29.96238],[48.44785,30.00148],[48.4494,30.04456],[48.43384,30.08233],[48.38869,30.11062],[48.38714,30.13485],[48.41671,30.17254],[48.41117,30.19846],[48.26393,30.3408],[48.24385,30.33846],[48.21279,30.31644],[48.19425,30.32796],[48.18321,30.39703],[48.14585,30.44133],[48.02443,30.4789],[48.03221,30.9967],[47.68219,31.00004],[47.6804,31.39086],[47.86337,31.78422],[47.64771,32.07666],[47.52474,32.15972],[47.57144,32.20583],[47.37529,32.47808],[47.17218,32.45393],[46.46788,32.91992],[46.32298,32.9731],[46.17198,32.95612],[46.09103,32.98354],[46.15175,33.07229],[46.03966,33.09577],[46.05367,33.13097],[46.11905,33.11924],[46.20623,33.20395],[45.99919,33.5082],[45.86687,33.49263],[45.96183,33.55751],[45.89801,33.63661],[45.77814,33.60938],[45.50261,33.94968],[45.42789,33.9458],[45.41077,33.97421],[45.47264,34.03099],[45.56176,34.15088],[45.58667,34.30147],[45.53552,34.35148],[45.49171,34.3439],[45.46697,34.38221],[45.43879,34.45949],[45.51883,34.47692],[45.53219,34.60441],[45.59074,34.55558],[45.60224,34.55057],[45.73923,34.54416],[45.70031,34.69277],[45.65672,34.7222],[45.68284,34.76624],[45.70031,34.82322],[45.73641,34.83975],[45.79682,34.85133],[45.78904,34.91135],[45.86532,34.89858],[45.89477,34.95805],[45.87864,35.03441],[45.92173,35.0465],[45.92203,35.09538],[45.93108,35.08148],[45.94756,35.09188],[46.06508,35.03699],[46.07747,35.0838],[46.11763,35.07551],[46.19116,35.11097],[46.15642,35.1268],[46.16229,35.16984],[46.19738,35.18536],[46.18457,35.22561],[46.11367,35.23729],[46.15474,35.2883],[46.13152,35.32548],[46.05358,35.38568],[45.98453,35.49848],[46.01518,35.52012],[45.97584,35.58132],[46.03028,35.57416],[46.01307,35.59756],[46.0165,35.61501],[45.99452,35.63574],[46.0117,35.65059],[46.01631,35.69139],[46.23736,35.71414],[46.34166,35.78363],[46.32921,35.82655],[46.17198,35.8013],[46.08325,35.8581],[45.94711,35.82218],[45.89784,35.83708],[45.81442,35.82107],[45.76145,35.79898],[45.6645,35.92872],[45.60018,35.96069],[45.55245,35.99943],[45.46594,36.00042],[45.38275,35.97156],[45.33916,35.99424],[45.37652,36.06222],[45.37312,36.09917],[45.32235,36.17383],[45.30038,36.27769],[45.26261,36.3001],[45.27394,36.35846],[45.23953,36.43257],[45.11811,36.40751],[45.00759,36.5402],[45.06985,36.62645],[45.06985,36.6814],[45.01537,36.75128],[44.84725,36.77622],[44.83479,36.81362],[44.90173,36.86096],[44.91199,36.91468],[44.89862,37.01897],[44.81611,37.04383],[44.75229,37.11958],[44.78319,37.1431],[44.76698,37.16162],[44.63179,37.19229],[44.42631,37.05825],[44.38117,37.05825],[44.35315,37.04955],[44.35937,37.02843],[44.30645,36.97373],[44.25975,36.98119],[44.18503,37.09551],[44.22239,37.15756],[44.27998,37.16501],[44.2613,37.25055],[44.13521,37.32486],[44.02002,37.33229],[43.90949,37.22453],[43.84878,37.22205],[43.82699,37.19477],[43.8052,37.22825],[43.7009,37.23692],[43.63085,37.21957],[43.56702,37.25675],[43.50787,37.24436],[43.33508,37.33105],[43.30083,37.30629],[43.11403,37.37436],[42.93705,37.32015],[42.78887,37.38615]]]]}},{type:"Feature",properties:{iso1A2:"IR",iso1A3:"IRN",iso1N3:"364",wikidata:"Q794",nameEn:"Iran",groups:["034","142"],callingCodes:["98"]},geometry:{type:"MultiPolygon",coordinates:[[[[44.96746,39.42998],[44.88916,39.59653],[44.81043,39.62677],[44.71806,39.71124],[44.65422,39.72163],[44.6137,39.78393],[44.47298,39.68788],[44.48111,39.61579],[44.41849,39.56659],[44.42832,39.4131],[44.37921,39.4131],[44.29818,39.378],[44.22452,39.4169],[44.03667,39.39223],[44.1043,39.19842],[44.20946,39.13975],[44.18863,38.93881],[44.30322,38.81581],[44.26155,38.71427],[44.28065,38.6465],[44.32058,38.62752],[44.3207,38.49799],[44.3119,38.37887],[44.38309,38.36117],[44.44386,38.38295],[44.50115,38.33939],[44.42476,38.25763],[44.22509,37.88859],[44.3883,37.85433],[44.45948,37.77065],[44.55498,37.783],[44.62096,37.71985],[44.56887,37.6429],[44.61401,37.60165],[44.58449,37.45018],[44.81021,37.2915],[44.75986,37.21549],[44.7868,37.16644],[44.78319,37.1431],[44.75229,37.11958],[44.81611,37.04383],[44.89862,37.01897],[44.91199,36.91468],[44.90173,36.86096],[44.83479,36.81362],[44.84725,36.77622],[45.01537,36.75128],[45.06985,36.6814],[45.06985,36.62645],[45.00759,36.5402],[45.11811,36.40751],[45.23953,36.43257],[45.27394,36.35846],[45.26261,36.3001],[45.30038,36.27769],[45.32235,36.17383],[45.37312,36.09917],[45.37652,36.06222],[45.33916,35.99424],[45.38275,35.97156],[45.46594,36.00042],[45.55245,35.99943],[45.60018,35.96069],[45.6645,35.92872],[45.76145,35.79898],[45.81442,35.82107],[45.89784,35.83708],[45.94711,35.82218],[46.08325,35.8581],[46.17198,35.8013],[46.32921,35.82655],[46.34166,35.78363],[46.23736,35.71414],[46.01631,35.69139],[46.0117,35.65059],[45.99452,35.63574],[46.0165,35.61501],[46.01307,35.59756],[46.03028,35.57416],[45.97584,35.58132],[46.01518,35.52012],[45.98453,35.49848],[46.05358,35.38568],[46.13152,35.32548],[46.15474,35.2883],[46.11367,35.23729],[46.18457,35.22561],[46.19738,35.18536],[46.16229,35.16984],[46.15642,35.1268],[46.19116,35.11097],[46.11763,35.07551],[46.07747,35.0838],[46.06508,35.03699],[45.94756,35.09188],[45.93108,35.08148],[45.92203,35.09538],[45.92173,35.0465],[45.87864,35.03441],[45.89477,34.95805],[45.86532,34.89858],[45.78904,34.91135],[45.79682,34.85133],[45.73641,34.83975],[45.70031,34.82322],[45.68284,34.76624],[45.65672,34.7222],[45.70031,34.69277],[45.73923,34.54416],[45.60224,34.55057],[45.59074,34.55558],[45.53219,34.60441],[45.51883,34.47692],[45.43879,34.45949],[45.46697,34.38221],[45.49171,34.3439],[45.53552,34.35148],[45.58667,34.30147],[45.56176,34.15088],[45.47264,34.03099],[45.41077,33.97421],[45.42789,33.9458],[45.50261,33.94968],[45.77814,33.60938],[45.89801,33.63661],[45.96183,33.55751],[45.86687,33.49263],[45.99919,33.5082],[46.20623,33.20395],[46.11905,33.11924],[46.05367,33.13097],[46.03966,33.09577],[46.15175,33.07229],[46.09103,32.98354],[46.17198,32.95612],[46.32298,32.9731],[46.46788,32.91992],[47.17218,32.45393],[47.37529,32.47808],[47.57144,32.20583],[47.52474,32.15972],[47.64771,32.07666],[47.86337,31.78422],[47.6804,31.39086],[47.68219,31.00004],[48.03221,30.9967],[48.02443,30.4789],[48.14585,30.44133],[48.18321,30.39703],[48.19425,30.32796],[48.21279,30.31644],[48.24385,30.33846],[48.26393,30.3408],[48.41117,30.19846],[48.41671,30.17254],[48.38714,30.13485],[48.38869,30.11062],[48.43384,30.08233],[48.4494,30.04456],[48.44785,30.00148],[48.51011,29.96238],[48.61441,29.93675],[48.83867,29.78572],[49.98877,27.87827],[50.37726,27.89227],[54.39838,25.68383],[55.14145,25.62624],[55.81777,26.18798],[56.2644,26.58649],[56.68954,26.76645],[56.79239,26.41236],[56.82555,25.7713],[56.86325,25.03856],[61.5251,24.57287],[61.57592,25.0492],[61.6433,25.27541],[61.683,25.66638],[61.83968,25.7538],[61.83831,26.07249],[61.89391,26.26251],[62.05117,26.31647],[62.21304,26.26601],[62.31484,26.528],[62.77352,26.64099],[63.1889,26.65072],[63.18688,26.83844],[63.25005,26.84212],[63.25005,27.08692],[63.32283,27.14437],[63.19649,27.25674],[62.80604,27.22412],[62.79684,27.34381],[62.84905,27.47627],[62.7638,28.02992],[62.79412,28.28108],[62.59499,28.24842],[62.40259,28.42703],[61.93581,28.55284],[61.65978,28.77937],[61.53765,29.00507],[61.31508,29.38903],[60.87231,29.86514],[61.80829,30.84224],[61.78268,30.92724],[61.8335,30.97669],[61.83257,31.0452],[61.80957,31.12576],[61.80569,31.16167],[61.70929,31.37391],[60.84541,31.49561],[60.86191,32.22565],[60.56485,33.12944],[60.88908,33.50219],[60.91133,33.55596],[60.69573,33.56054],[60.57762,33.59772],[60.5485,33.73422],[60.5838,33.80793],[60.50209,34.13992],[60.66502,34.31539],[60.91321,34.30411],[60.72316,34.52857],[60.99922,34.63064],[61.00197,34.70631],[61.06926,34.82139],[61.12831,35.09938],[61.0991,35.27845],[61.18187,35.30249],[61.27371,35.61482],[61.22719,35.67038],[61.26152,35.80749],[61.22444,35.92879],[61.12007,35.95992],[61.22719,36.12759],[61.1393,36.38782],[61.18187,36.55348],[61.14516,36.64644],[60.34767,36.63214],[60.00768,37.04102],[59.74678,37.12499],[59.55178,37.13594],[59.39385,37.34257],[59.39797,37.47892],[59.33507,37.53146],[59.22905,37.51161],[58.9338,37.67374],[58.6921,37.64548],[58.5479,37.70526],[58.47786,37.6433],[58.39959,37.63134],[58.22999,37.6856],[58.21399,37.77281],[57.79534,37.89299],[57.35042,37.98546],[57.37236,38.09321],[57.21169,38.28965],[57.03453,38.18717],[56.73928,38.27887],[56.62255,38.24005],[56.43303,38.26054],[56.32454,38.18502],[56.33278,38.08132],[55.97847,38.08024],[55.76561,38.12238],[55.44152,38.08564],[55.13412,37.94705],[54.851,37.75739],[54.77684,37.62264],[54.81804,37.61285],[54.77822,37.51597],[54.67247,37.43532],[54.58664,37.45809],[54.36211,37.34912],[54.24565,37.32047],[53.89734,37.3464],[48.88288,38.43975],[48.84969,38.45015],[48.81072,38.44853],[48.78979,38.45026],[48.70001,38.40564],[48.62217,38.40198],[48.58793,38.45076],[48.45084,38.61013],[48.3146,38.59958],[48.24773,38.71883],[48.02581,38.82705],[48.01409,38.90333],[48.07734,38.91616],[48.08627,38.94434],[48.28437,38.97186],[48.33884,39.03022],[48.31239,39.09278],[48.15361,39.19419],[48.12404,39.25208],[48.15984,39.30028],[48.37385,39.37584],[48.34264,39.42935],[47.98977,39.70999],[47.84774,39.66285],[47.50099,39.49615],[47.38978,39.45999],[47.31301,39.37492],[47.05927,39.24846],[47.05771,39.20143],[46.95341,39.13505],[46.92539,39.16644],[46.83822,39.13143],[46.75752,39.03231],[46.53497,38.86548],[46.34059,38.92076],[46.20601,38.85262],[46.14785,38.84206],[46.06766,38.87861],[46.00228,38.87376],[45.94624,38.89072],[45.90266,38.87739],[45.83883,38.90768],[45.65172,38.95199],[45.6155,38.94304],[45.6131,38.964],[45.44966,38.99243],[45.44811,39.04927],[45.40452,39.07224],[45.40148,39.09007],[45.30489,39.18333],[45.16168,39.21952],[45.08751,39.35052],[45.05932,39.36435],[44.96746,39.42998]]]]}},{type:"Feature",properties:{iso1A2:"IS",iso1A3:"ISL",iso1N3:"352",wikidata:"Q189",nameEn:"Iceland",groups:["154","150"],callingCodes:["354"]},geometry:{type:"MultiPolygon",coordinates:[[[[-33.15676,62.62995],[-8.25539,63.0423],[-15.70914,69.67442],[-33.15676,62.62995]]]]}},{type:"Feature",properties:{iso1A2:"IT",iso1A3:"ITA",iso1N3:"380",wikidata:"Q38",nameEn:"Italy",groups:["EU","039","150"],callingCodes:["39"]},geometry:{type:"MultiPolygon",coordinates:[[[[8.95861,45.96485],[8.97604,45.96151],[8.97741,45.98317],[8.96668,45.98436],[8.95861,45.96485]]],[[[7.63035,43.57419],[9.56115,43.20816],[10.09675,41.44089],[7.60802,41.05927],[7.89009,38.19924],[11.2718,37.6713],[12.13667,34.20326],[14.02721,36.53141],[17.67657,35.68918],[18.83516,40.36999],[16.15283,42.18525],[13.12821,44.48877],[13.05142,45.33128],[13.45644,45.59464],[13.6076,45.64761],[13.7198,45.59352],[13.74587,45.59811],[13.78445,45.5825],[13.84106,45.58185],[13.86771,45.59898],[13.8695,45.60835],[13.9191,45.6322],[13.87933,45.65207],[13.83422,45.68703],[13.83332,45.70855],[13.8235,45.7176],[13.66986,45.79955],[13.59784,45.8072],[13.58858,45.83503],[13.57563,45.8425],[13.58644,45.88173],[13.59565,45.89446],[13.60857,45.89907],[13.61931,45.91782],[13.63815,45.93607],[13.6329,45.94894],[13.64307,45.98326],[13.63458,45.98947],[13.62074,45.98388],[13.58903,45.99009],[13.56759,45.96991],[13.52963,45.96588],[13.50104,45.98078],[13.47474,46.00546],[13.49702,46.01832],[13.50998,46.04498],[13.49568,46.04839],[13.50104,46.05986],[13.57072,46.09022],[13.64053,46.13587],[13.66472,46.17392],[13.64451,46.18966],[13.56682,46.18703],[13.56114,46.2054],[13.47587,46.22725],[13.42218,46.20758],[13.37671,46.29668],[13.44808,46.33507],[13.43418,46.35992],[13.47019,46.3621],[13.5763,46.40915],[13.5763,46.42613],[13.59777,46.44137],[13.68684,46.43881],[13.7148,46.5222],[13.64088,46.53438],[13.27627,46.56059],[12.94445,46.60401],[12.59992,46.6595],[12.38708,46.71529],[12.27591,46.88651],[12.2006,46.88854],[12.11675,47.01241],[12.21781,47.03996],[12.19254,47.09331],[11.74789,46.98484],[11.50739,47.00644],[11.33355,46.99862],[11.10618,46.92966],[11.00764,46.76896],[10.72974,46.78972],[10.75753,46.82258],[10.66405,46.87614],[10.54783,46.84505],[10.47197,46.85698],[10.38659,46.67847],[10.40475,46.63671],[10.44686,46.64162],[10.49375,46.62049],[10.46136,46.53164],[10.25309,46.57432],[10.23674,46.63484],[10.10307,46.61003],[10.03715,46.44479],[10.165,46.41051],[10.10506,46.3372],[10.17862,46.25626],[10.14439,46.22992],[10.07055,46.21668],[9.95249,46.38045],[9.73086,46.35071],[9.71273,46.29266],[9.57015,46.2958],[9.46117,46.37481],[9.45936,46.50873],[9.40487,46.46621],[9.36128,46.5081],[9.28136,46.49685],[9.25502,46.43743],[9.29226,46.32717],[9.24503,46.23616],[9.01618,46.04928],[8.99257,45.9698],[9.09065,45.89906],[9.06642,45.8761],[9.04546,45.84968],[9.04059,45.8464],[9.03505,45.83976],[9.03793,45.83548],[9.03279,45.82865],[9.0298,45.82127],[9.00324,45.82055],[8.99663,45.83466],[8.9621,45.83707],[8.94737,45.84285],[8.91129,45.8388],[8.93504,45.86245],[8.94372,45.86587],[8.93649,45.86775],[8.88904,45.95465],[8.86688,45.96135],[8.85121,45.97239],[8.8319,45.9879],[8.79362,45.99207],[8.78585,45.98973],[8.79414,46.00913],[8.85617,46.0748],[8.80778,46.10085],[8.75697,46.10395],[8.62242,46.12112],[8.45032,46.26869],[8.46317,46.43712],[8.42464,46.46367],[8.30648,46.41587],[8.31162,46.38044],[8.08814,46.26692],[8.16866,46.17817],[8.11383,46.11577],[8.02906,46.10331],[7.98881,45.99867],[7.9049,45.99945],[7.85949,45.91485],[7.56343,45.97421],[7.10685,45.85653],[7.04151,45.92435],[6.95315,45.85163],[6.80785,45.83265],[6.80785,45.71864],[6.98948,45.63869],[7.00037,45.509],[7.18019,45.40071],[7.10572,45.32924],[7.13115,45.25386],[7.07074,45.21228],[6.96706,45.20841],[6.85144,45.13226],[6.7697,45.16044],[6.62803,45.11175],[6.66981,45.02324],[6.74791,45.01939],[6.74519,44.93661],[6.75518,44.89915],[6.90774,44.84322],[6.93499,44.8664],[7.02217,44.82519],[7.00401,44.78782],[7.07484,44.68073],[7.00582,44.69364],[6.95133,44.66264],[6.96042,44.62129],[6.85507,44.53072],[6.86233,44.49834],[6.94504,44.43112],[6.88784,44.42043],[6.89171,44.36637],[6.98221,44.28289],[7.00764,44.23736],[7.16929,44.20352],[7.27827,44.1462],[7.34547,44.14359],[7.36364,44.11882],[7.62155,44.14881],[7.63245,44.17877],[7.68694,44.17487],[7.66878,44.12795],[7.72508,44.07578],[7.6597,44.03009],[7.66848,43.99943],[7.65266,43.9763],[7.60771,43.95772],[7.56858,43.94506],[7.56075,43.89932],[7.51162,43.88301],[7.49355,43.86551],[7.50423,43.84345],[7.53006,43.78405],[7.63035,43.57419]],[[12.45181,41.90056],[12.44834,41.90095],[12.44582,41.90194],[12.44815,41.90326],[12.44984,41.90545],[12.45091,41.90625],[12.45543,41.90738],[12.45561,41.90629],[12.45762,41.9058],[12.45755,41.9033],[12.45826,41.90281],[12.45834,41.90174],[12.4577,41.90115],[12.45691,41.90125],[12.45626,41.90172],[12.45435,41.90143],[12.45446,41.90028],[12.45181,41.90056]],[[12.45648,43.89369],[12.44184,43.90498],[12.41641,43.89991],[12.40935,43.9024],[12.41233,43.90956],[12.40733,43.92379],[12.41551,43.92984],[12.41165,43.93769],[12.40506,43.94325],[12.40415,43.95485],[12.41414,43.95273],[12.42005,43.9578],[12.43662,43.95698],[12.44684,43.96597],[12.46205,43.97463],[12.47853,43.98052],[12.49406,43.98492],[12.50678,43.99113],[12.51463,43.99122],[12.5154,43.98508],[12.51064,43.98165],[12.51109,43.97201],[12.50622,43.97131],[12.50875,43.96198],[12.50655,43.95796],[12.51427,43.94897],[12.51553,43.94096],[12.50496,43.93017],[12.50269,43.92363],[12.49724,43.92248],[12.49247,43.91774],[12.49429,43.90973],[12.48771,43.89706],[12.45648,43.89369]]]]}},{type:"Feature",properties:{iso1A2:"JE",iso1A3:"JEY",iso1N3:"832",wikidata:"Q785",nameEn:"Jersey",country:"GB",groups:["830","154","150"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["44 01534"]},geometry:{type:"MultiPolygon",coordinates:[[[[-2.00491,48.86706],[-1.83944,49.23037],[-2.09454,49.46288],[-2.65349,49.15373],[-2.00491,48.86706]]]]}},{type:"Feature",properties:{iso1A2:"JM",iso1A3:"JAM",iso1N3:"388",wikidata:"Q766",nameEn:"Jamaica",aliases:["JA"],groups:["029","003","419","019"],driveSide:"left",callingCodes:["1 876","1 658"]},geometry:{type:"MultiPolygon",coordinates:[[[[-75.50728,17.08879],[-76.34192,18.86145],[-78.75694,18.78765],[-78.34606,16.57862],[-75.50728,17.08879]]]]}},{type:"Feature",properties:{iso1A2:"JO",iso1A3:"JOR",iso1N3:"400",wikidata:"Q810",nameEn:"Jordan",groups:["145","142"],callingCodes:["962"]},geometry:{type:"MultiPolygon",coordinates:[[[[39.04251,32.30203],[38.98762,32.47694],[39.08202,32.50304],[38.79171,33.37328],[36.83946,32.31293],[36.40959,32.37908],[36.23948,32.50108],[36.20875,32.49529],[36.20379,32.52751],[36.08074,32.51463],[36.02239,32.65911],[35.96633,32.66237],[35.93307,32.71966],[35.88405,32.71321],[35.75983,32.74803],[35.68467,32.70715],[35.66527,32.681],[35.61669,32.67999],[35.59813,32.65159],[35.56614,32.64393],[35.57485,32.48669],[35.55494,32.42687],[35.55807,32.38674],[35.57111,32.21877],[35.52012,32.04076],[35.54375,31.96587],[35.52758,31.9131],[35.55941,31.76535],[35.47672,31.49578],[35.40316,31.25535],[35.43658,31.12444],[35.41371,30.95565],[35.33984,30.8802],[35.33456,30.81224],[35.29311,30.71365],[35.21379,30.60401],[35.19595,30.50297],[35.16218,30.43535],[35.19183,30.34636],[35.14108,30.07374],[35.02147,29.66343],[34.98207,29.58147],[34.97718,29.54294],[34.92298,29.45305],[34.88293,29.37455],[34.95987,29.35727],[36.07081,29.18469],[36.50005,29.49696],[36.75083,29.86903],[37.4971,29.99949],[37.66395,30.33245],[37.99354,30.49998],[36.99791,31.50081],[38.99233,31.99721],[39.29903,32.23259],[39.26157,32.35555],[39.04251,32.30203]]]]}},{type:"Feature",properties:{iso1A2:"JP",iso1A3:"JPN",iso1N3:"392",wikidata:"Q17",nameEn:"Japan",groups:["030","142"],driveSide:"left",callingCodes:["81"]},geometry:{type:"MultiPolygon",coordinates:[[[[145.82361,43.38904],[145.23667,43.76813],[145.82343,44.571],[140.9182,45.92937],[133.61399,37.41],[129.2669,34.87122],[122.26612,25.98197],[123.92912,17.8782],[155.16731,23.60141],[145.82361,43.38904]]]]}},{type:"Feature",properties:{iso1A2:"KE",iso1A3:"KEN",iso1N3:"404",wikidata:"Q114",nameEn:"Kenya",groups:["014","202","002"],driveSide:"left",callingCodes:["254"]},geometry:{type:"MultiPolygon",coordinates:[[[[35.9419,4.61933],[35.51424,4.61643],[35.42366,4.76969],[35.47843,4.91872],[35.30992,4.90402],[35.34151,5.02364],[34.47601,4.72162],[33.9873,4.23316],[34.06046,4.15235],[34.15429,3.80464],[34.45815,3.67385],[34.44922,3.51627],[34.39112,3.48802],[34.41794,3.44342],[34.40006,3.37949],[34.45815,3.18319],[34.56242,3.11478],[34.60114,2.93034],[34.65774,2.8753],[34.73967,2.85447],[34.78137,2.76223],[34.77244,2.70272],[34.95267,2.47209],[34.90947,2.42447],[34.98692,1.97348],[34.9899,1.6668],[34.92734,1.56109],[34.87819,1.5596],[34.7918,1.36752],[34.82606,1.30944],[34.82606,1.26626],[34.80223,1.22754],[34.67562,1.21265],[34.58029,1.14712],[34.57427,1.09868],[34.52369,1.10692],[34.43349,0.85254],[34.40041,0.80266],[34.31516,0.75693],[34.27345,0.63182],[34.20196,0.62289],[34.13493,0.58118],[34.11408,0.48884],[34.08727,0.44713],[34.10067,0.36372],[33.90936,0.10581],[33.98449,-0.13079],[33.9264,-0.54188],[33.93107,-0.99298],[34.02286,-1.00779],[34.03084,-1.05101],[34.0824,-1.02264],[37.67199,-3.06222],[37.71745,-3.304],[37.5903,-3.42735],[37.63099,-3.50723],[37.75036,-3.54243],[37.81321,-3.69179],[39.21631,-4.67835],[39.44306,-4.93877],[39.62121,-4.68136],[41.75542,-1.85308],[41.56362,-1.66375],[41.56,-1.59812],[41.00099,-0.83068],[40.98767,2.82959],[41.31368,3.14314],[41.89488,3.97375],[41.1754,3.94079],[40.77498,4.27683],[39.86043,3.86974],[39.76808,3.67058],[39.58339,3.47434],[39.55132,3.39634],[39.51551,3.40895],[39.49444,3.45521],[39.19954,3.47834],[39.07736,3.5267],[38.91938,3.51198],[38.52336,3.62551],[38.45812,3.60445],[38.14168,3.62487],[37.07724,4.33503],[36.84474,4.44518],[36.03924,4.44406],[35.95449,4.53244],[35.9419,4.61933]]]]}},{type:"Feature",properties:{iso1A2:"KG",iso1A3:"KGZ",iso1N3:"417",wikidata:"Q813",nameEn:"Kyrgyzstan",groups:["143","142"],callingCodes:["996"]},geometry:{type:"MultiPolygon",coordinates:[[[[74.88756,42.98612],[74.75,42.99029],[74.70331,43.02519],[74.64615,43.05881],[74.57491,43.13702],[74.22489,43.24657],[73.55634,43.03071],[73.50992,42.82356],[73.44393,42.43098],[71.88792,42.83578],[71.62405,42.76613],[71.53272,42.8014],[71.2724,42.77853],[71.22785,42.69248],[71.17807,42.67381],[71.15232,42.60486],[70.97717,42.50147],[70.85973,42.30188],[70.94483,42.26238],[71.13263,42.28356],[71.28719,42.18033],[70.69777,41.92554],[70.17682,41.5455],[70.48909,41.40335],[70.67586,41.47953],[70.78572,41.36419],[70.77885,41.24813],[70.86263,41.23833],[70.9615,41.16393],[71.02193,41.19494],[71.11806,41.15359],[71.25813,41.18796],[71.27187,41.11015],[71.34877,41.16807],[71.40198,41.09436],[71.46148,41.13958],[71.43814,41.19644],[71.46688,41.31883],[71.57227,41.29175],[71.6787,41.42111],[71.65914,41.49599],[71.73054,41.54713],[71.71132,41.43012],[71.76625,41.4466],[71.83914,41.3546],[71.91457,41.2982],[71.85964,41.19081],[72.07249,41.11739],[72.10745,41.15483],[72.16433,41.16483],[72.17594,41.15522],[72.14864,41.13363],[72.1792,41.10621],[72.21061,41.05607],[72.17594,41.02377],[72.18339,40.99571],[72.324,41.03381],[72.34026,41.04539],[72.34757,41.06104],[72.36138,41.04384],[72.38511,41.02785],[72.45206,41.03018],[72.48742,40.97136],[72.55109,40.96046],[72.59136,40.86947],[72.68157,40.84942],[72.84291,40.85512],[72.94454,40.8094],[73.01869,40.84681],[73.13267,40.83512],[73.13412,40.79122],[73.0612,40.76678],[72.99133,40.76457],[72.93296,40.73089],[72.8722,40.71111],[72.85372,40.7116],[72.84754,40.67229],[72.80137,40.67856],[72.74866,40.60873],[72.74894,40.59592],[72.75982,40.57273],[72.74862,40.57131],[72.74768,40.58051],[72.73995,40.58409],[72.69579,40.59778],[72.66713,40.59076],[72.66713,40.5219],[72.47795,40.5532],[72.40517,40.61917],[72.34406,40.60144],[72.41714,40.55736],[72.38384,40.51535],[72.41513,40.50856],[72.44191,40.48222],[72.40346,40.4007],[72.24368,40.46091],[72.18648,40.49893],[71.96401,40.31907],[72.05464,40.27586],[71.85002,40.25647],[71.82646,40.21872],[71.73054,40.14818],[71.71719,40.17886],[71.69621,40.18492],[71.70569,40.20391],[71.68386,40.26984],[71.61931,40.26775],[71.61725,40.20615],[71.51549,40.22986],[71.51215,40.26943],[71.4246,40.28619],[71.36663,40.31593],[71.13042,40.34106],[71.05901,40.28765],[70.95789,40.28761],[70.9818,40.22392],[70.80495,40.16813],[70.7928,40.12797],[70.65827,40.0981],[70.65946,39.9878],[70.58912,39.95211],[70.55033,39.96619],[70.47557,39.93216],[70.57384,39.99394],[70.58297,40.00891],[70.01283,40.23288],[69.67001,40.10639],[69.64704,40.12165],[69.57615,40.10524],[69.55555,40.12296],[69.53794,40.11833],[69.53855,40.0887],[69.5057,40.03277],[69.53615,39.93991],[69.43557,39.92877],[69.43134,39.98431],[69.35649,40.01994],[69.26938,39.8127],[69.3594,39.52516],[69.68677,39.59281],[69.87491,39.53882],[70.11111,39.58223],[70.2869,39.53141],[70.44757,39.60128],[70.64087,39.58792],[70.7854,39.38933],[71.06418,39.41586],[71.08752,39.50704],[71.49814,39.61397],[71.55856,39.57588],[71.5517,39.45722],[71.62688,39.44056],[71.76816,39.45456],[71.80164,39.40631],[71.7522,39.32031],[71.79202,39.27355],[71.90601,39.27674],[72.04059,39.36704],[72.09689,39.26823],[72.17242,39.2661],[72.23834,39.17248],[72.33173,39.33093],[72.62027,39.39696],[72.85934,39.35116],[73.18454,39.35536],[73.31912,39.38615],[73.45096,39.46677],[73.59831,39.46425],[73.87018,39.47879],[73.94683,39.60733],[73.92354,39.69565],[73.9051,39.75073],[73.83006,39.76136],[73.97049,40.04378],[74.25533,40.13191],[74.35063,40.09742],[74.69875,40.34668],[74.85996,40.32857],[74.78168,40.44886],[74.82013,40.52197],[75.08243,40.43945],[75.22834,40.45382],[75.5854,40.66874],[75.69663,40.28642],[75.91361,40.2948],[75.96168,40.38064],[76.33659,40.3482],[76.5261,40.46114],[76.75681,40.95354],[76.99302,41.0696],[77.28004,41.0033],[77.3693,41.0375],[77.52723,41.00227],[77.76206,41.01574],[77.81287,41.14307],[78.12873,41.23091],[78.15757,41.38565],[78.3732,41.39603],[79.92977,42.04113],[80.17842,42.03211],[80.17807,42.21166],[79.97364,42.42816],[79.52921,42.44778],[79.19763,42.804],[78.91502,42.76839],[78.48469,42.89649],[75.82823,42.94848],[75.72174,42.79672],[75.29966,42.86183],[75.22619,42.85528],[74.88756,42.98612]],[[70.74189,39.86319],[70.63105,39.77923],[70.59667,39.83542],[70.54998,39.85137],[70.52631,39.86989],[70.53651,39.89155],[70.74189,39.86319]],[[71.86463,39.98598],[71.84316,39.95582],[71.7504,39.93701],[71.71511,39.96348],[71.78838,40.01404],[71.86463,39.98598]],[[71.21139,40.03369],[71.1427,39.95026],[71.23067,39.93581],[71.16101,39.88423],[71.10531,39.91354],[71.04979,39.89808],[71.10501,39.95568],[71.09063,39.99],[71.11668,39.99291],[71.11037,40.01984],[71.01035,40.05481],[71.00236,40.18154],[71.06305,40.1771],[71.12218,40.03052],[71.21139,40.03369]]]]}},{type:"Feature",properties:{iso1A2:"KH",iso1A3:"KHM",iso1N3:"116",wikidata:"Q424",nameEn:"Cambodia",groups:["035","142"],callingCodes:["855"]},geometry:{type:"MultiPolygon",coordinates:[[[[105.87328,11.55953],[105.81645,11.56876],[105.80867,11.60536],[105.8507,11.66635],[105.88962,11.67854],[105.95188,11.63738],[106.00792,11.7197],[106.02038,11.77457],[106.06708,11.77761],[106.13158,11.73283],[106.18539,11.75171],[106.26478,11.72122],[106.30525,11.67549],[106.37219,11.69836],[106.44691,11.66787],[106.45158,11.68616],[106.41577,11.76999],[106.44535,11.8279],[106.44068,11.86294],[106.4687,11.86751],[106.4111,11.97413],[106.70687,11.96956],[106.79405,12.0807],[106.92325,12.06548],[106.99953,12.08983],[107.15831,12.27547],[107.34511,12.33327],[107.42917,12.24657],[107.4463,12.29373],[107.55059,12.36824],[107.5755,12.52177],[107.55993,12.7982],[107.49611,12.88926],[107.49144,13.01215],[107.62843,13.3668],[107.61909,13.52577],[107.53503,13.73908],[107.45252,13.78897],[107.46498,13.91593],[107.44318,13.99751],[107.38247,13.99147],[107.35757,14.02319],[107.37158,14.07906],[107.33577,14.11832],[107.40427,14.24509],[107.39493,14.32655],[107.44941,14.41552],[107.48521,14.40346],[107.52569,14.54665],[107.52102,14.59034],[107.55371,14.628],[107.54361,14.69092],[107.47238,14.61523],[107.44435,14.52785],[107.37897,14.54443],[107.3276,14.58812],[107.29803,14.58963],[107.26534,14.54292],[107.256,14.48716],[107.21241,14.48716],[107.17038,14.41782],[107.09722,14.3937],[107.03962,14.45099],[107.04585,14.41782],[106.98825,14.36806],[106.9649,14.3198],[106.90574,14.33639],[106.8497,14.29416],[106.80767,14.31226],[106.73762,14.42687],[106.63333,14.44194],[106.59908,14.50977],[106.57106,14.50525],[106.54148,14.59565],[106.50723,14.58963],[106.45898,14.55045],[106.47766,14.50977],[106.43874,14.52032],[106.40916,14.45249],[106.32355,14.44043],[106.25194,14.48415],[106.21302,14.36203],[106.00131,14.36957],[105.99509,14.32734],[106.02311,14.30623],[106.04801,14.20363],[106.10872,14.18401],[106.11962,14.11307],[106.18656,14.06324],[106.16632,14.01794],[106.10094,13.98471],[106.10405,13.9137],[105.90791,13.92881],[105.78182,14.02247],[105.78338,14.08438],[105.5561,14.15684],[105.44869,14.10703],[105.36775,14.09948],[105.2759,14.17496],[105.20894,14.34967],[105.17748,14.34432],[105.14012,14.23873],[105.08408,14.20402],[105.02804,14.23722],[104.97667,14.38806],[104.69335,14.42726],[104.55014,14.36091],[104.27616,14.39861],[103.93836,14.3398],[103.70175,14.38052],[103.71109,14.4348],[103.53518,14.42575],[103.39353,14.35639],[103.16469,14.33075],[102.93275,14.19044],[102.91251,14.01531],[102.77864,13.93374],[102.72727,13.77806],[102.56848,13.69366],[102.5481,13.6589],[102.58635,13.6286],[102.62483,13.60883],[102.57573,13.60461],[102.5358,13.56933],[102.44601,13.5637],[102.36859,13.57488],[102.33828,13.55613],[102.361,13.50551],[102.35563,13.47307],[102.35692,13.38274],[102.34611,13.35618],[102.36001,13.31142],[102.36146,13.26006],[102.43422,13.09061],[102.46011,13.08057],[102.52275,12.99813],[102.48694,12.97537],[102.49335,12.92711],[102.53053,12.77506],[102.4994,12.71736],[102.51963,12.66117],[102.57567,12.65358],[102.7796,12.43781],[102.78116,12.40284],[102.73134,12.37091],[102.70176,12.1686],[102.77026,12.06815],[102.78427,11.98746],[102.83957,11.8519],[102.90973,11.75613],[102.91449,11.65512],[102.52395,11.25257],[102.47649,9.66162],[103.99198,10.48391],[104.43778,10.42386],[104.47963,10.43046],[104.49869,10.4057],[104.59018,10.53073],[104.87933,10.52833],[104.95094,10.64003],[105.09571,10.72722],[105.02722,10.89236],[105.08326,10.95656],[105.11449,10.96332],[105.34011,10.86179],[105.42884,10.96878],[105.50045,10.94586],[105.77751,11.03671],[105.86376,10.89839],[105.84603,10.85873],[105.93403,10.83853],[105.94535,10.9168],[106.06708,10.8098],[106.18539,10.79451],[106.14301,10.98176],[106.20095,10.97795],[106.1757,11.07301],[106.1527,11.10476],[106.10444,11.07879],[105.86782,11.28343],[105.88962,11.43605],[105.87328,11.55953]]]]}},{type:"Feature",properties:{iso1A2:"KI",iso1A3:"KIR",iso1N3:"296",wikidata:"Q710",nameEn:"Kiribati",groups:["057","009"],driveSide:"left",callingCodes:["686"]},geometry:{type:"MultiPolygon",coordinates:[[[[169,3.9],[169,-3.5],[178,-3.5],[178,3.9],[169,3.9]]],[[[-158.62058,-1.35506],[-161.04969,-1.36251],[-175.33482,-1.40631],[-175.31804,-7.54825],[-174.18707,-7.54408],[-167.75329,-7.52784],[-156.50903,-7.4975],[-156.4957,-12.32002],[-149.61166,-12.30171],[-149.6249,-7.51261],[-149.65979,5.27712],[-161.06795,5.2462],[-161.05669,1.11722],[-158.62734,1.1296],[-158.62058,-1.35506]]]]}},{type:"Feature",properties:{iso1A2:"KM",iso1A3:"COM",iso1N3:"174",wikidata:"Q970",nameEn:"Comoros",groups:["014","202","002"],callingCodes:["269"]},geometry:{type:"MultiPolygon",coordinates:[[[[42.93552,-11.11413],[42.99868,-12.65261],[44.75722,-12.58368],[44.69407,-11.04481],[42.93552,-11.11413]]]]}},{type:"Feature",properties:{iso1A2:"KN",iso1A3:"KNA",iso1N3:"659",wikidata:"Q763",nameEn:"St. Kitts and Nevis",groups:["029","003","419","019"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["1 869"]},geometry:{type:"MultiPolygon",coordinates:[[[[-62.27053,17.22145],[-62.76692,17.64353],[-63.11114,17.23125],[-62.62949,16.82364],[-62.27053,17.22145]]]]}},{type:"Feature",properties:{iso1A2:"KP",iso1A3:"PRK",iso1N3:"408",wikidata:"Q423",nameEn:"North Korea",groups:["030","142"],callingCodes:["850"]},geometry:{type:"MultiPolygon",coordinates:[[[[130.26095,42.9027],[130.09764,42.91425],[130.12957,42.98361],[129.96409,42.97306],[129.95082,43.01051],[129.8865,43.00395],[129.85261,42.96494],[129.83277,42.86746],[129.80719,42.79218],[129.7835,42.76521],[129.77183,42.69435],[129.75294,42.59409],[129.72541,42.43739],[129.60482,42.44461],[129.54701,42.37254],[129.42882,42.44702],[129.28541,42.41574],[129.22423,42.3553],[129.22285,42.26491],[129.15178,42.17224],[128.96068,42.06657],[128.94007,42.03537],[128.04487,42.01769],[128.15119,41.74568],[128.30716,41.60322],[128.20061,41.40895],[128.18546,41.41279],[128.12967,41.37931],[128.03311,41.39232],[128.02633,41.42103],[127.92943,41.44291],[127.29712,41.49473],[127.17841,41.59714],[126.90729,41.79955],[126.60631,41.65565],[126.53189,41.35206],[126.242,41.15454],[126.00335,40.92835],[125.76869,40.87908],[125.71172,40.85223],[124.86913,40.45387],[124.40719,40.13655],[124.38556,40.11047],[124.3322,40.05573],[124.37089,40.03004],[124.35029,39.95639],[124.23201,39.9248],[124.17532,39.8232],[123.90497,38.79949],[123.85601,37.49093],[124.67666,38.05679],[124.84224,37.977],[124.87921,37.80827],[125.06408,37.66334],[125.37112,37.62643],[125.81159,37.72949],[126.13074,37.70512],[126.18776,37.74728],[126.19097,37.81462],[126.24402,37.83113],[126.43239,37.84095],[126.46818,37.80873],[126.56709,37.76857],[126.59918,37.76364],[126.66067,37.7897],[126.68793,37.83728],[126.68793,37.9175],[126.67023,37.95852],[126.84961,38.0344],[126.88106,38.10246],[126.95887,38.1347],[126.95338,38.17735],[127.04479,38.25518],[127.15749,38.30722],[127.38727,38.33227],[127.49672,38.30647],[127.55013,38.32257],[128.02917,38.31861],[128.27652,38.41657],[128.31105,38.58462],[128.37487,38.62345],[128.65655,38.61914],[131.95041,41.5445],[130.65022,42.32281],[130.66367,42.38024],[130.64181,42.41422],[130.60805,42.4317],[130.56835,42.43281],[130.55143,42.52158],[130.50123,42.61636],[130.44361,42.54849],[130.41826,42.6011],[130.2385,42.71127],[130.23068,42.80125],[130.26095,42.9027]]]]}},{type:"Feature",properties:{iso1A2:"KR",iso1A3:"KOR",iso1N3:"410",wikidata:"Q884",nameEn:"South Korea",groups:["030","142"],callingCodes:["82"]},geometry:{type:"MultiPolygon",coordinates:[[[[133.61399,37.41],[128.65655,38.61914],[128.37487,38.62345],[128.31105,38.58462],[128.27652,38.41657],[128.02917,38.31861],[127.55013,38.32257],[127.49672,38.30647],[127.38727,38.33227],[127.15749,38.30722],[127.04479,38.25518],[126.95338,38.17735],[126.95887,38.1347],[126.88106,38.10246],[126.84961,38.0344],[126.67023,37.95852],[126.68793,37.9175],[126.68793,37.83728],[126.66067,37.7897],[126.59918,37.76364],[126.56709,37.76857],[126.46818,37.80873],[126.43239,37.84095],[126.24402,37.83113],[126.19097,37.81462],[126.18776,37.74728],[126.13074,37.70512],[125.81159,37.72949],[125.37112,37.62643],[125.06408,37.66334],[124.87921,37.80827],[124.84224,37.977],[124.67666,38.05679],[123.85601,37.49093],[122.80525,33.30571],[125.99728,32.63328],[129.2669,34.87122],[133.61399,37.41]]]]}},{type:"Feature",properties:{iso1A2:"KW",iso1A3:"KWT",iso1N3:"414",wikidata:"Q817",nameEn:"Kuwait",groups:["145","142"],callingCodes:["965"]},geometry:{type:"MultiPolygon",coordinates:[[[[49.00421,28.81495],[48.59531,29.66815],[48.40479,29.85763],[48.17332,30.02448],[48.06782,30.02906],[48.01114,29.98906],[47.7095,30.10453],[47.37192,30.10421],[47.15166,30.01044],[46.89695,29.50584],[46.5527,29.10283],[47.46202,29.0014],[47.58376,28.83382],[47.59863,28.66798],[47.70561,28.5221],[48.42991,28.53628],[49.00421,28.81495]]]]}},{type:"Feature",properties:{iso1A2:"KY",iso1A3:"CYM",iso1N3:"136",wikidata:"Q5785",nameEn:"Cayman Islands",country:"GB",groups:["029","003","419","019"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["1 345"]},geometry:{type:"MultiPolygon",coordinates:[[[[-82.11509,19.60401],[-80.36068,18.11751],[-79.32727,20.06742],[-82.11509,19.60401]]]]}},{type:"Feature",properties:{iso1A2:"KZ",iso1A3:"KAZ",iso1N3:"398",wikidata:"Q232",nameEn:"Kazakhstan",groups:["143","142"],callingCodes:["7"]},geometry:{type:"MultiPolygon",coordinates:[[[[68.90865,55.38148],[68.19206,55.18823],[68.26661,55.09226],[68.21308,54.98645],[65.20174,54.55216],[65.24663,54.35721],[65.11033,54.33028],[64.97216,54.4212],[63.97686,54.29763],[64.02715,54.22679],[63.91224,54.20013],[63.80604,54.27079],[62.58651,54.05871],[62.56876,53.94047],[62.45931,53.90737],[62.38535,54.03961],[62.00966,54.04134],[62.03913,53.94768],[61.65318,54.02445],[61.56941,53.95703],[61.47603,54.08048],[61.3706,54.08464],[61.26863,53.92797],[60.99796,53.93699],[61.14283,53.90063],[61.22574,53.80268],[60.90626,53.62937],[61.55706,53.57144],[61.57185,53.50112],[61.37957,53.45887],[61.29082,53.50992],[61.14291,53.41481],[61.19024,53.30536],[62.14574,53.09626],[62.12799,52.99133],[62.0422,52.96105],[61.23462,53.03227],[61.05842,52.92217],[60.71989,52.75923],[60.71693,52.66245],[60.84118,52.63912],[60.84709,52.52228],[60.98021,52.50068],[61.05417,52.35096],[60.78201,52.22067],[60.72581,52.15538],[60.48915,52.15175],[60.19925,51.99173],[59.99809,51.98263],[60.09867,51.87135],[60.50986,51.7964],[60.36787,51.66815],[60.5424,51.61675],[60.92401,51.61124],[60.95655,51.48615],[61.50677,51.40687],[61.55114,51.32746],[61.6813,51.25716],[61.56889,51.23679],[61.4431,50.80679],[60.81833,50.6629],[60.31914,50.67705],[60.17262,50.83312],[60.01288,50.8163],[59.81172,50.54451],[59.51886,50.49937],[59.48928,50.64216],[58.87974,50.70852],[58.3208,51.15151],[57.75578,51.13852],[57.74986,50.93017],[57.44221,50.88354],[57.17302,51.11253],[56.17906,50.93204],[56.11398,50.7471],[55.67774,50.54508],[54.72067,51.03261],[54.56685,51.01958],[54.71476,50.61214],[54.55797,50.52006],[54.41894,50.61214],[54.46331,50.85554],[54.12248,51.11542],[53.69299,51.23466],[53.46165,51.49445],[52.54329,51.48444],[52.36119,51.74161],[51.8246,51.67916],[51.77431,51.49536],[51.301,51.48799],[51.26254,51.68466],[50.59695,51.61859],[50.26859,51.28677],[49.97277,51.2405],[49.76866,51.11067],[49.39001,51.09396],[49.41959,50.85927],[49.12673,50.78639],[48.86936,50.61589],[48.57946,50.63278],[48.90782,50.02281],[48.68352,49.89546],[48.42564,49.82283],[48.24519,49.86099],[48.10044,50.09242],[47.58551,50.47867],[47.30448,50.30894],[47.34589,50.09308],[47.18319,49.93721],[46.9078,49.86707],[46.78398,49.34026],[46.98795,49.23531],[47.04416,49.17152],[47.01458,49.07085],[46.91104,48.99715],[46.78392,48.95352],[46.49011,48.43019],[47.11516,48.27188],[47.12107,47.83687],[47.38731,47.68176],[47.41689,47.83687],[47.64973,47.76559],[48.15348,47.74545],[48.45173,47.40818],[48.52326,47.4102],[49.01136,46.72716],[48.51142,46.69268],[48.54988,46.56267],[49.16518,46.38542],[49.32259,46.26944],[49.88945,46.04554],[49.2134,44.84989],[52.26048,41.69249],[52.47884,41.78034],[52.97575,42.1308],[54.20635,42.38477],[54.95182,41.92424],[55.45471,41.25609],[56.00314,41.32584],[55.97584,44.99322],[55.97584,44.99328],[55.97584,44.99338],[55.97584,44.99343],[55.97584,44.99348],[55.97584,44.99353],[55.97584,44.99359],[55.97584,44.99369],[55.97584,44.99374],[55.97584,44.99384],[55.97584,44.9939],[55.97584,44.994],[55.97584,44.99405],[55.97584,44.99415],[55.97584,44.99421],[55.97584,44.99426],[55.97584,44.99431],[55.97584,44.99436],[55.97584,44.99441],[55.97594,44.99446],[55.97605,44.99452],[55.97605,44.99457],[55.97605,44.99462],[55.97605,44.99467],[55.97605,44.99477],[55.97615,44.99477],[55.97615,44.99483],[55.97615,44.99493],[55.97615,44.99498],[55.97615,44.99503],[55.97615,44.99508],[55.97625,44.99514],[55.97636,44.99519],[55.97636,44.99524],[55.97646,44.99529],[55.97646,44.99534],[55.97656,44.99539],[55.97667,44.99545],[55.97677,44.9955],[55.97677,44.99555],[55.97677,44.9956],[55.97687,44.9956],[55.97698,44.99565],[55.97698,44.9957],[55.97708,44.99576],[55.97718,44.99581],[55.97729,44.99586],[55.97739,44.99586],[55.97739,44.99591],[55.97749,44.99591],[55.9776,44.99591],[55.9777,44.99596],[55.9777,44.99601],[55.9778,44.99607],[55.97791,44.99607],[55.97801,44.99607],[55.97801,44.99612],[55.97811,44.99617],[55.97822,44.99617],[55.97832,44.99622],[55.97842,44.99622],[58.59711,45.58671],[61.01475,44.41383],[62.01711,43.51008],[63.34656,43.64003],[64.53885,43.56941],[64.96464,43.74748],[65.18666,43.48835],[65.53277,43.31856],[65.85194,42.85481],[66.09482,42.93426],[66.00546,41.94455],[66.53302,41.87388],[66.69129,41.1311],[67.9644,41.14611],[67.98511,41.02794],[68.08273,41.08148],[68.1271,41.0324],[67.96736,40.83798],[68.49983,40.56437],[68.63,40.59358],[68.58444,40.91447],[68.49983,40.99669],[68.62221,41.03019],[68.65662,40.93861],[68.73945,40.96989],[68.7217,41.05025],[69.01308,41.22804],[69.05006,41.36183],[69.15137,41.43078],[69.17701,41.43769],[69.18528,41.45175],[69.20439,41.45391],[69.22671,41.46298],[69.23332,41.45847],[69.25059,41.46693],[69.29778,41.43673],[69.35554,41.47211],[69.37468,41.46555],[69.45081,41.46246],[69.39485,41.51518],[69.45751,41.56863],[69.49545,41.545],[70.94483,42.26238],[70.85973,42.30188],[70.97717,42.50147],[71.15232,42.60486],[71.17807,42.67381],[71.22785,42.69248],[71.2724,42.77853],[71.53272,42.8014],[71.62405,42.76613],[71.88792,42.83578],[73.44393,42.43098],[73.50992,42.82356],[73.55634,43.03071],[74.22489,43.24657],[74.57491,43.13702],[74.64615,43.05881],[74.70331,43.02519],[74.75,42.99029],[74.88756,42.98612],[75.22619,42.85528],[75.29966,42.86183],[75.72174,42.79672],[75.82823,42.94848],[78.48469,42.89649],[78.91502,42.76839],[79.19763,42.804],[79.52921,42.44778],[79.97364,42.42816],[80.17807,42.21166],[80.26841,42.23797],[80.16892,42.61137],[80.26886,42.8366],[80.38169,42.83142],[80.58999,42.9011],[80.3735,43.01557],[80.62913,43.141],[80.78817,43.14235],[80.77771,43.30065],[80.69718,43.32589],[80.75156,43.44948],[80.40031,44.10986],[80.40229,44.23319],[80.38384,44.63073],[79.8987,44.89957],[80.11169,45.03352],[81.73278,45.3504],[82.51374,45.1755],[82.58474,45.40027],[82.21792,45.56619],[83.04622,47.19053],[83.92184,46.98912],[84.73077,47.01394],[84.93995,46.87399],[85.22443,47.04816],[85.54294,47.06171],[85.69696,47.2898],[85.61067,47.49753],[85.5169,48.05493],[85.73581,48.3939],[86.38069,48.46064],[86.75343,48.70331],[86.73568,48.99918],[86.87238,49.12432],[87.28386,49.11626],[87.31465,49.23603],[87.03071,49.25142],[86.82606,49.51796],[86.61307,49.60239],[86.79056,49.74787],[86.63674,49.80136],[86.18709,49.50259],[85.24047,49.60239],[84.99198,50.06793],[84.29385,50.27257],[83.8442,50.87375],[83.14607,51.00796],[82.55443,50.75412],[81.94999,50.79307],[81.46581,50.77658],[81.41248,50.97524],[81.06091,50.94833],[81.16999,51.15662],[80.80318,51.28262],[80.44819,51.20855],[80.4127,50.95581],[80.08138,50.77658],[79.11255,52.01171],[77.90383,53.29807],[76.54243,53.99329],[76.44076,54.16017],[76.82266,54.1798],[76.91052,54.4677],[75.3668,54.07439],[75.43398,53.98652],[75.07405,53.80831],[73.39218,53.44623],[73.25412,53.61532],[73.68921,53.86522],[73.74778,54.07194],[73.37963,53.96132],[72.71026,54.1161],[72.43415,53.92685],[72.17477,54.36303],[71.96141,54.17736],[71.10379,54.13326],[71.08706,54.33376],[71.24185,54.64965],[71.08288,54.71253],[70.96009,55.10558],[70.76493,55.3027],[70.19179,55.1476],[69.74917,55.35545],[69.34224,55.36344],[68.90865,55.38148]]]]}},{type:"Feature",properties:{iso1A2:"LA",iso1A3:"LAO",iso1N3:"418",wikidata:"Q819",nameEn:"Laos",groups:["035","142"],callingCodes:["856"]},geometry:{type:"MultiPolygon",coordinates:[[[[102.1245,22.43372],[102.03633,22.46164],[101.98487,22.42766],[101.91344,22.44417],[101.90714,22.38688],[101.86828,22.38397],[101.7685,22.50337],[101.68973,22.46843],[101.61306,22.27515],[101.56789,22.28876],[101.53638,22.24794],[101.60675,22.13513],[101.57525,22.13026],[101.62566,21.96574],[101.7791,21.83019],[101.74555,21.72852],[101.83257,21.61562],[101.80001,21.57461],[101.7475,21.5873],[101.7727,21.51794],[101.74224,21.48276],[101.74014,21.30967],[101.84412,21.25291],[101.83887,21.20983],[101.76745,21.21571],[101.79266,21.19025],[101.7622,21.14813],[101.70548,21.14911],[101.66977,21.20004],[101.60886,21.17947],[101.59491,21.18621],[101.6068,21.23329],[101.54563,21.25668],[101.29326,21.17254],[101.2229,21.23271],[101.26912,21.36482],[101.19349,21.41959],[101.2124,21.56422],[101.15156,21.56129],[101.16198,21.52808],[101.00234,21.39612],[100.80173,21.2934],[100.72716,21.31786],[100.63578,21.05639],[100.55281,21.02796],[100.50974,20.88574],[100.64628,20.88279],[100.60112,20.8347],[100.51079,20.82194],[100.36375,20.82783],[100.1957,20.68247],[100.08404,20.36626],[100.09999,20.31614],[100.09337,20.26293],[100.11785,20.24787],[100.1712,20.24324],[100.16668,20.2986],[100.22076,20.31598],[100.25769,20.3992],[100.33383,20.4028],[100.37439,20.35156],[100.41473,20.25625],[100.44992,20.23644],[100.4537,20.19971],[100.47567,20.19133],[100.51052,20.14928],[100.55218,20.17741],[100.58808,20.15791],[100.5094,19.87904],[100.398,19.75047],[100.49604,19.53504],[100.58219,19.49164],[100.64606,19.55884],[100.77231,19.48324],[100.90302,19.61901],[101.08928,19.59748],[101.26545,19.59242],[101.26991,19.48324],[101.21347,19.46223],[101.20604,19.35296],[101.24911,19.33334],[101.261,19.12717],[101.35606,19.04716],[101.25803,18.89545],[101.22832,18.73377],[101.27585,18.68875],[101.06047,18.43247],[101.18227,18.34367],[101.15108,18.25624],[101.19118,18.2125],[101.1793,18.0544],[101.02185,17.87637],[100.96541,17.57926],[101.15108,17.47586],[101.44667,17.7392],[101.72294,17.92867],[101.78087,18.07559],[101.88485,18.02474],[102.11359,18.21532],[102.45523,17.97106],[102.59234,17.96127],[102.60971,17.95411],[102.61432,17.92273],[102.5896,17.84889],[102.59485,17.83537],[102.68194,17.80151],[102.69946,17.81686],[102.67543,17.84529],[102.68538,17.86653],[102.75954,17.89561],[102.79044,17.93612],[102.81988,17.94233],[102.86323,17.97531],[102.95812,18.0054],[102.9912,17.9949],[103.01998,17.97095],[103.0566,18.00144],[103.07823,18.03833],[103.07343,18.12351],[103.1493,18.17799],[103.14994,18.23172],[103.17093,18.2618],[103.29757,18.30475],[103.23818,18.34875],[103.24779,18.37807],[103.30977,18.4341],[103.41044,18.4486],[103.47773,18.42841],[103.60957,18.40528],[103.699,18.34125],[103.82449,18.33979],[103.85642,18.28666],[103.93916,18.33914],[103.97725,18.33631],[104.06533,18.21656],[104.10927,18.10826],[104.21776,17.99335],[104.2757,17.86139],[104.35432,17.82871],[104.45404,17.66788],[104.69867,17.53038],[104.80061,17.39367],[104.80716,17.19025],[104.73712,17.01404],[104.7373,16.91125],[104.76442,16.84752],[104.7397,16.81005],[104.76099,16.69302],[104.73349,16.565],[104.88057,16.37311],[105.00262,16.25627],[105.06204,16.09792],[105.42001,16.00657],[105.38508,15.987],[105.34115,15.92737],[105.37959,15.84074],[105.42285,15.76971],[105.46573,15.74742],[105.61756,15.68792],[105.60446,15.53301],[105.58191,15.41031],[105.47635,15.3796],[105.4692,15.33709],[105.50662,15.32054],[105.58043,15.32724],[105.46661,15.13132],[105.61162,15.00037],[105.5121,14.80802],[105.53864,14.55731],[105.43783,14.43865],[105.20894,14.34967],[105.2759,14.17496],[105.36775,14.09948],[105.44869,14.10703],[105.5561,14.15684],[105.78338,14.08438],[105.78182,14.02247],[105.90791,13.92881],[106.10405,13.9137],[106.10094,13.98471],[106.16632,14.01794],[106.18656,14.06324],[106.11962,14.11307],[106.10872,14.18401],[106.04801,14.20363],[106.02311,14.30623],[105.99509,14.32734],[106.00131,14.36957],[106.21302,14.36203],[106.25194,14.48415],[106.32355,14.44043],[106.40916,14.45249],[106.43874,14.52032],[106.47766,14.50977],[106.45898,14.55045],[106.50723,14.58963],[106.54148,14.59565],[106.57106,14.50525],[106.59908,14.50977],[106.63333,14.44194],[106.73762,14.42687],[106.80767,14.31226],[106.8497,14.29416],[106.90574,14.33639],[106.9649,14.3198],[106.98825,14.36806],[107.04585,14.41782],[107.03962,14.45099],[107.09722,14.3937],[107.17038,14.41782],[107.21241,14.48716],[107.256,14.48716],[107.26534,14.54292],[107.29803,14.58963],[107.3276,14.58812],[107.37897,14.54443],[107.44435,14.52785],[107.47238,14.61523],[107.54361,14.69092],[107.51579,14.79282],[107.59285,14.87795],[107.48277,14.93751],[107.46516,15.00982],[107.61486,15.0566],[107.61926,15.13949],[107.58844,15.20111],[107.62587,15.2266],[107.60605,15.37524],[107.62367,15.42193],[107.53341,15.40496],[107.50699,15.48771],[107.3815,15.49832],[107.34408,15.62345],[107.27583,15.62769],[107.27143,15.71459],[107.21859,15.74638],[107.21419,15.83747],[107.34188,15.89464],[107.39471,15.88829],[107.46296,16.01106],[107.44975,16.08511],[107.33968,16.05549],[107.25822,16.13587],[107.14595,16.17816],[107.15035,16.26271],[107.09091,16.3092],[107.02597,16.31132],[106.97385,16.30204],[106.96638,16.34938],[106.88067,16.43594],[106.88727,16.52671],[106.84104,16.55415],[106.74418,16.41904],[106.65832,16.47816],[106.66052,16.56892],[106.61477,16.60713],[106.58267,16.6012],[106.59013,16.62259],[106.55485,16.68704],[106.55265,16.86831],[106.52183,16.87884],[106.51963,16.92097],[106.54824,16.92729],[106.55045,17.0031],[106.50862,16.9673],[106.43597,17.01362],[106.31929,17.20509],[106.29287,17.3018],[106.24444,17.24714],[106.18991,17.28227],[106.09019,17.36399],[105.85744,17.63221],[105.76612,17.67147],[105.60381,17.89356],[105.64784,17.96687],[105.46292,18.22008],[105.38366,18.15315],[105.15942,18.38691],[105.10408,18.43533],[105.1327,18.58355],[105.19654,18.64196],[105.12829,18.70453],[104.64617,18.85668],[104.5361,18.97747],[103.87125,19.31854],[104.06058,19.43484],[104.10832,19.51575],[104.05617,19.61743],[104.06498,19.66926],[104.23229,19.70242],[104.41281,19.70035],[104.53169,19.61743],[104.64837,19.62365],[104.68359,19.72729],[104.8355,19.80395],[104.8465,19.91783],[104.9874,20.09573],[104.91695,20.15567],[104.86852,20.14121],[104.61315,20.24452],[104.62195,20.36633],[104.72102,20.40554],[104.66158,20.47774],[104.47886,20.37459],[104.40621,20.3849],[104.38199,20.47155],[104.63957,20.6653],[104.27412,20.91433],[104.11121,20.96779],[103.98024,20.91531],[103.82282,20.8732],[103.73478,20.6669],[103.68633,20.66324],[103.45737,20.82382],[103.38032,20.79501],[103.21497,20.89832],[103.12055,20.89994],[103.03469,21.05821],[102.97745,21.05821],[102.89825,21.24707],[102.80794,21.25736],[102.88939,21.3107],[102.94223,21.46034],[102.86297,21.4255],[102.98846,21.58936],[102.97965,21.74076],[102.86077,21.71213],[102.85637,21.84501],[102.81894,21.83888],[102.82115,21.73667],[102.74189,21.66713],[102.67145,21.65894],[102.62301,21.91447],[102.49092,21.99002],[102.51734,22.02676],[102.18712,22.30403],[102.14099,22.40092],[102.1245,22.43372]]]]}},{type:"Feature",properties:{iso1A2:"LB",iso1A3:"LBN",iso1N3:"422",wikidata:"Q822",nameEn:"Lebanon",aliases:["RL"],groups:["145","142"],callingCodes:["961"]},geometry:{type:"MultiPolygon",coordinates:[[[[35.94816,33.47886],[35.94465,33.52774],[36.05723,33.57904],[35.9341,33.6596],[36.06778,33.82927],[36.14517,33.85118],[36.3967,33.83365],[36.38263,33.86579],[36.28589,33.91981],[36.41078,34.05253],[36.50576,34.05982],[36.5128,34.09916],[36.62537,34.20251],[36.59195,34.2316],[36.58667,34.27667],[36.60778,34.31009],[36.56556,34.31881],[36.53039,34.3798],[36.55853,34.41609],[36.46179,34.46541],[36.4442,34.50165],[36.34745,34.5002],[36.3369,34.52629],[36.39846,34.55672],[36.41429,34.61175],[36.45299,34.59438],[36.46003,34.6378],[36.42941,34.62505],[36.35384,34.65447],[36.35135,34.68516],[36.32399,34.69334],[36.29165,34.62991],[35.98718,34.64977],[35.97386,34.63322],[35.48515,34.70851],[34.78515,33.20368],[35.10645,33.09318],[35.1924,33.08743],[35.31429,33.10515],[35.35223,33.05617],[35.43059,33.06659],[35.448,33.09264],[35.50272,33.09056],[35.50335,33.114],[35.52573,33.11921],[35.54228,33.19865],[35.5362,33.23196],[35.54808,33.236],[35.54544,33.25513],[35.55555,33.25844],[35.56523,33.28969],[35.58326,33.28381],[35.58502,33.26653],[35.62283,33.24226],[35.62019,33.27278],[35.77477,33.33609],[35.81324,33.36354],[35.82577,33.40479],[35.88668,33.43183],[35.94816,33.47886]]]]}},{type:"Feature",properties:{iso1A2:"LC",iso1A3:"LCA",iso1N3:"662",wikidata:"Q760",nameEn:"St. Lucia",aliases:["WL"],groups:["029","003","419","019"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["1 758"]},geometry:{type:"MultiPolygon",coordinates:[[[[-60.5958,14.23076],[-61.26561,14.25664],[-61.43129,13.68336],[-60.70539,13.41452],[-60.5958,14.23076]]]]}},{type:"Feature",properties:{iso1A2:"LI",iso1A3:"LIE",iso1N3:"438",wikidata:"Q347",nameEn:"Liechtenstein",aliases:["FL"],groups:["155","150"],callingCodes:["423"]},geometry:{type:"MultiPolygon",coordinates:[[[[9.60717,47.06091],[9.61216,47.07732],[9.63395,47.08443],[9.62623,47.14685],[9.56539,47.17124],[9.58264,47.20673],[9.56981,47.21926],[9.55176,47.22585],[9.56766,47.24281],[9.53116,47.27029],[9.52406,47.24959],[9.50318,47.22153],[9.4891,47.19346],[9.48774,47.17402],[9.51044,47.13727],[9.52089,47.10019],[9.51362,47.08505],[9.47139,47.06402],[9.47548,47.05257],[9.54041,47.06495],[9.55721,47.04762],[9.60717,47.06091]]]]}},{type:"Feature",properties:{iso1A2:"LK",iso1A3:"LKA",iso1N3:"144",wikidata:"Q854",nameEn:"Sri Lanka",groups:["034","142"],driveSide:"left",callingCodes:["94"]},geometry:{type:"MultiPolygon",coordinates:[[[[76.25812,4.62435],[85.15017,5.21497],[80.48418,10.20786],[79.42124,9.80115],[79.50447,8.91876],[76.25812,4.62435]]]]}},{type:"Feature",properties:{iso1A2:"LR",iso1A3:"LBR",iso1N3:"430",wikidata:"Q1014",nameEn:"Liberia",groups:["011","202","002"],callingCodes:["231"]},geometry:{type:"MultiPolygon",coordinates:[[[[-8.47114,7.55676],[-8.55874,7.62525],[-8.55874,7.70167],[-8.67814,7.69428],[-8.72789,7.51429],[-8.8448,7.35149],[-8.85724,7.26019],[-8.93435,7.2824],[-9.09107,7.1985],[-9.18311,7.30461],[-9.20798,7.38109],[-9.305,7.42056],[-9.41943,7.41809],[-9.48161,7.37122],[-9.37465,7.62032],[-9.35724,7.74111],[-9.44928,7.9284],[-9.41445,8.02448],[-9.50898,8.18455],[-9.47415,8.35195],[-9.77763,8.54633],[-10.05873,8.42578],[-10.05375,8.50697],[-10.14579,8.52665],[-10.203,8.47991],[-10.27575,8.48711],[-10.30084,8.30008],[-10.31635,8.28554],[-10.29839,8.21283],[-10.35227,8.15223],[-10.45023,8.15627],[-10.51554,8.1393],[-10.57523,8.04829],[-10.60492,8.04072],[-10.60422,7.7739],[-11.29417,7.21576],[-11.4027,6.97746],[-11.50429,6.92704],[-12.15048,6.15992],[-7.52774,3.7105],[-7.53259,4.35145],[-7.59349,4.8909],[-7.53876,4.94294],[-7.55369,5.08667],[-7.48901,5.14118],[-7.46165,5.26256],[-7.36463,5.32944],[-7.43428,5.42355],[-7.37209,5.61173],[-7.43926,5.74787],[-7.43677,5.84687],[-7.46165,5.84934],[-7.48155,5.80974],[-7.67309,5.94337],[-7.70294,5.90625],[-7.78254,5.99037],[-7.79747,6.07696],[-7.8497,6.08932],[-7.83478,6.20309],[-7.90692,6.27728],[-8.00642,6.31684],[-8.17557,6.28222],[-8.3298,6.36381],[-8.38453,6.35887],[-8.45666,6.49977],[-8.48652,6.43797],[-8.59456,6.50612],[-8.31736,6.82837],[-8.29249,7.1691],[-8.37458,7.25794],[-8.41935,7.51203],[-8.47114,7.55676]]]]}},{type:"Feature",properties:{iso1A2:"LS",iso1A3:"LSO",iso1N3:"426",wikidata:"Q1013",nameEn:"Lesotho",groups:["018","202","002"],driveSide:"left",callingCodes:["266"]},geometry:{type:"MultiPolygon",coordinates:[[[[29.33204,-29.45598],[29.44883,-29.3772],[29.40524,-29.21246],[28.68043,-28.58744],[28.65091,-28.57025],[28.40612,-28.6215],[28.30518,-28.69531],[28.2348,-28.69471],[28.1317,-28.7293],[28.02503,-28.85991],[27.98675,-28.8787],[27.9392,-28.84864],[27.88933,-28.88156],[27.8907,-28.91612],[27.75458,-28.89839],[27.55974,-29.18954],[27.5158,-29.2261],[27.54258,-29.25575],[27.48679,-29.29349],[27.45125,-29.29708],[27.47254,-29.31968],[27.4358,-29.33465],[27.33464,-29.48161],[27.01016,-29.65439],[27.09489,-29.72796],[27.22719,-30.00718],[27.29603,-30.05473],[27.32555,-30.14785],[27.40778,-30.14577],[27.37293,-30.19401],[27.36649,-30.27246],[27.38108,-30.33456],[27.45452,-30.32239],[27.56901,-30.42504],[27.56781,-30.44562],[27.62137,-30.50509],[27.6521,-30.51707],[27.67819,-30.53437],[27.69467,-30.55862],[27.74814,-30.60635],[28.12073,-30.68072],[28.2319,-30.28476],[28.399,-30.1592],[28.68627,-30.12885],[28.80222,-30.10579],[28.9338,-30.05072],[29.16548,-29.91706],[29.12553,-29.76266],[29.28545,-29.58456],[29.33204,-29.45598]]]]}},{type:"Feature",properties:{iso1A2:"LT",iso1A3:"LTU",iso1N3:"440",wikidata:"Q37",nameEn:"Lithuania",groups:["EU","154","150"],callingCodes:["370"]},geometry:{type:"MultiPolygon",coordinates:[[[[24.89005,56.46666],[24.83686,56.41565],[24.70022,56.40483],[24.57353,56.31525],[24.58143,56.29125],[24.42746,56.26522],[24.32334,56.30226],[24.13139,56.24881],[24.02657,56.3231],[23.75726,56.37282],[23.49803,56.34307],[23.40486,56.37689],[23.31606,56.3827],[23.17312,56.36795],[23.09531,56.30511],[22.96988,56.41213],[22.83048,56.367],[22.69354,56.36284],[22.56441,56.39305],[22.3361,56.4016],[22.09728,56.42851],[22.00548,56.41508],[21.74558,56.33181],[21.57888,56.31406],[21.49736,56.29106],[21.24644,56.16917],[21.15016,56.07818],[20.68447,56.04073],[20.60454,55.40986],[20.95181,55.27994],[21.26425,55.24456],[21.35465,55.28427],[21.38446,55.29348],[21.46766,55.21115],[21.51095,55.18507],[21.55605,55.20311],[21.64954,55.1791],[21.85521,55.09493],[21.96505,55.07353],[21.99543,55.08691],[22.03984,55.07888],[22.02582,55.05078],[22.06087,55.02935],[22.11697,55.02131],[22.14267,55.05345],[22.31562,55.0655],[22.47688,55.04408],[22.58907,55.07085],[22.60075,55.01863],[22.65451,54.97037],[22.68723,54.9811],[22.76422,54.92521],[22.85083,54.88711],[22.87317,54.79492],[22.73631,54.72952],[22.73397,54.66604],[22.75467,54.6483],[22.74225,54.64339],[22.7522,54.63525],[22.68021,54.58486],[22.71293,54.56454],[22.67788,54.532],[22.70208,54.45312],[22.7253,54.41732],[22.79705,54.36264],[22.83756,54.40827],[23.00584,54.38514],[22.99649,54.35927],[23.05726,54.34565],[23.04323,54.31567],[23.104,54.29794],[23.13905,54.31567],[23.15526,54.31076],[23.15938,54.29894],[23.24656,54.25701],[23.3494,54.25155],[23.39525,54.21672],[23.42418,54.17911],[23.45223,54.17775],[23.49196,54.14764],[23.52702,54.04622],[23.48261,53.98855],[23.51284,53.95052],[23.61677,53.92691],[23.71726,53.93379],[23.80543,53.89558],[23.81309,53.94205],[23.95098,53.9613],[23.98837,53.92554],[24.19638,53.96405],[24.34128,53.90076],[24.44411,53.90076],[24.62275,54.00217],[24.69652,54.01901],[24.69185,53.96543],[24.74279,53.96663],[24.85311,54.02862],[24.77131,54.11091],[24.96894,54.17589],[24.991,54.14241],[25.0728,54.13419],[25.19199,54.219],[25.22705,54.26271],[25.35559,54.26544],[25.509,54.30267],[25.56823,54.25212],[25.51452,54.17799],[25.54724,54.14925],[25.64875,54.1259],[25.71084,54.16704],[25.78563,54.15747],[25.78553,54.23327],[25.68513,54.31727],[25.55425,54.31591],[25.5376,54.33158],[25.63371,54.42075],[25.62203,54.4656],[25.64813,54.48704],[25.68045,54.5321],[25.75977,54.57252],[25.74122,54.80108],[25.89462,54.93438],[25.99129,54.95705],[26.05907,54.94631],[26.13386,54.98924],[26.20397,54.99729],[26.26941,55.08032],[26.23202,55.10439],[26.30628,55.12536],[26.35121,55.1525],[26.46249,55.12814],[26.51481,55.16051],[26.54753,55.14181],[26.69243,55.16718],[26.68075,55.19787],[26.72983,55.21788],[26.73017,55.24226],[26.835,55.28182],[26.83266,55.30444],[26.80929,55.31642],[26.6714,55.33902],[26.5709,55.32572],[26.44937,55.34832],[26.5522,55.40277],[26.55094,55.5093],[26.63167,55.57887],[26.63231,55.67968],[26.58248,55.6754],[26.46661,55.70375],[26.39561,55.71156],[26.18509,55.86813],[26.03815,55.95884],[25.90047,56.0013],[25.85893,56.00188],[25.81773,56.05444],[25.69246,56.08892],[25.68588,56.14725],[25.53621,56.16663],[25.39751,56.15707],[25.23099,56.19147],[25.09325,56.1878],[25.05762,56.26742],[24.89005,56.46666]]]]}},{type:"Feature",properties:{iso1A2:"LU",iso1A3:"LUX",iso1N3:"442",wikidata:"Q32",nameEn:"Luxembourg",groups:["EU","155","150"],callingCodes:["352"]},geometry:{type:"MultiPolygon",coordinates:[[[[6.1379,50.12964],[6.1137,50.13668],[6.12028,50.16374],[6.08577,50.17246],[6.06406,50.15344],[6.03093,50.16362],[6.02488,50.18283],[5.96453,50.17259],[5.95929,50.13295],[5.89488,50.11476],[5.8857,50.07824],[5.85474,50.06342],[5.86904,50.04614],[5.8551,50.02683],[5.81866,50.01286],[5.82331,49.99662],[5.83968,49.9892],[5.83467,49.97823],[5.81163,49.97142],[5.80833,49.96451],[5.77291,49.96056],[5.77314,49.93646],[5.73621,49.89796],[5.78415,49.87922],[5.75269,49.8711],[5.75861,49.85631],[5.74567,49.85368],[5.75884,49.84811],[5.74953,49.84709],[5.74975,49.83933],[5.74076,49.83823],[5.7404,49.83452],[5.74844,49.82435],[5.74364,49.82058],[5.74953,49.81428],[5.75409,49.79239],[5.78871,49.7962],[5.82245,49.75048],[5.83149,49.74729],[5.82562,49.72395],[5.84193,49.72161],[5.86503,49.72739],[5.88677,49.70951],[5.86527,49.69291],[5.86175,49.67862],[5.9069,49.66377],[5.90164,49.6511],[5.90599,49.63853],[5.88552,49.63507],[5.88393,49.62802],[5.87609,49.62047],[5.8762,49.60898],[5.84826,49.5969],[5.84971,49.58674],[5.86986,49.58756],[5.87256,49.57539],[5.8424,49.56082],[5.84692,49.55663],[5.84143,49.5533],[5.81838,49.54777],[5.80871,49.5425],[5.81664,49.53775],[5.83648,49.5425],[5.84466,49.53027],[5.83467,49.52717],[5.83389,49.52152],[5.86571,49.50015],[5.94128,49.50034],[5.94224,49.49608],[5.96876,49.49053],[5.97693,49.45513],[6.02648,49.45451],[6.02743,49.44845],[6.04176,49.44801],[6.05553,49.46663],[6.07887,49.46399],[6.08373,49.45594],[6.10072,49.45268],[6.09845,49.46351],[6.10325,49.4707],[6.12346,49.4735],[6.12814,49.49365],[6.14321,49.48796],[6.16115,49.49297],[6.15366,49.50226],[6.17386,49.50934],[6.19543,49.50536],[6.2409,49.51408],[6.25029,49.50609],[6.27875,49.503],[6.28818,49.48465],[6.3687,49.4593],[6.36778,49.46937],[6.36907,49.48931],[6.36788,49.50377],[6.35666,49.52931],[6.38072,49.55171],[6.38228,49.55855],[6.35825,49.57053],[6.36676,49.57813],[6.38024,49.57593],[6.38342,49.5799],[6.37464,49.58886],[6.385,49.59946],[6.39822,49.60081],[6.41861,49.61723],[6.4413,49.65722],[6.43768,49.66021],[6.42726,49.66078],[6.42937,49.66857],[6.44654,49.67799],[6.46048,49.69092],[6.48014,49.69767],[6.49785,49.71118],[6.50647,49.71353],[6.5042,49.71808],[6.49694,49.72205],[6.49535,49.72645],[6.50261,49.72718],[6.51397,49.72058],[6.51805,49.72425],[6.50193,49.73291],[6.50174,49.75292],[6.51646,49.75961],[6.51828,49.76855],[6.51056,49.77515],[6.51669,49.78336],[6.50534,49.78952],[6.52169,49.79787],[6.53122,49.80666],[6.52121,49.81338],[6.51215,49.80124],[6.50647,49.80916],[6.48718,49.81267],[6.47111,49.82263],[6.45425,49.81164],[6.44131,49.81443],[6.42905,49.81091],[6.42521,49.81591],[6.40022,49.82029],[6.36576,49.85032],[6.34267,49.84974],[6.33585,49.83785],[6.32098,49.83728],[6.32303,49.85133],[6.30963,49.87021],[6.29692,49.86685],[6.28874,49.87592],[6.26146,49.88203],[6.23496,49.89972],[6.22926,49.92096],[6.21882,49.92403],[6.22608,49.929],[6.22094,49.94955],[6.19856,49.95053],[6.19089,49.96991],[6.18045,49.96611],[6.18554,49.95622],[6.17872,49.9537],[6.16466,49.97086],[6.1701,49.98518],[6.14147,49.99563],[6.14948,50.00908],[6.13806,50.01056],[6.1295,50.01849],[6.13273,50.02019],[6.13794,50.01466],[6.14666,50.02207],[6.13044,50.02929],[6.13458,50.04141],[6.11274,50.05916],[6.12055,50.09171],[6.1379,50.12964]]]]}},{type:"Feature",properties:{iso1A2:"LV",iso1A3:"LVA",iso1N3:"428",wikidata:"Q211",nameEn:"Latvia",groups:["EU","154","150"],callingCodes:["371"]},geometry:{type:"MultiPolygon",coordinates:[[[[27.34698,57.52242],[26.90364,57.62823],[26.54675,57.51813],[26.46527,57.56885],[26.29253,57.59244],[26.1866,57.6849],[26.2029,57.7206],[26.08098,57.76619],[26.0543,57.76105],[26.03332,57.7718],[26.02415,57.76865],[26.02069,57.77169],[26.0266,57.77441],[26.027,57.78158],[26.02456,57.78342],[26.0324,57.79037],[26.05949,57.84744],[25.73499,57.90193],[25.29581,58.08288],[25.28237,57.98539],[25.19484,58.0831],[24.3579,57.87471],[24.26221,57.91787],[23.20055,57.56697],[22.80496,57.87798],[19.84909,57.57876],[19.64795,57.06466],[20.68447,56.04073],[21.15016,56.07818],[21.24644,56.16917],[21.49736,56.29106],[21.57888,56.31406],[21.74558,56.33181],[22.00548,56.41508],[22.09728,56.42851],[22.3361,56.4016],[22.56441,56.39305],[22.69354,56.36284],[22.83048,56.367],[22.96988,56.41213],[23.09531,56.30511],[23.17312,56.36795],[23.31606,56.3827],[23.40486,56.37689],[23.49803,56.34307],[23.75726,56.37282],[24.02657,56.3231],[24.13139,56.24881],[24.32334,56.30226],[24.42746,56.26522],[24.58143,56.29125],[24.57353,56.31525],[24.70022,56.40483],[24.83686,56.41565],[24.89005,56.46666],[25.05762,56.26742],[25.09325,56.1878],[25.23099,56.19147],[25.39751,56.15707],[25.53621,56.16663],[25.68588,56.14725],[25.69246,56.08892],[25.81773,56.05444],[25.85893,56.00188],[25.90047,56.0013],[26.03815,55.95884],[26.18509,55.86813],[26.39561,55.71156],[26.46661,55.70375],[26.58248,55.6754],[26.63231,55.67968],[26.64888,55.70515],[26.71802,55.70645],[26.76872,55.67658],[26.87448,55.7172],[26.97153,55.8102],[27.1559,55.85032],[27.27804,55.78299],[27.3541,55.8089],[27.61683,55.78558],[27.63065,55.89687],[27.97865,56.11849],[28.15217,56.16964],[28.23716,56.27588],[28.16599,56.37806],[28.19057,56.44637],[28.10069,56.524],[28.13526,56.57989],[28.04768,56.59004],[27.86101,56.88204],[27.66511,56.83921],[27.86101,57.29402],[27.52453,57.42826],[27.56832,57.53728],[27.34698,57.52242]]]]}},{type:"Feature",properties:{iso1A2:"LY",iso1A3:"LBY",iso1N3:"434",wikidata:"Q1016",nameEn:"Libya",groups:["015","002"],callingCodes:["218"]},geometry:{type:"MultiPolygon",coordinates:[[[[22.5213,33.45682],[11.66543,33.34642],[11.56255,33.16754],[11.55852,33.1409],[11.51549,33.09826],[11.46037,32.6307],[11.57828,32.48013],[11.53898,32.4138],[11.04234,32.2145],[10.7315,31.97235],[10.62788,31.96629],[10.48497,31.72956],[10.31364,31.72648],[10.12239,31.42098],[10.29516,30.90337],[9.88152,30.34074],[9.76848,30.34366],[9.55544,30.23971],[9.3876,30.16738],[9.78136,29.40961],[9.89569,26.57696],[9.51696,26.39148],[9.38834,26.19288],[10.03146,25.35635],[10.02432,24.98124],[10.33159,24.5465],[10.85323,24.5595],[11.41061,24.21456],[11.62498,24.26669],[11.96886,23.51735],[13.5631,23.16574],[14.22918,22.61719],[14.99751,23.00539],[15.99566,23.49639],[23.99539,19.49944],[23.99715,20.00038],[24.99794,19.99661],[24.99885,21.99535],[24.99968,29.24574],[24.71117,30.17441],[25.01077,30.73861],[24.83101,31.31921],[25.06041,31.57937],[25.14001,31.67534],[25.63787,31.9359],[22.5213,33.45682]]]]}},{type:"Feature",properties:{iso1A2:"MA",iso1A3:"MAR",iso1N3:"504",wikidata:"Q1028",nameEn:"Morocco",groups:["015","002"],callingCodes:["212"]},geometry:{type:"MultiPolygon",coordinates:[[[[-2.27707,35.35051],[-2.85819,35.63219],[-5.10878,36.05227],[-5.64962,35.93752],[-7.27694,35.93599],[-14.43883,27.02969],[-17.27295,21.93519],[-17.21511,21.34226],[-17.02707,21.34022],[-16.9978,21.36239],[-16.44269,21.39745],[-14.78487,21.36587],[-14.47329,21.63839],[-14.48112,22.00886],[-14.1291,22.41636],[-14.10361,22.75501],[-13.75627,23.77231],[-13.00628,24.01923],[-12.92147,24.39502],[-12.12281,25.13682],[-12.06001,26.04442],[-11.62052,26.05229],[-11.38635,26.611],[-11.23622,26.72023],[-11.35695,26.8505],[-10.68417,26.90984],[-9.81998,26.71379],[-9.56957,26.90042],[-9.08698,26.98639],[-8.71787,26.9898],[-8.77527,27.66663],[-8.66879,27.6666],[-8.6715,28.71194],[-7.61585,29.36252],[-6.95824,29.50924],[-6.78351,29.44634],[-6.69965,29.51623],[-5.75616,29.61407],[-5.72121,29.52322],[-5.58831,29.48103],[-5.21671,29.95253],[-4.6058,30.28343],[-4.31774,30.53229],[-3.64735,30.67539],[-3.65418,30.85566],[-3.54944,31.0503],[-3.77103,31.14984],[-3.77647,31.31912],[-3.66386,31.39202],[-3.66314,31.6339],[-2.82784,31.79459],[-2.93873,32.06557],[-2.46166,32.16603],[-1.22829,32.07832],[-1.15735,32.12096],[-1.24453,32.1917],[-1.24998,32.32993],[-0.9912,32.52467],[-1.37794,32.73628],[-1.54244,32.95499],[-1.46249,33.0499],[-1.67067,33.27084],[-1.59508,33.59929],[-1.73494,33.71721],[-1.64666,34.10405],[-1.78042,34.39018],[-1.69788,34.48056],[-1.84569,34.61907],[-1.73707,34.74226],[-1.97469,34.886],[-1.97833,34.93218],[-2.04734,34.93218],[-2.21445,35.04378],[-2.21248,35.08532],[-2.27707,35.35051]],[[-2.92224,35.3401],[-2.92181,35.28599],[-2.92674,35.27313],[-2.93893,35.26737],[-2.95065,35.26576],[-2.95431,35.2728],[-2.96516,35.27967],[-2.96826,35.28296],[-2.96507,35.28801],[-2.97035,35.28852],[-2.96978,35.29459],[-2.96648,35.30475],[-2.96038,35.31609],[-2.92224,35.3401]],[[-3.90602,35.21494],[-3.90288,35.22024],[-3.88617,35.21406],[-3.88926,35.20841],[-3.90602,35.21494]],[[-4.30191,35.17419],[-4.29436,35.17149],[-4.30112,35.17058],[-4.30191,35.17419]],[[-2.41312,35.17111],[-2.44887,35.17075],[-2.44896,35.18777],[-2.41265,35.1877],[-2.41312,35.17111]],[[-5.38491,35.92591],[-5.27635,35.91222],[-5.27056,35.88794],[-5.34379,35.8711],[-5.35844,35.87375],[-5.37338,35.88417],[-5.38491,35.92591]]]]}},{type:"Feature",properties:{iso1A2:"MC",iso1A3:"MCO",iso1N3:"492",wikidata:"Q235",nameEn:"Monaco",groups:["155","150"],callingCodes:["377"]},geometry:{type:"MultiPolygon",coordinates:[[[[7.47823,43.73341],[7.4379,43.74963],[7.4389,43.75151],[7.43708,43.75197],[7.43624,43.75014],[7.43013,43.74895],[7.42809,43.74396],[7.42443,43.74087],[7.42299,43.74176],[7.42062,43.73977],[7.41233,43.73439],[7.41298,43.73311],[7.41291,43.73168],[7.41113,43.73156],[7.40903,43.7296],[7.42422,43.72209],[7.47823,43.73341]]]]}},{type:"Feature",properties:{iso1A2:"MD",iso1A3:"MDA",iso1N3:"498",wikidata:"Q217",nameEn:"Moldova",groups:["151","150"],callingCodes:["373"]},geometry:{type:"MultiPolygon",coordinates:[[[[27.74422,48.45926],[27.6658,48.44034],[27.59027,48.46311],[27.5889,48.49224],[27.46942,48.454],[27.44333,48.41209],[27.37741,48.41026],[27.37604,48.44398],[27.32159,48.4434],[27.27855,48.37534],[27.13434,48.37288],[27.08078,48.43214],[27.0231,48.42485],[27.03821,48.37653],[26.93384,48.36558],[26.85556,48.41095],[26.71274,48.40388],[26.82809,48.31629],[26.79239,48.29071],[26.6839,48.35828],[26.62823,48.25804],[26.81161,48.25049],[26.87708,48.19919],[26.94265,48.1969],[26.98042,48.15752],[26.96119,48.13003],[27.04118,48.12522],[27.02985,48.09083],[27.15622,47.98538],[27.1618,47.92391],[27.29069,47.73722],[27.25519,47.71366],[27.32202,47.64009],[27.3979,47.59473],[27.47942,47.48113],[27.55731,47.46637],[27.60263,47.32507],[27.68706,47.28962],[27.73172,47.29248],[27.81892,47.1381],[28.09095,46.97621],[28.12173,46.82283],[28.24808,46.64305],[28.22281,46.50481],[28.25769,46.43334],[28.18902,46.35283],[28.19864,46.31869],[28.10937,46.22852],[28.13684,46.18099],[28.08612,46.01105],[28.13111,45.92819],[28.16568,45.6421],[28.08927,45.6051],[28.18741,45.47358],[28.21139,45.46895],[28.30201,45.54744],[28.41836,45.51715],[28.43072,45.48538],[28.51449,45.49982],[28.49252,45.56716],[28.54196,45.58062],[28.51587,45.6613],[28.47879,45.66994],[28.52823,45.73803],[28.70401,45.78019],[28.69852,45.81753],[28.78503,45.83475],[28.74383,45.96664],[28.98004,46.00385],[29.00613,46.04962],[28.94643,46.09176],[29.06656,46.19716],[28.94953,46.25852],[28.98478,46.31803],[29.004,46.31495],[28.9306,46.45699],[29.01241,46.46177],[29.02409,46.49582],[29.23547,46.55435],[29.24886,46.37912],[29.35357,46.49505],[29.49914,46.45889],[29.5939,46.35472],[29.6763,46.36041],[29.66359,46.4215],[29.74496,46.45605],[29.88329,46.35851],[29.94114,46.40114],[30.09103,46.38694],[30.16794,46.40967],[30.02511,46.45132],[29.88916,46.54302],[29.94409,46.56002],[29.9743,46.75325],[29.94522,46.80055],[29.98814,46.82358],[29.87405,46.88199],[29.75458,46.8604],[29.72986,46.92234],[29.57056,46.94766],[29.62137,47.05069],[29.61038,47.09932],[29.53044,47.07851],[29.49732,47.12878],[29.57696,47.13581],[29.54996,47.24962],[29.59665,47.25521],[29.5733,47.36508],[29.48678,47.36043],[29.47854,47.30366],[29.39889,47.30179],[29.3261,47.44664],[29.18603,47.43387],[29.11743,47.55001],[29.22414,47.60012],[29.22242,47.73607],[29.27255,47.79953],[29.20663,47.80367],[29.27804,47.88893],[29.19839,47.89261],[29.1723,47.99013],[28.9306,47.96255],[28.8414,48.03392],[28.85232,48.12506],[28.69896,48.13106],[28.53921,48.17453],[28.48428,48.0737],[28.42454,48.12047],[28.43701,48.15832],[28.38712,48.17567],[28.34009,48.13147],[28.30609,48.14018],[28.30586,48.1597],[28.34912,48.1787],[28.36996,48.20543],[28.35519,48.24957],[28.32508,48.23384],[28.2856,48.23202],[28.19314,48.20749],[28.17666,48.25963],[28.07504,48.23494],[28.09873,48.3124],[28.04527,48.32661],[27.95883,48.32368],[27.88391,48.36699],[27.87533,48.4037],[27.81902,48.41874],[27.79225,48.44244],[27.74422,48.45926]]]]}},{type:"Feature",properties:{iso1A2:"ME",iso1A3:"MNE",iso1N3:"499",wikidata:"Q236",nameEn:"Montenegro",groups:["039","150"],callingCodes:["382"]},geometry:{type:"MultiPolygon",coordinates:[[[[19.22807,43.5264],[19.15685,43.53943],[19.13933,43.5282],[19.04934,43.50384],[19.01078,43.55806],[18.91379,43.50299],[18.95469,43.49367],[18.96053,43.45042],[19.01078,43.43854],[19.04071,43.397],[19.08673,43.31453],[19.08206,43.29668],[19.04233,43.30008],[19.00844,43.24988],[18.95001,43.29327],[18.95819,43.32899],[18.90911,43.36383],[18.83912,43.34795],[18.84794,43.33735],[18.85342,43.32426],[18.76538,43.29838],[18.6976,43.25243],[18.71747,43.2286],[18.66605,43.2056],[18.64735,43.14766],[18.66254,43.03928],[18.52232,43.01451],[18.49076,42.95553],[18.49661,42.89306],[18.4935,42.86433],[18.47633,42.85829],[18.45921,42.81682],[18.47324,42.74992],[18.56789,42.72074],[18.55221,42.69045],[18.54603,42.69171],[18.54841,42.68328],[18.57373,42.64429],[18.52232,42.62279],[18.55504,42.58409],[18.53751,42.57376],[18.49778,42.58409],[18.43735,42.55921],[18.44307,42.51077],[18.43588,42.48556],[18.52152,42.42302],[18.54128,42.39171],[18.45131,42.21682],[19.26406,41.74971],[19.37597,41.84849],[19.37451,41.8842],[19.33812,41.90669],[19.34601,41.95675],[19.37691,41.96977],[19.36867,42.02564],[19.37548,42.06835],[19.40687,42.10024],[19.28623,42.17745],[19.42,42.33019],[19.42352,42.36546],[19.4836,42.40831],[19.65972,42.62774],[19.73244,42.66299],[19.77375,42.58517],[19.74731,42.57422],[19.76549,42.50237],[19.82333,42.46581],[19.9324,42.51699],[20.00842,42.5109],[20.01834,42.54622],[20.07761,42.55582],[20.0969,42.65559],[20.02915,42.71147],[20.02088,42.74789],[20.04898,42.77701],[20.2539,42.76245],[20.27869,42.81945],[20.35692,42.8335],[20.34528,42.90676],[20.16415,42.97177],[20.14896,42.99058],[20.12325,42.96237],[20.05431,42.99571],[20.04729,43.02732],[19.98887,43.0538],[19.96549,43.11098],[19.92576,43.08539],[19.79255,43.11951],[19.76918,43.16044],[19.64063,43.19027],[19.62661,43.2286],[19.54598,43.25158],[19.52962,43.31623],[19.48171,43.32644],[19.44315,43.38846],[19.22229,43.47926],[19.22807,43.5264]]]]}},{type:"Feature",properties:{iso1A2:"MF",iso1A3:"MAF",iso1N3:"663",wikidata:"Q126125",nameEn:"Saint-Martin",country:"FR",groups:["EU","029","003","419","019"],callingCodes:["590"]},geometry:{type:"MultiPolygon",coordinates:[[[[-62.93924,18.02904],[-62.75637,18.13489],[-62.86666,18.19278],[-63.35989,18.06012],[-63.33064,17.9615],[-63.13584,18.0541],[-63.11096,18.05368],[-63.09686,18.04608],[-63.07759,18.04943],[-63.0579,18.06614],[-63.04039,18.05619],[-63.02323,18.05757],[-62.93924,18.02904]]]]}},{type:"Feature",properties:{iso1A2:"MG",iso1A3:"MDG",iso1N3:"450",wikidata:"Q1019",nameEn:"Madagascar",aliases:["RM"],groups:["014","202","002"],callingCodes:["261"]},geometry:{type:"MultiPolygon",coordinates:[[[[51.94557,-12.74579],[49.10033,-10.96054],[43.72277,-16.09877],[40.40841,-23.17181],[45.90777,-29.77366],[51.94557,-12.74579]]]]}},{type:"Feature",properties:{iso1A2:"MH",iso1A3:"MHL",iso1N3:"584",wikidata:"Q709",nameEn:"Marshall Islands",groups:["057","009"],roadSpeedUnit:"mph",callingCodes:["692"]},geometry:{type:"MultiPolygon",coordinates:[[[[169,3.9],[173.53711,5.70687],[169.29099,15.77133],[159.04653,10.59067],[169,3.9]]]]}},{type:"Feature",properties:{iso1A2:"MK",iso1A3:"MKD",iso1N3:"807",wikidata:"Q221",nameEn:"North Macedonia",groups:["039","150"],callingCodes:["389"]},geometry:{type:"MultiPolygon",coordinates:[[[[22.34773,42.31725],[22.29275,42.34913],[22.29605,42.37477],[22.16384,42.32103],[22.02908,42.29848],[21.94405,42.34669],[21.91595,42.30392],[21.84654,42.3247],[21.77176,42.2648],[21.70111,42.23789],[21.58992,42.25915],[21.52145,42.24465],[21.50823,42.27156],[21.43882,42.2789],[21.43882,42.23609],[21.38428,42.24465],[21.30496,42.1418],[21.29913,42.13954],[21.31983,42.10993],[21.22728,42.08909],[21.16614,42.19815],[21.11491,42.20794],[20.75464,42.05229],[20.76786,41.91839],[20.68523,41.85318],[20.59524,41.8818],[20.55976,41.87068],[20.57144,41.7897],[20.53405,41.78099],[20.51301,41.72433],[20.52937,41.69292],[20.51769,41.65975],[20.55508,41.58113],[20.52103,41.56473],[20.45809,41.5549],[20.45331,41.51436],[20.49039,41.49277],[20.51301,41.442],[20.55976,41.4087],[20.52119,41.34381],[20.49432,41.33679],[20.51068,41.2323],[20.59715,41.13644],[20.58546,41.11179],[20.59832,41.09066],[20.63454,41.0889],[20.65558,41.08009],[20.71634,40.91781],[20.73504,40.9081],[20.81567,40.89662],[20.83671,40.92752],[20.94305,40.92399],[20.97693,40.90103],[20.97887,40.85475],[21.15262,40.85546],[21.21105,40.8855],[21.25779,40.86165],[21.35595,40.87578],[21.41555,40.9173],[21.53007,40.90759],[21.57448,40.86076],[21.69601,40.9429],[21.7556,40.92525],[21.91102,41.04786],[21.90869,41.09191],[22.06527,41.15617],[22.1424,41.12449],[22.17629,41.15969],[22.26744,41.16409],[22.42285,41.11921],[22.5549,41.13065],[22.58295,41.11568],[22.62852,41.14385],[22.65306,41.18168],[22.71266,41.13945],[22.74538,41.16321],[22.76408,41.32225],[22.81199,41.3398],[22.93334,41.34104],[22.96331,41.35782],[22.95513,41.63265],[23.03342,41.71034],[23.01239,41.76527],[22.96682,41.77137],[22.90254,41.87587],[22.86749,42.02275],[22.67701,42.06614],[22.51224,42.15457],[22.50289,42.19527],[22.47251,42.20393],[22.38136,42.30339],[22.34773,42.31725]]]]}},{type:"Feature",properties:{iso1A2:"ML",iso1A3:"MLI",iso1N3:"466",wikidata:"Q912",nameEn:"Mali",groups:["011","202","002"],callingCodes:["223"]},geometry:{type:"MultiPolygon",coordinates:[[[[-4.83423,24.99935],[-6.57191,25.0002],[-5.60725,16.49919],[-5.33435,16.33354],[-5.50165,15.50061],[-9.32979,15.50032],[-9.31106,15.69412],[-9.33314,15.7044],[-9.44673,15.60553],[-9.40447,15.4396],[-10.71721,15.4223],[-10.90932,15.11001],[-11.43483,15.62339],[-11.70705,15.51558],[-11.94903,14.76143],[-12.23936,14.76324],[-11.93043,13.84505],[-12.06897,13.71049],[-11.83345,13.33333],[-11.63025,13.39174],[-11.39935,12.97808],[-11.37536,12.40788],[-11.50006,12.17826],[-11.24136,12.01286],[-10.99758,12.24634],[-10.80355,12.1053],[-10.71897,11.91552],[-10.30604,12.24634],[-9.714,12.0226],[-9.63938,12.18312],[-9.32097,12.29009],[-9.38067,12.48446],[-9.13689,12.50875],[-8.94784,12.34842],[-8.80854,11.66715],[-8.40058,11.37466],[-8.66923,10.99397],[-8.35083,11.06234],[-8.2667,10.91762],[-8.32614,10.69273],[-8.22711,10.41722],[-8.10207,10.44649],[-7.9578,10.2703],[-7.97971,10.17117],[-7.92107,10.15577],[-7.63048,10.46334],[-7.54462,10.40921],[-7.52261,10.4655],[-7.44555,10.44602],[-7.3707,10.24677],[-7.13331,10.24877],[-7.0603,10.14711],[-7.00966,10.15794],[-6.97444,10.21644],[-7.01186,10.25111],[-6.93921,10.35291],[-6.68164,10.35074],[-6.63541,10.66893],[-6.52974,10.59104],[-6.42847,10.5694],[-6.40646,10.69922],[-6.325,10.68624],[-6.24795,10.74248],[-6.1731,10.46983],[-6.18851,10.24244],[-5.99478,10.19694],[-5.78124,10.43952],[-5.65135,10.46767],[-5.51058,10.43177],[-5.46643,10.56074],[-5.47083,10.75329],[-5.41579,10.84628],[-5.49284,11.07538],[-5.32994,11.13371],[-5.32553,11.21578],[-5.25949,11.24816],[-5.25509,11.36905],[-5.20665,11.43811],[-5.22867,11.60421],[-5.29251,11.61715],[-5.26389,11.75728],[-5.40258,11.8327],[-5.26389,11.84778],[-5.07897,11.97918],[-4.72893,12.01579],[-4.70692,12.06746],[-4.62987,12.06531],[-4.62546,12.13204],[-4.54841,12.1385],[-4.57703,12.19875],[-4.41412,12.31922],[-4.47356,12.71252],[-4.238,12.71467],[-4.21819,12.95722],[-4.34477,13.12927],[-3.96501,13.49778],[-3.90558,13.44375],[-3.96282,13.38164],[-3.7911,13.36665],[-3.54454,13.1781],[-3.4313,13.1588],[-3.43507,13.27272],[-3.23599,13.29035],[-3.28396,13.5422],[-3.26407,13.70699],[-2.88189,13.64921],[-2.90831,13.81174],[-2.84667,14.05532],[-2.66175,14.14713],[-2.47587,14.29671],[-2.10223,14.14878],[-1.9992,14.19011],[-1.97945,14.47709],[-1.68083,14.50023],[-1.32166,14.72774],[-1.05875,14.7921],[-0.72004,15.08655],[-0.24673,15.07805],[0.06588,14.96961],[0.23859,15.00135],[0.72632,14.95898],[0.96711,14.98275],[1.31275,15.27978],[3.01806,15.34571],[3.03134,15.42221],[3.50368,15.35934],[4.19893,16.39923],[4.21787,17.00118],[4.26762,17.00432],[4.26651,19.14224],[3.36082,18.9745],[3.12501,19.1366],[3.24648,19.81703],[1.20992,20.73533],[1.15698,21.12843],[-4.83423,24.99935]]]]}},{type:"Feature",properties:{iso1A2:"MM",iso1A3:"MMR",iso1N3:"104",wikidata:"Q836",nameEn:"Myanmar",aliases:["Burma","BU"],groups:["035","142"],callingCodes:["95"]},geometry:{type:"MultiPolygon",coordinates:[[[[92.62187,21.87037],[92.59775,21.6092],[92.68152,21.28454],[92.60187,21.24615],[92.55105,21.3856],[92.43158,21.37025],[92.37939,21.47764],[92.20087,21.337],[92.17752,21.17445],[92.26071,21.05697],[92.37665,20.72172],[92.28464,20.63179],[92.31348,20.57137],[92.4302,20.5688],[92.39837,20.38919],[92.61042,13.76986],[94.6371,13.81803],[97.63455,9.60854],[98.12555,9.44056],[98.33094,9.91973],[98.47298,9.95782],[98.52291,9.92389],[98.55174,9.92804],[98.7391,10.31488],[98.81944,10.52761],[98.77275,10.62548],[98.78511,10.68351],[98.86819,10.78336],[99.0069,10.85485],[98.99701,10.92962],[99.02337,10.97217],[99.06938,10.94857],[99.32756,11.28545],[99.31573,11.32081],[99.39485,11.3925],[99.47598,11.62434],[99.5672,11.62732],[99.64108,11.78948],[99.64891,11.82699],[99.53424,12.02317],[99.56445,12.14805],[99.47519,12.1353],[99.409,12.60603],[99.29254,12.68921],[99.18905,12.84799],[99.18748,12.9898],[99.10646,13.05804],[99.12225,13.19847],[99.20617,13.20575],[99.16695,13.72621],[98.97356,14.04868],[98.56762,14.37701],[98.24874,14.83013],[98.18821,15.13125],[98.22,15.21327],[98.30446,15.30667],[98.40522,15.25268],[98.41906,15.27103],[98.39351,15.34177],[98.4866,15.39154],[98.56027,15.33471],[98.58598,15.46821],[98.541,15.65406],[98.59853,15.87197],[98.57019,16.04578],[98.69585,16.13353],[98.8376,16.11706],[98.92656,16.36425],[98.84485,16.42354],[98.68074,16.27068],[98.63817,16.47424],[98.57912,16.55983],[98.5695,16.62826],[98.51113,16.64503],[98.51833,16.676],[98.51472,16.68521],[98.51579,16.69433],[98.51043,16.70107],[98.49713,16.69022],[98.50253,16.7139],[98.46994,16.73613],[98.53833,16.81934],[98.49603,16.8446],[98.52624,16.89979],[98.39441,17.06266],[98.34566,17.04822],[98.10439,17.33847],[98.11185,17.36829],[97.91829,17.54504],[97.76407,17.71595],[97.66794,17.88005],[97.73723,17.97912],[97.60841,18.23846],[97.64116,18.29778],[97.56219,18.33885],[97.50383,18.26844],[97.34522,18.54596],[97.36444,18.57138],[97.5258,18.4939],[97.76752,18.58097],[97.73836,18.88478],[97.66487,18.9371],[97.73654,18.9812],[97.73797,19.04261],[97.83479,19.09972],[97.84024,19.22217],[97.78606,19.26769],[97.84186,19.29526],[97.78769,19.39429],[97.88423,19.5041],[97.84715,19.55782],[98.04364,19.65755],[98.03314,19.80941],[98.13829,19.78541],[98.24884,19.67876],[98.51182,19.71303],[98.56065,19.67807],[98.83661,19.80931],[98.98679,19.7419],[99.0735,20.10298],[99.20328,20.12877],[99.416,20.08614],[99.52943,20.14811],[99.5569,20.20676],[99.46077,20.36198],[99.46008,20.39673],[99.68255,20.32077],[99.81096,20.33687],[99.86383,20.44371],[99.88211,20.44488],[99.88451,20.44596],[99.89168,20.44548],[99.89301,20.44311],[99.89692,20.44789],[99.90499,20.4487],[99.91616,20.44986],[99.95721,20.46301],[100.08404,20.36626],[100.1957,20.68247],[100.36375,20.82783],[100.51079,20.82194],[100.60112,20.8347],[100.64628,20.88279],[100.50974,20.88574],[100.55281,21.02796],[100.63578,21.05639],[100.72716,21.31786],[100.80173,21.2934],[101.00234,21.39612],[101.16198,21.52808],[101.15156,21.56129],[101.11744,21.77659],[100.87265,21.67396],[100.72143,21.51898],[100.57861,21.45637],[100.4811,21.46148],[100.42892,21.54325],[100.35201,21.53176],[100.25863,21.47043],[100.18447,21.51898],[100.1625,21.48704],[100.12542,21.50365],[100.10757,21.59945],[100.17486,21.65306],[100.12679,21.70539],[100.04956,21.66843],[99.98654,21.71064],[99.94003,21.82782],[99.99084,21.97053],[99.96612,22.05965],[99.85351,22.04183],[99.47585,22.13345],[99.33166,22.09656],[99.1552,22.15874],[99.19176,22.16983],[99.17318,22.18025],[99.28771,22.4105],[99.37972,22.50188],[99.38247,22.57544],[99.31243,22.73893],[99.45654,22.85726],[99.43537,22.94086],[99.54218,22.90014],[99.52214,23.08218],[99.34127,23.13099],[99.25741,23.09025],[99.04601,23.12215],[99.05975,23.16382],[98.88597,23.18656],[98.92515,23.29535],[98.93958,23.31414],[98.87573,23.33038],[98.92104,23.36946],[98.87683,23.48995],[98.82877,23.47908],[98.80294,23.5345],[98.88396,23.59555],[98.81775,23.694],[98.82933,23.72921],[98.79607,23.77947],[98.68209,23.80492],[98.67797,23.9644],[98.89632,24.10612],[98.87998,24.15624],[98.85319,24.13042],[98.59256,24.08371],[98.54476,24.13119],[98.20666,24.11406],[98.07806,24.07988],[98.06703,24.08028],[98.0607,24.07812],[98.05671,24.07961],[98.05302,24.07408],[98.04709,24.07616],[97.99583,24.04932],[97.98691,24.03897],[97.93951,24.01953],[97.90998,24.02094],[97.88616,24.00463],[97.88414,23.99405],[97.88814,23.98605],[97.89683,23.98389],[97.89676,23.97931],[97.8955,23.97758],[97.88811,23.97446],[97.86545,23.97723],[97.84328,23.97603],[97.79416,23.95663],[97.79456,23.94836],[97.72302,23.89288],[97.64667,23.84574],[97.5247,23.94032],[97.62363,24.00506],[97.72903,24.12606],[97.75305,24.16902],[97.72799,24.18883],[97.72998,24.2302],[97.76799,24.26365],[97.71941,24.29652],[97.66723,24.30027],[97.65624,24.33781],[97.7098,24.35658],[97.66998,24.45288],[97.60029,24.4401],[97.52757,24.43748],[97.56286,24.54535],[97.56525,24.72838],[97.54675,24.74202],[97.5542,24.74943],[97.56383,24.75535],[97.56648,24.76475],[97.64354,24.79171],[97.70181,24.84557],[97.73127,24.83015],[97.76481,24.8289],[97.79949,24.85655],[97.72903,24.91332],[97.72216,25.08508],[97.77023,25.11492],[97.83614,25.2715],[97.92541,25.20815],[98.14925,25.41547],[98.12591,25.50722],[98.18084,25.56298],[98.16848,25.62739],[98.25774,25.6051],[98.31268,25.55307],[98.40606,25.61129],[98.54064,25.85129],[98.63128,25.79937],[98.70818,25.86241],[98.60763,26.01512],[98.57085,26.11547],[98.63128,26.15492],[98.66884,26.09165],[98.7329,26.17218],[98.67797,26.24487],[98.72741,26.36183],[98.77547,26.61994],[98.7333,26.85615],[98.69582,27.56499],[98.43353,27.67086],[98.42529,27.55404],[98.32641,27.51385],[98.13964,27.9478],[98.15337,28.12114],[97.90069,28.3776],[97.79632,28.33168],[97.70705,28.5056],[97.56835,28.55628],[97.50518,28.49716],[97.47085,28.2688],[97.41729,28.29783],[97.34547,28.21385],[97.31292,28.06784],[97.35412,28.06663],[97.38845,28.01329],[97.35824,27.87256],[97.29919,27.92233],[96.90112,27.62149],[96.91431,27.45752],[97.17422,27.14052],[97.14675,27.09041],[96.89132,27.17474],[96.85287,27.2065],[96.88445,27.25046],[96.73888,27.36638],[96.55761,27.29928],[96.40779,27.29818],[96.15591,27.24572],[96.04949,27.19428],[95.93002,27.04149],[95.81603,27.01335],[95.437,26.7083],[95.30339,26.65372],[95.23513,26.68499],[95.05798,26.45408],[95.12801,26.38397],[95.11428,26.1019],[95.18556,26.07338],[94.80117,25.49359],[94.68032,25.47003],[94.57458,25.20318],[94.74212,25.13606],[94.73937,25.00545],[94.60204,24.70889],[94.5526,24.70764],[94.50729,24.59281],[94.45279,24.56656],[94.32362,24.27692],[94.30215,24.23752],[94.14081,23.83333],[93.92089,23.95812],[93.80279,23.92549],[93.75952,24.0003],[93.62871,24.00922],[93.50616,23.94432],[93.46633,23.97067],[93.41415,24.07854],[93.34735,24.10151],[93.32351,24.04468],[93.36059,23.93176],[93.3908,23.92925],[93.3908,23.7622],[93.43475,23.68299],[93.38805,23.4728],[93.39981,23.38828],[93.38781,23.36139],[93.36862,23.35426],[93.38478,23.13698],[93.2878,23.00464],[93.12988,23.05772],[93.134,22.92498],[93.09417,22.69459],[93.134,22.59573],[93.11477,22.54374],[93.13537,22.45873],[93.18206,22.43716],[93.19991,22.25425],[93.14224,22.24535],[93.15734,22.18687],[93.04885,22.20595],[92.99255,22.05965],[92.99804,21.98964],[92.93899,22.02656],[92.89504,21.95143],[92.86208,22.05456],[92.70416,22.16017],[92.67532,22.03547],[92.60949,21.97638],[92.62187,21.87037]]]]}},{type:"Feature",properties:{iso1A2:"MN",iso1A3:"MNG",iso1N3:"496",wikidata:"Q711",nameEn:"Mongolia",groups:["030","142"],callingCodes:["976"]},geometry:{type:"MultiPolygon",coordinates:[[[[102.14032,51.35566],[101.5044,51.50467],[101.39085,51.45753],[100.61116,51.73028],[99.89203,51.74903],[99.75578,51.90108],[99.27888,51.96876],[98.87768,52.14563],[98.74142,51.8637],[98.33222,51.71832],[98.22053,51.46579],[98.05257,51.46696],[97.83305,51.00248],[98.01472,50.86652],[97.9693,50.78044],[98.06393,50.61262],[98.31373,50.4996],[98.29481,50.33561],[97.85197,49.91339],[97.76871,49.99861],[97.56432,49.92801],[97.56811,49.84265],[97.24639,49.74737],[96.97388,49.88413],[95.80056,50.04239],[95.74757,49.97915],[95.02465,49.96941],[94.97166,50.04725],[94.6121,50.04239],[94.49477,50.17832],[94.39258,50.22193],[94.30823,50.57498],[92.99595,50.63183],[93.01109,50.79001],[92.44714,50.78762],[92.07173,50.69585],[91.86048,50.73734],[89.59711,49.90851],[89.70687,49.72535],[88.82499,49.44808],[88.42449,49.48821],[88.17223,49.46934],[88.15543,49.30314],[87.98977,49.18147],[87.81333,49.17354],[87.88171,48.95853],[87.73822,48.89582],[88.0788,48.71436],[87.96361,48.58478],[88.58939,48.34531],[88.58316,48.21893],[88.8011,48.11302],[88.93186,48.10263],[89.0711,47.98528],[89.55453,48.0423],[89.76624,47.82745],[90.06512,47.88177],[90.10871,47.7375],[90.33598,47.68303],[90.48854,47.41826],[90.48542,47.30438],[90.76108,46.99399],[90.84035,46.99525],[91.03649,46.72916],[91.0147,46.58171],[91.07696,46.57315],[90.89639,46.30711],[90.99672,46.14207],[91.03026,46.04194],[90.70907,45.73437],[90.65114,45.49314],[90.89169,45.19667],[91.64048,45.07408],[93.51161,44.95964],[94.10003,44.71016],[94.71959,44.35284],[95.01191,44.25274],[95.39772,44.2805],[95.32891,44.02407],[95.52594,43.99353],[95.89543,43.2528],[96.35658,42.90363],[96.37926,42.72055],[97.1777,42.7964],[99.50671,42.56535],[100.33297,42.68231],[100.84979,42.67087],[101.28833,42.58524],[101.80515,42.50074],[102.07645,42.22519],[102.42826,42.15137],[102.72403,42.14675],[103.3685,41.89696],[103.92804,41.78246],[104.52258,41.8706],[104.51667,41.66113],[104.91272,41.64619],[105.01119,41.58382],[105.24708,41.7442],[106.76517,42.28741],[107.24774,42.36107],[107.29755,42.41395],[107.49681,42.46221],[107.57258,42.40898],[108.23156,42.45532],[108.84489,42.40246],[109.00679,42.45302],[109.452,42.44842],[109.89402,42.63111],[110.08401,42.6411],[110.4327,42.78293],[111.0149,43.3289],[111.59087,43.51207],[111.79758,43.6637],[111.93776,43.68709],[111.96289,43.81596],[111.40498,44.3461],[111.76275,44.98032],[111.98695,45.09074],[112.4164,45.06858],[112.74662,44.86297],[113.63821,44.74326],[113.909,44.91444],[114.08071,44.92847],[114.5166,45.27189],[114.54801,45.38337],[114.74612,45.43585],[114.94546,45.37377],[115.35757,45.39106],[115.69688,45.45761],[115.91898,45.6227],[116.16989,45.68603],[116.27366,45.78637],[116.24012,45.8778],[116.26678,45.96479],[116.58612,46.30211],[116.75551,46.33083],[116.83166,46.38637],[117.07252,46.35818],[117.36609,46.36335],[117.41782,46.57862],[117.60748,46.59771],[117.69554,46.50991],[118.30534,46.73519],[118.78747,46.68689],[118.8337,46.77742],[118.89974,46.77139],[118.92616,46.72765],[119.00541,46.74273],[119.10448,46.65516],[119.24978,46.64761],[119.30261,46.6083],[119.37306,46.61132],[119.42827,46.63783],[119.65265,46.62342],[119.68127,46.59015],[119.77373,46.62947],[119.80455,46.67631],[119.89261,46.66423],[119.91242,46.90091],[119.85518,46.92196],[119.71209,47.19192],[119.62403,47.24575],[119.56019,47.24874],[119.54918,47.29505],[119.31964,47.42617],[119.35892,47.48104],[119.13995,47.53997],[119.12343,47.66458],[118.7564,47.76947],[118.55766,47.99277],[118.29654,48.00246],[118.22677,48.03853],[118.11009,48.04],[118.03676,48.00982],[117.80196,48.01661],[117.50181,47.77216],[117.37875,47.63627],[117.08918,47.82242],[116.87527,47.88836],[116.67405,47.89039],[116.4465,47.83662],[116.2527,47.87766],[116.08431,47.80693],[115.94296,47.67741],[115.57128,47.91988],[115.52082,48.15367],[115.811,48.25699],[115.78876,48.51781],[116.06565,48.81716],[116.03781,48.87014],[116.71193,49.83813],[116.62502,49.92919],[116.22402,50.04477],[115.73602,49.87688],[115.26068,49.97367],[114.9703,50.19254],[114.325,50.28098],[113.20216,49.83356],[113.02647,49.60772],[110.64493,49.1816],[110.39891,49.25083],[110.24373,49.16676],[109.51325,49.22859],[109.18017,49.34709],[108.53969,49.32325],[108.27937,49.53167],[107.95387,49.66659],[107.96116,49.93191],[107.36407,49.97612],[107.1174,50.04239],[107.00007,50.1977],[106.80326,50.30177],[106.58373,50.34044],[106.51122,50.34408],[106.49628,50.32436],[106.47156,50.31909],[106.07865,50.33474],[106.05562,50.40582],[105.32528,50.4648],[103.70343,50.13952],[102.71178,50.38873],[102.32194,50.67982],[102.14032,51.35566]]]]}},{type:"Feature",properties:{iso1A2:"MO",iso1A3:"MAC",iso1N3:"446",wikidata:"Q14773",nameEn:"Macau",aliases:["Macao"],country:"CN",groups:["030","142"],driveSide:"left",callingCodes:["853"]},geometry:{type:"MultiPolygon",coordinates:[[[[113.54942,22.14519],[113.54839,22.10909],[113.57191,22.07696],[113.63011,22.10782],[113.60504,22.20464],[113.57123,22.20416],[113.56865,22.20973],[113.5508,22.21672],[113.54333,22.21688],[113.54093,22.21314],[113.53593,22.2137],[113.53301,22.21235],[113.53552,22.20607],[113.52659,22.18271],[113.54093,22.15497],[113.54942,22.14519]]]]}},{type:"Feature",properties:{iso1A2:"MP",iso1A3:"MNP",iso1N3:"580",wikidata:"Q16644",nameEn:"Northern Mariana Islands",country:"US",groups:["057","009"],roadSpeedUnit:"mph",callingCodes:["1 670"]},geometry:{type:"MultiPolygon",coordinates:[[[[143.82485,13.92273],[146.25931,13.85876],[146.6755,21.00809],[144.18594,21.03576],[143.82485,13.92273]]]]}},{type:"Feature",properties:{iso1A2:"MQ",iso1A3:"MTQ",iso1N3:"474",wikidata:"Q17054",nameEn:"Martinique",country:"FR",groups:["EU","029","003","419","019"],callingCodes:["596"]},geometry:{type:"MultiPolygon",coordinates:[[[[-60.5958,14.23076],[-60.69955,15.22234],[-61.51867,14.96709],[-61.26561,14.25664],[-60.5958,14.23076]]]]}},{type:"Feature",properties:{iso1A2:"MR",iso1A3:"MRT",iso1N3:"478",wikidata:"Q1025",nameEn:"Mauritania",groups:["011","202","002"],callingCodes:["222"]},geometry:{type:"MultiPolygon",coordinates:[[[[-5.60725,16.49919],[-6.57191,25.0002],[-4.83423,24.99935],[-8.66674,27.31569],[-8.66721,25.99918],[-12.0002,25.9986],[-12.00251,23.4538],[-12.14969,23.41935],[-12.36213,23.3187],[-12.5741,23.28975],[-13.00412,23.02297],[-13.10753,22.89493],[-13.15313,22.75649],[-13.08438,22.53866],[-13.01525,21.33343],[-16.95474,21.33997],[-16.99806,21.12142],[-17.0357,21.05368],[-17.0396,20.9961],[-17.06781,20.92697],[-17.0695,20.85742],[-17.0471,20.76408],[-17.15288,16.07139],[-16.50854,16.09032],[-16.48967,16.0496],[-16.44814,16.09753],[-16.4429,16.20605],[-16.27016,16.51565],[-15.6509,16.50315],[-15.00557,16.64997],[-14.32144,16.61495],[-13.80075,16.13961],[-13.43135,16.09022],[-13.11029,15.52116],[-12.23936,14.76324],[-11.94903,14.76143],[-11.70705,15.51558],[-11.43483,15.62339],[-10.90932,15.11001],[-10.71721,15.4223],[-9.40447,15.4396],[-9.44673,15.60553],[-9.33314,15.7044],[-9.31106,15.69412],[-9.32979,15.50032],[-5.50165,15.50061],[-5.33435,16.33354],[-5.60725,16.49919]]]]}},{type:"Feature",properties:{iso1A2:"MS",iso1A3:"MSR",iso1N3:"500",wikidata:"Q13353",nameEn:"Montserrat",country:"GB",groups:["029","003","419","019"],driveSide:"left",callingCodes:["1 664"]},geometry:{type:"MultiPolygon",coordinates:[[[[-61.83929,16.66647],[-62.14123,17.02632],[-62.52079,16.69392],[-62.17275,16.35721],[-61.83929,16.66647]]]]}},{type:"Feature",properties:{iso1A2:"MT",iso1A3:"MLT",iso1N3:"470",wikidata:"Q233",nameEn:"Malta",groups:["EU","039","150"],driveSide:"left",callingCodes:["356"]},geometry:{type:"MultiPolygon",coordinates:[[[[15.70991,35.79901],[14.07544,36.41525],[13.27636,35.20764],[15.70991,35.79901]]]]}},{type:"Feature",properties:{iso1A2:"MU",iso1A3:"MUS",iso1N3:"480",wikidata:"Q1027",nameEn:"Mauritius",groups:["014","202","002"],driveSide:"left",callingCodes:["230"]},geometry:{type:"MultiPolygon",coordinates:[[[[56.73473,-21.9174],[64.11105,-21.5783],[63.47388,-9.1938],[56.09755,-9.55401],[56.73473,-21.9174]]]]}},{type:"Feature",properties:{iso1A2:"MV",iso1A3:"MDV",iso1N3:"462",wikidata:"Q826",nameEn:"Maldives",groups:["034","142"],driveSide:"left",callingCodes:["960"]},geometry:{type:"MultiPolygon",coordinates:[[[[71.27292,7.36038],[73.37814,-3.88401],[74.6203,7.39289],[71.27292,7.36038]]]]}},{type:"Feature",properties:{iso1A2:"MW",iso1A3:"MWI",iso1N3:"454",wikidata:"Q1020",nameEn:"Malawi",groups:["014","202","002"],driveSide:"left",callingCodes:["265"]},geometry:{type:"MultiPolygon",coordinates:[[[[33.48052,-9.62442],[33.31581,-9.48554],[33.14925,-9.49322],[32.99397,-9.36712],[32.95389,-9.40138],[33.00476,-9.5133],[33.00256,-9.63053],[33.05485,-9.61316],[33.10163,-9.66525],[33.12144,-9.58929],[33.2095,-9.61099],[33.31517,-9.82364],[33.36581,-9.81063],[33.37902,-9.9104],[33.31297,-10.05133],[33.53863,-10.20148],[33.54797,-10.36077],[33.70675,-10.56896],[33.47636,-10.78465],[33.28022,-10.84428],[33.25998,-10.88862],[33.39697,-11.15296],[33.29267,-11.3789],[33.29267,-11.43536],[33.23663,-11.40637],[33.24252,-11.59302],[33.32692,-11.59248],[33.33937,-11.91252],[33.25998,-12.14242],[33.3705,-12.34931],[33.47636,-12.32498],[33.54485,-12.35996],[33.37517,-12.54085],[33.28177,-12.54692],[33.18837,-12.61377],[33.05917,-12.59554],[32.94397,-12.76868],[32.96733,-12.88251],[33.02181,-12.88707],[32.98289,-13.12671],[33.0078,-13.19492],[32.86113,-13.47292],[32.84176,-13.52794],[32.73683,-13.57682],[32.68436,-13.55769],[32.66468,-13.60019],[32.68654,-13.64268],[32.7828,-13.64805],[32.84528,-13.71576],[32.76962,-13.77224],[32.79015,-13.80755],[32.88985,-13.82956],[32.99042,-13.95689],[33.02977,-14.05022],[33.07568,-13.98447],[33.16749,-13.93992],[33.24249,-14.00019],[33.66677,-14.61306],[33.7247,-14.4989],[33.88503,-14.51652],[33.92898,-14.47929],[34.08588,-14.48893],[34.18733,-14.43823],[34.22355,-14.43607],[34.34453,-14.3985],[34.35843,-14.38652],[34.39277,-14.39467],[34.4192,-14.43191],[34.44641,-14.47746],[34.45053,-14.49873],[34.47628,-14.53363],[34.48932,-14.53646],[34.49636,-14.55091],[34.52366,-14.5667],[34.53962,-14.59776],[34.55112,-14.64494],[34.53516,-14.67782],[34.52057,-14.68263],[34.54503,-14.74672],[34.567,-14.77345],[34.61522,-14.99583],[34.57503,-15.30619],[34.43126,-15.44778],[34.44981,-15.60864],[34.25195,-15.90321],[34.43126,-16.04737],[34.40344,-16.20923],[35.04805,-16.83167],[35.13771,-16.81687],[35.17017,-16.93521],[35.04805,-17.00027],[35.0923,-17.13235],[35.3062,-17.1244],[35.27065,-16.93817],[35.30929,-16.82871],[35.27219,-16.69402],[35.14235,-16.56812],[35.25828,-16.4792],[35.30157,-16.2211],[35.43355,-16.11371],[35.52365,-16.15414],[35.70107,-16.10147],[35.80487,-16.03907],[35.85303,-15.41913],[35.78799,-15.17428],[35.91812,-14.89514],[35.87212,-14.89478],[35.86945,-14.67481],[35.5299,-14.27714],[35.47989,-14.15594],[34.86229,-13.48958],[34.60253,-13.48487],[34.37831,-12.17408],[34.46088,-12.0174],[34.70739,-12.15652],[34.82903,-12.04837],[34.57917,-11.87849],[34.64241,-11.57499],[34.96296,-11.57354],[34.91153,-11.39799],[34.79375,-11.32245],[34.63305,-11.11731],[34.61161,-11.01611],[34.67047,-10.93796],[34.65946,-10.6828],[34.57581,-10.56271],[34.51911,-10.12279],[34.54499,-10.0678],[34.03865,-9.49398],[33.95829,-9.54066],[33.9638,-9.62206],[33.93298,-9.71647],[33.76677,-9.58516],[33.48052,-9.62442]]]]}},{type:"Feature",properties:{iso1A2:"MX",iso1A3:"MEX",iso1N3:"484",wikidata:"Q96",nameEn:"Mexico",groups:["013","003","419","019"],callingCodes:["52"]},geometry:{type:"MultiPolygon",coordinates:[[[[-117.1243,32.53427],[-118.48109,32.5991],[-120.12904,18.41089],[-92.37213,14.39277],[-92.2261,14.53423],[-92.1454,14.6804],[-92.18161,14.84147],[-92.1423,14.88647],[-92.1454,14.98143],[-92.0621,15.07406],[-92.20983,15.26077],[-91.73182,16.07371],[-90.44567,16.07573],[-90.40499,16.40524],[-90.61212,16.49832],[-90.69064,16.70697],[-91.04436,16.92175],[-91.43809,17.25373],[-90.99199,17.25192],[-90.98678,17.81655],[-89.14985,17.81563],[-89.15105,17.95104],[-89.03839,18.0067],[-88.8716,17.89535],[-88.71505,18.0707],[-88.48242,18.49164],[-88.3268,18.49048],[-88.29909,18.47591],[-88.26593,18.47617],[-88.03238,18.41778],[-88.03165,18.16657],[-87.90671,18.15213],[-87.87604,18.18313],[-87.86657,18.19971],[-87.85693,18.18266],[-87.84815,18.18511],[-86.92368,17.61462],[-85.9092,21.8218],[-96.92418,25.97377],[-97.13927,25.96583],[-97.35946,25.92189],[-97.37332,25.83854],[-97.42511,25.83969],[-97.45669,25.86874],[-97.49828,25.89877],[-97.52025,25.88518],[-97.66511,26.01708],[-97.95155,26.0625],[-97.97017,26.05232],[-98.24603,26.07191],[-98.27075,26.09457],[-98.30491,26.10475],[-98.35126,26.15129],[-99.00546,26.3925],[-99.03053,26.41249],[-99.08477,26.39849],[-99.53573,27.30926],[-99.49744,27.43746],[-99.482,27.47128],[-99.48045,27.49016],[-99.50208,27.50021],[-99.52955,27.49747],[-99.51478,27.55836],[-99.55409,27.61314],[-100.50029,28.66117],[-100.51222,28.70679],[-100.5075,28.74066],[-100.52313,28.75598],[-100.59809,28.88197],[-100.63689,28.90812],[-100.67294,29.09744],[-100.79696,29.24688],[-100.87982,29.296],[-100.94056,29.33371],[-100.94579,29.34523],[-100.96725,29.3477],[-101.01128,29.36947],[-101.05686,29.44738],[-101.47277,29.7744],[-102.60596,29.8192],[-103.15787,28.93865],[-104.37752,29.54255],[-104.39363,29.55396],[-104.3969,29.57105],[-104.5171,29.64671],[-104.77674,30.4236],[-106.00363,31.39181],[-106.09025,31.40569],[-106.20346,31.46305],[-106.23711,31.51262],[-106.24612,31.54193],[-106.28084,31.56173],[-106.30305,31.62154],[-106.33419,31.66303],[-106.34864,31.69663],[-106.3718,31.71165],[-106.38003,31.73151],[-106.41773,31.75196],[-106.43419,31.75478],[-106.45244,31.76523],[-106.46726,31.75998],[-106.47298,31.75054],[-106.48815,31.74769],[-106.50111,31.75714],[-106.50962,31.76155],[-106.51251,31.76922],[-106.52266,31.77509],[-106.529,31.784],[-108.20899,31.78534],[-108.20979,31.33316],[-109.05235,31.3333],[-111.07523,31.33232],[-112.34553,31.7357],[-114.82011,32.49609],[-114.79524,32.55731],[-114.81141,32.55543],[-114.80584,32.62028],[-114.76736,32.64094],[-114.71871,32.71894],[-115.88053,32.63624],[-117.1243,32.53427]]]]}},{type:"Feature",properties:{iso1A2:"MY",iso1A3:"MYS",iso1N3:"458",wikidata:"Q833",nameEn:"Malaysia",groups:["035","142"],driveSide:"left",callingCodes:["60"]},geometry:{type:"MultiPolygon",coordinates:[[[[114.08532,4.64632],[109.55486,8.10026],[104.81582,8.03101],[102.46318,7.22462],[102.09086,6.23546],[102.08127,6.22679],[102.07732,6.193],[102.09182,6.14161],[102.01835,6.05407],[101.99209,6.04075],[101.97114,6.01992],[101.9714,6.00575],[101.94712,5.98421],[101.92819,5.85511],[101.91776,5.84269],[101.89188,5.8386],[101.80144,5.74505],[101.75074,5.79091],[101.69773,5.75881],[101.58019,5.93534],[101.25524,5.78633],[101.25755,5.71065],[101.14062,5.61613],[100.98815,5.79464],[101.02708,5.91013],[101.087,5.9193],[101.12388,6.11411],[101.06165,6.14161],[101.12618,6.19431],[101.10313,6.25617],[100.85884,6.24929],[100.81045,6.45086],[100.74822,6.46231],[100.74361,6.50811],[100.66986,6.45086],[100.43027,6.52389],[100.42351,6.51762],[100.41791,6.5189],[100.41152,6.52299],[100.35413,6.54932],[100.31929,6.65413],[100.32607,6.65933],[100.32671,6.66526],[100.31884,6.66423],[100.31618,6.66781],[100.30828,6.66462],[100.29651,6.68439],[100.19511,6.72559],[100.12,6.42105],[100.0756,6.4045],[99.91873,6.50233],[99.50117,6.44501],[99.31854,5.99868],[99.75778,3.86466],[103.03657,1.30383],[103.56591,1.19719],[103.62738,1.35255],[103.67468,1.43166],[103.7219,1.46108],[103.74161,1.4502],[103.76395,1.45183],[103.81181,1.47953],[103.86383,1.46288],[103.89565,1.42841],[103.93384,1.42926],[104.00131,1.42405],[104.02277,1.4438],[104.04622,1.44691],[104.07348,1.43322],[104.08871,1.42015],[104.09162,1.39694],[104.08072,1.35998],[104.12282,1.27714],[104.34728,1.33529],[104.56723,1.44271],[105.01437,3.24936],[108.10426,5.42408],[109.71058,2.32059],[109.64506,2.08014],[109.62558,1.99182],[109.53794,1.91771],[109.57923,1.80624],[109.66397,1.79972],[109.66397,1.60425],[110.35354,0.98869],[110.49182,0.88088],[110.62374,0.873],[111.22979,1.08326],[111.55434,0.97864],[111.82846,0.99349],[111.94553,1.12016],[112.15679,1.17004],[112.2127,1.44135],[112.48648,1.56516],[113.021,1.57819],[113.01448,1.42832],[113.64677,1.23933],[114.03788,1.44787],[114.57892,1.5],[114.80706,1.92351],[114.80706,2.21665],[115.1721,2.49671],[115.11343,2.82879],[115.53713,3.14776],[115.58276,3.93499],[115.90217,4.37708],[117.25801,4.35108],[117.47313,4.18857],[117.67641,4.16535],[117.89538,4.16637],[118.07935,4.15511],[118.8663,4.44172],[118.75416,4.59798],[119.44841,5.09568],[119.34756,5.53889],[117.89159,6.25755],[117.43832,7.3895],[117.17735,7.52841],[116.79524,7.43869],[115.02521,5.35005],[115.16236,5.01011],[115.15092,4.87604],[115.20737,4.8256],[115.27819,4.63661],[115.2851,4.42295],[115.36346,4.33563],[115.31275,4.30806],[115.09978,4.39123],[115.07737,4.53418],[115.04064,4.63706],[115.02278,4.74137],[115.02955,4.82087],[115.05038,4.90275],[114.99417,4.88201],[114.96982,4.81146],[114.88841,4.81905],[114.8266,4.75062],[114.77303,4.72871],[114.83189,4.42387],[114.88039,4.4257],[114.78539,4.12205],[114.64211,4.00694],[114.49922,4.13108],[114.4416,4.27588],[114.32176,4.2552],[114.32176,4.34942],[114.26876,4.49878],[114.15813,4.57],[114.07448,4.58441],[114.08532,4.64632]]]]}},{type:"Feature",properties:{iso1A2:"MZ",iso1A3:"MOZ",iso1N3:"508",wikidata:"Q1029",nameEn:"Mozambique",groups:["014","202","002"],driveSide:"left",callingCodes:["258"]},geometry:{type:"MultiPolygon",coordinates:[[[[40.74206,-10.25691],[40.44265,-10.4618],[40.00295,-10.80255],[39.58249,-10.96043],[39.24395,-11.17433],[38.88996,-11.16978],[38.47258,-11.4199],[38.21598,-11.27289],[37.93618,-11.26228],[37.8388,-11.3123],[37.76614,-11.53352],[37.3936,-11.68949],[36.80309,-11.56836],[36.62068,-11.72884],[36.19094,-11.70008],[36.19094,-11.57593],[35.82767,-11.41081],[35.63599,-11.55927],[34.96296,-11.57354],[34.64241,-11.57499],[34.57917,-11.87849],[34.82903,-12.04837],[34.70739,-12.15652],[34.46088,-12.0174],[34.37831,-12.17408],[34.60253,-13.48487],[34.86229,-13.48958],[35.47989,-14.15594],[35.5299,-14.27714],[35.86945,-14.67481],[35.87212,-14.89478],[35.91812,-14.89514],[35.78799,-15.17428],[35.85303,-15.41913],[35.80487,-16.03907],[35.70107,-16.10147],[35.52365,-16.15414],[35.43355,-16.11371],[35.30157,-16.2211],[35.25828,-16.4792],[35.14235,-16.56812],[35.27219,-16.69402],[35.30929,-16.82871],[35.27065,-16.93817],[35.3062,-17.1244],[35.0923,-17.13235],[35.04805,-17.00027],[35.17017,-16.93521],[35.13771,-16.81687],[35.04805,-16.83167],[34.40344,-16.20923],[34.43126,-16.04737],[34.25195,-15.90321],[34.44981,-15.60864],[34.43126,-15.44778],[34.57503,-15.30619],[34.61522,-14.99583],[34.567,-14.77345],[34.54503,-14.74672],[34.52057,-14.68263],[34.53516,-14.67782],[34.55112,-14.64494],[34.53962,-14.59776],[34.52366,-14.5667],[34.49636,-14.55091],[34.48932,-14.53646],[34.47628,-14.53363],[34.45053,-14.49873],[34.44641,-14.47746],[34.4192,-14.43191],[34.39277,-14.39467],[34.35843,-14.38652],[34.34453,-14.3985],[34.22355,-14.43607],[34.18733,-14.43823],[34.08588,-14.48893],[33.92898,-14.47929],[33.88503,-14.51652],[33.7247,-14.4989],[33.66677,-14.61306],[33.24249,-14.00019],[30.22098,-14.99447],[30.41902,-15.62269],[30.42568,-15.9962],[30.91597,-15.99924],[30.97761,-16.05848],[31.13171,-15.98019],[31.30563,-16.01193],[31.42451,-16.15154],[31.67988,-16.19595],[31.90223,-16.34388],[31.91324,-16.41569],[32.02772,-16.43892],[32.28529,-16.43892],[32.42838,-16.4727],[32.71017,-16.59932],[32.69917,-16.66893],[32.78943,-16.70267],[32.97655,-16.70689],[32.91051,-16.89446],[32.84113,-16.92259],[32.96554,-17.11971],[33.00517,-17.30477],[33.0426,-17.3468],[32.96554,-17.48964],[32.98536,-17.55891],[33.0492,-17.60298],[32.94133,-17.99705],[33.03159,-18.35054],[33.02278,-18.4696],[32.88629,-18.51344],[32.88629,-18.58023],[32.95013,-18.69079],[32.9017,-18.7992],[32.82465,-18.77419],[32.70137,-18.84712],[32.73439,-18.92628],[32.69917,-18.94293],[32.72118,-19.02204],[32.84006,-19.0262],[32.87088,-19.09279],[32.85107,-19.29238],[32.77966,-19.36098],[32.78282,-19.47513],[32.84446,-19.48343],[32.84666,-19.68462],[32.95013,-19.67219],[33.06461,-19.77787],[33.01178,-20.02007],[32.93032,-20.03868],[32.85987,-20.16686],[32.85987,-20.27841],[32.66174,-20.56106],[32.55167,-20.56312],[32.48122,-20.63319],[32.51644,-20.91929],[32.37115,-21.133],[32.48236,-21.32873],[32.41234,-21.31246],[31.38336,-22.36919],[31.30611,-22.422],[31.55779,-23.176],[31.56539,-23.47268],[31.67942,-23.60858],[31.70223,-23.72695],[31.77445,-23.90082],[31.87707,-23.95293],[31.90368,-24.18892],[31.9835,-24.29983],[32.03196,-25.10785],[32.01676,-25.38117],[31.97875,-25.46356],[32.00631,-25.65044],[31.92649,-25.84216],[31.974,-25.95387],[32.00916,-25.999],[32.08599,-26.00978],[32.10435,-26.15656],[32.07352,-26.40185],[32.13409,-26.5317],[32.13315,-26.84345],[32.19409,-26.84032],[32.22302,-26.84136],[32.29584,-26.852],[32.35222,-26.86027],[34.51034,-26.91792],[42.99868,-12.65261],[40.74206,-10.25691]]]]}},{type:"Feature",properties:{iso1A2:"NA",iso1A3:"NAM",iso1N3:"516",wikidata:"Q1030",nameEn:"Namibia",groups:["018","202","002"],driveSide:"left",callingCodes:["264"]},geometry:{type:"MultiPolygon",coordinates:[[[[14.28743,-17.38814],[13.95896,-17.43141],[13.36212,-16.98048],[12.97145,-16.98567],[12.52111,-17.24495],[12.07076,-17.15165],[11.75063,-17.25013],[10.5065,-17.25284],[12.51595,-32.27486],[16.45332,-28.63117],[16.46592,-28.57126],[16.59922,-28.53246],[16.90446,-28.057],[17.15405,-28.08573],[17.4579,-28.68718],[18.99885,-28.89165],[19.99882,-28.42622],[19.99817,-24.76768],[19.99912,-21.99991],[20.99751,-22.00026],[20.99904,-18.31743],[21.45556,-18.31795],[23.0996,-18.00075],[23.29618,-17.99855],[23.61088,-18.4881],[24.19416,-18.01919],[24.40577,-17.95726],[24.57485,-18.07151],[24.6303,-17.9863],[24.71887,-17.9218],[24.73364,-17.89338],[24.95586,-17.79674],[25.05895,-17.84452],[25.16882,-17.78253],[25.26433,-17.79571],[25.00198,-17.58221],[24.70864,-17.49501],[24.5621,-17.52963],[24.38712,-17.46818],[24.32811,-17.49082],[24.23619,-17.47489],[23.47474,-17.62877],[21.42741,-18.02787],[21.14283,-17.94318],[18.84226,-17.80375],[18.39229,-17.38927],[14.28743,-17.38814]]]]}},{type:"Feature",properties:{iso1A2:"NC",iso1A3:"NCL",iso1N3:"540",wikidata:"Q33788",nameEn:"New Caledonia",country:"FR",groups:["054","009"],callingCodes:["687"]},geometry:{type:"MultiPolygon",coordinates:[[[[158.65519,-23.4036],[174.90025,-23.53966],[162.93363,-17.28904],[157.83842,-18.82563],[158.65519,-23.4036]]]]}},{type:"Feature",properties:{iso1A2:"NE",iso1A3:"NER",iso1N3:"562",wikidata:"Q1032",nameEn:"Niger",aliases:["RN"],groups:["011","202","002"],callingCodes:["227"]},geometry:{type:"MultiPolygon",coordinates:[[[[14.22918,22.61719],[13.5631,23.16574],[11.96886,23.51735],[7.48273,20.87258],[7.38361,20.79165],[5.8153,19.45101],[4.26651,19.14224],[4.26762,17.00432],[4.21787,17.00118],[4.19893,16.39923],[3.50368,15.35934],[3.03134,15.42221],[3.01806,15.34571],[1.31275,15.27978],[0.96711,14.98275],[0.72632,14.95898],[0.23859,15.00135],[0.16936,14.51654],[0.38051,14.05575],[0.61924,13.68491],[0.77377,13.6866],[0.77637,13.64442],[0.99514,13.5668],[1.02813,13.46635],[1.20088,13.38951],[1.24429,13.39373],[1.28509,13.35488],[1.24516,13.33968],[1.21217,13.37853],[1.18873,13.31771],[0.99253,13.37515],[0.99167,13.10727],[2.26349,12.41915],[2.05785,12.35539],[2.39723,11.89473],[2.45824,11.98672],[2.39657,12.10952],[2.37783,12.24804],[2.6593,12.30631],[2.83978,12.40585],[3.25352,12.01467],[3.31613,11.88495],[3.48187,11.86092],[3.59375,11.70269],[3.61075,11.69181],[3.67988,11.75429],[3.67122,11.80865],[3.63063,11.83042],[3.61955,11.91847],[3.67775,11.97599],[3.63136,12.11826],[3.66364,12.25884],[3.65111,12.52223],[3.94339,12.74979],[4.10006,12.98862],[4.14367,13.17189],[4.14186,13.47586],[4.23456,13.47725],[4.4668,13.68286],[4.87425,13.78],[4.9368,13.7345],[5.07396,13.75052],[5.21026,13.73627],[5.27797,13.75474],[5.35437,13.83567],[5.52957,13.8845],[6.15771,13.64564],[6.27411,13.67835],[6.43053,13.6006],[6.69617,13.34057],[6.94445,12.99825],[7.0521,13.00076],[7.12676,13.02445],[7.22399,13.1293],[7.39241,13.09717],[7.81085,13.34902],[8.07997,13.30847],[8.25185,13.20369],[8.41853,13.06166],[8.49493,13.07519],[8.60431,13.01768],[8.64251,12.93985],[8.97413,12.83661],[9.65995,12.80614],[10.00373,13.18171],[10.19993,13.27129],[10.46731,13.28819],[10.66004,13.36422],[11.4535,13.37773],[11.88236,13.2527],[12.04209,13.14452],[12.16189,13.10056],[12.19315,13.12423],[12.47095,13.06673],[12.58033,13.27805],[12.6793,13.29157],[12.87376,13.48919],[13.05085,13.53984],[13.19844,13.52802],[13.33213,13.71195],[13.6302,13.71094],[13.47559,14.40881],[13.48259,14.46704],[13.68573,14.55276],[13.67878,14.64013],[13.809,14.72915],[13.78991,14.87519],[13.86301,15.04043],[14.37425,15.72591],[15.50373,16.89649],[15.6032,18.77402],[15.75098,19.93002],[15.99632,20.35364],[15.6721,20.70069],[15.59841,20.74039],[15.56004,20.79488],[15.55382,20.86507],[15.57248,20.92138],[15.62515,20.95395],[15.28332,21.44557],[15.20213,21.49365],[15.19692,21.99339],[14.99751,23.00539],[14.22918,22.61719]]]]}},{type:"Feature",properties:{iso1A2:"NF",iso1A3:"NFK",iso1N3:"574",wikidata:"Q31057",nameEn:"Norfolk Island",country:"AU",groups:["053","009"],driveSide:"left",callingCodes:["672 3"]},geometry:{type:"MultiPolygon",coordinates:[[[[169.82316,-28.16667],[166.29505,-28.29175],[167.94076,-30.60745],[169.82316,-28.16667]]]]}},{type:"Feature",properties:{iso1A2:"NG",iso1A3:"NGA",iso1N3:"566",wikidata:"Q1033",nameEn:"Nigeria",groups:["011","202","002"],callingCodes:["234"]},geometry:{type:"MultiPolygon",coordinates:[[[[6.15771,13.64564],[5.52957,13.8845],[5.35437,13.83567],[5.27797,13.75474],[5.21026,13.73627],[5.07396,13.75052],[4.9368,13.7345],[4.87425,13.78],[4.4668,13.68286],[4.23456,13.47725],[4.14186,13.47586],[4.14367,13.17189],[4.10006,12.98862],[3.94339,12.74979],[3.65111,12.52223],[3.66364,12.25884],[3.63136,12.11826],[3.67775,11.97599],[3.61955,11.91847],[3.63063,11.83042],[3.67122,11.80865],[3.67988,11.75429],[3.61075,11.69181],[3.59375,11.70269],[3.49175,11.29765],[3.71505,11.13015],[3.84243,10.59316],[3.78292,10.40538],[3.6844,10.46351],[3.57275,10.27185],[3.66908,10.18136],[3.54429,9.87739],[3.35383,9.83641],[3.32099,9.78032],[3.34726,9.70696],[3.25093,9.61632],[3.13928,9.47167],[3.14147,9.28375],[3.08017,9.10006],[2.77907,9.06924],[2.67523,7.87825],[2.73095,7.7755],[2.73405,7.5423],[2.78668,7.5116],[2.79442,7.43486],[2.74489,7.42565],[2.76965,7.13543],[2.71702,6.95722],[2.74024,6.92802],[2.73405,6.78508],[2.78823,6.76356],[2.78204,6.70514],[2.7325,6.64057],[2.74334,6.57291],[2.70464,6.50831],[2.70566,6.38038],[2.74181,6.13349],[5.87055,3.78489],[8.34397,4.30689],[8.60302,4.87353],[8.78027,5.1243],[8.92029,5.58403],[8.83687,5.68483],[8.88156,5.78857],[8.84209,5.82562],[9.51757,6.43874],[9.70674,6.51717],[9.77824,6.79088],[9.86314,6.77756],[10.15135,7.03781],[10.21466,6.88996],[10.53639,6.93432],[10.57214,7.16345],[10.59746,7.14719],[10.60789,7.06885],[10.83727,6.9358],[10.8179,6.83377],[10.94302,6.69325],[11.09644,6.68437],[11.09495,6.51717],[11.42041,6.53789],[11.42264,6.5882],[11.51499,6.60892],[11.57755,6.74059],[11.55818,6.86186],[11.63117,6.9905],[11.87396,7.09398],[11.84864,7.26098],[11.93205,7.47812],[12.01844,7.52981],[11.99908,7.67302],[12.20909,7.97553],[12.19271,8.10826],[12.24782,8.17904],[12.26123,8.43696],[12.4489,8.52536],[12.44146,8.6152],[12.68722,8.65938],[12.71701,8.7595],[12.79,8.75361],[12.81085,8.91992],[12.90022,9.11411],[12.91958,9.33905],[12.85628,9.36698],[13.02385,9.49334],[13.22642,9.57266],[13.25472,9.76795],[13.29941,9.8296],[13.25025,9.86042],[13.24132,9.91031],[13.27409,9.93232],[13.286,9.9822],[13.25323,10.00127],[13.25025,10.03647],[13.34111,10.12299],[13.43644,10.13326],[13.5705,10.53183],[13.54964,10.61236],[13.73434,10.9255],[13.70753,10.94451],[13.7403,11.00593],[13.78945,11.00154],[13.97489,11.30258],[14.17821,11.23831],[14.6124,11.51283],[14.64591,11.66166],[14.55207,11.72001],[14.61612,11.7798],[14.6474,12.17466],[14.4843,12.35223],[14.22215,12.36533],[14.17523,12.41916],[14.20204,12.53405],[14.08251,13.0797],[13.6302,13.71094],[13.33213,13.71195],[13.19844,13.52802],[13.05085,13.53984],[12.87376,13.48919],[12.6793,13.29157],[12.58033,13.27805],[12.47095,13.06673],[12.19315,13.12423],[12.16189,13.10056],[12.04209,13.14452],[11.88236,13.2527],[11.4535,13.37773],[10.66004,13.36422],[10.46731,13.28819],[10.19993,13.27129],[10.00373,13.18171],[9.65995,12.80614],[8.97413,12.83661],[8.64251,12.93985],[8.60431,13.01768],[8.49493,13.07519],[8.41853,13.06166],[8.25185,13.20369],[8.07997,13.30847],[7.81085,13.34902],[7.39241,13.09717],[7.22399,13.1293],[7.12676,13.02445],[7.0521,13.00076],[6.94445,12.99825],[6.69617,13.34057],[6.43053,13.6006],[6.27411,13.67835],[6.15771,13.64564]]]]}},{type:"Feature",properties:{iso1A2:"NI",iso1A3:"NIC",iso1N3:"558",wikidata:"Q811",nameEn:"Nicaragua",groups:["013","003","419","019"],callingCodes:["505"]},geometry:{type:"MultiPolygon",coordinates:[[[[-83.13724,15.00002],[-83.49268,15.01158],[-83.62101,14.89448],[-83.89551,14.76697],[-84.10584,14.76353],[-84.48373,14.63249],[-84.70119,14.68078],[-84.82596,14.82212],[-84.90082,14.80489],[-85.1575,14.53934],[-85.18602,14.24929],[-85.32149,14.2562],[-85.45762,14.11304],[-85.73964,13.9698],[-85.75477,13.8499],[-86.03458,13.99181],[-86.00685,14.08474],[-86.14801,14.04317],[-86.35219,13.77157],[-86.76812,13.79605],[-86.71267,13.30348],[-86.87066,13.30641],[-86.93383,13.18677],[-86.93197,13.05313],[-87.03785,12.98682],[-87.06306,13.00892],[-87.37107,12.98646],[-87.55124,13.12523],[-87.7346,13.13228],[-88.11443,12.63306],[-86.14524,11.09059],[-85.71223,11.06868],[-85.60529,11.22607],[-84.92439,10.9497],[-84.68197,11.07568],[-83.90838,10.71161],[-83.66597,10.79916],[-83.68276,11.01562],[-82.56142,11.91792],[-82.06974,14.49418],[-83.04763,15.03256],[-83.13724,15.00002]]]]}},{type:"Feature",properties:{iso1A2:"NL",iso1A3:"NLD",iso1N3:"528",wikidata:"Q55",nameEn:"Netherlands",groups:["EU","155","150"],callingCodes:["31"]},geometry:{type:"MultiPolygon",coordinates:[[[[5.45168,54.20039],[2.56575,51.85301],[3.36263,51.37112],[3.38696,51.33436],[3.35847,51.31572],[3.38289,51.27331],[3.41704,51.25933],[3.43488,51.24135],[3.52698,51.2458],[3.51502,51.28697],[3.58939,51.30064],[3.78999,51.25766],[3.78783,51.2151],[3.90125,51.20371],[3.97889,51.22537],[4.01957,51.24504],[4.05165,51.24171],[4.16721,51.29348],[4.24024,51.35371],[4.21923,51.37443],[4.33265,51.37687],[4.34086,51.35738],[4.39292,51.35547],[4.43777,51.36989],[4.38064,51.41965],[4.39747,51.43316],[4.38122,51.44905],[4.47736,51.4778],[4.5388,51.48184],[4.54675,51.47265],[4.52846,51.45002],[4.53521,51.4243],[4.57489,51.4324],[4.65442,51.42352],[4.72935,51.48424],[4.74578,51.48937],[4.77321,51.50529],[4.78803,51.50284],[4.84139,51.4799],[4.82409,51.44736],[4.82946,51.4213],[4.78314,51.43319],[4.76577,51.43046],[4.77229,51.41337],[4.78941,51.41102],[4.84988,51.41502],[4.90016,51.41404],[4.92152,51.39487],[5.00393,51.44406],[5.0106,51.47167],[5.03281,51.48679],[5.04774,51.47022],[5.07891,51.4715],[5.10456,51.43163],[5.07102,51.39469],[5.13105,51.34791],[5.13377,51.31592],[5.16222,51.31035],[5.2002,51.32243],[5.24244,51.30495],[5.22542,51.26888],[5.23814,51.26064],[5.26461,51.26693],[5.29716,51.26104],[5.33886,51.26314],[5.347,51.27502],[5.41672,51.26248],[5.4407,51.28169],[5.46519,51.2849],[5.48476,51.30053],[5.515,51.29462],[5.5569,51.26544],[5.5603,51.22249],[5.65145,51.19788],[5.65528,51.18736],[5.70344,51.1829],[5.74617,51.18928],[5.77735,51.17845],[5.77697,51.1522],[5.82564,51.16753],[5.85508,51.14445],[5.80798,51.11661],[5.8109,51.10861],[5.83226,51.10585],[5.82921,51.09328],[5.79903,51.09371],[5.79835,51.05834],[5.77258,51.06196],[5.75961,51.03113],[5.77688,51.02483],[5.76242,50.99703],[5.71864,50.96092],[5.72875,50.95428],[5.74752,50.96202],[5.75927,50.95601],[5.74644,50.94723],[5.72545,50.92312],[5.72644,50.91167],[5.71626,50.90796],[5.69858,50.91046],[5.67886,50.88142],[5.64504,50.87107],[5.64009,50.84742],[5.65259,50.82309],[5.70118,50.80764],[5.68995,50.79641],[5.70107,50.7827],[5.68091,50.75804],[5.69469,50.75529],[5.72216,50.76398],[5.73904,50.75674],[5.74356,50.7691],[5.76533,50.78159],[5.77513,50.78308],[5.80673,50.7558],[5.84548,50.76542],[5.84888,50.75448],[5.88734,50.77092],[5.89129,50.75125],[5.89132,50.75124],[5.95942,50.7622],[5.97545,50.75441],[6.01976,50.75398],[6.02624,50.77453],[5.97497,50.79992],[5.98404,50.80988],[6.00462,50.80065],[6.02328,50.81694],[6.01921,50.84435],[6.05623,50.8572],[6.05702,50.85179],[6.07431,50.84674],[6.07693,50.86025],[6.08805,50.87223],[6.07486,50.89307],[6.09297,50.92066],[6.01615,50.93367],[6.02697,50.98303],[5.95282,50.98728],[5.90296,50.97356],[5.90493,51.00198],[5.87849,51.01969],[5.86735,51.05182],[5.9134,51.06736],[5.9541,51.03496],[5.98292,51.07469],[6.16706,51.15677],[6.17384,51.19589],[6.07889,51.17038],[6.07889,51.24432],[6.16977,51.33169],[6.22674,51.36135],[6.22641,51.39948],[6.20654,51.40049],[6.21724,51.48568],[6.18017,51.54096],[6.09055,51.60564],[6.11759,51.65609],[6.02767,51.6742],[6.04091,51.71821],[5.95003,51.7493],[5.98665,51.76944],[5.94568,51.82786],[5.99848,51.83195],[6.06705,51.86136],[6.10337,51.84829],[6.16902,51.84094],[6.11551,51.89769],[6.15349,51.90439],[6.21443,51.86801],[6.29872,51.86801],[6.30593,51.84998],[6.40704,51.82771],[6.38815,51.87257],[6.47179,51.85395],[6.50231,51.86313],[6.58556,51.89386],[6.68386,51.91861],[6.72319,51.89518],[6.82357,51.96711],[6.83035,51.9905],[6.68128,52.05052],[6.76117,52.11895],[6.83984,52.11728],[6.97189,52.20329],[6.9897,52.2271],[7.03729,52.22695],[7.06365,52.23789],[7.02703,52.27941],[7.07044,52.37805],[7.03417,52.40237],[6.99041,52.47235],[6.94293,52.43597],[6.69507,52.488],[6.71641,52.62905],[6.77307,52.65375],[7.04557,52.63318],[7.07253,52.81083],[7.21694,53.00742],[7.17898,53.13817],[7.22681,53.18165],[7.21679,53.20058],[7.19052,53.31866],[7.00198,53.32672],[6.91025,53.44221],[5.45168,54.20039]],[[4.93295,51.44945],[4.95244,51.45207],[4.9524,51.45014],[4.93909,51.44632],[4.93295,51.44945]],[[4.91493,51.4353],[4.91935,51.43634],[4.92227,51.44252],[4.91811,51.44621],[4.92287,51.44741],[4.92811,51.4437],[4.92566,51.44273],[4.92815,51.43856],[4.92879,51.44161],[4.93544,51.44634],[4.94025,51.44193],[4.93416,51.44185],[4.93471,51.43861],[4.94265,51.44003],[4.93986,51.43064],[4.92952,51.42984],[4.92652,51.43329],[4.91493,51.4353]]]]}},{type:"Feature",properties:{iso1A2:"NO",iso1A3:"NOR",iso1N3:"578",wikidata:"Q20",nameEn:"Norway",groups:["154","150"],callingCodes:["47"]},geometry:{type:"MultiPolygon",coordinates:[[[[10.40861,58.38489],[10.64958,58.89391],[11.08911,58.98745],[11.15367,59.07862],[11.34459,59.11672],[11.4601,58.99022],[11.45199,58.89604],[11.65732,58.90177],[11.8213,59.24985],[11.69297,59.59442],[11.92112,59.69531],[11.87121,59.86039],[12.15641,59.8926],[12.36317,59.99259],[12.52003,60.13846],[12.59133,60.50559],[12.2277,61.02442],[12.69115,61.06584],[12.86939,61.35427],[12.57707,61.56547],[12.40595,61.57226],[12.14746,61.7147],[12.29187,62.25699],[12.07085,62.6297],[12.19919,63.00104],[11.98529,63.27487],[12.19919,63.47935],[12.14928,63.59373],[12.74105,64.02171],[13.23411,64.09087],[13.98222,64.00953],[14.16051,64.18725],[14.11117,64.46674],[13.64276,64.58402],[14.50926,65.31786],[14.53778,66.12399],[15.05113,66.15572],[15.49318,66.28509],[15.37197,66.48217],[16.35589,67.06419],[16.39154,67.21653],[16.09922,67.4364],[16.12774,67.52106],[16.38441,67.52923],[16.7409,67.91037],[17.30416,68.11591],[17.90787,67.96537],[18.13836,68.20874],[18.1241,68.53721],[18.39503,68.58672],[18.63032,68.50849],[18.97255,68.52416],[19.93508,68.35911],[20.22027,68.48759],[19.95647,68.55546],[20.22027,68.67246],[20.33435,68.80174],[20.28444,68.93283],[20.0695,69.04469],[20.55258,69.06069],[20.72171,69.11874],[21.05775,69.0356],[21.11099,69.10291],[20.98641,69.18809],[21.00732,69.22755],[21.27827,69.31281],[21.63833,69.27485],[22.27276,68.89514],[22.38367,68.71561],[22.53321,68.74393],[23.13064,68.64684],[23.68017,68.70276],[23.781,68.84514],[24.02299,68.81601],[24.18432,68.73936],[24.74898,68.65143],[24.90023,68.55579],[24.93048,68.61102],[25.10189,68.63307],[25.12206,68.78684],[25.42455,68.90328],[25.61613,68.89602],[25.75729,68.99383],[25.69679,69.27039],[25.96904,69.68397],[26.40261,69.91377],[26.64461,69.96565],[27.05802,69.92069],[27.57226,70.06215],[27.95542,70.0965],[27.97558,69.99671],[28.32849,69.88605],[28.36883,69.81658],[29.12697,69.69193],[29.31664,69.47994],[28.8629,69.22395],[28.81248,69.11997],[28.91738,69.04774],[29.0444,69.0119],[29.26623,69.13794],[29.27631,69.2811],[29.97205,69.41623],[30.16363,69.65244],[30.52662,69.54699],[30.95011,69.54699],[30.84095,69.80584],[31.59909,70.16571],[32.07813,72.01005],[18.46509,71.28681],[-0.3751,61.32236],[7.28637,57.35913],[10.40861,58.38489]]]]}},{type:"Feature",properties:{iso1A2:"NP",iso1A3:"NPL",iso1N3:"524",wikidata:"Q837",nameEn:"Nepal",groups:["034","142"],driveSide:"left",callingCodes:["977"]},geometry:{type:"MultiPolygon",coordinates:[[[[88.13378,27.88015],[87.82681,27.95248],[87.72718,27.80938],[87.56996,27.84517],[87.11696,27.84104],[87.03757,27.94835],[86.75582,28.04182],[86.74181,28.10638],[86.56265,28.09569],[86.51609,27.96623],[86.42736,27.91122],[86.22966,27.9786],[86.18607,28.17364],[86.088,28.09264],[86.08333,28.02121],[86.12069,27.93047],[86.06309,27.90021],[85.94946,27.9401],[85.97813,27.99023],[85.90743,28.05144],[85.84672,28.18187],[85.74864,28.23126],[85.71907,28.38064],[85.69105,28.38475],[85.60854,28.25045],[85.59765,28.30529],[85.4233,28.32996],[85.38127,28.28336],[85.10729,28.34092],[85.18668,28.54076],[85.19135,28.62825],[85.06059,28.68562],[84.85511,28.58041],[84.62317,28.73887],[84.47528,28.74023],[84.2231,28.89571],[84.24801,29.02783],[84.18107,29.23451],[83.97559,29.33091],[83.82303,29.30513],[83.63156,29.16249],[83.44787,29.30513],[83.28131,29.56813],[83.07116,29.61957],[82.73024,29.81695],[82.5341,29.9735],[82.38622,30.02608],[82.16984,30.0692],[82.19475,30.16884],[82.10757,30.23745],[82.10135,30.35439],[81.99082,30.33423],[81.62033,30.44703],[81.41018,30.42153],[81.39928,30.21862],[81.33355,30.15303],[81.2623,30.14596],[81.29032,30.08806],[81.24362,30.0126],[81.12842,30.01395],[81.03953,30.20059],[80.93695,30.18229],[80.8778,30.13384],[80.67076,29.95732],[80.60226,29.95732],[80.56957,29.88176],[80.56247,29.86661],[80.48997,29.79566],[80.43458,29.80466],[80.41554,29.79451],[80.36803,29.73865],[80.38428,29.68513],[80.41858,29.63581],[80.37939,29.57098],[80.24322,29.44299],[80.31428,29.30784],[80.28626,29.20327],[80.24112,29.21414],[80.26602,29.13938],[80.23178,29.11626],[80.18085,29.13649],[80.05743,28.91479],[80.06957,28.82763],[80.12125,28.82346],[80.37188,28.63371],[80.44504,28.63098],[80.52443,28.54897],[80.50575,28.6706],[80.55142,28.69182],[80.89648,28.47237],[81.08507,28.38346],[81.19847,28.36284],[81.32923,28.13521],[81.38683,28.17638],[81.48179,28.12148],[81.47867,28.08303],[81.91223,27.84995],[81.97214,27.93322],[82.06554,27.92222],[82.46405,27.6716],[82.70378,27.72122],[82.74119,27.49838],[82.93261,27.50328],[82.94938,27.46036],[83.19413,27.45632],[83.27197,27.38309],[83.2673,27.36235],[83.29999,27.32778],[83.35136,27.33885],[83.38872,27.39276],[83.39495,27.4798],[83.61288,27.47013],[83.85595,27.35797],[83.86182,27.4241],[83.93306,27.44939],[84.02229,27.43836],[84.10791,27.52399],[84.21376,27.45218],[84.25735,27.44941],[84.29315,27.39],[84.62161,27.33885],[84.69166,27.21294],[84.64496,27.04669],[84.793,26.9968],[84.82913,27.01989],[84.85754,26.98984],[84.96687,26.95599],[84.97186,26.9149],[85.00536,26.89523],[85.05592,26.88991],[85.02635,26.85381],[85.15883,26.86966],[85.19291,26.86909],[85.18046,26.80519],[85.21159,26.75933],[85.34302,26.74954],[85.47752,26.79292],[85.56471,26.84133],[85.5757,26.85955],[85.59461,26.85161],[85.61621,26.86721],[85.66239,26.84822],[85.73483,26.79613],[85.72315,26.67471],[85.76907,26.63076],[85.83126,26.61134],[85.85126,26.60866],[85.8492,26.56667],[86.02729,26.66756],[86.13596,26.60651],[86.22513,26.58863],[86.26235,26.61886],[86.31564,26.61925],[86.49726,26.54218],[86.54258,26.53819],[86.57073,26.49825],[86.61313,26.48658],[86.62686,26.46891],[86.69124,26.45169],[86.74025,26.42386],[86.76797,26.45892],[86.82898,26.43919],[86.94543,26.52076],[86.95912,26.52076],[87.01559,26.53228],[87.04691,26.58685],[87.0707,26.58571],[87.09147,26.45039],[87.14751,26.40542],[87.18863,26.40558],[87.24682,26.4143],[87.26587,26.40592],[87.26568,26.37294],[87.34568,26.34787],[87.37314,26.40815],[87.46566,26.44058],[87.51571,26.43106],[87.55274,26.40596],[87.59175,26.38342],[87.66803,26.40294],[87.67893,26.43501],[87.76004,26.40711],[87.7918,26.46737],[87.84193,26.43663],[87.89085,26.48565],[87.90115,26.44923],[88.00895,26.36029],[88.09414,26.43732],[88.09963,26.54195],[88.16452,26.64111],[88.1659,26.68177],[88.19107,26.75516],[88.12302,26.95324],[88.13422,26.98705],[88.11719,26.98758],[87.9887,27.11045],[88.01587,27.21388],[88.01646,27.21612],[88.07277,27.43007],[88.04008,27.49223],[88.19107,27.79285],[88.1973,27.85067],[88.13378,27.88015]]]]}},{type:"Feature",properties:{iso1A2:"NR",iso1A3:"NRU",iso1N3:"520",wikidata:"Q697",nameEn:"Nauru",groups:["057","009"],driveSide:"left",callingCodes:["674"]},geometry:{type:"MultiPolygon",coordinates:[[[[166.95155,0.14829],[166.21778,-0.7977],[167.60042,-0.88259],[166.95155,0.14829]]]]}},{type:"Feature",properties:{iso1A2:"NU",iso1A3:"NIU",iso1N3:"570",wikidata:"Q34020",nameEn:"Niue",country:"NZ",groups:["061","009"],driveSide:"left",callingCodes:["683"]},geometry:{type:"MultiPolygon",coordinates:[[[[-173.13438,-14.94228],[-173.11048,-23.23027],[-167.73129,-23.22266],[-167.73854,-14.92809],[-171.14262,-14.93704],[-173.13438,-14.94228]]]]}},{type:"Feature",properties:{iso1A2:"NZ",iso1A3:"NZL",iso1N3:"554",wikidata:"Q664",nameEn:"New Zealand",groups:["053","009"],driveSide:"left",callingCodes:["64"]},geometry:{type:"MultiPolygon",coordinates:[[[[-180,-24.21376],[-179.93224,-45.18423],[-155.99562,-45.16785],[-180,-24.21376]]],[[[161.96603,-56.07661],[179.49541,-50.04657],[179.49541,-36.79303],[169.6687,-29.09191],[161.96603,-56.07661]]]]}},{type:"Feature",properties:{iso1A2:"OM",iso1A3:"OMN",iso1N3:"512",wikidata:"Q842",nameEn:"Oman",groups:["145","142"],callingCodes:["968"]},geometry:{type:"MultiPolygon",coordinates:[[[[56.82555,25.7713],[56.79239,26.41236],[56.68954,26.76645],[56.2644,26.58649],[55.81777,26.18798],[56.08666,26.05038],[56.15498,26.06828],[56.19334,25.9795],[56.13963,25.82765],[56.17416,25.77239],[56.13579,25.73524],[56.14826,25.66351],[56.18363,25.65508],[56.20473,25.61119],[56.25365,25.60211],[56.26636,25.60643],[56.25341,25.61443],[56.26534,25.62825],[56.82555,25.7713]]],[[[56.26062,25.33108],[56.23362,25.31253],[56.25008,25.28843],[56.24465,25.27505],[56.20838,25.25668],[56.20872,25.24104],[56.24341,25.22867],[56.27628,25.23404],[56.34438,25.26653],[56.35172,25.30681],[56.3111,25.30107],[56.3005,25.31815],[56.26062,25.33108]],[[56.28423,25.26344],[56.27086,25.26128],[56.2716,25.27916],[56.28102,25.28486],[56.29379,25.2754],[56.28423,25.26344]]],[[[61.45114,22.55394],[56.86325,25.03856],[56.3227,24.97284],[56.34873,24.93205],[56.30269,24.88334],[56.20568,24.85063],[56.20062,24.78565],[56.13684,24.73699],[56.06128,24.74457],[56.03535,24.81161],[55.97836,24.87673],[55.97467,24.89639],[56.05106,24.87461],[56.05715,24.95727],[55.96316,25.00857],[55.90849,24.96771],[55.85094,24.96858],[55.81116,24.9116],[55.81348,24.80102],[55.83408,24.77858],[55.83271,24.68567],[55.76461,24.5287],[55.83271,24.41521],[55.83395,24.32776],[55.80747,24.31069],[55.79145,24.27914],[55.76781,24.26209],[55.75939,24.26114],[55.75382,24.2466],[55.75257,24.23466],[55.76558,24.23227],[55.77658,24.23476],[55.83367,24.20193],[55.95472,24.2172],[56.01799,24.07426],[55.8308,24.01633],[55.73301,24.05994],[55.48677,23.94946],[55.57358,23.669],[55.22634,23.10378],[55.2137,22.71065],[55.66469,21.99658],[54.99756,20.00083],[52.00311,19.00083],[52.78009,17.35124],[52.74267,17.29519],[52.81185,17.28568],[53.09917,16.67084],[53.32998,16.16312],[56.66759,17.24021],[61.45114,22.55394]]]]}},{type:"Feature",properties:{iso1A2:"PA",iso1A3:"PAN",iso1N3:"591",wikidata:"Q804",nameEn:"Panama",groups:["013","003","419","019"],callingCodes:["507"]},geometry:{type:"MultiPolygon",coordinates:[[[[-77.32389,8.81247],[-77.58292,9.22278],[-78.79327,9.93766],[-82.51044,9.65379],[-82.56507,9.57279],[-82.61345,9.49881],[-82.66667,9.49746],[-82.77206,9.59573],[-82.87919,9.62645],[-82.84871,9.4973],[-82.93516,9.46741],[-82.93516,9.07687],[-82.72126,8.97125],[-82.88253,8.83331],[-82.91377,8.774],[-82.92068,8.74832],[-82.8794,8.6981],[-82.82739,8.60153],[-82.83975,8.54755],[-82.83322,8.52464],[-82.8382,8.48117],[-82.8679,8.44042],[-82.93056,8.43465],[-83.05209,8.33394],[-82.9388,8.26634],[-82.88641,8.10219],[-82.89137,8.05755],[-82.89978,8.04083],[-82.94503,7.93865],[-82.13751,6.97312],[-78.06168,7.07793],[-77.89178,7.22681],[-77.81426,7.48319],[-77.72157,7.47612],[-77.72514,7.72348],[-77.57185,7.51147],[-77.17257,7.97422],[-77.45064,8.49991],[-77.32389,8.81247]]]]}},{type:"Feature",properties:{iso1A2:"PE",iso1A3:"PER",iso1N3:"604",wikidata:"Q419",nameEn:"Peru",groups:["005","419","019"],callingCodes:["51"]},geometry:{type:"MultiPolygon",coordinates:[[[[-74.26675,-0.97229],[-74.42701,-0.50218],[-75.18513,-0.0308],[-75.25764,-0.11943],[-75.40192,-0.17196],[-75.61997,-0.10012],[-75.60169,-0.18708],[-75.53615,-0.19213],[-75.22862,-0.60048],[-75.22862,-0.95588],[-75.3872,-0.9374],[-75.57429,-1.55961],[-76.05203,-2.12179],[-76.6324,-2.58397],[-77.94147,-3.05454],[-78.19369,-3.36431],[-78.14324,-3.47653],[-78.22642,-3.51113],[-78.24589,-3.39907],[-78.34362,-3.38633],[-78.68394,-4.60754],[-78.85149,-4.66795],[-79.01659,-5.01481],[-79.1162,-4.97774],[-79.26248,-4.95167],[-79.59402,-4.46848],[-79.79722,-4.47558],[-80.13945,-4.29786],[-80.39256,-4.48269],[-80.46386,-4.41516],[-80.32114,-4.21323],[-80.45023,-4.20938],[-80.4822,-4.05477],[-80.46386,-4.01342],[-80.13232,-3.90317],[-80.19926,-3.68894],[-80.18741,-3.63994],[-80.19848,-3.59249],[-80.21642,-3.5888],[-80.20535,-3.51667],[-80.22629,-3.501],[-80.23651,-3.48652],[-80.24586,-3.48677],[-80.24475,-3.47846],[-80.24123,-3.46124],[-80.20647,-3.431],[-80.30602,-3.39149],[-84.52388,-3.36941],[-85.71054,-21.15413],[-70.59118,-18.35072],[-70.378,-18.3495],[-70.31267,-18.31258],[-70.16394,-18.31737],[-69.96732,-18.25992],[-69.81607,-18.12582],[-69.75305,-17.94605],[-69.82868,-17.72048],[-69.79087,-17.65563],[-69.66483,-17.65083],[-69.46897,-17.4988],[-69.46863,-17.37466],[-69.62883,-17.28142],[-69.16896,-16.72233],[-69.00853,-16.66769],[-69.04027,-16.57214],[-68.98358,-16.42165],[-68.79464,-16.33272],[-68.96238,-16.194],[-69.09986,-16.22693],[-69.20291,-16.16668],[-69.40336,-15.61358],[-69.14856,-15.23478],[-69.36254,-14.94634],[-68.88135,-14.18639],[-69.05265,-13.68546],[-68.8864,-13.40792],[-68.85615,-12.87769],[-68.65044,-12.50689],[-68.98115,-11.8979],[-69.57156,-10.94555],[-69.57835,-10.94051],[-69.90896,-10.92744],[-70.38791,-11.07096],[-70.51395,-10.92249],[-70.64134,-11.0108],[-70.62487,-9.80666],[-70.55429,-9.76692],[-70.58453,-9.58303],[-70.53373,-9.42628],[-71.23394,-9.9668],[-72.14742,-9.98049],[-72.31883,-9.5184],[-72.72216,-9.41397],[-73.21498,-9.40904],[-72.92886,-9.04074],[-73.76576,-7.89884],[-73.65485,-7.77897],[-73.96938,-7.58465],[-73.77011,-7.28944],[-73.73986,-6.87919],[-73.12983,-6.43852],[-73.24579,-6.05764],[-72.83973,-5.14765],[-72.64391,-5.0391],[-71.87003,-4.51661],[-70.96814,-4.36915],[-70.77601,-4.15717],[-70.33236,-4.15214],[-70.19582,-4.3607],[-70.11305,-4.27281],[-70.00888,-4.37833],[-69.94708,-4.2431],[-70.3374,-3.79505],[-70.52393,-3.87553],[-70.71396,-3.7921],[-70.04609,-2.73906],[-70.94377,-2.23142],[-71.75223,-2.15058],[-72.92587,-2.44514],[-73.65312,-1.26222],[-74.26675,-0.97229]]]]}},{type:"Feature",properties:{iso1A2:"PF",iso1A3:"PYF",iso1N3:"258",wikidata:"Q30971",nameEn:"French Polynesia",country:"FR",groups:["061","009"],callingCodes:["689"]},geometry:{type:"MultiPolygon",coordinates:[[[[-149.6249,-7.51261],[-149.61166,-12.30171],[-156.4957,-12.32002],[-156.46451,-23.21255],[-156.44843,-28.52556],[-133.59543,-28.4709],[-133.61511,-21.93325],[-133.65593,-7.46952],[-149.6249,-7.51261]]]]}},{type:"Feature",properties:{iso1A2:"PG",iso1A3:"PNG",iso1N3:"598",wikidata:"Q691",nameEn:"Papua New Guinea",groups:["054","009"],driveSide:"left",callingCodes:["675"]},geometry:{type:"MultiPolygon",coordinates:[[[[141.03157,2.12829],[140.99813,-6.3233],[140.85295,-6.72996],[141.01763,-6.90181],[141.00782,-9.1242],[140.88922,-9.34945],[142.0601,-9.56571],[142.0953,-9.23534],[142.1462,-9.19923],[142.23304,-9.19253],[142.31447,-9.24611],[142.5723,-9.35994],[142.81927,-9.31709],[144.30183,-9.48146],[155.22803,-12.9001],[154.74815,-7.33315],[155.60735,-6.92266],[155.69784,-6.92661],[155.92557,-6.84664],[156.03993,-6.65703],[156.03296,-6.55528],[160.43769,-4.17974],[141.03157,2.12829]]]]}},{type:"Feature",properties:{iso1A2:"PH",iso1A3:"PHL",iso1N3:"608",wikidata:"Q928",nameEn:"Philippines",aliases:["PI","RP"],groups:["035","142"],callingCodes:["63"]},geometry:{type:"MultiPolygon",coordinates:[[[[129.19694,7.84182],[121.8109,21.77688],[120.69238,21.52331],[118.82252,14.67191],[115.39742,10.92666],[116.79524,7.43869],[117.17735,7.52841],[117.43832,7.3895],[117.89159,6.25755],[119.34756,5.53889],[119.44841,5.09568],[118.75416,4.59798],[118.8663,4.44172],[118.07935,4.15511],[118.41402,3.99509],[124.97752,4.82064],[129.19694,7.84182]]]]}},{type:"Feature",properties:{iso1A2:"PK",iso1A3:"PAK",iso1N3:"586",wikidata:"Q843",nameEn:"Pakistan",groups:["034","142"],driveSide:"left",callingCodes:["92"]},geometry:{type:"MultiPolygon",coordinates:[[[[75.72737,36.7529],[75.45562,36.71971],[75.40481,36.95382],[75.13839,37.02622],[74.56453,37.03023],[74.53739,36.96224],[74.43389,37.00977],[74.04856,36.82648],[73.82685,36.91421],[72.6323,36.84601],[72.18135,36.71838],[71.80267,36.49924],[71.60491,36.39429],[71.19505,36.04134],[71.37969,35.95865],[71.55273,35.71483],[71.49917,35.6267],[71.65435,35.4479],[71.54294,35.31037],[71.5541,35.28776],[71.67495,35.21262],[71.52938,35.09023],[71.55273,35.02615],[71.49917,35.00478],[71.50329,34.97328],[71.29472,34.87728],[71.28356,34.80882],[71.08718,34.69034],[71.11602,34.63047],[71.0089,34.54568],[71.02401,34.44835],[71.17662,34.36769],[71.12815,34.26619],[71.13078,34.16503],[71.09453,34.13524],[71.09307,34.11961],[71.06933,34.10564],[71.07345,34.06242],[70.88119,33.97933],[70.54336,33.9463],[69.90203,34.04194],[69.87307,33.9689],[69.85671,33.93719],[70.00503,33.73528],[70.14236,33.71701],[70.14785,33.6553],[70.20141,33.64387],[70.17062,33.53535],[70.32775,33.34496],[70.13686,33.21064],[70.07369,33.22557],[70.02563,33.14282],[69.85259,33.09451],[69.79766,33.13247],[69.71526,33.09911],[69.57656,33.09911],[69.49004,33.01509],[69.49854,32.88843],[69.5436,32.8768],[69.47082,32.85834],[69.38018,32.76601],[69.43649,32.7302],[69.44747,32.6678],[69.38155,32.56601],[69.2868,32.53938],[69.23599,32.45946],[69.27932,32.29119],[69.27032,32.14141],[69.3225,31.93186],[69.20577,31.85957],[69.11514,31.70782],[69.00939,31.62249],[68.95995,31.64822],[68.91078,31.59687],[68.79997,31.61665],[68.6956,31.75687],[68.57475,31.83158],[68.44222,31.76446],[68.27605,31.75863],[68.25614,31.80357],[68.1655,31.82691],[68.00071,31.6564],[67.86887,31.63536],[67.72056,31.52304],[67.58323,31.52772],[67.62374,31.40473],[67.7748,31.4188],[67.78854,31.33203],[67.29964,31.19586],[67.03323,31.24519],[67.04147,31.31561],[66.83273,31.26867],[66.72561,31.20526],[66.68166,31.07597],[66.58175,30.97532],[66.42645,30.95309],[66.39194,30.9408],[66.28413,30.57001],[66.34869,30.404],[66.23609,30.06321],[66.36042,29.9583],[66.24175,29.85181],[65.04005,29.53957],[64.62116,29.58903],[64.19796,29.50407],[64.12966,29.39157],[63.5876,29.50456],[62.47751,29.40782],[60.87231,29.86514],[61.31508,29.38903],[61.53765,29.00507],[61.65978,28.77937],[61.93581,28.55284],[62.40259,28.42703],[62.59499,28.24842],[62.79412,28.28108],[62.7638,28.02992],[62.84905,27.47627],[62.79684,27.34381],[62.80604,27.22412],[63.19649,27.25674],[63.32283,27.14437],[63.25005,27.08692],[63.25005,26.84212],[63.18688,26.83844],[63.1889,26.65072],[62.77352,26.64099],[62.31484,26.528],[62.21304,26.26601],[62.05117,26.31647],[61.89391,26.26251],[61.83831,26.07249],[61.83968,25.7538],[61.683,25.66638],[61.6433,25.27541],[61.57592,25.0492],[61.5251,24.57287],[68.11329,23.53945],[68.20763,23.85849],[68.39339,23.96838],[68.74643,23.97027],[68.7416,24.31904],[68.90914,24.33156],[68.97781,24.26021],[69.07806,24.29777],[69.19341,24.25646],[69.29778,24.28712],[69.59579,24.29777],[69.73335,24.17007],[70.03428,24.172],[70.11712,24.30915],[70.5667,24.43787],[70.57906,24.27774],[70.71502,24.23517],[70.88393,24.27398],[70.85784,24.30903],[70.94985,24.3791],[71.04461,24.34657],[71.12838,24.42662],[71.00341,24.46038],[70.97594,24.60904],[71.09405,24.69017],[70.94002,24.92843],[70.89148,25.15064],[70.66695,25.39314],[70.67382,25.68186],[70.60378,25.71898],[70.53649,25.68928],[70.37444,25.67443],[70.2687,25.71156],[70.0985,25.93238],[70.08193,26.08094],[70.17532,26.24118],[70.17532,26.55362],[70.05584,26.60398],[69.88555,26.56836],[69.50904,26.74892],[69.58519,27.18109],[70.03136,27.56627],[70.12502,27.8057],[70.37307,28.01208],[70.60927,28.02178],[70.79054,27.68423],[71.89921,27.96035],[71.9244,28.11555],[72.20329,28.3869],[72.29495,28.66367],[72.40402,28.78283],[72.94272,29.02487],[73.01337,29.16422],[73.05886,29.1878],[73.28094,29.56646],[73.3962,29.94707],[73.58665,30.01848],[73.80299,30.06969],[73.97225,30.19829],[73.95736,30.28466],[73.88993,30.36305],[74.5616,31.04153],[74.67971,31.05479],[74.6852,31.12771],[74.60006,31.13711],[74.60281,31.10419],[74.56023,31.08303],[74.51629,31.13829],[74.53223,31.30321],[74.59773,31.4136],[74.64713,31.45605],[74.59319,31.50197],[74.61517,31.55698],[74.57498,31.60382],[74.47771,31.72227],[74.58907,31.87824],[74.79919,31.95983],[74.86236,32.04485],[74.9269,32.0658],[75.00793,32.03786],[75.25649,32.10187],[75.38046,32.26836],[75.28259,32.36556],[75.03265,32.49538],[74.97634,32.45367],[74.84725,32.49075],[74.68362,32.49298],[74.67431,32.56676],[74.65251,32.56416],[74.64424,32.60985],[74.69542,32.66792],[74.65345,32.71225],[74.7113,32.84219],[74.64675,32.82604],[74.6289,32.75561],[74.45312,32.77755],[74.41467,32.90563],[74.31227,32.92795],[74.34875,32.97823],[74.31854,33.02891],[74.17571,33.07495],[74.15374,33.13477],[74.02144,33.18908],[74.01366,33.25199],[74.08782,33.26232],[74.17983,33.3679],[74.18121,33.4745],[74.10115,33.56392],[74.03576,33.56718],[73.97367,33.64061],[73.98968,33.66155],[73.96423,33.73071],[74.00891,33.75437],[74.05898,33.82089],[74.14001,33.83002],[74.26086,33.92237],[74.25262,34.01577],[74.21554,34.03853],[73.91341,34.01235],[73.88732,34.05105],[73.90677,34.10504],[73.98208,34.2522],[73.90517,34.35317],[73.8475,34.32935],[73.74862,34.34183],[73.74999,34.3781],[73.88732,34.48911],[73.89419,34.54568],[73.93951,34.57169],[73.93401,34.63386],[73.96423,34.68244],[74.12897,34.70073],[74.31239,34.79626],[74.58083,34.77386],[74.6663,34.703],[75.01479,34.64629],[75.38009,34.55021],[75.75438,34.51827],[76.04614,34.67566],[76.15463,34.6429],[76.47186,34.78965],[76.67648,34.76371],[76.74377,34.84039],[76.74514,34.92488],[76.87193,34.96906],[76.99251,34.93349],[77.11796,35.05419],[76.93465,35.39866],[76.85088,35.39754],[76.75475,35.52617],[76.77323,35.66062],[76.50961,35.8908],[76.33453,35.84296],[76.14913,35.82848],[76.15325,35.9264],[75.93028,36.13136],[76.00906,36.17511],[76.0324,36.41198],[75.92391,36.56986],[75.72737,36.7529]]]]}},{type:"Feature",properties:{iso1A2:"PL",iso1A3:"POL",iso1N3:"616",wikidata:"Q36",nameEn:"Poland",groups:["EU","151","150"],callingCodes:["48"]},geometry:{type:"MultiPolygon",coordinates:[[[[18.57853,55.25302],[14.20811,54.12784],[14.22634,53.9291],[14.20647,53.91671],[14.18544,53.91258],[14.20823,53.90776],[14.21323,53.8664],[14.27249,53.74464],[14.26782,53.69866],[14.2836,53.67721],[14.27133,53.66613],[14.28477,53.65955],[14.2853,53.63392],[14.31904,53.61581],[14.30416,53.55499],[14.3273,53.50587],[14.35209,53.49506],[14.4215,53.27724],[14.44133,53.27427],[14.45125,53.26241],[14.40662,53.21098],[14.37853,53.20405],[14.36696,53.16444],[14.38679,53.13669],[14.35044,53.05829],[14.25954,53.00264],[14.14056,52.95786],[14.15873,52.87715],[14.12256,52.84311],[14.13806,52.82392],[14.22071,52.81175],[14.61073,52.59847],[14.6289,52.57136],[14.60081,52.53116],[14.63056,52.48993],[14.54423,52.42568],[14.55228,52.35264],[14.56378,52.33838],[14.58149,52.28007],[14.70139,52.25038],[14.71319,52.22144],[14.68344,52.19612],[14.70616,52.16927],[14.67683,52.13936],[14.6917,52.10283],[14.72971,52.09167],[14.76026,52.06624],[14.71339,52.00337],[14.70488,51.97679],[14.7139,51.95643],[14.71836,51.95606],[14.72163,51.95188],[14.7177,51.94048],[14.70601,51.92944],[14.6933,51.9044],[14.6588,51.88359],[14.59089,51.83302],[14.60493,51.80473],[14.64625,51.79472],[14.66386,51.73282],[14.69065,51.70842],[14.75392,51.67445],[14.75759,51.62318],[14.7727,51.61263],[14.71125,51.56209],[14.73047,51.54606],[14.72652,51.53902],[14.73219,51.52922],[14.94749,51.47155],[14.9652,51.44793],[14.96899,51.38367],[14.98008,51.33449],[15.04288,51.28387],[15.01242,51.21285],[15.0047,51.16874],[14.99311,51.16249],[14.99414,51.15813],[15.00083,51.14974],[14.99646,51.14365],[14.99079,51.14284],[14.99689,51.12205],[14.98229,51.11354],[14.97938,51.07742],[14.95529,51.04552],[14.92942,50.99744],[14.89252,50.94999],[14.89681,50.9422],[14.81664,50.88148],[14.82803,50.86966],[14.99852,50.86817],[15.01088,50.97984],[14.96419,50.99108],[15.02433,51.0242],[15.03895,51.0123],[15.06218,51.02269],[15.10152,51.01095],[15.11937,50.99021],[15.16744,51.01959],[15.1743,50.9833],[15.2361,50.99886],[15.27043,50.97724],[15.2773,50.8907],[15.36656,50.83956],[15.3803,50.77187],[15.43798,50.80833],[15.73186,50.73885],[15.81683,50.75666],[15.87331,50.67188],[15.97219,50.69799],[16.0175,50.63009],[15.98317,50.61528],[16.02437,50.60046],[16.10265,50.66405],[16.20839,50.63096],[16.23174,50.67101],[16.33611,50.66579],[16.44597,50.58041],[16.34572,50.49575],[16.31413,50.50274],[16.19526,50.43291],[16.21585,50.40627],[16.22821,50.41054],[16.28118,50.36891],[16.30289,50.38292],[16.36495,50.37679],[16.3622,50.34875],[16.39379,50.3207],[16.42674,50.32509],[16.56407,50.21009],[16.55446,50.16613],[16.63137,50.1142],[16.7014,50.09659],[16.8456,50.20834],[16.98018,50.24172],[17.00353,50.21449],[17.02825,50.23118],[16.99803,50.25753],[17.02138,50.27772],[16.99803,50.30316],[16.94448,50.31281],[16.90877,50.38642],[16.85933,50.41093],[16.89229,50.45117],[17.1224,50.39494],[17.14498,50.38117],[17.19579,50.38817],[17.19991,50.3654],[17.27681,50.32246],[17.34273,50.32947],[17.34548,50.2628],[17.3702,50.28123],[17.58889,50.27837],[17.67764,50.28977],[17.69292,50.32859],[17.74648,50.29966],[17.72176,50.25665],[17.76296,50.23382],[17.70528,50.18812],[17.59404,50.16437],[17.66683,50.10275],[17.6888,50.12037],[17.7506,50.07896],[17.77669,50.02253],[17.86886,49.97452],[18.00191,50.01723],[18.04585,50.01194],[18.04585,50.03311],[18.00396,50.04954],[18.03212,50.06574],[18.07898,50.04535],[18.10628,50.00223],[18.20241,49.99958],[18.21752,49.97309],[18.27107,49.96779],[18.27794,49.93863],[18.31914,49.91565],[18.33278,49.92415],[18.33562,49.94747],[18.41604,49.93498],[18.53423,49.89906],[18.54495,49.9079],[18.54299,49.92537],[18.57697,49.91565],[18.57045,49.87849],[18.60341,49.86256],[18.57183,49.83334],[18.61278,49.7618],[18.61368,49.75426],[18.62645,49.75002],[18.62943,49.74603],[18.62676,49.71983],[18.69817,49.70473],[18.72838,49.68163],[18.80479,49.6815],[18.84786,49.5446],[18.84521,49.51672],[18.94536,49.52143],[18.97283,49.49914],[18.9742,49.39557],[19.18019,49.41165],[19.25435,49.53391],[19.36009,49.53747],[19.37795,49.574],[19.45348,49.61583],[19.52626,49.57311],[19.53313,49.52856],[19.57845,49.46077],[19.64162,49.45184],[19.6375,49.40897],[19.72127,49.39288],[19.78581,49.41701],[19.82237,49.27806],[19.75286,49.20751],[19.86409,49.19316],[19.90529,49.23532],[19.98494,49.22904],[20.08238,49.1813],[20.13738,49.31685],[20.21977,49.35265],[20.31453,49.34817],[20.31728,49.39914],[20.39939,49.3896],[20.46422,49.41612],[20.5631,49.375],[20.61666,49.41791],[20.72274,49.41813],[20.77971,49.35383],[20.9229,49.29626],[20.98733,49.30774],[21.09799,49.37176],[21.041,49.41791],[21.12477,49.43666],[21.19756,49.4054],[21.27858,49.45988],[21.43376,49.41433],[21.62328,49.4447],[21.77983,49.35443],[21.82927,49.39467],[21.96385,49.3437],[22.04427,49.22136],[22.56155,49.08865],[22.89122,49.00725],[22.86336,49.10513],[22.72009,49.20288],[22.748,49.32759],[22.69444,49.49378],[22.64534,49.53094],[22.78304,49.65543],[22.80261,49.69098],[22.83179,49.69875],[22.99329,49.84249],[23.28221,50.0957],[23.67635,50.33385],[23.71382,50.38248],[23.79445,50.40481],[23.99563,50.41289],[24.03668,50.44507],[24.07048,50.5071],[24.0996,50.60752],[24.0595,50.71625],[23.95925,50.79271],[23.99254,50.83847],[24.0952,50.83262],[24.14524,50.86128],[24.04576,50.90196],[23.92217,51.00836],[23.90376,51.07697],[23.80678,51.18405],[23.63858,51.32182],[23.69905,51.40871],[23.62751,51.50512],[23.56236,51.53673],[23.57053,51.55938],[23.53198,51.74298],[23.62691,51.78208],[23.61523,51.92066],[23.68733,51.9906],[23.64066,52.07626],[23.61,52.11264],[23.54314,52.12148],[23.47859,52.18215],[23.20071,52.22848],[23.18196,52.28812],[23.34141,52.44845],[23.45112,52.53774],[23.58296,52.59868],[23.73615,52.6149],[23.93763,52.71332],[23.91805,52.94016],[23.94689,52.95919],[23.92184,53.02079],[23.87548,53.0831],[23.91393,53.16469],[23.85657,53.22923],[23.81995,53.24131],[23.62004,53.60942],[23.51284,53.95052],[23.48261,53.98855],[23.52702,54.04622],[23.49196,54.14764],[23.45223,54.17775],[23.42418,54.17911],[23.39525,54.21672],[23.3494,54.25155],[23.24656,54.25701],[23.15938,54.29894],[23.15526,54.31076],[23.13905,54.31567],[23.104,54.29794],[23.04323,54.31567],[23.05726,54.34565],[22.99649,54.35927],[23.00584,54.38514],[22.83756,54.40827],[22.79705,54.36264],[21.41123,54.32395],[20.63871,54.3706],[19.8038,54.44203],[19.64312,54.45423],[18.57853,55.25302]]]]}},{type:"Feature",properties:{iso1A2:"PM",iso1A3:"SPM",iso1N3:"666",wikidata:"Q34617",nameEn:"Saint Pierre and Miquelon",country:"FR",groups:["021","003","019"],callingCodes:["508"]},geometry:{type:"MultiPolygon",coordinates:[[[[-56.72993,46.65575],[-55.90758,46.6223],[-56.27503,47.39728],[-56.72993,46.65575]]]]}},{type:"Feature",properties:{iso1A2:"PN",iso1A3:"PCN",iso1N3:"612",wikidata:"Q35672",nameEn:"Pitcairn Islands",country:"GB",groups:["061","009"],driveSide:"left",callingCodes:["64"]},geometry:{type:"MultiPolygon",coordinates:[[[[-133.59543,-28.4709],[-122.0366,-24.55017],[-133.61511,-21.93325],[-133.59543,-28.4709]]]]}},{type:"Feature",properties:{iso1A2:"PR",iso1A3:"PRI",iso1N3:"630",wikidata:"Q1183",nameEn:"Puerto Rico",country:"US",groups:["029","003","419","019"],roadSpeedUnit:"mph",callingCodes:["1 787","1 939"]},geometry:{type:"MultiPolygon",coordinates:[[[[-65.27974,17.56928],[-65.02435,18.73231],[-67.99519,18.97186],[-68.20301,17.83927],[-65.27974,17.56928]]]]}},{type:"Feature",properties:{iso1A2:"PS",iso1A3:"PSE",iso1N3:"275",wikidata:"Q23792",nameEn:"Palestine",country:"IL",groups:["145","142"],callingCodes:["970"]},geometry:{type:"MultiPolygon",coordinates:[[[[34.052,31.46619],[34.21853,31.32363],[34.23572,31.2966],[34.24012,31.29591],[34.26742,31.21998],[34.29417,31.24194],[34.36523,31.28963],[34.37381,31.30598],[34.36505,31.36404],[34.40077,31.40926],[34.48892,31.48365],[34.56797,31.54197],[34.48681,31.59711],[34.29262,31.70393],[34.052,31.46619]]],[[[35.47672,31.49578],[35.55941,31.76535],[35.52758,31.9131],[35.54375,31.96587],[35.52012,32.04076],[35.57111,32.21877],[35.55807,32.38674],[35.42078,32.41562],[35.41048,32.43706],[35.41598,32.45593],[35.42034,32.46009],[35.40224,32.50136],[35.35212,32.52047],[35.30685,32.51024],[35.29306,32.50947],[35.25049,32.52453],[35.2244,32.55289],[35.15937,32.50466],[35.10882,32.4757],[35.10024,32.47856],[35.09236,32.47614],[35.08564,32.46948],[35.07059,32.4585],[35.05423,32.41754],[35.05311,32.4024],[35.0421,32.38242],[35.05142,32.3667],[35.04243,32.35008],[35.01772,32.33863],[35.01119,32.28684],[35.02939,32.2671],[35.01841,32.23981],[34.98885,32.20758],[34.95703,32.19522],[34.96009,32.17503],[34.99039,32.14626],[34.98507,32.12606],[34.99437,32.10962],[34.9863,32.09551],[35.00261,32.027],[34.98682,31.96935],[35.00124,31.93264],[35.03489,31.92448],[35.03978,31.89276],[35.03489,31.85919],[34.99712,31.85569],[34.9724,31.83352],[35.01978,31.82944],[35.05617,31.85685],[35.07677,31.85627],[35.14174,31.81325],[35.18603,31.80901],[35.18169,31.82542],[35.19461,31.82687],[35.21469,31.81835],[35.216,31.83894],[35.21128,31.863],[35.20381,31.86716],[35.20673,31.88151],[35.20791,31.8821],[35.20945,31.8815],[35.21016,31.88237],[35.21276,31.88153],[35.2136,31.88241],[35.22014,31.88264],[35.22294,31.87889],[35.22567,31.86745],[35.22817,31.8638],[35.2249,31.85433],[35.2304,31.84222],[35.24816,31.8458],[35.25753,31.8387],[35.251,31.83085],[35.26404,31.82567],[35.25573,31.81362],[35.26058,31.79064],[35.25225,31.7678],[35.26319,31.74846],[35.25182,31.73945],[35.24981,31.72543],[35.2438,31.7201],[35.24315,31.71244],[35.23972,31.70896],[35.22392,31.71899],[35.21937,31.71578],[35.20538,31.72388],[35.18023,31.72067],[35.16478,31.73242],[35.15474,31.73352],[35.15119,31.73634],[35.13931,31.73012],[35.12933,31.7325],[35.11895,31.71454],[35.10782,31.71594],[35.08226,31.69107],[35.00879,31.65426],[34.95249,31.59813],[34.9415,31.55601],[34.94356,31.50743],[34.93258,31.47816],[34.89756,31.43891],[34.87833,31.39321],[34.88932,31.37093],[34.92571,31.34337],[35.02459,31.35979],[35.13033,31.3551],[35.22921,31.37445],[35.39675,31.49572],[35.47672,31.49578]]]]}},{type:"Feature",properties:{iso1A2:"PT",iso1A3:"PRT",iso1N3:"620",wikidata:"Q45",nameEn:"Portugal",groups:["EU","039","150"],callingCodes:["351"]},geometry:{type:"MultiPolygon",coordinates:[[[[-6.19128,41.57638],[-6.29863,41.66432],[-6.44204,41.68258],[-6.49907,41.65823],[-6.54633,41.68623],[-6.56426,41.74219],[-6.51374,41.8758],[-6.56752,41.88429],[-6.5447,41.94371],[-6.58544,41.96674],[-6.61967,41.94008],[-6.75004,41.94129],[-6.76959,41.98734],[-6.81196,41.99097],[-6.82174,41.94493],[-6.94396,41.94403],[-6.95537,41.96553],[-6.98144,41.9728],[-7.01078,41.94977],[-7.07596,41.94977],[-7.08574,41.97401],[-7.14115,41.98855],[-7.18549,41.97515],[-7.18677,41.88793],[-7.32366,41.8406],[-7.37092,41.85031],[-7.42864,41.80589],[-7.42854,41.83262],[-7.44759,41.84451],[-7.45566,41.86488],[-7.49803,41.87095],[-7.52737,41.83939],[-7.62188,41.83089],[-7.58603,41.87944],[-7.65774,41.88308],[-7.69848,41.90977],[-7.84188,41.88065],[-7.88055,41.84571],[-7.88751,41.92553],[-7.90707,41.92432],[-7.92336,41.8758],[-7.9804,41.87337],[-8.01136,41.83453],[-8.0961,41.81024],[-8.16455,41.81753],[-8.16944,41.87944],[-8.19551,41.87459],[-8.2185,41.91237],[-8.16232,41.9828],[-8.08796,42.01398],[-8.08847,42.05767],[-8.11729,42.08537],[-8.18178,42.06436],[-8.19406,42.12141],[-8.18947,42.13853],[-8.1986,42.15402],[-8.22406,42.1328],[-8.24681,42.13993],[-8.2732,42.12396],[-8.29809,42.106],[-8.32161,42.10218],[-8.33912,42.08358],[-8.36353,42.09065],[-8.38323,42.07683],[-8.40143,42.08052],[-8.42512,42.07199],[-8.44123,42.08218],[-8.48185,42.0811],[-8.52837,42.07658],[-8.5252,42.06264],[-8.54563,42.0537],[-8.58086,42.05147],[-8.59493,42.05708],[-8.63791,42.04691],[-8.64626,42.03668],[-8.65832,42.02972],[-8.6681,41.99703],[-8.69071,41.98862],[-8.7478,41.96282],[-8.74606,41.9469],[-8.75712,41.92833],[-8.81794,41.90375],[-8.87157,41.86488],[-9.14112,41.86623],[-36.43765,41.39418],[-15.92339,29.50503],[-7.37282,36.96896],[-7.39769,37.16868],[-7.41133,37.20314],[-7.41854,37.23813],[-7.43227,37.25152],[-7.43974,37.38913],[-7.46878,37.47127],[-7.51759,37.56119],[-7.41981,37.75729],[-7.33441,37.81193],[-7.27314,37.90145],[-7.24544,37.98884],[-7.12648,38.00296],[-7.10366,38.04404],[-7.05966,38.01966],[-7.00375,38.01914],[-6.93418,38.21454],[-7.09389,38.17227],[-7.15581,38.27597],[-7.32529,38.44336],[-7.265,38.61674],[-7.26174,38.72107],[-7.03848,38.87221],[-7.051,38.907],[-6.95211,39.0243],[-6.97004,39.07619],[-7.04011,39.11919],[-7.10692,39.10275],[-7.14929,39.11287],[-7.12811,39.17101],[-7.23566,39.20132],[-7.23403,39.27579],[-7.3149,39.34857],[-7.2927,39.45847],[-7.49477,39.58794],[-7.54121,39.66717],[-7.33507,39.64569],[-7.24707,39.66576],[-7.01613,39.66877],[-6.97492,39.81488],[-6.91463,39.86618],[-6.86737,40.01986],[-6.94233,40.10716],[-7.00589,40.12087],[-7.02544,40.18564],[-7.00426,40.23169],[-6.86085,40.26776],[-6.86085,40.2976],[-6.80218,40.33239],[-6.78426,40.36468],[-6.84618,40.42177],[-6.84944,40.46394],[-6.7973,40.51723],[-6.80218,40.55067],[-6.84292,40.56801],[-6.79567,40.65955],[-6.82826,40.74603],[-6.82337,40.84472],[-6.79892,40.84842],[-6.80707,40.88047],[-6.84292,40.89771],[-6.8527,40.93958],[-6.9357,41.02888],[-6.913,41.03922],[-6.88843,41.03027],[-6.84781,41.02692],[-6.80942,41.03629],[-6.79241,41.05397],[-6.75655,41.10187],[-6.77319,41.13049],[-6.69711,41.1858],[-6.68286,41.21641],[-6.65046,41.24725],[-6.55937,41.24417],[-6.38551,41.35274],[-6.38553,41.38655],[-6.3306,41.37677],[-6.26777,41.48796],[-6.19128,41.57638]]]]}},{type:"Feature",properties:{iso1A2:"PW",iso1A3:"PLW",iso1N3:"585",wikidata:"Q695",nameEn:"Palau",groups:["057","009"],roadSpeedUnit:"mph",callingCodes:["680"]},geometry:{type:"MultiPolygon",coordinates:[[[[128.97621,3.08804],[134.40878,1.79674],[136.27107,6.73747],[136.04605,12.45908],[128.97621,3.08804]]]]}},{type:"Feature",properties:{iso1A2:"PY",iso1A3:"PRY",iso1N3:"600",wikidata:"Q733",nameEn:"Paraguay",groups:["005","419","019"],callingCodes:["595"]},geometry:{type:"MultiPolygon",coordinates:[[[[-58.16225,-20.16193],[-58.23216,-19.80058],[-59.06965,-19.29148],[-60.00638,-19.2981],[-61.73723,-19.63958],[-61.93912,-20.10053],[-62.26883,-20.55311],[-62.2757,-21.06657],[-62.64455,-22.25091],[-62.51761,-22.37684],[-62.22768,-22.55807],[-61.9756,-23.0507],[-61.0782,-23.62932],[-60.99754,-23.80934],[-60.28163,-24.04436],[-60.03367,-24.00701],[-59.45482,-24.34787],[-59.33886,-24.49935],[-58.33055,-24.97099],[-58.25492,-24.92528],[-57.80821,-25.13863],[-57.57431,-25.47269],[-57.87176,-25.93604],[-58.1188,-26.16704],[-58.3198,-26.83443],[-58.65321,-27.14028],[-58.59549,-27.29973],[-58.04205,-27.2387],[-56.85337,-27.5165],[-56.18313,-27.29851],[-55.89195,-27.3467],[-55.74475,-27.44485],[-55.59094,-27.32444],[-55.62322,-27.1941],[-55.39611,-26.97679],[-55.25243,-26.93808],[-55.16948,-26.96068],[-55.06351,-26.80195],[-55.00584,-26.78754],[-54.80868,-26.55669],[-54.70732,-26.45099],[-54.69333,-26.37705],[-54.67359,-25.98607],[-54.60664,-25.9691],[-54.62063,-25.91213],[-54.59398,-25.59224],[-54.59509,-25.53696],[-54.60196,-25.48397],[-54.62033,-25.46026],[-54.4423,-25.13381],[-54.28207,-24.07305],[-54.32807,-24.01865],[-54.6238,-23.83078],[-55.02691,-23.97317],[-55.0518,-23.98666],[-55.12292,-23.99669],[-55.41784,-23.9657],[-55.44117,-23.9185],[-55.43585,-23.87157],[-55.5555,-23.28237],[-55.52288,-23.2595],[-55.5446,-23.22811],[-55.63849,-22.95122],[-55.62493,-22.62765],[-55.68742,-22.58407],[-55.6986,-22.56268],[-55.72366,-22.5519],[-55.741,-22.52018],[-55.74941,-22.46436],[-55.8331,-22.29008],[-56.23206,-22.25347],[-56.45893,-22.08072],[-56.5212,-22.11556],[-56.6508,-22.28387],[-57.98625,-22.09157],[-57.94642,-21.73799],[-57.88239,-21.6868],[-57.93492,-21.65505],[-57.84536,-20.93155],[-58.16225,-20.16193]]]]}},{type:"Feature",properties:{iso1A2:"QA",iso1A3:"QAT",iso1N3:"634",wikidata:"Q846",nameEn:"Qatar",groups:["145","142"],callingCodes:["974"]},geometry:{type:"MultiPolygon",coordinates:[[[[50.92992,24.54396],[51.09638,24.46907],[51.29972,24.50747],[51.39468,24.62785],[51.58834,24.66608],[51.83108,24.71675],[51.83682,26.70231],[50.93865,26.30758],[50.81266,25.88946],[50.86149,25.6965],[50.7801,25.595],[50.80824,25.54641],[50.57069,25.57887],[50.8133,24.74049],[50.92992,24.54396]]]]}},{type:"Feature",properties:{iso1A2:"RE",iso1A3:"REU",iso1N3:"638",wikidata:"Q17070",nameEn:"Réunion",country:"FR",groups:["EU","014","202","002"],callingCodes:["262"]},geometry:{type:"MultiPolygon",coordinates:[[[[53.37984,-21.23941],[56.73473,-21.9174],[56.62373,-20.2711],[53.37984,-21.23941]]]]}},{type:"Feature",properties:{iso1A2:"RO",iso1A3:"ROU",iso1N3:"642",wikidata:"Q218",nameEn:"Romania",groups:["EU","151","150"],callingCodes:["40"]},geometry:{type:"MultiPolygon",coordinates:[[[[27.15622,47.98538],[27.02985,48.09083],[27.04118,48.12522],[26.96119,48.13003],[26.98042,48.15752],[26.94265,48.1969],[26.87708,48.19919],[26.81161,48.25049],[26.62823,48.25804],[26.55202,48.22445],[26.33504,48.18418],[26.17711,47.99246],[26.05901,47.9897],[25.77723,47.93919],[25.63878,47.94924],[25.23778,47.89403],[25.11144,47.75203],[24.88896,47.7234],[24.81893,47.82031],[24.70632,47.84428],[24.61994,47.95062],[24.43578,47.97131],[24.34926,47.9244],[24.22566,47.90231],[24.11281,47.91487],[24.06466,47.95317],[24.02999,47.95087],[24.00801,47.968],[23.98553,47.96076],[23.96337,47.96672],[23.94192,47.94868],[23.89352,47.94512],[23.8602,47.9329],[23.80904,47.98142],[23.75188,47.99705],[23.66262,47.98786],[23.63894,48.00293],[23.5653,48.00499],[23.52803,48.01818],[23.4979,47.96858],[23.33577,48.0237],[23.27397,48.08245],[23.15999,48.12188],[23.1133,48.08061],[23.08858,48.00716],[23.0158,47.99338],[22.92241,48.02002],[22.94301,47.96672],[22.89849,47.95851],[22.77991,47.87211],[22.76617,47.8417],[22.67247,47.7871],[22.46559,47.76583],[22.41979,47.7391],[22.31816,47.76126],[22.00917,47.50492],[22.03389,47.42508],[22.01055,47.37767],[21.94463,47.38046],[21.78395,47.11104],[21.648,47.03902],[21.68645,46.99595],[21.59581,46.91628],[21.59307,46.86935],[21.52028,46.84118],[21.48935,46.7577],[21.5151,46.72147],[21.43926,46.65109],[21.33214,46.63035],[21.26929,46.4993],[21.28061,46.44941],[21.16872,46.30118],[21.06572,46.24897],[20.86797,46.28884],[20.74574,46.25467],[20.76085,46.21002],[20.63863,46.12728],[20.49718,46.18721],[20.45377,46.14405],[20.35573,46.16629],[20.28324,46.1438],[20.26068,46.12332],[20.35862,45.99356],[20.54818,45.89939],[20.65645,45.82801],[20.70069,45.7493],[20.77416,45.75601],[20.78446,45.78522],[20.82364,45.77738],[20.80361,45.65875],[20.76798,45.60969],[20.83321,45.53567],[20.77217,45.49788],[20.86026,45.47295],[20.87948,45.42743],[21.09894,45.30144],[21.17612,45.32566],[21.20392,45.2677],[21.29398,45.24148],[21.48278,45.19557],[21.51299,45.15345],[21.4505,45.04294],[21.35855,45.01941],[21.54938,44.9327],[21.56328,44.89502],[21.48202,44.87199],[21.44013,44.87613],[21.35643,44.86364],[21.38802,44.78133],[21.55007,44.77304],[21.60019,44.75208],[21.61942,44.67059],[21.67504,44.67107],[21.71692,44.65349],[21.7795,44.66165],[21.99364,44.63395],[22.08016,44.49844],[22.13234,44.47444],[22.18315,44.48179],[22.30844,44.6619],[22.45301,44.7194],[22.61917,44.61489],[22.69196,44.61587],[22.76749,44.54446],[22.70981,44.51852],[22.61368,44.55719],[22.56493,44.53419],[22.54021,44.47836],[22.45436,44.47258],[22.56012,44.30712],[22.68166,44.28206],[22.67173,44.21564],[23.04988,44.07694],[23.01674,44.01946],[22.87873,43.9844],[22.83753,43.88055],[22.85314,43.84452],[23.05288,43.79494],[23.26772,43.84843],[23.4507,43.84936],[23.61687,43.79289],[23.73978,43.80627],[24.18149,43.68218],[24.35364,43.70211],[24.50264,43.76314],[24.62281,43.74082],[24.73542,43.68523],[24.96682,43.72693],[25.10718,43.6831],[25.17144,43.70261],[25.39528,43.61866],[25.72792,43.69263],[25.94911,43.85745],[26.05584,43.90925],[26.10115,43.96908],[26.38764,44.04356],[26.62712,44.05698],[26.95141,44.13555],[27.26845,44.12602],[27.39757,44.0141],[27.60834,44.01206],[27.64542,44.04958],[27.73468,43.95326],[27.92008,44.00761],[27.99558,43.84193],[28.23293,43.76],[29.24336,43.70874],[30.04414,45.08461],[29.69272,45.19227],[29.65428,45.25629],[29.68175,45.26885],[29.59798,45.38857],[29.42632,45.44545],[29.24779,45.43388],[28.96077,45.33164],[28.94292,45.28045],[28.81383,45.3384],[28.78911,45.24179],[28.71358,45.22631],[28.5735,45.24759],[28.34554,45.32102],[28.28504,45.43907],[28.21139,45.46895],[28.18741,45.47358],[28.08927,45.6051],[28.16568,45.6421],[28.13111,45.92819],[28.08612,46.01105],[28.13684,46.18099],[28.10937,46.22852],[28.19864,46.31869],[28.18902,46.35283],[28.25769,46.43334],[28.22281,46.50481],[28.24808,46.64305],[28.12173,46.82283],[28.09095,46.97621],[27.81892,47.1381],[27.73172,47.29248],[27.68706,47.28962],[27.60263,47.32507],[27.55731,47.46637],[27.47942,47.48113],[27.3979,47.59473],[27.32202,47.64009],[27.25519,47.71366],[27.29069,47.73722],[27.1618,47.92391],[27.15622,47.98538]]]]}},{type:"Feature",properties:{iso1A2:"RS",iso1A3:"SRB",iso1N3:"688",wikidata:"Q403",nameEn:"Serbia",groups:["039","150"],callingCodes:["381"]},geometry:{type:"MultiPolygon",coordinates:[[[[19.66007,46.19005],[19.56113,46.16824],[19.52473,46.1171],[19.28826,45.99694],[19.14543,45.9998],[19.10388,46.04015],[19.0791,45.96458],[19.01284,45.96529],[18.99712,45.93537],[18.81394,45.91329],[18.85783,45.85493],[18.90305,45.71863],[18.96691,45.66731],[18.88776,45.57253],[18.94562,45.53712],[19.07471,45.53086],[19.08364,45.48804],[18.99918,45.49333],[18.97446,45.37528],[19.10774,45.29547],[19.28208,45.23813],[19.41941,45.23475],[19.43589,45.17137],[19.19144,45.17863],[19.14063,45.12972],[19.07952,45.14668],[19.1011,45.01191],[19.05205,44.97692],[19.15573,44.95409],[19.06853,44.89915],[19.02871,44.92541],[18.98957,44.90645],[19.01994,44.85493],[19.18183,44.92055],[19.36722,44.88164],[19.32543,44.74058],[19.26388,44.65412],[19.16699,44.52197],[19.13369,44.52521],[19.12278,44.50132],[19.14837,44.45253],[19.14681,44.41463],[19.11785,44.40313],[19.10749,44.39421],[19.10704,44.38249],[19.10365,44.37795],[19.10298,44.36924],[19.11865,44.36712],[19.1083,44.3558],[19.11547,44.34218],[19.13556,44.338],[19.13332,44.31492],[19.16741,44.28648],[19.18328,44.28383],[19.20508,44.2917],[19.23306,44.26097],[19.26945,44.26957],[19.32464,44.27185],[19.34773,44.23244],[19.3588,44.18353],[19.40927,44.16722],[19.43905,44.13088],[19.47338,44.15034],[19.48386,44.14332],[19.47321,44.1193],[19.51167,44.08158],[19.55999,44.06894],[19.57467,44.04716],[19.61991,44.05254],[19.61836,44.01464],[19.56498,43.99922],[19.52515,43.95573],[19.38439,43.96611],[19.24363,44.01502],[19.23465,43.98764],[19.3986,43.79668],[19.5176,43.71403],[19.50455,43.58385],[19.42696,43.57987],[19.41941,43.54056],[19.36653,43.60921],[19.33426,43.58833],[19.2553,43.5938],[19.24774,43.53061],[19.22807,43.5264],[19.22229,43.47926],[19.44315,43.38846],[19.48171,43.32644],[19.52962,43.31623],[19.54598,43.25158],[19.62661,43.2286],[19.64063,43.19027],[19.76918,43.16044],[19.79255,43.11951],[19.92576,43.08539],[19.96549,43.11098],[19.98887,43.0538],[20.04729,43.02732],[20.05431,42.99571],[20.12325,42.96237],[20.14896,42.99058],[20.16415,42.97177],[20.34528,42.90676],[20.35692,42.8335],[20.40594,42.84853],[20.43734,42.83157],[20.53484,42.8885],[20.48692,42.93208],[20.59929,43.01067],[20.64557,43.00826],[20.69515,43.09641],[20.59929,43.20492],[20.68688,43.21335],[20.73811,43.25068],[20.82145,43.26769],[20.88685,43.21697],[20.83727,43.17842],[20.96287,43.12416],[21.00749,43.13984],[21.05378,43.10707],[21.08952,43.13471],[21.14465,43.11089],[21.16734,42.99694],[21.2041,43.02277],[21.23877,43.00848],[21.23534,42.95523],[21.2719,42.8994],[21.32974,42.90424],[21.36941,42.87397],[21.44047,42.87276],[21.39045,42.74888],[21.47498,42.74695],[21.59154,42.72643],[21.58755,42.70418],[21.6626,42.67813],[21.75025,42.70125],[21.79413,42.65923],[21.75672,42.62695],[21.7327,42.55041],[21.70522,42.54176],[21.7035,42.51899],[21.62556,42.45106],[21.64209,42.41081],[21.62887,42.37664],[21.59029,42.38042],[21.57021,42.3647],[21.53467,42.36809],[21.5264,42.33634],[21.56772,42.30946],[21.58992,42.25915],[21.70111,42.23789],[21.77176,42.2648],[21.84654,42.3247],[21.91595,42.30392],[21.94405,42.34669],[22.02908,42.29848],[22.16384,42.32103],[22.29605,42.37477],[22.29275,42.34913],[22.34773,42.31725],[22.45919,42.33822],[22.47498,42.3915],[22.51961,42.3991],[22.55669,42.50144],[22.43983,42.56851],[22.4997,42.74144],[22.43309,42.82057],[22.54302,42.87774],[22.74826,42.88701],[22.78397,42.98253],[22.89521,43.03625],[22.98104,43.11199],[23.00806,43.19279],[22.89727,43.22417],[22.82036,43.33665],[22.53397,43.47225],[22.47582,43.6558],[22.41043,43.69566],[22.35558,43.81281],[22.41449,44.00514],[22.61688,44.06534],[22.61711,44.16938],[22.67173,44.21564],[22.68166,44.28206],[22.56012,44.30712],[22.45436,44.47258],[22.54021,44.47836],[22.56493,44.53419],[22.61368,44.55719],[22.70981,44.51852],[22.76749,44.54446],[22.69196,44.61587],[22.61917,44.61489],[22.45301,44.7194],[22.30844,44.6619],[22.18315,44.48179],[22.13234,44.47444],[22.08016,44.49844],[21.99364,44.63395],[21.7795,44.66165],[21.71692,44.65349],[21.67504,44.67107],[21.61942,44.67059],[21.60019,44.75208],[21.55007,44.77304],[21.38802,44.78133],[21.35643,44.86364],[21.44013,44.87613],[21.48202,44.87199],[21.56328,44.89502],[21.54938,44.9327],[21.35855,45.01941],[21.4505,45.04294],[21.51299,45.15345],[21.48278,45.19557],[21.29398,45.24148],[21.20392,45.2677],[21.17612,45.32566],[21.09894,45.30144],[20.87948,45.42743],[20.86026,45.47295],[20.77217,45.49788],[20.83321,45.53567],[20.76798,45.60969],[20.80361,45.65875],[20.82364,45.77738],[20.78446,45.78522],[20.77416,45.75601],[20.70069,45.7493],[20.65645,45.82801],[20.54818,45.89939],[20.35862,45.99356],[20.26068,46.12332],[20.09713,46.17315],[20.03533,46.14509],[20.01816,46.17696],[19.93508,46.17553],[19.81491,46.1313],[19.66007,46.19005]]]]}},{type:"Feature",properties:{iso1A2:"RU",iso1A3:"RUS",iso1N3:"643",wikidata:"Q159",nameEn:"Russia",groups:["151","150"],callingCodes:["7"]},geometry:{type:"MultiPolygon",coordinates:[[[[-179.99933,64.74703],[-172.76104,63.77445],[-169.03888,65.48473],[-168.95635,65.98512],[-168.25765,71.99091],[-179.9843,71.90735],[-179.99933,64.74703]]],[[[39.81147,43.06294],[40.0078,43.38551],[40.00853,43.40578],[40.01552,43.42025],[40.01007,43.42411],[40.03312,43.44262],[40.04445,43.47776],[40.10657,43.57344],[40.65957,43.56212],[41.64935,43.22331],[42.40563,43.23226],[42.66667,43.13917],[42.75889,43.19651],[43.03322,43.08883],[43.0419,43.02413],[43.81453,42.74297],[43.73119,42.62043],[43.95517,42.55396],[44.54202,42.75699],[44.70002,42.74679],[44.80941,42.61277],[44.88754,42.74934],[45.15318,42.70598],[45.36501,42.55268],[45.78692,42.48358],[45.61676,42.20768],[46.42738,41.91323],[46.5332,41.87389],[46.58924,41.80547],[46.75269,41.8623],[46.8134,41.76252],[47.00955,41.63583],[46.99554,41.59743],[47.03757,41.55434],[47.10762,41.59044],[47.34579,41.27884],[47.49004,41.26366],[47.54504,41.20275],[47.62288,41.22969],[47.75831,41.19455],[47.87973,41.21798],[48.07587,41.49957],[48.22064,41.51472],[48.2878,41.56221],[48.40277,41.60441],[48.42301,41.65444],[48.55078,41.77917],[48.5867,41.84306],[48.80971,41.95365],[49.2134,44.84989],[49.88945,46.04554],[49.32259,46.26944],[49.16518,46.38542],[48.54988,46.56267],[48.51142,46.69268],[49.01136,46.72716],[48.52326,47.4102],[48.45173,47.40818],[48.15348,47.74545],[47.64973,47.76559],[47.41689,47.83687],[47.38731,47.68176],[47.12107,47.83687],[47.11516,48.27188],[46.49011,48.43019],[46.78392,48.95352],[46.91104,48.99715],[47.01458,49.07085],[47.04416,49.17152],[46.98795,49.23531],[46.78398,49.34026],[46.9078,49.86707],[47.18319,49.93721],[47.34589,50.09308],[47.30448,50.30894],[47.58551,50.47867],[48.10044,50.09242],[48.24519,49.86099],[48.42564,49.82283],[48.68352,49.89546],[48.90782,50.02281],[48.57946,50.63278],[48.86936,50.61589],[49.12673,50.78639],[49.41959,50.85927],[49.39001,51.09396],[49.76866,51.11067],[49.97277,51.2405],[50.26859,51.28677],[50.59695,51.61859],[51.26254,51.68466],[51.301,51.48799],[51.77431,51.49536],[51.8246,51.67916],[52.36119,51.74161],[52.54329,51.48444],[53.46165,51.49445],[53.69299,51.23466],[54.12248,51.11542],[54.46331,50.85554],[54.41894,50.61214],[54.55797,50.52006],[54.71476,50.61214],[54.56685,51.01958],[54.72067,51.03261],[55.67774,50.54508],[56.11398,50.7471],[56.17906,50.93204],[57.17302,51.11253],[57.44221,50.88354],[57.74986,50.93017],[57.75578,51.13852],[58.3208,51.15151],[58.87974,50.70852],[59.48928,50.64216],[59.51886,50.49937],[59.81172,50.54451],[60.01288,50.8163],[60.17262,50.83312],[60.31914,50.67705],[60.81833,50.6629],[61.4431,50.80679],[61.56889,51.23679],[61.6813,51.25716],[61.55114,51.32746],[61.50677,51.40687],[60.95655,51.48615],[60.92401,51.61124],[60.5424,51.61675],[60.36787,51.66815],[60.50986,51.7964],[60.09867,51.87135],[59.99809,51.98263],[60.19925,51.99173],[60.48915,52.15175],[60.72581,52.15538],[60.78201,52.22067],[61.05417,52.35096],[60.98021,52.50068],[60.84709,52.52228],[60.84118,52.63912],[60.71693,52.66245],[60.71989,52.75923],[61.05842,52.92217],[61.23462,53.03227],[62.0422,52.96105],[62.12799,52.99133],[62.14574,53.09626],[61.19024,53.30536],[61.14291,53.41481],[61.29082,53.50992],[61.37957,53.45887],[61.57185,53.50112],[61.55706,53.57144],[60.90626,53.62937],[61.22574,53.80268],[61.14283,53.90063],[60.99796,53.93699],[61.26863,53.92797],[61.3706,54.08464],[61.47603,54.08048],[61.56941,53.95703],[61.65318,54.02445],[62.03913,53.94768],[62.00966,54.04134],[62.38535,54.03961],[62.45931,53.90737],[62.56876,53.94047],[62.58651,54.05871],[63.80604,54.27079],[63.91224,54.20013],[64.02715,54.22679],[63.97686,54.29763],[64.97216,54.4212],[65.11033,54.33028],[65.24663,54.35721],[65.20174,54.55216],[68.21308,54.98645],[68.26661,55.09226],[68.19206,55.18823],[68.90865,55.38148],[69.34224,55.36344],[69.74917,55.35545],[70.19179,55.1476],[70.76493,55.3027],[70.96009,55.10558],[71.08288,54.71253],[71.24185,54.64965],[71.08706,54.33376],[71.10379,54.13326],[71.96141,54.17736],[72.17477,54.36303],[72.43415,53.92685],[72.71026,54.1161],[73.37963,53.96132],[73.74778,54.07194],[73.68921,53.86522],[73.25412,53.61532],[73.39218,53.44623],[75.07405,53.80831],[75.43398,53.98652],[75.3668,54.07439],[76.91052,54.4677],[76.82266,54.1798],[76.44076,54.16017],[76.54243,53.99329],[77.90383,53.29807],[79.11255,52.01171],[80.08138,50.77658],[80.4127,50.95581],[80.44819,51.20855],[80.80318,51.28262],[81.16999,51.15662],[81.06091,50.94833],[81.41248,50.97524],[81.46581,50.77658],[81.94999,50.79307],[82.55443,50.75412],[83.14607,51.00796],[83.8442,50.87375],[84.29385,50.27257],[84.99198,50.06793],[85.24047,49.60239],[86.18709,49.50259],[86.63674,49.80136],[86.79056,49.74787],[86.61307,49.60239],[86.82606,49.51796],[87.03071,49.25142],[87.31465,49.23603],[87.28386,49.11626],[87.478,49.07403],[87.48983,49.13794],[87.81333,49.17354],[87.98977,49.18147],[88.15543,49.30314],[88.17223,49.46934],[88.42449,49.48821],[88.82499,49.44808],[89.70687,49.72535],[89.59711,49.90851],[91.86048,50.73734],[92.07173,50.69585],[92.44714,50.78762],[93.01109,50.79001],[92.99595,50.63183],[94.30823,50.57498],[94.39258,50.22193],[94.49477,50.17832],[94.6121,50.04239],[94.97166,50.04725],[95.02465,49.96941],[95.74757,49.97915],[95.80056,50.04239],[96.97388,49.88413],[97.24639,49.74737],[97.56811,49.84265],[97.56432,49.92801],[97.76871,49.99861],[97.85197,49.91339],[98.29481,50.33561],[98.31373,50.4996],[98.06393,50.61262],[97.9693,50.78044],[98.01472,50.86652],[97.83305,51.00248],[98.05257,51.46696],[98.22053,51.46579],[98.33222,51.71832],[98.74142,51.8637],[98.87768,52.14563],[99.27888,51.96876],[99.75578,51.90108],[99.89203,51.74903],[100.61116,51.73028],[101.39085,51.45753],[101.5044,51.50467],[102.14032,51.35566],[102.32194,50.67982],[102.71178,50.38873],[103.70343,50.13952],[105.32528,50.4648],[106.05562,50.40582],[106.07865,50.33474],[106.47156,50.31909],[106.49628,50.32436],[106.51122,50.34408],[106.58373,50.34044],[106.80326,50.30177],[107.00007,50.1977],[107.1174,50.04239],[107.36407,49.97612],[107.96116,49.93191],[107.95387,49.66659],[108.27937,49.53167],[108.53969,49.32325],[109.18017,49.34709],[109.51325,49.22859],[110.24373,49.16676],[110.39891,49.25083],[110.64493,49.1816],[113.02647,49.60772],[113.20216,49.83356],[114.325,50.28098],[114.9703,50.19254],[115.26068,49.97367],[115.73602,49.87688],[116.22402,50.04477],[116.62502,49.92919],[116.71193,49.83813],[117.07142,49.68482],[117.27597,49.62544],[117.48208,49.62324],[117.82343,49.52696],[118.61623,49.93809],[119.11003,50.00276],[119.27996,50.13348],[119.38598,50.35162],[119.13553,50.37412],[120.10963,51.671],[120.65907,51.93544],[120.77337,52.20805],[120.61346,52.32447],[120.71673,52.54099],[120.46454,52.63811],[120.04049,52.58773],[120.0451,52.7359],[120.85633,53.28499],[121.39213,53.31888],[122.35063,53.49565],[122.85966,53.47395],[123.26989,53.54843],[123.86158,53.49391],[124.46078,53.21881],[125.17522,53.20225],[125.6131,53.07229],[126.558,52.13738],[126.44606,51.98254],[126.68349,51.70607],[126.90369,51.3238],[126.93135,51.0841],[127.14586,50.91152],[127.28165,50.72075],[127.36335,50.58306],[127.28765,50.46585],[127.36009,50.43787],[127.37384,50.28393],[127.60515,50.23503],[127.49299,50.01251],[127.53516,49.84306],[127.83476,49.5748],[128.72896,49.58676],[129.11153,49.36813],[129.23232,49.40353],[129.35317,49.3481],[129.40398,49.44194],[129.50685,49.42398],[129.67598,49.29596],[129.85416,49.11067],[130.2355,48.86741],[130.43232,48.90844],[130.66946,48.88251],[130.52147,48.61745],[130.84462,48.30942],[130.65103,48.10052],[130.90915,47.90623],[130.95985,47.6957],[131.09871,47.6852],[131.2635,47.73325],[131.90448,47.68011],[132.57309,47.71741],[132.66989,47.96491],[134.49516,48.42884],[134.75328,48.36763],[134.67098,48.1564],[134.55508,47.98651],[134.7671,47.72051],[134.50898,47.4812],[134.20016,47.33458],[134.03538,46.75668],[133.84104,46.46681],[133.91496,46.4274],[133.88097,46.25066],[133.68047,46.14697],[133.72695,46.05576],[133.67569,45.9759],[133.60442,45.90053],[133.48457,45.86203],[133.41083,45.57723],[133.19419,45.51913],[133.09279,45.25693],[133.12293,45.1332],[132.96373,45.0212],[132.83978,45.05916],[131.99417,45.2567],[131.86903,45.33636],[131.76532,45.22609],[131.66852,45.2196],[131.68466,45.12374],[131.48415,44.99513],[130.95639,44.85154],[131.1108,44.70266],[131.30365,44.04262],[131.25484,44.03131],[131.23583,43.96085],[131.26176,43.94011],[131.21105,43.82383],[131.19492,43.53047],[131.29402,43.46695],[131.30324,43.39498],[131.19031,43.21385],[131.20414,43.13654],[131.10274,43.04734],[131.135,42.94114],[131.02668,42.91246],[131.02438,42.86518],[130.66524,42.84753],[130.44361,42.76205],[130.40213,42.70788],[130.56576,42.68925],[130.62107,42.58413],[130.55143,42.52158],[130.56835,42.43281],[130.60805,42.4317],[130.64181,42.41422],[130.66367,42.38024],[130.65022,42.32281],[131.95041,41.5445],[140.9182,45.92937],[145.82343,44.571],[145.23667,43.76813],[153.94307,38.42848],[180,62.52334],[180,71.53642],[155.31937,81.93282],[36.48095,82.16765],[32.07813,72.01005],[31.59909,70.16571],[30.84095,69.80584],[30.95011,69.54699],[30.52662,69.54699],[30.16363,69.65244],[29.97205,69.41623],[29.27631,69.2811],[29.26623,69.13794],[29.0444,69.0119],[28.91738,69.04774],[28.45957,68.91417],[28.78224,68.86696],[28.43941,68.53366],[28.62982,68.19816],[29.34179,68.06655],[29.66955,67.79872],[30.02041,67.67523],[29.91155,67.51507],[28.9839,66.94139],[29.91155,66.13863],[30.16363,65.66935],[29.97205,65.70256],[29.74013,65.64025],[29.84096,65.56945],[29.68972,65.31803],[29.61914,65.23791],[29.8813,65.22101],[29.84096,65.1109],[29.61914,65.05993],[29.68972,64.80789],[30.05271,64.79072],[30.12329,64.64862],[30.01238,64.57513],[30.06279,64.35782],[30.4762,64.25728],[30.55687,64.09036],[30.25437,63.83364],[29.98213,63.75795],[30.49637,63.46666],[31.23244,63.22239],[31.29294,63.09035],[31.58535,62.91642],[31.38369,62.66284],[31.10136,62.43042],[29.01829,61.17448],[28.82816,61.1233],[28.47974,60.93365],[27.77352,60.52722],[27.71177,60.3893],[27.44953,60.22766],[26.32936,60.00121],[26.90044,59.63819],[27.85643,59.58538],[28.04187,59.47017],[28.19061,59.39962],[28.21137,59.38058],[28.20537,59.36491],[28.19284,59.35791],[28.14215,59.28934],[28.00689,59.28351],[27.90911,59.24353],[27.87978,59.18097],[27.80482,59.1116],[27.74429,58.98351],[27.36366,58.78381],[27.55489,58.39525],[27.48541,58.22615],[27.62393,58.09462],[27.67282,57.92627],[27.81841,57.89244],[27.78526,57.83963],[27.56689,57.83356],[27.50171,57.78842],[27.52615,57.72843],[27.3746,57.66834],[27.40393,57.62125],[27.31919,57.57672],[27.34698,57.52242],[27.56832,57.53728],[27.52453,57.42826],[27.86101,57.29402],[27.66511,56.83921],[27.86101,56.88204],[28.04768,56.59004],[28.13526,56.57989],[28.10069,56.524],[28.19057,56.44637],[28.16599,56.37806],[28.23716,56.27588],[28.15217,56.16964],[28.30571,56.06035],[28.36888,56.05805],[28.37987,56.11399],[28.43068,56.09407],[28.5529,56.11705],[28.68337,56.10173],[28.63668,56.07262],[28.73418,55.97131],[29.08299,56.03427],[29.21717,55.98971],[29.44692,55.95978],[29.3604,55.75862],[29.51283,55.70294],[29.61446,55.77716],[29.80672,55.79569],[29.97975,55.87281],[30.12136,55.8358],[30.27776,55.86819],[30.30987,55.83592],[30.48257,55.81066],[30.51346,55.78982],[30.51037,55.76568],[30.63344,55.73079],[30.67464,55.64176],[30.72957,55.66268],[30.7845,55.58514],[30.86003,55.63169],[30.93419,55.6185],[30.95204,55.50667],[30.90123,55.46621],[30.93144,55.3914],[30.8257,55.3313],[30.81946,55.27931],[30.87944,55.28223],[30.97369,55.17134],[31.02071,55.06167],[31.00972,55.02783],[30.94243,55.03964],[30.9081,55.02232],[30.95754,54.98609],[30.93144,54.9585],[30.81759,54.94064],[30.8264,54.90062],[30.75165,54.80699],[30.95479,54.74346],[30.97127,54.71967],[31.0262,54.70698],[30.98226,54.68872],[30.99187,54.67046],[31.19339,54.66947],[31.21399,54.63113],[31.08543,54.50361],[31.22945,54.46585],[31.3177,54.34067],[31.30791,54.25315],[31.57002,54.14535],[31.89599,54.0837],[31.88744,54.03653],[31.85019,53.91801],[31.77028,53.80015],[31.89137,53.78099],[32.12621,53.81586],[32.36663,53.7166],[32.45717,53.74039],[32.50112,53.68594],[32.40499,53.6656],[32.47777,53.5548],[32.74968,53.45597],[32.73257,53.33494],[32.51725,53.28431],[32.40773,53.18856],[32.15368,53.07594],[31.82373,53.10042],[31.787,53.18033],[31.62496,53.22886],[31.56316,53.19432],[31.40523,53.21406],[31.36403,53.13504],[31.3915,53.09712],[31.33519,53.08805],[31.32283,53.04101],[31.24147,53.031],[31.35667,52.97854],[31.592,52.79011],[31.57277,52.71613],[31.50406,52.69707],[31.63869,52.55361],[31.56316,52.51518],[31.61397,52.48843],[31.62084,52.33849],[31.57971,52.32146],[31.70735,52.26711],[31.6895,52.1973],[31.77877,52.18636],[31.7822,52.11406],[31.81722,52.09955],[31.85018,52.11305],[31.96141,52.08015],[31.92159,52.05144],[32.08813,52.03319],[32.23331,52.08085],[32.2777,52.10266],[32.34044,52.1434],[32.33083,52.23685],[32.38988,52.24946],[32.3528,52.32842],[32.54781,52.32423],[32.69475,52.25535],[32.85405,52.27888],[32.89937,52.2461],[33.18913,52.3754],[33.51323,52.35779],[33.48027,52.31499],[33.55718,52.30324],[33.78789,52.37204],[34.05239,52.20132],[34.11199,52.14087],[34.09413,52.00835],[34.41136,51.82793],[34.42922,51.72852],[34.07765,51.67065],[34.17599,51.63253],[34.30562,51.5205],[34.22048,51.4187],[34.33446,51.363],[34.23009,51.26429],[34.31661,51.23936],[34.38802,51.2746],[34.6613,51.25053],[34.6874,51.18],[34.82472,51.17483],[34.97304,51.2342],[35.14058,51.23162],[35.12685,51.16191],[35.20375,51.04723],[35.31774,51.08434],[35.40837,51.04119],[35.32598,50.94524],[35.39307,50.92145],[35.41367,50.80227],[35.47704,50.77274],[35.48116,50.66405],[35.39464,50.64751],[35.47463,50.49247],[35.58003,50.45117],[35.61711,50.35707],[35.73659,50.35489],[35.80388,50.41356],[35.8926,50.43829],[36.06893,50.45205],[36.20763,50.3943],[36.30101,50.29088],[36.47817,50.31457],[36.58371,50.28563],[36.56655,50.2413],[36.64571,50.218],[36.69377,50.26982],[36.91762,50.34963],[37.08468,50.34935],[37.48204,50.46079],[37.47243,50.36277],[37.62486,50.29966],[37.62879,50.24481],[37.61113,50.21976],[37.75807,50.07896],[37.79515,50.08425],[37.90776,50.04194],[38.02999,49.94482],[38.02999,49.90592],[38.21675,49.98104],[38.18517,50.08161],[38.32524,50.08866],[38.35408,50.00664],[38.65688,49.97176],[38.68677,50.00904],[38.73311,49.90238],[38.90477,49.86787],[38.9391,49.79524],[39.1808,49.88911],[39.27968,49.75976],[39.44496,49.76067],[39.59142,49.73758],[39.65047,49.61761],[39.84548,49.56064],[40.13249,49.61672],[40.16683,49.56865],[40.03636,49.52321],[40.03087,49.45452],[40.1141,49.38798],[40.14912,49.37681],[40.18331,49.34996],[40.22176,49.25683],[40.01988,49.1761],[39.93437,49.05709],[39.6836,49.05121],[39.6683,48.99454],[39.71353,48.98959],[39.72649,48.9754],[39.74874,48.98675],[39.78368,48.91596],[39.98967,48.86901],[40.03636,48.91957],[40.08168,48.87443],[39.97182,48.79398],[39.79466,48.83739],[39.73104,48.7325],[39.71765,48.68673],[39.67226,48.59368],[39.79764,48.58668],[39.84548,48.57821],[39.86196,48.46633],[39.88794,48.44226],[39.94847,48.35055],[39.84136,48.33321],[39.84273,48.30947],[39.90041,48.3049],[39.91465,48.26743],[39.95248,48.29972],[39.9693,48.29904],[39.97325,48.31399],[39.99241,48.31768],[40.00752,48.22445],[39.94847,48.22811],[39.83724,48.06501],[39.88256,48.04482],[39.77544,48.04206],[39.82213,47.96396],[39.73935,47.82876],[38.87979,47.87719],[38.79628,47.81109],[38.76379,47.69346],[38.35062,47.61631],[38.28679,47.53552],[38.28954,47.39255],[38.22225,47.30788],[38.33074,47.30508],[38.32112,47.2585],[38.23049,47.2324],[38.22955,47.12069],[38.3384,46.98085],[38.12112,46.86078],[37.62608,46.82615],[35.23066,45.79231],[34.96015,45.75634],[34.79905,45.81009],[34.80153,45.90047],[34.75479,45.90705],[34.66679,45.97136],[34.60861,45.99347],[34.55889,45.99347],[34.52011,45.95097],[34.48729,45.94267],[34.44155,45.95995],[34.41221,46.00245],[34.33912,46.06114],[34.25111,46.0532],[34.181,46.06804],[34.12929,46.10494],[34.07311,46.11769],[34.05272,46.10838],[33.91549,46.15938],[33.85234,46.19863],[33.79715,46.20482],[33.74047,46.18555],[33.646,46.23028],[33.61517,46.22615],[33.63854,46.14147],[33.61467,46.13561],[33.57318,46.10317],[33.59087,46.06013],[33.54017,46.0123],[31.62627,45.50633],[32.99857,44.48323],[33.66142,43.9825],[39.81147,43.06294]]],[[[21.46766,55.21115],[21.38446,55.29348],[21.35465,55.28427],[21.26425,55.24456],[20.95181,55.27994],[20.60454,55.40986],[18.57853,55.25302],[19.64312,54.45423],[19.8038,54.44203],[20.63871,54.3706],[21.41123,54.32395],[22.79705,54.36264],[22.7253,54.41732],[22.70208,54.45312],[22.67788,54.532],[22.71293,54.56454],[22.68021,54.58486],[22.7522,54.63525],[22.74225,54.64339],[22.75467,54.6483],[22.73397,54.66604],[22.73631,54.72952],[22.87317,54.79492],[22.85083,54.88711],[22.76422,54.92521],[22.68723,54.9811],[22.65451,54.97037],[22.60075,55.01863],[22.58907,55.07085],[22.47688,55.04408],[22.31562,55.0655],[22.14267,55.05345],[22.11697,55.02131],[22.06087,55.02935],[22.02582,55.05078],[22.03984,55.07888],[21.99543,55.08691],[21.96505,55.07353],[21.85521,55.09493],[21.64954,55.1791],[21.55605,55.20311],[21.51095,55.18507],[21.46766,55.21115]]]]}},{type:"Feature",properties:{iso1A2:"RW",iso1A3:"RWA",iso1N3:"646",wikidata:"Q1037",nameEn:"Rwanda",groups:["014","202","002"],callingCodes:["250"]},geometry:{type:"MultiPolygon",coordinates:[[[[30.47194,-1.0555],[30.35212,-1.06896],[30.16369,-1.34303],[29.912,-1.48269],[29.82657,-1.31187],[29.59061,-1.39016],[29.53062,-1.40499],[29.45038,-1.5054],[29.36322,-1.50887],[29.24323,-1.66826],[29.24458,-1.69663],[29.11847,-1.90576],[29.17562,-2.12278],[29.105,-2.27043],[29.00051,-2.29001],[28.95642,-2.37321],[28.89601,-2.37321],[28.86826,-2.41888],[28.86846,-2.44866],[28.89132,-2.47557],[28.89342,-2.49017],[28.88846,-2.50493],[28.87497,-2.50887],[28.86209,-2.5231],[28.86193,-2.53185],[28.87943,-2.55165],[28.89288,-2.55848],[28.90226,-2.62385],[28.89793,-2.66111],[28.94346,-2.69124],[29.00357,-2.70596],[29.04081,-2.7416],[29.0562,-2.58632],[29.32234,-2.6483],[29.36805,-2.82933],[29.88237,-2.75105],[29.95911,-2.33348],[30.14034,-2.43626],[30.42933,-2.31064],[30.54501,-2.41404],[30.83915,-2.35795],[30.89303,-2.08223],[30.80802,-1.91477],[30.84079,-1.64652],[30.71974,-1.43244],[30.57123,-1.33264],[30.50889,-1.16412],[30.45116,-1.10641],[30.47194,-1.0555]]]]}},{type:"Feature",properties:{iso1A2:"SA",iso1A3:"SAU",iso1N3:"682",wikidata:"Q851",nameEn:"Saudi Arabia",groups:["145","142"],callingCodes:["966"]},geometry:{type:"MultiPolygon",coordinates:[[[[40.01521,32.05667],[39.29903,32.23259],[38.99233,31.99721],[36.99791,31.50081],[37.99354,30.49998],[37.66395,30.33245],[37.4971,29.99949],[36.75083,29.86903],[36.50005,29.49696],[36.07081,29.18469],[34.95987,29.35727],[34.88293,29.37455],[34.46254,27.99552],[34.51305,27.70027],[37.8565,22.00903],[39.63762,18.37348],[41.37609,16.19728],[42.15205,16.40211],[42.76801,16.40371],[42.94625,16.39721],[42.94351,16.49467],[42.97215,16.51093],[43.11601,16.53166],[43.15274,16.67248],[43.22066,16.65179],[43.21325,16.74416],[43.25857,16.75304],[43.26303,16.79479],[43.24801,16.80613],[43.22956,16.80613],[43.22012,16.83932],[43.18338,16.84852],[43.1398,16.90696],[43.19328,16.94703],[43.1813,16.98438],[43.18233,17.02673],[43.23967,17.03428],[43.17787,17.14717],[43.20156,17.25901],[43.32653,17.31179],[43.22533,17.38343],[43.29185,17.53224],[43.43005,17.56148],[43.70631,17.35762],[44.50126,17.47475],[46.31018,17.20464],[46.76494,17.29151],[47.00571,16.94765],[47.48245,17.10808],[47.58351,17.50366],[48.19996,18.20584],[49.04884,18.59899],[52.00311,19.00083],[54.99756,20.00083],[55.66469,21.99658],[55.2137,22.71065],[55.13599,22.63334],[52.56622,22.94341],[51.59617,24.12041],[51.58871,24.27256],[51.41644,24.39615],[51.58834,24.66608],[51.39468,24.62785],[51.29972,24.50747],[51.09638,24.46907],[50.92992,24.54396],[50.8133,24.74049],[50.57069,25.57887],[50.302,25.87592],[50.26923,26.08243],[50.38162,26.53976],[50.71771,26.73086],[50.37726,27.89227],[49.98877,27.87827],[49.00421,28.81495],[48.42991,28.53628],[47.70561,28.5221],[47.59863,28.66798],[47.58376,28.83382],[47.46202,29.0014],[46.5527,29.10283],[46.42415,29.05947],[44.72255,29.19736],[42.97796,30.48295],[42.97601,30.72204],[40.01521,32.05667]]]]}},{type:"Feature",properties:{iso1A2:"SB",iso1A3:"SLB",iso1N3:"090",wikidata:"Q685",nameEn:"Solomon Islands",groups:["054","009"],driveSide:"left",callingCodes:["677"]},geometry:{type:"MultiPolygon",coordinates:[[[[174,-12.72535],[160.43769,-4.17974],[156.03296,-6.55528],[156.03993,-6.65703],[155.92557,-6.84664],[155.69784,-6.92661],[155.60735,-6.92266],[154.74815,-7.33315],[160.04026,-13.08769],[174,-12.72535]]]]}},{type:"Feature",properties:{iso1A2:"SC",iso1A3:"SYC",iso1N3:"690",wikidata:"Q1042",nameEn:"Seychelles",groups:["014","202","002"],driveSide:"left",callingCodes:["248"]},geometry:{type:"MultiPolygon",coordinates:[[[[43.75112,-10.38913],[54.83239,-10.93575],[66.3222,5.65313],[43.75112,-10.38913]]]]}},{type:"Feature",properties:{iso1A2:"SD",iso1A3:"SDN",iso1N3:"729",wikidata:"Q1049",nameEn:"Sudan",groups:["015","002"],callingCodes:["249"]},geometry:{type:"MultiPolygon",coordinates:[[[[37.8565,22.00903],[34.0765,22.00501],[33.99686,21.76784],[33.57251,21.72406],[33.17563,22.00405],[24.99885,21.99535],[24.99794,19.99661],[23.99715,20.00038],[23.99539,19.49944],[23.99997,15.69575],[23.62785,15.7804],[23.38812,15.69649],[23.10792,15.71297],[22.93201,15.55107],[22.92579,15.47007],[22.99584,15.40105],[22.99584,15.22989],[22.66115,14.86308],[22.70474,14.69149],[22.38562,14.58907],[22.44944,14.24986],[22.55997,14.23024],[22.5553,14.11704],[22.22995,13.96754],[22.08674,13.77863],[22.29689,13.3731],[22.1599,13.19281],[22.02914,13.13976],[21.94819,13.05637],[21.81432,12.81362],[21.89371,12.68001],[21.98711,12.63292],[22.15679,12.66634],[22.22684,12.74682],[22.46345,12.61925],[22.38873,12.45514],[22.50548,12.16769],[22.48369,12.02766],[22.64092,12.07485],[22.54907,11.64372],[22.7997,11.40424],[22.93124,11.41645],[22.97249,11.21955],[22.87758,10.91915],[23.02221,10.69235],[23.3128,10.45214],[23.67164,9.86923],[23.69155,9.67566],[24.09319,9.66572],[24.12744,9.73784],[24.49389,9.79962],[24.84653,9.80643],[24.97739,9.9081],[25.05688,10.06776],[25.0918,10.33718],[25.78141,10.42599],[25.93163,10.38159],[25.93241,10.17941],[26.21338,9.91545],[26.35815,9.57946],[26.70685,9.48735],[27.14427,9.62858],[27.90704,9.61323],[28.99983,9.67155],[29.06988,9.74826],[29.53844,9.75133],[29.54,10.07949],[29.94629,10.29245],[30.00389,10.28633],[30.53005,9.95992],[30.82893,9.71451],[30.84605,9.7498],[31.28504,9.75287],[31.77539,10.28939],[31.99177,10.65065],[32.46967,11.04662],[32.39358,11.18207],[32.39578,11.70208],[32.10079,11.95203],[32.73921,11.95203],[32.73921,12.22757],[33.25876,12.22111],[33.13988,11.43248],[33.26977,10.83632],[33.24645,10.77913],[33.52294,10.64382],[33.66604,10.44254],[33.80913,10.32994],[33.90159,10.17179],[33.96984,10.15446],[33.99185,9.99623],[33.96323,9.80972],[33.9082,9.762],[33.87958,9.49937],[34.10229,9.50238],[34.08717,9.55243],[34.13186,9.7492],[34.20484,9.9033],[34.22718,10.02506],[34.32102,10.11599],[34.34783,10.23914],[34.2823,10.53508],[34.4372,10.781],[34.59062,10.89072],[34.77383,10.74588],[34.77532,10.69027],[34.86618,10.74588],[34.86916,10.78832],[34.97491,10.86147],[34.97789,10.91559],[34.93172,10.95946],[35.01215,11.19626],[34.95704,11.24448],[35.09556,11.56278],[35.05832,11.71158],[35.11492,11.85156],[35.24302,11.91132],[35.70476,12.67101],[36.01458,12.72478],[36.14268,12.70879],[36.16651,12.88019],[36.13374,12.92665],[36.24545,13.36759],[36.38993,13.56459],[36.48824,13.83954],[36.44653,13.95666],[36.54376,14.25597],[36.44337,15.14963],[36.54276,15.23478],[36.69761,15.75323],[36.76371,15.80831],[36.92193,16.23451],[36.99777,17.07172],[37.42694,17.04041],[37.50967,17.32199],[38.13362,17.53906],[38.37133,17.66269],[38.45916,17.87167],[38.57727,17.98125],[39.63762,18.37348],[37.8565,22.00903]]]]}},{type:"Feature",properties:{iso1A2:"SE",iso1A3:"SWE",iso1N3:"752",wikidata:"Q34",nameEn:"Sweden",groups:["EU","154","150"],callingCodes:["46"]},geometry:{type:"MultiPolygon",coordinates:[[[[24.15791,65.85385],[23.90497,66.15802],[23.71339,66.21299],[23.64982,66.30603],[23.67591,66.3862],[23.63776,66.43568],[23.85959,66.56434],[23.89488,66.772],[23.98059,66.79585],[23.98563,66.84149],[23.56214,67.17038],[23.58735,67.20752],[23.54701,67.25435],[23.75372,67.29914],[23.75372,67.43688],[23.39577,67.46974],[23.54701,67.59306],[23.45627,67.85297],[23.65793,67.9497],[23.40081,68.05545],[23.26469,68.15134],[23.15377,68.14759],[23.10336,68.26551],[22.73028,68.40881],[22.00429,68.50692],[21.03001,68.88969],[20.90649,68.89696],[20.85104,68.93142],[20.91658,68.96764],[20.78802,69.03087],[20.55258,69.06069],[20.0695,69.04469],[20.28444,68.93283],[20.33435,68.80174],[20.22027,68.67246],[19.95647,68.55546],[20.22027,68.48759],[19.93508,68.35911],[18.97255,68.52416],[18.63032,68.50849],[18.39503,68.58672],[18.1241,68.53721],[18.13836,68.20874],[17.90787,67.96537],[17.30416,68.11591],[16.7409,67.91037],[16.38441,67.52923],[16.12774,67.52106],[16.09922,67.4364],[16.39154,67.21653],[16.35589,67.06419],[15.37197,66.48217],[15.49318,66.28509],[15.05113,66.15572],[14.53778,66.12399],[14.50926,65.31786],[13.64276,64.58402],[14.11117,64.46674],[14.16051,64.18725],[13.98222,64.00953],[13.23411,64.09087],[12.74105,64.02171],[12.14928,63.59373],[12.19919,63.47935],[11.98529,63.27487],[12.19919,63.00104],[12.07085,62.6297],[12.29187,62.25699],[12.14746,61.7147],[12.40595,61.57226],[12.57707,61.56547],[12.86939,61.35427],[12.69115,61.06584],[12.2277,61.02442],[12.59133,60.50559],[12.52003,60.13846],[12.36317,59.99259],[12.15641,59.8926],[11.87121,59.86039],[11.92112,59.69531],[11.69297,59.59442],[11.8213,59.24985],[11.65732,58.90177],[11.45199,58.89604],[11.4601,58.99022],[11.34459,59.11672],[11.15367,59.07862],[11.08911,58.98745],[10.64958,58.89391],[10.40861,58.38489],[12.16597,56.60205],[12.07466,56.29488],[12.65312,56.04345],[12.6372,55.91371],[12.88472,55.63369],[12.60345,55.42675],[12.84405,55.13257],[14.28399,55.1553],[14.89259,55.5623],[15.79951,55.54655],[19.64795,57.06466],[19.84909,57.57876],[20.5104,59.15546],[19.08191,60.19152],[19.23413,60.61414],[20.15877,63.06556],[24.14112,65.39731],[24.15107,65.81427],[24.14798,65.83466],[24.15791,65.85385]]]]}},{type:"Feature",properties:{iso1A2:"SG",iso1A3:"SGP",iso1N3:"702",wikidata:"Q334",nameEn:"Singapore",groups:["035","142"],driveSide:"left",callingCodes:["65"]},geometry:{type:"MultiPolygon",coordinates:[[[[104.00131,1.42405],[103.93384,1.42926],[103.89565,1.42841],[103.86383,1.46288],[103.81181,1.47953],[103.76395,1.45183],[103.74161,1.4502],[103.7219,1.46108],[103.67468,1.43166],[103.62738,1.35255],[103.56591,1.19719],[103.66049,1.18825],[103.74084,1.12902],[104.03085,1.26954],[104.12282,1.27714],[104.08072,1.35998],[104.09162,1.39694],[104.08871,1.42015],[104.07348,1.43322],[104.04622,1.44691],[104.02277,1.4438],[104.00131,1.42405]]]]}},{type:"Feature",properties:{iso1A2:"SH",iso1A3:"SHN",iso1N3:"654",wikidata:"Q34497",nameEn:"Saint Helena, Ascension and Tristan da Cunha",country:"GB",groups:["011","202","002"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["290"]},geometry:{type:"MultiPolygon",coordinates:[[[[-14.82771,-8.70814],[-13.48367,-36.6746],[-11.55782,-36.60319],[-11.48092,-37.8367],[-13.41694,-37.88844],[-13.29655,-40.02846],[-9.34669,-41.00353],[-4.97086,-15.55882],[-13.33271,-8.07391],[-14.82771,-8.70814]]]]}},{type:"Feature",properties:{iso1A2:"SI",iso1A3:"SVN",iso1N3:"705",wikidata:"Q215",nameEn:"Slovenia",groups:["EU","039","150"],callingCodes:["386"]},geometry:{type:"MultiPolygon",coordinates:[[[[16.50139,46.56684],[16.39217,46.63673],[16.38594,46.6549],[16.41863,46.66238],[16.42641,46.69228],[16.37816,46.69975],[16.30966,46.7787],[16.31303,46.79838],[16.3408,46.80641],[16.34547,46.83836],[16.2941,46.87137],[16.2365,46.87775],[16.21892,46.86961],[16.15711,46.85434],[16.14365,46.8547],[16.10983,46.867],[16.05786,46.83927],[15.99054,46.82772],[15.99126,46.78199],[15.98432,46.74991],[15.99769,46.7266],[16.02808,46.71094],[16.04347,46.68694],[16.04036,46.6549],[15.99988,46.67947],[15.98512,46.68463],[15.94864,46.68769],[15.87691,46.7211],[15.8162,46.71897],[15.78518,46.70712],[15.76771,46.69863],[15.73823,46.70011],[15.72279,46.69548],[15.69523,46.69823],[15.67411,46.70735],[15.6543,46.70616],[15.6543,46.69228],[15.6365,46.6894],[15.63255,46.68069],[15.62317,46.67947],[15.59826,46.68908],[15.54533,46.66985],[15.55333,46.64988],[15.54431,46.6312],[15.46906,46.61321],[15.45514,46.63697],[15.41235,46.65556],[15.23711,46.63994],[15.14215,46.66131],[15.01451,46.641],[14.98024,46.6009],[14.96002,46.63459],[14.92283,46.60848],[14.87129,46.61],[14.86419,46.59411],[14.83549,46.56614],[14.81836,46.51046],[14.72185,46.49974],[14.66892,46.44936],[14.5942,46.43434],[14.56463,46.37208],[14.52176,46.42617],[14.45877,46.41717],[14.42608,46.44614],[14.314,46.43327],[14.28326,46.44315],[14.15989,46.43327],[14.12097,46.47724],[14.04002,46.49117],[14.00422,46.48474],[13.89837,46.52331],[13.7148,46.5222],[13.68684,46.43881],[13.59777,46.44137],[13.5763,46.42613],[13.5763,46.40915],[13.47019,46.3621],[13.43418,46.35992],[13.44808,46.33507],[13.37671,46.29668],[13.42218,46.20758],[13.47587,46.22725],[13.56114,46.2054],[13.56682,46.18703],[13.64451,46.18966],[13.66472,46.17392],[13.64053,46.13587],[13.57072,46.09022],[13.50104,46.05986],[13.49568,46.04839],[13.50998,46.04498],[13.49702,46.01832],[13.47474,46.00546],[13.50104,45.98078],[13.52963,45.96588],[13.56759,45.96991],[13.58903,45.99009],[13.62074,45.98388],[13.63458,45.98947],[13.64307,45.98326],[13.6329,45.94894],[13.63815,45.93607],[13.61931,45.91782],[13.60857,45.89907],[13.59565,45.89446],[13.58644,45.88173],[13.57563,45.8425],[13.58858,45.83503],[13.59784,45.8072],[13.66986,45.79955],[13.8235,45.7176],[13.83332,45.70855],[13.83422,45.68703],[13.87933,45.65207],[13.9191,45.6322],[13.8695,45.60835],[13.86771,45.59898],[13.84106,45.58185],[13.78445,45.5825],[13.74587,45.59811],[13.7198,45.59352],[13.6076,45.64761],[13.45644,45.59464],[13.56979,45.4895],[13.62902,45.45898],[13.67398,45.4436],[13.7785,45.46787],[13.81742,45.43729],[13.88124,45.42637],[13.90771,45.45149],[13.97309,45.45258],[13.99488,45.47551],[13.96063,45.50825],[14.00578,45.52352],[14.07116,45.48752],[14.20348,45.46896],[14.22371,45.50388],[14.24239,45.50607],[14.26611,45.48239],[14.27681,45.4902],[14.32487,45.47142],[14.36693,45.48642],[14.49769,45.54424],[14.5008,45.60852],[14.53816,45.6205],[14.57397,45.67165],[14.60977,45.66403],[14.59576,45.62812],[14.69694,45.57366],[14.68605,45.53006],[14.71718,45.53442],[14.80124,45.49515],[14.81992,45.45913],[14.90554,45.47769],[14.92266,45.52788],[15.02385,45.48533],[15.05187,45.49079],[15.16862,45.42309],[15.27758,45.46678],[15.33051,45.45258],[15.38188,45.48752],[15.30249,45.53224],[15.29837,45.5841],[15.27747,45.60504],[15.31027,45.6303],[15.34695,45.63382],[15.34214,45.64702],[15.38952,45.63682],[15.4057,45.64727],[15.34919,45.71623],[15.30872,45.69014],[15.25423,45.72275],[15.40836,45.79491],[15.47531,45.79802],[15.47325,45.8253],[15.52234,45.82195],[15.57952,45.84953],[15.64185,45.82915],[15.66662,45.84085],[15.70411,45.8465],[15.68232,45.86819],[15.68383,45.88867],[15.67967,45.90455],[15.70636,45.92116],[15.70327,46.00015],[15.71246,46.01196],[15.72977,46.04682],[15.62317,46.09103],[15.6083,46.11992],[15.59909,46.14761],[15.64904,46.19229],[15.6434,46.21396],[15.67395,46.22478],[15.75436,46.21969],[15.75479,46.20336],[15.78817,46.21719],[15.79284,46.25811],[15.97965,46.30652],[16.07616,46.3463],[16.07314,46.36458],[16.05065,46.3833],[16.05281,46.39141],[16.14859,46.40547],[16.18824,46.38282],[16.30233,46.37837],[16.30162,46.40437],[16.27329,46.41467],[16.27398,46.42875],[16.25124,46.48067],[16.23961,46.49653],[16.26759,46.50566],[16.26733,46.51505],[16.29793,46.5121],[16.37193,46.55008],[16.38771,46.53608],[16.44036,46.5171],[16.5007,46.49644],[16.52604,46.47831],[16.59527,46.47524],[16.52604,46.5051],[16.52885,46.53303],[16.50139,46.56684]]]]}},{type:"Feature",properties:{iso1A2:"SJ",iso1A3:"SJM",iso1N3:"744",wikidata:"Q842829",nameEn:"Svalbard and Jan Mayen",country:"NO",groups:["154","150"],callingCodes:["47 79"]},geometry:{type:"MultiPolygon",coordinates:[[[[-7.49892,77.24208],[32.07813,72.01005],[36.85549,84.09565],[-7.49892,77.24208]]],[[[-9.18243,72.23144],[-10.71459,70.09565],[-5.93364,70.76368],[-9.18243,72.23144]]]]}},{type:"Feature",properties:{iso1A2:"SK",iso1A3:"SVK",iso1N3:"703",wikidata:"Q214",nameEn:"Slovakia",groups:["EU","151","150"],callingCodes:["421"]},geometry:{type:"MultiPolygon",coordinates:[[[[19.82237,49.27806],[19.78581,49.41701],[19.72127,49.39288],[19.6375,49.40897],[19.64162,49.45184],[19.57845,49.46077],[19.53313,49.52856],[19.52626,49.57311],[19.45348,49.61583],[19.37795,49.574],[19.36009,49.53747],[19.25435,49.53391],[19.18019,49.41165],[18.9742,49.39557],[18.97283,49.49914],[18.94536,49.52143],[18.84521,49.51672],[18.74761,49.492],[18.67757,49.50895],[18.6144,49.49824],[18.57183,49.51162],[18.53063,49.49022],[18.54848,49.47059],[18.44686,49.39467],[18.4084,49.40003],[18.4139,49.36517],[18.36446,49.3267],[18.18456,49.28909],[18.15022,49.24518],[18.1104,49.08624],[18.06885,49.03157],[17.91814,49.01784],[17.87831,48.92679],[17.77944,48.92318],[17.73126,48.87885],[17.7094,48.86721],[17.5295,48.81117],[17.45671,48.85004],[17.3853,48.80936],[17.29054,48.85546],[17.19355,48.87602],[17.11202,48.82925],[17.00215,48.70887],[16.93955,48.60371],[16.94611,48.53614],[16.85204,48.44968],[16.8497,48.38321],[16.83588,48.3844],[16.83317,48.38138],[16.84243,48.35258],[16.90903,48.32519],[16.89461,48.31332],[16.97701,48.17385],[17.02919,48.13996],[17.05735,48.14179],[17.09168,48.09366],[17.07039,48.0317],[17.16001,48.00636],[17.23699,48.02094],[17.71215,47.7548],[18.02938,47.75665],[18.29305,47.73541],[18.56496,47.76588],[18.66521,47.76772],[18.74074,47.8157],[18.8506,47.82308],[18.76821,47.87469],[18.76134,47.97499],[18.82176,48.04206],[19.01952,48.07052],[19.23924,48.0595],[19.28182,48.08336],[19.47957,48.09437],[19.52489,48.19791],[19.63338,48.25006],[19.92452,48.1283],[20.24312,48.2784],[20.29943,48.26104],[20.5215,48.53336],[20.83248,48.5824],[21.11516,48.49546],[21.44063,48.58456],[21.6068,48.50365],[21.67134,48.3989],[21.72525,48.34628],[21.8279,48.33321],[21.83339,48.36242],[22.14689,48.4005],[22.16023,48.56548],[22.21379,48.6218],[22.34151,48.68893],[22.42934,48.92857],[22.48296,48.99172],[22.54338,49.01424],[22.56155,49.08865],[22.04427,49.22136],[21.96385,49.3437],[21.82927,49.39467],[21.77983,49.35443],[21.62328,49.4447],[21.43376,49.41433],[21.27858,49.45988],[21.19756,49.4054],[21.12477,49.43666],[21.041,49.41791],[21.09799,49.37176],[20.98733,49.30774],[20.9229,49.29626],[20.77971,49.35383],[20.72274,49.41813],[20.61666,49.41791],[20.5631,49.375],[20.46422,49.41612],[20.39939,49.3896],[20.31728,49.39914],[20.31453,49.34817],[20.21977,49.35265],[20.13738,49.31685],[20.08238,49.1813],[19.98494,49.22904],[19.90529,49.23532],[19.86409,49.19316],[19.75286,49.20751],[19.82237,49.27806]]]]}},{type:"Feature",properties:{iso1A2:"SL",iso1A3:"SLE",iso1N3:"694",wikidata:"Q1044",nameEn:"Sierra Leone",groups:["011","202","002"],callingCodes:["232"]},geometry:{type:"MultiPolygon",coordinates:[[[[-10.27575,8.48711],[-10.37257,8.48941],[-10.54891,8.31174],[-10.63934,8.35326],[-10.70565,8.29235],[-10.61422,8.5314],[-10.47707,8.67669],[-10.56197,8.81225],[-10.5783,9.06386],[-10.74484,9.07998],[-10.6534,9.29919],[-11.2118,10.00098],[-11.89624,9.99763],[-11.91023,9.93927],[-12.12634,9.87203],[-12.24262,9.92386],[-12.47254,9.86834],[-12.76788,9.3133],[-12.94095,9.26335],[-13.08953,9.0409],[-13.18586,9.0925],[-13.29911,9.04245],[-14.36218,8.64107],[-12.15048,6.15992],[-11.50429,6.92704],[-11.4027,6.97746],[-11.29417,7.21576],[-10.60422,7.7739],[-10.60492,8.04072],[-10.57523,8.04829],[-10.51554,8.1393],[-10.45023,8.15627],[-10.35227,8.15223],[-10.29839,8.21283],[-10.31635,8.28554],[-10.30084,8.30008],[-10.27575,8.48711]]]]}},{type:"Feature",properties:{iso1A2:"SM",iso1A3:"SMR",iso1N3:"674",wikidata:"Q238",nameEn:"San Marino",groups:["039","150"],callingCodes:["378"]},geometry:{type:"MultiPolygon",coordinates:[[[[12.45648,43.89369],[12.48771,43.89706],[12.49429,43.90973],[12.49247,43.91774],[12.49724,43.92248],[12.50269,43.92363],[12.50496,43.93017],[12.51553,43.94096],[12.51427,43.94897],[12.50655,43.95796],[12.50875,43.96198],[12.50622,43.97131],[12.51109,43.97201],[12.51064,43.98165],[12.5154,43.98508],[12.51463,43.99122],[12.50678,43.99113],[12.49406,43.98492],[12.47853,43.98052],[12.46205,43.97463],[12.44684,43.96597],[12.43662,43.95698],[12.42005,43.9578],[12.41414,43.95273],[12.40415,43.95485],[12.40506,43.94325],[12.41165,43.93769],[12.41551,43.92984],[12.40733,43.92379],[12.41233,43.90956],[12.40935,43.9024],[12.41641,43.89991],[12.44184,43.90498],[12.45648,43.89369]]]]}},{type:"Feature",properties:{iso1A2:"SN",iso1A3:"SEN",iso1N3:"686",wikidata:"Q1041",nameEn:"Senegal",groups:["011","202","002"],callingCodes:["221"]},geometry:{type:"MultiPolygon",coordinates:[[[[-14.32144,16.61495],[-15.00557,16.64997],[-15.6509,16.50315],[-16.27016,16.51565],[-16.4429,16.20605],[-16.44814,16.09753],[-16.48967,16.0496],[-16.50854,16.09032],[-17.15288,16.07139],[-18.35085,14.63444],[-17.43598,13.59273],[-15.47902,13.58758],[-15.36504,13.79313],[-14.93719,13.80173],[-14.34721,13.46578],[-13.8955,13.59126],[-13.79409,13.34472],[-14.36795,13.23033],[-15.14917,13.57989],[-15.26908,13.37768],[-15.80478,13.34832],[-15.80355,13.16729],[-16.69343,13.16791],[-16.74676,13.06025],[-17.43966,13.04579],[-17.4623,11.92379],[-16.70562,12.34803],[-16.38191,12.36449],[-16.20591,12.46157],[-15.67302,12.42974],[-15.17582,12.6847],[-13.70523,12.68013],[-13.05296,12.64003],[-13.06603,12.49342],[-12.87336,12.51892],[-12.35415,12.32758],[-11.91331,12.42008],[-11.46267,12.44559],[-11.37536,12.40788],[-11.39935,12.97808],[-11.63025,13.39174],[-11.83345,13.33333],[-12.06897,13.71049],[-11.93043,13.84505],[-12.23936,14.76324],[-13.11029,15.52116],[-13.43135,16.09022],[-13.80075,16.13961],[-14.32144,16.61495]]]]}},{type:"Feature",properties:{iso1A2:"SO",iso1A3:"SOM",iso1N3:"706",wikidata:"Q1045",nameEn:"Somalia",groups:["014","202","002"],callingCodes:["252"]},geometry:{type:"MultiPolygon",coordinates:[[[[48.95249,11.56816],[43.42425,11.70983],[42.95776,10.98533],[42.69452,10.62672],[42.87643,10.18441],[43.0937,9.90579],[43.23518,9.84605],[43.32613,9.59205],[44.19222,8.93028],[46.99339,7.9989],[47.92477,8.00111],[47.97917,8.00124],[44.98104,4.91821],[44.02436,4.9451],[43.40263,4.79289],[43.04177,4.57923],[42.97746,4.44032],[42.84526,4.28357],[42.55853,4.20518],[42.07619,4.17667],[41.89488,3.97375],[41.31368,3.14314],[40.98767,2.82959],[41.00099,-0.83068],[41.56,-1.59812],[41.56362,-1.66375],[41.75542,-1.85308],[49.16337,2.78611],[52.253,11.68582],[51.12877,12.56479],[48.95249,11.56816]]]]}},{type:"Feature",properties:{iso1A2:"SR",iso1A3:"SUR",iso1N3:"740",wikidata:"Q730",nameEn:"Suriname",groups:["005","419","019"],driveSide:"left",callingCodes:["597"]},geometry:{type:"MultiPolygon",coordinates:[[[[-54.26916,5.26909],[-54.01877,5.52789],[-54.01074,5.68785],[-53.7094,6.2264],[-56.84822,6.73257],[-57.31629,5.33714],[-57.22536,5.15605],[-57.37442,5.0208],[-57.8699,4.89394],[-58.0307,3.95513],[-57.35891,3.32121],[-56.70519,2.02964],[-56.55439,2.02003],[-56.47045,1.95135],[-55.99278,1.83137],[-55.89863,1.89861],[-55.92159,2.05236],[-56.13054,2.27723],[-55.96292,2.53188],[-55.71493,2.40342],[-55.01919,2.564],[-54.6084,2.32856],[-54.42864,2.42442],[-54.28534,2.67798],[-53.9849,3.58697],[-53.98914,3.627],[-54.05128,3.63557],[-54.19367,3.84387],[-54.38444,4.13222],[-54.4717,4.91964],[-54.26916,5.26909]]]]}},{type:"Feature",properties:{iso1A2:"SS",iso1A3:"SSD",iso1N3:"728",wikidata:"Q958",nameEn:"South Sudan",groups:["014","202","002"],callingCodes:["211"]},geometry:{type:"MultiPolygon",coordinates:[[[[34.10229,9.50238],[33.87958,9.49937],[33.9082,9.762],[33.96323,9.80972],[33.99185,9.99623],[33.96984,10.15446],[33.90159,10.17179],[33.80913,10.32994],[33.66604,10.44254],[33.52294,10.64382],[33.24645,10.77913],[33.26977,10.83632],[33.13988,11.43248],[33.25876,12.22111],[32.73921,12.22757],[32.73921,11.95203],[32.10079,11.95203],[32.39578,11.70208],[32.39358,11.18207],[32.46967,11.04662],[31.99177,10.65065],[31.77539,10.28939],[31.28504,9.75287],[30.84605,9.7498],[30.82893,9.71451],[30.53005,9.95992],[30.00389,10.28633],[29.94629,10.29245],[29.54,10.07949],[29.53844,9.75133],[29.06988,9.74826],[28.99983,9.67155],[27.90704,9.61323],[27.14427,9.62858],[26.70685,9.48735],[26.35815,9.57946],[26.21338,9.91545],[25.93241,10.17941],[25.93163,10.38159],[25.78141,10.42599],[25.0918,10.33718],[25.05688,10.06776],[24.97739,9.9081],[24.84653,9.80643],[24.49389,9.79962],[24.12744,9.73784],[24.09319,9.66572],[23.69155,9.67566],[23.62179,9.53823],[23.64981,9.44303],[23.64358,9.28637],[23.56263,9.19418],[23.4848,9.16959],[23.44744,8.99128],[23.59065,8.99743],[23.51905,8.71749],[24.25691,8.69288],[24.13238,8.36959],[24.35965,8.26177],[24.85156,8.16933],[24.98855,7.96588],[25.25319,7.8487],[25.29214,7.66675],[25.20649,7.61115],[25.20337,7.50312],[25.35281,7.42595],[25.37461,7.33024],[25.90076,7.09549],[26.38022,6.63493],[26.32729,6.36272],[26.58259,6.1987],[26.51721,6.09655],[27.22705,5.71254],[27.22705,5.62889],[27.28621,5.56382],[27.23017,5.37167],[27.26886,5.25876],[27.44012,5.07349],[27.56656,4.89375],[27.65462,4.89375],[27.76469,4.79284],[27.79551,4.59976],[28.20719,4.35614],[28.6651,4.42638],[28.8126,4.48784],[29.03054,4.48784],[29.22207,4.34297],[29.43341,4.50101],[29.49726,4.7007],[29.82087,4.56246],[29.79666,4.37809],[30.06964,4.13221],[30.1621,4.10586],[30.22374,3.93896],[30.27658,3.95653],[30.47691,3.83353],[30.55396,3.84451],[30.57378,3.74567],[30.56277,3.62703],[30.78512,3.67097],[30.80713,3.60506],[30.85997,3.5743],[30.85153,3.48867],[30.97601,3.693],[31.16666,3.79853],[31.29476,3.8015],[31.50478,3.67814],[31.50776,3.63652],[31.72075,3.74354],[31.81459,3.82083],[31.86821,3.78664],[31.96205,3.6499],[31.95907,3.57408],[32.05187,3.589],[32.08491,3.56287],[32.08866,3.53543],[32.19888,3.50867],[32.20782,3.6053],[32.41337,3.748],[32.72021,3.77327],[32.89746,3.81339],[33.02852,3.89296],[33.18356,3.77812],[33.51264,3.75068],[33.9873,4.23316],[34.47601,4.72162],[35.34151,5.02364],[35.30992,4.90402],[35.47843,4.91872],[35.42366,4.76969],[35.51424,4.61643],[35.9419,4.61933],[35.82118,4.77382],[35.81968,5.10757],[35.8576,5.33413],[35.50792,5.42431],[35.29938,5.34042],[35.31188,5.50106],[35.13058,5.62118],[35.12611,5.68937],[35.00546,5.89387],[34.96227,6.26415],[35.01738,6.46991],[34.87736,6.60161],[34.77459,6.5957],[34.65096,6.72589],[34.53776,6.74808],[34.53925,6.82794],[34.47669,6.91076],[34.35753,6.91963],[34.19369,7.04382],[34.19369,7.12807],[34.01495,7.25664],[34.03878,7.27437],[34.02984,7.36449],[33.87642,7.5491],[33.71407,7.65983],[33.44745,7.7543],[33.32531,7.71297],[33.24637,7.77939],[33.04944,7.78989],[33.0006,7.90333],[33.08401,8.05822],[33.18083,8.13047],[33.1853,8.29264],[33.19721,8.40317],[33.3119,8.45474],[33.54575,8.47094],[33.66938,8.44442],[33.71407,8.3678],[33.87195,8.41938],[33.89579,8.4842],[34.01346,8.50041],[34.14453,8.60204],[34.14304,9.04654],[34.10229,9.50238]]]]}},{type:"Feature",properties:{iso1A2:"ST",iso1A3:"STP",iso1N3:"678",wikidata:"Q1039",nameEn:"São Tomé and Principe",groups:["017","202","002"],callingCodes:["239"]},geometry:{type:"MultiPolygon",coordinates:[[[[5.9107,-0.09539],[6.69416,-0.53945],[8.0168,1.79377],[7.23334,2.23756],[5.9107,-0.09539]]]]}},{type:"Feature",properties:{iso1A2:"SV",iso1A3:"SLV",iso1N3:"222",wikidata:"Q792",nameEn:"El Salvador",groups:["013","003","419","019"],callingCodes:["503"]},geometry:{type:"MultiPolygon",coordinates:[[[[-89.34776,14.43013],[-89.39028,14.44561],[-89.57441,14.41637],[-89.58814,14.33165],[-89.50614,14.26084],[-89.52397,14.22628],[-89.61844,14.21937],[-89.70756,14.1537],[-89.75569,14.07073],[-89.73251,14.04133],[-89.76103,14.02923],[-89.81807,14.07073],[-89.88937,14.0396],[-90.10505,13.85104],[-90.11344,13.73679],[-90.55276,12.8866],[-88.11443,12.63306],[-87.7346,13.13228],[-87.55124,13.12523],[-87.69751,13.25228],[-87.73714,13.32715],[-87.80177,13.35689],[-87.84675,13.41078],[-87.83467,13.44655],[-87.77354,13.45767],[-87.73841,13.44169],[-87.72115,13.46083],[-87.71657,13.50577],[-87.78148,13.52906],[-87.73106,13.75443],[-87.68821,13.80829],[-87.7966,13.91353],[-88.00331,13.86948],[-88.07641,13.98447],[-88.23018,13.99915],[-88.25791,13.91108],[-88.48982,13.86458],[-88.49738,13.97224],[-88.70661,14.04317],[-88.73182,14.10919],[-88.815,14.11652],[-88.85785,14.17763],[-88.94608,14.20207],[-89.04187,14.33644],[-89.34776,14.43013]]]]}},{type:"Feature",properties:{iso1A2:"SX",iso1A3:"SXM",iso1N3:"534",wikidata:"Q26273",nameEn:"Sint Maarten",country:"NL",groups:["029","003","419","019"],callingCodes:["1 721"]},geometry:{type:"MultiPolygon",coordinates:[[[[-63.29212,17.90532],[-63.07669,17.79659],[-62.93924,18.02904],[-63.02323,18.05757],[-63.04039,18.05619],[-63.0579,18.06614],[-63.07759,18.04943],[-63.09686,18.04608],[-63.11096,18.05368],[-63.13584,18.0541],[-63.33064,17.9615],[-63.29212,17.90532]]]]}},{type:"Feature",properties:{iso1A2:"SY",iso1A3:"SYR",iso1N3:"760",wikidata:"Q858",nameEn:"Syria",groups:["145","142"],callingCodes:["963"]},geometry:{type:"MultiPolygon",coordinates:[[[[42.23683,37.2863],[42.21548,37.28026],[42.20454,37.28715],[42.22381,37.30238],[42.22257,37.31395],[42.2112,37.32491],[42.19301,37.31323],[42.18225,37.28569],[42.00894,37.17209],[41.515,37.08084],[41.21937,37.07665],[40.90856,37.13147],[40.69136,37.0996],[39.81589,36.75538],[39.21538,36.66834],[39.03217,36.70911],[38.74042,36.70629],[38.55908,36.84429],[38.38859,36.90064],[38.21064,36.91842],[37.81974,36.76055],[37.68048,36.75065],[37.49103,36.66904],[37.47253,36.63243],[37.21988,36.6736],[37.16177,36.66069],[37.10894,36.6704],[37.08279,36.63495],[37.02088,36.66422],[37.01647,36.69512],[37.04619,36.71101],[37.04399,36.73483],[36.99886,36.74012],[36.99557,36.75997],[36.66727,36.82901],[36.61581,36.74629],[36.62681,36.71189],[36.57398,36.65186],[36.58829,36.58295],[36.54206,36.49539],[36.6081,36.33772],[36.65653,36.33861],[36.68672,36.23677],[36.6125,36.22592],[36.50463,36.2419],[36.4617,36.20461],[36.39206,36.22088],[36.37474,36.01163],[36.33956,35.98687],[36.30099,36.00985],[36.28338,36.00273],[36.29769,35.96086],[36.27678,35.94839],[36.25366,35.96264],[36.19973,35.95195],[36.17441,35.92076],[36.1623,35.80925],[36.14029,35.81015],[36.13919,35.83692],[36.11827,35.85923],[35.99829,35.88242],[36.01844,35.92403],[36.00514,35.94113],[35.98499,35.94107],[35.931,35.92109],[35.51152,36.10954],[35.48515,34.70851],[35.97386,34.63322],[35.98718,34.64977],[36.29165,34.62991],[36.32399,34.69334],[36.35135,34.68516],[36.35384,34.65447],[36.42941,34.62505],[36.46003,34.6378],[36.45299,34.59438],[36.41429,34.61175],[36.39846,34.55672],[36.3369,34.52629],[36.34745,34.5002],[36.4442,34.50165],[36.46179,34.46541],[36.55853,34.41609],[36.53039,34.3798],[36.56556,34.31881],[36.60778,34.31009],[36.58667,34.27667],[36.59195,34.2316],[36.62537,34.20251],[36.5128,34.09916],[36.50576,34.05982],[36.41078,34.05253],[36.28589,33.91981],[36.38263,33.86579],[36.3967,33.83365],[36.14517,33.85118],[36.06778,33.82927],[35.9341,33.6596],[36.05723,33.57904],[35.94465,33.52774],[35.94816,33.47886],[35.88668,33.43183],[35.82577,33.40479],[35.81324,33.36354],[35.77477,33.33609],[35.813,33.3172],[35.77513,33.27342],[35.81295,33.24841],[35.81647,33.2028],[35.83846,33.19397],[35.84285,33.16673],[35.81911,33.1336],[35.81911,33.11077],[35.84802,33.1031],[35.87188,32.98028],[35.89298,32.9456],[35.87012,32.91976],[35.84021,32.8725],[35.83758,32.82817],[35.78745,32.77938],[35.75983,32.74803],[35.88405,32.71321],[35.93307,32.71966],[35.96633,32.66237],[36.02239,32.65911],[36.08074,32.51463],[36.20379,32.52751],[36.20875,32.49529],[36.23948,32.50108],[36.40959,32.37908],[36.83946,32.31293],[38.79171,33.37328],[40.64314,34.31604],[40.97676,34.39788],[41.12388,34.65742],[41.2345,34.80049],[41.21654,35.1508],[41.26569,35.42708],[41.38184,35.62502],[41.37027,35.84095],[41.2564,36.06012],[41.28864,36.35368],[41.40058,36.52502],[41.81736,36.58782],[42.36697,37.0627],[42.35724,37.10998],[42.32313,37.17814],[42.34735,37.22548],[42.2824,37.2798],[42.26039,37.27017],[42.23683,37.2863]]]]}},{type:"Feature",properties:{iso1A2:"SZ",iso1A3:"SWZ",iso1N3:"748",wikidata:"Q1050",nameEn:"Eswatini",aliases:["Swaziland"],groups:["018","202","002"],driveSide:"left",callingCodes:["268"]},geometry:{type:"MultiPolygon",coordinates:[[[[31.86881,-25.99973],[31.4175,-25.71886],[31.31237,-25.7431],[31.13073,-25.91558],[30.95819,-26.26303],[30.78927,-26.48271],[30.81101,-26.84722],[30.88826,-26.79622],[30.97757,-26.92706],[30.96088,-27.0245],[31.15027,-27.20151],[31.49834,-27.31549],[31.97592,-27.31675],[31.97463,-27.11057],[32.00893,-26.8096],[32.09664,-26.80721],[32.13315,-26.84345],[32.13409,-26.5317],[32.07352,-26.40185],[32.10435,-26.15656],[32.08599,-26.00978],[32.00916,-25.999],[31.974,-25.95387],[31.86881,-25.99973]]]]}},{type:"Feature",properties:{iso1A2:"TA",iso1A3:"TAA",wikidata:"Q220982",nameEn:"Tristan da Cunha",country:"GB",groups:["SH","011","202","002"],isoStatus:"excRes",driveSide:"left",roadSpeedUnit:"mph",callingCodes:["290 8","44 20"]},geometry:{type:"MultiPolygon",coordinates:[[[[-13.48367,-36.6746],[-13.41694,-37.88844],[-11.48092,-37.8367],[-11.55782,-36.60319],[-13.48367,-36.6746]]]]}},{type:"Feature",properties:{iso1A2:"TC",iso1A3:"TCA",iso1N3:"796",wikidata:"Q18221",nameEn:"Turks and Caicos Islands",country:"GB",groups:["029","003","419","019"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["1 649"]},geometry:{type:"MultiPolygon",coordinates:[[[[-72.41726,22.40371],[-72.72017,21.48055],[-71.46138,20.64433],[-70.63262,21.53631],[-72.41726,22.40371]]]]}},{type:"Feature",properties:{iso1A2:"TD",iso1A3:"TCD",iso1N3:"148",wikidata:"Q657",nameEn:"Chad",groups:["017","202","002"],callingCodes:["235"]},geometry:{type:"MultiPolygon",coordinates:[[[[23.99539,19.49944],[15.99566,23.49639],[14.99751,23.00539],[15.19692,21.99339],[15.20213,21.49365],[15.28332,21.44557],[15.62515,20.95395],[15.57248,20.92138],[15.55382,20.86507],[15.56004,20.79488],[15.59841,20.74039],[15.6721,20.70069],[15.99632,20.35364],[15.75098,19.93002],[15.6032,18.77402],[15.50373,16.89649],[14.37425,15.72591],[13.86301,15.04043],[13.78991,14.87519],[13.809,14.72915],[13.67878,14.64013],[13.68573,14.55276],[13.48259,14.46704],[13.47559,14.40881],[13.6302,13.71094],[14.08251,13.0797],[14.46881,13.08259],[14.56101,12.91036],[14.55058,12.78256],[14.83314,12.62963],[14.90827,12.3269],[14.89019,12.16593],[14.96952,12.0925],[15.00146,12.1223],[15.0349,12.10698],[15.05786,12.0608],[15.04808,11.8731],[15.11579,11.79313],[15.06595,11.71126],[15.13149,11.5537],[15.0585,11.40481],[15.10021,11.04101],[15.04957,11.02347],[15.09127,10.87431],[15.06737,10.80921],[15.15532,10.62846],[15.14936,10.53915],[15.23724,10.47764],[15.30874,10.31063],[15.50535,10.1098],[15.68761,9.99344],[15.41408,9.92876],[15.24618,9.99246],[15.14043,9.99246],[15.05999,9.94845],[14.95722,9.97926],[14.80082,9.93818],[14.4673,10.00264],[14.20411,10.00055],[14.1317,9.82413],[14.01793,9.73169],[13.97544,9.6365],[14.37094,9.2954],[14.35707,9.19611],[14.83566,8.80557],[15.09484,8.65982],[15.20426,8.50892],[15.50743,7.79302],[15.59272,7.7696],[15.56964,7.58936],[15.49743,7.52179],[15.73118,7.52006],[15.79942,7.44149],[16.40703,7.68809],[16.41583,7.77971],[16.58315,7.88657],[16.59415,7.76444],[16.658,7.75353],[16.6668,7.67281],[16.8143,7.53971],[17.67288,7.98905],[17.93926,7.95853],[18.02731,8.01085],[18.6085,8.05009],[18.64153,8.08714],[18.62612,8.14163],[18.67455,8.22226],[18.79783,8.25929],[19.11044,8.68172],[18.86388,8.87971],[19.06421,9.00367],[20.36748,9.11019],[20.82979,9.44696],[21.26348,9.97642],[21.34934,9.95907],[21.52766,10.2105],[21.63553,10.217],[21.71479,10.29932],[21.72139,10.64136],[22.45889,11.00246],[22.87758,10.91915],[22.97249,11.21955],[22.93124,11.41645],[22.7997,11.40424],[22.54907,11.64372],[22.64092,12.07485],[22.48369,12.02766],[22.50548,12.16769],[22.38873,12.45514],[22.46345,12.61925],[22.22684,12.74682],[22.15679,12.66634],[21.98711,12.63292],[21.89371,12.68001],[21.81432,12.81362],[21.94819,13.05637],[22.02914,13.13976],[22.1599,13.19281],[22.29689,13.3731],[22.08674,13.77863],[22.22995,13.96754],[22.5553,14.11704],[22.55997,14.23024],[22.44944,14.24986],[22.38562,14.58907],[22.70474,14.69149],[22.66115,14.86308],[22.99584,15.22989],[22.99584,15.40105],[22.92579,15.47007],[22.93201,15.55107],[23.10792,15.71297],[23.38812,15.69649],[23.62785,15.7804],[23.99997,15.69575],[23.99539,19.49944]]]]}},{type:"Feature",properties:{iso1A2:"TF",iso1A3:"ATF",iso1N3:"260",wikidata:"Q129003",nameEn:"French Southern and Antarctic Lands",country:"FR",groups:["014","202","002"]},geometry:{type:"MultiPolygon",coordinates:[[[[53.53458,-16.36909],[54.96649,-16.28353],[54.61476,-15.02273],[53.53458,-16.36909]]],[[[39.10324,-21.48967],[40.40841,-23.17181],[43.72277,-16.09877],[41.06663,-17.08802],[39.10324,-21.48967]]],[[[46.52682,-10.83678],[47.29063,-12.45583],[48.86266,-10.8109],[46.52682,-10.83678]]],[[[80.15867,-36.04977],[46.31615,-46.28749],[70.67507,-51.14192],[80.15867,-36.04977]]]]}},{type:"Feature",properties:{iso1A2:"TG",iso1A3:"TGO",iso1N3:"768",wikidata:"Q945",nameEn:"Togo",groups:["011","202","002"],callingCodes:["228"]},geometry:{type:"MultiPolygon",coordinates:[[[[0.50388,11.01011],[-0.13493,11.14075],[-0.14462,11.10811],[-0.05733,11.08628],[-0.0275,11.11202],[-0.00514,11.10763],[0.00342,11.08317],[0.02395,11.06229],[0.03355,10.9807],[-0.0063,10.96417],[-0.00908,10.91644],[-0.02685,10.8783],[-0.0228,10.81916],[-0.07183,10.76794],[-0.07327,10.71845],[-0.09141,10.7147],[-0.05945,10.63458],[0.12886,10.53149],[0.18846,10.4096],[0.29453,10.41546],[0.33028,10.30408],[0.39584,10.31112],[0.35293,10.09412],[0.41371,10.06361],[0.41252,10.02018],[0.36366,10.03309],[0.32075,9.72781],[0.34816,9.71607],[0.34816,9.66907],[0.32313,9.6491],[0.28261,9.69022],[0.26712,9.66437],[0.29334,9.59387],[0.36008,9.6256],[0.38153,9.58682],[0.23851,9.57389],[0.2409,9.52335],[0.30406,9.521],[0.31241,9.50337],[0.2254,9.47869],[0.25758,9.42696],[0.33148,9.44812],[0.36485,9.49749],[0.49118,9.48339],[0.56388,9.40697],[0.45424,9.04581],[0.52455,8.87746],[0.37319,8.75262],[0.47211,8.59945],[0.64731,8.48866],[0.73432,8.29529],[0.63897,8.25873],[0.5913,8.19622],[0.61156,8.18324],[0.6056,8.13959],[0.58891,8.12779],[0.62943,7.85751],[0.58295,7.62368],[0.51979,7.58706],[0.52455,7.45354],[0.57223,7.39326],[0.62943,7.41099],[0.65327,7.31643],[0.59606,7.01252],[0.52217,6.9723],[0.52098,6.94391],[0.56508,6.92971],[0.52853,6.82921],[0.57406,6.80348],[0.58176,6.76049],[0.6497,6.73682],[0.63659,6.63857],[0.74862,6.56517],[0.71048,6.53083],[0.89283,6.33779],[0.99652,6.33779],[1.03108,6.24064],[1.05969,6.22998],[1.09187,6.17074],[1.19966,6.17069],[1.19771,6.11522],[1.27574,5.93551],[1.67336,6.02702],[1.62913,6.24075],[1.79826,6.28221],[1.76906,6.43189],[1.58105,6.68619],[1.61812,6.74843],[1.55877,6.99737],[1.64249,6.99562],[1.61838,9.0527],[1.5649,9.16941],[1.41746,9.3226],[1.33675,9.54765],[1.36624,9.5951],[1.35507,9.99525],[0.77666,10.37665],[0.80358,10.71459],[0.8804,10.803],[0.91245,10.99597],[0.66104,10.99964],[0.4958,10.93269],[0.50521,10.98035],[0.48852,10.98561],[0.50388,11.01011]]]]}},{type:"Feature",properties:{iso1A2:"TH",iso1A3:"THA",iso1N3:"764",wikidata:"Q869",nameEn:"Thailand",groups:["035","142"],driveSide:"left",callingCodes:["66"]},geometry:{type:"MultiPolygon",coordinates:[[[[100.08404,20.36626],[99.95721,20.46301],[99.91616,20.44986],[99.90499,20.4487],[99.89692,20.44789],[99.89301,20.44311],[99.89168,20.44548],[99.88451,20.44596],[99.88211,20.44488],[99.86383,20.44371],[99.81096,20.33687],[99.68255,20.32077],[99.46008,20.39673],[99.46077,20.36198],[99.5569,20.20676],[99.52943,20.14811],[99.416,20.08614],[99.20328,20.12877],[99.0735,20.10298],[98.98679,19.7419],[98.83661,19.80931],[98.56065,19.67807],[98.51182,19.71303],[98.24884,19.67876],[98.13829,19.78541],[98.03314,19.80941],[98.04364,19.65755],[97.84715,19.55782],[97.88423,19.5041],[97.78769,19.39429],[97.84186,19.29526],[97.78606,19.26769],[97.84024,19.22217],[97.83479,19.09972],[97.73797,19.04261],[97.73654,18.9812],[97.66487,18.9371],[97.73836,18.88478],[97.76752,18.58097],[97.5258,18.4939],[97.36444,18.57138],[97.34522,18.54596],[97.50383,18.26844],[97.56219,18.33885],[97.64116,18.29778],[97.60841,18.23846],[97.73723,17.97912],[97.66794,17.88005],[97.76407,17.71595],[97.91829,17.54504],[98.11185,17.36829],[98.10439,17.33847],[98.34566,17.04822],[98.39441,17.06266],[98.52624,16.89979],[98.49603,16.8446],[98.53833,16.81934],[98.46994,16.73613],[98.50253,16.7139],[98.49713,16.69022],[98.51043,16.70107],[98.51579,16.69433],[98.51472,16.68521],[98.51833,16.676],[98.51113,16.64503],[98.5695,16.62826],[98.57912,16.55983],[98.63817,16.47424],[98.68074,16.27068],[98.84485,16.42354],[98.92656,16.36425],[98.8376,16.11706],[98.69585,16.13353],[98.57019,16.04578],[98.59853,15.87197],[98.541,15.65406],[98.58598,15.46821],[98.56027,15.33471],[98.4866,15.39154],[98.39351,15.34177],[98.41906,15.27103],[98.40522,15.25268],[98.30446,15.30667],[98.22,15.21327],[98.18821,15.13125],[98.24874,14.83013],[98.56762,14.37701],[98.97356,14.04868],[99.16695,13.72621],[99.20617,13.20575],[99.12225,13.19847],[99.10646,13.05804],[99.18748,12.9898],[99.18905,12.84799],[99.29254,12.68921],[99.409,12.60603],[99.47519,12.1353],[99.56445,12.14805],[99.53424,12.02317],[99.64891,11.82699],[99.64108,11.78948],[99.5672,11.62732],[99.47598,11.62434],[99.39485,11.3925],[99.31573,11.32081],[99.32756,11.28545],[99.06938,10.94857],[99.02337,10.97217],[98.99701,10.92962],[99.0069,10.85485],[98.86819,10.78336],[98.78511,10.68351],[98.77275,10.62548],[98.81944,10.52761],[98.7391,10.31488],[98.55174,9.92804],[98.52291,9.92389],[98.47298,9.95782],[98.33094,9.91973],[98.12555,9.44056],[97.63455,9.60854],[97.19814,8.18901],[99.31854,5.99868],[99.50117,6.44501],[99.91873,6.50233],[100.0756,6.4045],[100.12,6.42105],[100.19511,6.72559],[100.29651,6.68439],[100.30828,6.66462],[100.31618,6.66781],[100.31884,6.66423],[100.32671,6.66526],[100.32607,6.65933],[100.31929,6.65413],[100.35413,6.54932],[100.41152,6.52299],[100.41791,6.5189],[100.42351,6.51762],[100.43027,6.52389],[100.66986,6.45086],[100.74361,6.50811],[100.74822,6.46231],[100.81045,6.45086],[100.85884,6.24929],[101.10313,6.25617],[101.12618,6.19431],[101.06165,6.14161],[101.12388,6.11411],[101.087,5.9193],[101.02708,5.91013],[100.98815,5.79464],[101.14062,5.61613],[101.25755,5.71065],[101.25524,5.78633],[101.58019,5.93534],[101.69773,5.75881],[101.75074,5.79091],[101.80144,5.74505],[101.89188,5.8386],[101.91776,5.84269],[101.92819,5.85511],[101.94712,5.98421],[101.9714,6.00575],[101.97114,6.01992],[101.99209,6.04075],[102.01835,6.05407],[102.09182,6.14161],[102.07732,6.193],[102.08127,6.22679],[102.09086,6.23546],[102.46318,7.22462],[102.47649,9.66162],[102.52395,11.25257],[102.91449,11.65512],[102.90973,11.75613],[102.83957,11.8519],[102.78427,11.98746],[102.77026,12.06815],[102.70176,12.1686],[102.73134,12.37091],[102.78116,12.40284],[102.7796,12.43781],[102.57567,12.65358],[102.51963,12.66117],[102.4994,12.71736],[102.53053,12.77506],[102.49335,12.92711],[102.48694,12.97537],[102.52275,12.99813],[102.46011,13.08057],[102.43422,13.09061],[102.36146,13.26006],[102.36001,13.31142],[102.34611,13.35618],[102.35692,13.38274],[102.35563,13.47307],[102.361,13.50551],[102.33828,13.55613],[102.36859,13.57488],[102.44601,13.5637],[102.5358,13.56933],[102.57573,13.60461],[102.62483,13.60883],[102.58635,13.6286],[102.5481,13.6589],[102.56848,13.69366],[102.72727,13.77806],[102.77864,13.93374],[102.91251,14.01531],[102.93275,14.19044],[103.16469,14.33075],[103.39353,14.35639],[103.53518,14.42575],[103.71109,14.4348],[103.70175,14.38052],[103.93836,14.3398],[104.27616,14.39861],[104.55014,14.36091],[104.69335,14.42726],[104.97667,14.38806],[105.02804,14.23722],[105.08408,14.20402],[105.14012,14.23873],[105.17748,14.34432],[105.20894,14.34967],[105.43783,14.43865],[105.53864,14.55731],[105.5121,14.80802],[105.61162,15.00037],[105.46661,15.13132],[105.58043,15.32724],[105.50662,15.32054],[105.4692,15.33709],[105.47635,15.3796],[105.58191,15.41031],[105.60446,15.53301],[105.61756,15.68792],[105.46573,15.74742],[105.42285,15.76971],[105.37959,15.84074],[105.34115,15.92737],[105.38508,15.987],[105.42001,16.00657],[105.06204,16.09792],[105.00262,16.25627],[104.88057,16.37311],[104.73349,16.565],[104.76099,16.69302],[104.7397,16.81005],[104.76442,16.84752],[104.7373,16.91125],[104.73712,17.01404],[104.80716,17.19025],[104.80061,17.39367],[104.69867,17.53038],[104.45404,17.66788],[104.35432,17.82871],[104.2757,17.86139],[104.21776,17.99335],[104.10927,18.10826],[104.06533,18.21656],[103.97725,18.33631],[103.93916,18.33914],[103.85642,18.28666],[103.82449,18.33979],[103.699,18.34125],[103.60957,18.40528],[103.47773,18.42841],[103.41044,18.4486],[103.30977,18.4341],[103.24779,18.37807],[103.23818,18.34875],[103.29757,18.30475],[103.17093,18.2618],[103.14994,18.23172],[103.1493,18.17799],[103.07343,18.12351],[103.07823,18.03833],[103.0566,18.00144],[103.01998,17.97095],[102.9912,17.9949],[102.95812,18.0054],[102.86323,17.97531],[102.81988,17.94233],[102.79044,17.93612],[102.75954,17.89561],[102.68538,17.86653],[102.67543,17.84529],[102.69946,17.81686],[102.68194,17.80151],[102.59485,17.83537],[102.5896,17.84889],[102.61432,17.92273],[102.60971,17.95411],[102.59234,17.96127],[102.45523,17.97106],[102.11359,18.21532],[101.88485,18.02474],[101.78087,18.07559],[101.72294,17.92867],[101.44667,17.7392],[101.15108,17.47586],[100.96541,17.57926],[101.02185,17.87637],[101.1793,18.0544],[101.19118,18.2125],[101.15108,18.25624],[101.18227,18.34367],[101.06047,18.43247],[101.27585,18.68875],[101.22832,18.73377],[101.25803,18.89545],[101.35606,19.04716],[101.261,19.12717],[101.24911,19.33334],[101.20604,19.35296],[101.21347,19.46223],[101.26991,19.48324],[101.26545,19.59242],[101.08928,19.59748],[100.90302,19.61901],[100.77231,19.48324],[100.64606,19.55884],[100.58219,19.49164],[100.49604,19.53504],[100.398,19.75047],[100.5094,19.87904],[100.58808,20.15791],[100.55218,20.17741],[100.51052,20.14928],[100.47567,20.19133],[100.4537,20.19971],[100.44992,20.23644],[100.41473,20.25625],[100.37439,20.35156],[100.33383,20.4028],[100.25769,20.3992],[100.22076,20.31598],[100.16668,20.2986],[100.1712,20.24324],[100.11785,20.24787],[100.09337,20.26293],[100.09999,20.31614],[100.08404,20.36626]]]]}},{type:"Feature",properties:{iso1A2:"TJ",iso1A3:"TJK",iso1N3:"762",wikidata:"Q863",nameEn:"Tajikistan",groups:["143","142"],callingCodes:["992"]},geometry:{type:"MultiPolygon",coordinates:[[[[70.45251,41.04438],[70.38028,41.02014],[70.36655,40.90296],[69.69434,40.62615],[69.59441,40.70181],[69.53021,40.77621],[69.38327,40.7918],[69.32834,40.70233],[69.3455,40.57988],[69.2643,40.57506],[69.21063,40.54469],[69.27066,40.49274],[69.28525,40.41894],[69.30774,40.36102],[69.33794,40.34819],[69.32833,40.29794],[69.30808,40.2821],[69.24817,40.30357],[69.25229,40.26362],[69.30104,40.24502],[69.30448,40.18774],[69.2074,40.21488],[69.15659,40.2162],[69.04544,40.22904],[68.85832,40.20885],[68.84357,40.18604],[68.79276,40.17555],[68.77902,40.20492],[68.5332,40.14826],[68.52771,40.11676],[68.62796,40.07789],[69.01523,40.15771],[69.01935,40.11466],[68.96579,40.06949],[68.84906,40.04952],[68.93695,39.91167],[68.88889,39.87163],[68.63071,39.85265],[68.61972,39.68905],[68.54166,39.53929],[68.12053,39.56317],[67.70992,39.66156],[67.62889,39.60234],[67.44899,39.57799],[67.46547,39.53564],[67.39681,39.52505],[67.46822,39.46146],[67.45998,39.315],[67.36522,39.31287],[67.33226,39.23739],[67.67833,39.14479],[67.68915,39.00775],[68.09704,39.02589],[68.19743,38.85985],[68.06948,38.82115],[68.12877,38.73677],[68.05598,38.71641],[68.0807,38.64136],[68.05873,38.56087],[68.11366,38.47169],[68.06274,38.39435],[68.13289,38.40822],[68.40343,38.19484],[68.27159,37.91477],[68.12635,37.93],[67.81566,37.43107],[67.8474,37.31594],[67.78329,37.1834],[67.7803,37.08978],[67.87917,37.0591],[68.02194,36.91923],[68.18542,37.02074],[68.27605,37.00977],[68.29253,37.10621],[68.41201,37.10402],[68.41888,37.13906],[68.61851,37.19815],[68.6798,37.27906],[68.81438,37.23862],[68.80889,37.32494],[68.91189,37.26704],[68.88168,37.33368],[68.96407,37.32603],[69.03274,37.25174],[69.25152,37.09426],[69.39529,37.16752],[69.45022,37.23315],[69.36645,37.40462],[69.44954,37.4869],[69.51888,37.5844],[69.80041,37.5746],[69.84435,37.60616],[69.93362,37.61378],[69.95971,37.5659],[70.15015,37.52519],[70.28243,37.66706],[70.27694,37.81258],[70.1863,37.84296],[70.17206,37.93276],[70.4898,38.12546],[70.54673,38.24541],[70.60407,38.28046],[70.61526,38.34774],[70.64966,38.34999],[70.69189,38.37031],[70.6761,38.39144],[70.67438,38.40597],[70.69807,38.41861],[70.72485,38.4131],[70.75455,38.4252],[70.77132,38.45548],[70.78581,38.45502],[70.78702,38.45031],[70.79766,38.44944],[70.80521,38.44447],[70.81697,38.44507],[70.82538,38.45394],[70.84376,38.44688],[70.88719,38.46826],[70.92728,38.43021],[70.98693,38.48862],[71.03545,38.44779],[71.0556,38.40176],[71.09542,38.42517],[71.10592,38.42077],[71.10957,38.40671],[71.1451,38.40106],[71.21291,38.32797],[71.33114,38.30339],[71.33869,38.27335],[71.37803,38.25641],[71.36444,38.15358],[71.29878,38.04429],[71.28922,38.01272],[71.27622,37.99946],[71.27278,37.96496],[71.24969,37.93031],[71.2809,37.91995],[71.296,37.93403],[71.32871,37.88564],[71.51565,37.95349],[71.58843,37.92425],[71.59255,37.79956],[71.55752,37.78677],[71.54324,37.77104],[71.53053,37.76534],[71.55234,37.73209],[71.54186,37.69691],[71.51972,37.61945],[71.5065,37.60912],[71.49693,37.53527],[71.50616,37.50733],[71.5256,37.47971],[71.49612,37.4279],[71.47685,37.40281],[71.4862,37.33405],[71.49821,37.31975],[71.50674,37.31502],[71.48536,37.26017],[71.4824,37.24921],[71.48339,37.23937],[71.47386,37.2269],[71.4555,37.21418],[71.4494,37.18137],[71.44127,37.11856],[71.43097,37.05855],[71.45578,37.03094],[71.46923,36.99925],[71.48481,36.93218],[71.51502,36.89128],[71.57195,36.74943],[71.67083,36.67346],[71.83229,36.68084],[72.31676,36.98115],[72.54095,37.00007],[72.66381,37.02014],[72.79693,37.22222],[73.06884,37.31729],[73.29633,37.46495],[73.77197,37.4417],[73.76647,37.33913],[73.61129,37.27469],[73.64974,37.23643],[73.82552,37.22659],[73.8564,37.26158],[74.20308,37.34208],[74.23339,37.41116],[74.41055,37.3948],[74.56161,37.37734],[74.68383,37.3948],[74.8294,37.3435],[74.88887,37.23275],[75.12328,37.31839],[75.09719,37.37297],[75.15899,37.41443],[75.06011,37.52779],[74.94338,37.55501],[74.8912,37.67576],[75.00935,37.77486],[74.92416,37.83428],[74.9063,38.03033],[74.82665,38.07359],[74.80331,38.19889],[74.69894,38.22155],[74.69619,38.42947],[74.51217,38.47034],[74.17022,38.65504],[73.97933,38.52945],[73.79806,38.61106],[73.80656,38.66449],[73.7033,38.84782],[73.7445,38.93867],[73.82964,38.91517],[73.81728,39.04007],[73.75823,39.023],[73.60638,39.24534],[73.54572,39.27567],[73.55396,39.3543],[73.5004,39.38402],[73.59241,39.40843],[73.59831,39.46425],[73.45096,39.46677],[73.31912,39.38615],[73.18454,39.35536],[72.85934,39.35116],[72.62027,39.39696],[72.33173,39.33093],[72.23834,39.17248],[72.17242,39.2661],[72.09689,39.26823],[72.04059,39.36704],[71.90601,39.27674],[71.79202,39.27355],[71.7522,39.32031],[71.80164,39.40631],[71.76816,39.45456],[71.62688,39.44056],[71.5517,39.45722],[71.55856,39.57588],[71.49814,39.61397],[71.08752,39.50704],[71.06418,39.41586],[70.7854,39.38933],[70.64087,39.58792],[70.44757,39.60128],[70.2869,39.53141],[70.11111,39.58223],[69.87491,39.53882],[69.68677,39.59281],[69.3594,39.52516],[69.26938,39.8127],[69.35649,40.01994],[69.43134,39.98431],[69.43557,39.92877],[69.53615,39.93991],[69.5057,40.03277],[69.53855,40.0887],[69.53794,40.11833],[69.55555,40.12296],[69.57615,40.10524],[69.64704,40.12165],[69.67001,40.10639],[70.01283,40.23288],[70.58297,40.00891],[70.57384,39.99394],[70.47557,39.93216],[70.55033,39.96619],[70.58912,39.95211],[70.65946,39.9878],[70.65827,40.0981],[70.7928,40.12797],[70.80495,40.16813],[70.9818,40.22392],[70.8607,40.217],[70.62342,40.17396],[70.56394,40.26421],[70.57149,40.3442],[70.37511,40.38605],[70.32626,40.45174],[70.49871,40.52503],[70.80009,40.72825],[70.45251,41.04438]]],[[[70.68112,40.90612],[70.6158,40.97661],[70.56077,41.00642],[70.54223,40.98787],[70.57501,40.98941],[70.6721,40.90555],[70.68112,40.90612]]],[[[70.74189,39.86319],[70.53651,39.89155],[70.52631,39.86989],[70.54998,39.85137],[70.59667,39.83542],[70.63105,39.77923],[70.74189,39.86319]]]]}},{type:"Feature",properties:{iso1A2:"TK",iso1A3:"TKL",iso1N3:"772",wikidata:"Q36823",nameEn:"Tokelau",country:"NZ",groups:["061","009"],driveSide:"left",callingCodes:["690"]},geometry:{type:"MultiPolygon",coordinates:[[[[-167.75195,-10.12005],[-167.75329,-7.52784],[-174.18707,-7.54408],[-174.17993,-10.13616],[-167.75195,-10.12005]]]]}},{type:"Feature",properties:{iso1A2:"TL",iso1A3:"TLS",iso1N3:"626",wikidata:"Q574",nameEn:"East Timor",aliases:["Timor-Leste","TP"],groups:["035","142"],driveSide:"left",callingCodes:["670"]},geometry:{type:"MultiPolygon",coordinates:[[[[124.46701,-9.13002],[124.94011,-8.85617],[124.97742,-9.08128],[125.11764,-8.96359],[125.18632,-9.03142],[125.18907,-9.16434],[125.09434,-9.19669],[125.04044,-9.17093],[124.97892,-9.19281],[125.09025,-9.46406],[125.68138,-9.85176],[127.55165,-9.05052],[127.42116,-8.22471],[125.87691,-8.31789],[125.65946,-8.06136],[125.31127,-8.22976],[124.92337,-8.75859],[124.33472,-9.11416],[124.04628,-9.22671],[124.04286,-9.34243],[124.10539,-9.41206],[124.14517,-9.42324],[124.21247,-9.36904],[124.28115,-9.42189],[124.28115,-9.50453],[124.3535,-9.48493],[124.35258,-9.43002],[124.38554,-9.3582],[124.45971,-9.30263],[124.46701,-9.13002]]]]}},{type:"Feature",properties:{iso1A2:"TM",iso1A3:"TKM",iso1N3:"795",wikidata:"Q874",nameEn:"Turkmenistan",groups:["143","142"],callingCodes:["993"]},geometry:{type:"MultiPolygon",coordinates:[[[[60.5078,41.21694],[60.06581,41.4363],[60.18117,41.60082],[60.06032,41.76287],[60.08504,41.80997],[60.33223,41.75058],[59.95046,41.97966],[60.0356,42.01028],[60.04659,42.08982],[59.96419,42.1428],[60.00539,42.212],[59.94633,42.27655],[59.4341,42.29738],[59.2955,42.37064],[59.17317,42.52248],[58.93422,42.5407],[58.6266,42.79314],[58.57991,42.64988],[58.27504,42.69632],[58.14321,42.62159],[58.29427,42.56497],[58.51674,42.30348],[58.40688,42.29535],[58.3492,42.43335],[57.99214,42.50021],[57.90975,42.4374],[57.92897,42.24047],[57.84932,42.18555],[57.6296,42.16519],[57.30275,42.14076],[57.03633,41.92043],[56.96218,41.80383],[57.03359,41.41777],[57.13796,41.36625],[57.03423,41.25435],[56.00314,41.32584],[55.45471,41.25609],[54.95182,41.92424],[54.20635,42.38477],[52.97575,42.1308],[52.47884,41.78034],[52.26048,41.69249],[51.7708,40.29239],[53.89734,37.3464],[54.24565,37.32047],[54.36211,37.34912],[54.58664,37.45809],[54.67247,37.43532],[54.77822,37.51597],[54.81804,37.61285],[54.77684,37.62264],[54.851,37.75739],[55.13412,37.94705],[55.44152,38.08564],[55.76561,38.12238],[55.97847,38.08024],[56.33278,38.08132],[56.32454,38.18502],[56.43303,38.26054],[56.62255,38.24005],[56.73928,38.27887],[57.03453,38.18717],[57.21169,38.28965],[57.37236,38.09321],[57.35042,37.98546],[57.79534,37.89299],[58.21399,37.77281],[58.22999,37.6856],[58.39959,37.63134],[58.47786,37.6433],[58.5479,37.70526],[58.6921,37.64548],[58.9338,37.67374],[59.22905,37.51161],[59.33507,37.53146],[59.39797,37.47892],[59.39385,37.34257],[59.55178,37.13594],[59.74678,37.12499],[60.00768,37.04102],[60.34767,36.63214],[61.14516,36.64644],[61.18187,36.55348],[61.1393,36.38782],[61.22719,36.12759],[61.12007,35.95992],[61.22444,35.92879],[61.26152,35.80749],[61.22719,35.67038],[61.27371,35.61482],[61.58742,35.43803],[61.77693,35.41341],[61.97743,35.4604],[62.05709,35.43803],[62.15871,35.33278],[62.29191,35.25964],[62.29878,35.13312],[62.48006,35.28796],[62.62288,35.22067],[62.74098,35.25432],[62.90853,35.37086],[63.0898,35.43131],[63.12276,35.53196],[63.10079,35.63024],[63.23262,35.67487],[63.10318,35.81782],[63.12276,35.86208],[63.29579,35.85985],[63.53475,35.90881],[63.56496,35.95106],[63.98519,36.03773],[64.05385,36.10433],[64.43288,36.24401],[64.57295,36.34362],[64.62514,36.44311],[64.61141,36.6351],[64.97945,37.21913],[65.51778,37.23881],[65.64263,37.34388],[65.64137,37.45061],[65.72274,37.55438],[66.30993,37.32409],[66.55743,37.35409],[66.52303,37.39827],[66.65761,37.45497],[66.52852,37.58568],[66.53676,37.80084],[66.67684,37.96776],[66.56697,38.0435],[66.41042,38.02403],[66.24013,38.16238],[65.83913,38.25733],[65.55873,38.29052],[64.32576,38.98691],[64.19086,38.95561],[63.70778,39.22349],[63.6913,39.27666],[62.43337,39.98528],[62.34273,40.43206],[62.11751,40.58242],[61.87856,41.12257],[61.4446,41.29407],[61.39732,41.19873],[61.33199,41.14946],[61.22212,41.14946],[61.03261,41.25691],[60.5078,41.21694]]]]}},{type:"Feature",properties:{iso1A2:"TN",iso1A3:"TUN",iso1N3:"788",wikidata:"Q948",nameEn:"Tunisia",groups:["015","002"],callingCodes:["216"]},geometry:{type:"MultiPolygon",coordinates:[[[[11.2718,37.6713],[7.89009,38.19924],[8.59123,37.14286],[8.64044,36.9401],[8.62972,36.86499],[8.67706,36.8364],[8.57613,36.78062],[8.46537,36.7706],[8.47609,36.66607],[8.16167,36.48817],[8.18936,36.44939],[8.40731,36.42208],[8.2626,35.91733],[8.26472,35.73669],[8.35371,35.66373],[8.36086,35.47774],[8.30329,35.29884],[8.47318,35.23376],[8.3555,35.10007],[8.30727,34.95378],[8.25189,34.92009],[8.29655,34.72798],[8.20482,34.57575],[7.86264,34.3987],[7.81242,34.21841],[7.74207,34.16492],[7.66174,34.20167],[7.52851,34.06493],[7.54088,33.7726],[7.73687,33.42114],[7.83028,33.18851],[8.11433,33.10175],[8.1179,33.05086],[8.31895,32.83483],[8.35999,32.50101],[9.07483,32.07865],[9.55544,30.23971],[9.76848,30.34366],[9.88152,30.34074],[10.29516,30.90337],[10.12239,31.42098],[10.31364,31.72648],[10.48497,31.72956],[10.62788,31.96629],[10.7315,31.97235],[11.04234,32.2145],[11.53898,32.4138],[11.57828,32.48013],[11.46037,32.6307],[11.51549,33.09826],[11.55852,33.1409],[11.56255,33.16754],[11.66543,33.34642],[11.2718,37.6713]]]]}},{type:"Feature",properties:{iso1A2:"TO",iso1A3:"TON",iso1N3:"776",wikidata:"Q678",nameEn:"Tonga",groups:["061","009"],driveSide:"left",callingCodes:["676"]},geometry:{type:"MultiPolygon",coordinates:[[[[-176.74538,-22.89767],[-180,-22.90585],[-180,-24.21376],[-173.10761,-24.19665],[-173.11048,-23.23027],[-173.13438,-14.94228],[-174.17905,-14.94502],[-176.76826,-14.95183],[-176.74538,-22.89767]]]]}},{type:"Feature",properties:{iso1A2:"TR",iso1A3:"TUR",iso1N3:"792",wikidata:"Q43",nameEn:"Turkey",groups:["145","142"],callingCodes:["90"]},geometry:{type:"MultiPolygon",coordinates:[[[[41.54366,41.52185],[40.89217,41.72528],[34.8305,42.4581],[28.32297,41.98371],[28.02971,41.98066],[27.91479,41.97902],[27.83492,41.99709],[27.81235,41.94803],[27.69949,41.97515],[27.55191,41.90928],[27.52379,41.93756],[27.45478,41.96591],[27.27411,42.10409],[27.22376,42.10152],[27.19251,42.06028],[27.08486,42.08735],[27.03277,42.0809],[26.95638,42.00741],[26.79143,41.97386],[26.62996,41.97644],[26.56051,41.92995],[26.57961,41.90024],[26.53968,41.82653],[26.36952,41.82265],[26.33589,41.76802],[26.32952,41.73637],[26.35957,41.71149],[26.47958,41.67037],[26.5209,41.62592],[26.59196,41.60491],[26.59742,41.48058],[26.61767,41.42281],[26.62997,41.34613],[26.5837,41.32131],[26.5209,41.33993],[26.39861,41.25053],[26.32259,41.24929],[26.31928,41.07386],[26.3606,41.02027],[26.33297,40.98388],[26.35894,40.94292],[26.32259,40.94042],[26.28623,40.93005],[26.29441,40.89119],[26.26169,40.9168],[26.20856,40.86048],[26.21351,40.83298],[26.15685,40.80709],[26.12854,40.77339],[26.12495,40.74283],[26.08638,40.73214],[26.0754,40.72772],[26.03489,40.73051],[25.94795,40.72797],[26.04292,40.3958],[25.61285,40.17161],[25.94257,39.39358],[26.43357,39.43096],[26.70773,39.0312],[26.61814,38.81372],[26.21136,38.65436],[26.32173,38.48731],[26.24183,38.44695],[26.21136,38.17558],[27.05537,37.9131],[27.16428,37.72343],[26.99377,37.69034],[26.95583,37.64989],[27.14757,37.32],[27.20312,36.94571],[27.45627,36.9008],[27.24613,36.71622],[27.46117,36.53789],[27.89482,36.69898],[27.95037,36.46155],[28.23708,36.56812],[29.30783,36.01033],[29.48192,36.18377],[29.61002,36.1731],[29.61805,36.14179],[29.69611,36.10365],[29.73302,35.92555],[32.82353,35.70297],[35.51152,36.10954],[35.931,35.92109],[35.98499,35.94107],[36.00514,35.94113],[36.01844,35.92403],[35.99829,35.88242],[36.11827,35.85923],[36.13919,35.83692],[36.14029,35.81015],[36.1623,35.80925],[36.17441,35.92076],[36.19973,35.95195],[36.25366,35.96264],[36.27678,35.94839],[36.29769,35.96086],[36.28338,36.00273],[36.30099,36.00985],[36.33956,35.98687],[36.37474,36.01163],[36.39206,36.22088],[36.4617,36.20461],[36.50463,36.2419],[36.6125,36.22592],[36.68672,36.23677],[36.65653,36.33861],[36.6081,36.33772],[36.54206,36.49539],[36.58829,36.58295],[36.57398,36.65186],[36.62681,36.71189],[36.61581,36.74629],[36.66727,36.82901],[36.99557,36.75997],[36.99886,36.74012],[37.04399,36.73483],[37.04619,36.71101],[37.01647,36.69512],[37.02088,36.66422],[37.08279,36.63495],[37.10894,36.6704],[37.16177,36.66069],[37.21988,36.6736],[37.47253,36.63243],[37.49103,36.66904],[37.68048,36.75065],[37.81974,36.76055],[38.21064,36.91842],[38.38859,36.90064],[38.55908,36.84429],[38.74042,36.70629],[39.03217,36.70911],[39.21538,36.66834],[39.81589,36.75538],[40.69136,37.0996],[40.90856,37.13147],[41.21937,37.07665],[41.515,37.08084],[42.00894,37.17209],[42.18225,37.28569],[42.19301,37.31323],[42.2112,37.32491],[42.22257,37.31395],[42.22381,37.30238],[42.20454,37.28715],[42.21548,37.28026],[42.23683,37.2863],[42.26039,37.27017],[42.2824,37.2798],[42.34735,37.22548],[42.32313,37.17814],[42.35724,37.10998],[42.56725,37.14878],[42.78887,37.38615],[42.93705,37.32015],[43.11403,37.37436],[43.30083,37.30629],[43.33508,37.33105],[43.50787,37.24436],[43.56702,37.25675],[43.63085,37.21957],[43.7009,37.23692],[43.8052,37.22825],[43.82699,37.19477],[43.84878,37.22205],[43.90949,37.22453],[44.02002,37.33229],[44.13521,37.32486],[44.2613,37.25055],[44.27998,37.16501],[44.22239,37.15756],[44.18503,37.09551],[44.25975,36.98119],[44.30645,36.97373],[44.35937,37.02843],[44.35315,37.04955],[44.38117,37.05825],[44.42631,37.05825],[44.63179,37.19229],[44.76698,37.16162],[44.78319,37.1431],[44.7868,37.16644],[44.75986,37.21549],[44.81021,37.2915],[44.58449,37.45018],[44.61401,37.60165],[44.56887,37.6429],[44.62096,37.71985],[44.55498,37.783],[44.45948,37.77065],[44.3883,37.85433],[44.22509,37.88859],[44.42476,38.25763],[44.50115,38.33939],[44.44386,38.38295],[44.38309,38.36117],[44.3119,38.37887],[44.3207,38.49799],[44.32058,38.62752],[44.28065,38.6465],[44.26155,38.71427],[44.30322,38.81581],[44.18863,38.93881],[44.20946,39.13975],[44.1043,39.19842],[44.03667,39.39223],[44.22452,39.4169],[44.29818,39.378],[44.37921,39.4131],[44.42832,39.4131],[44.41849,39.56659],[44.48111,39.61579],[44.47298,39.68788],[44.6137,39.78393],[44.65422,39.72163],[44.71806,39.71124],[44.81043,39.62677],[44.80977,39.65768],[44.75779,39.7148],[44.61845,39.8281],[44.46635,39.97733],[44.26973,40.04866],[44.1778,40.02845],[44.1057,40.03555],[43.92307,40.01787],[43.65688,40.11199],[43.65221,40.14889],[43.71136,40.16673],[43.59928,40.34019],[43.60862,40.43267],[43.54791,40.47413],[43.63664,40.54159],[43.7425,40.66805],[43.74872,40.7365],[43.67712,40.84846],[43.67712,40.93084],[43.58683,40.98961],[43.47319,41.02251],[43.44984,41.0988],[43.4717,41.12611],[43.44973,41.17666],[43.36118,41.2028],[43.23096,41.17536],[43.1945,41.25242],[43.13373,41.25503],[43.21707,41.30331],[43.02956,41.37891],[42.8785,41.50516],[42.84899,41.47265],[42.78995,41.50126],[42.84471,41.58912],[42.72794,41.59714],[42.59202,41.58183],[42.51772,41.43606],[42.26387,41.49346],[41.95134,41.52466],[41.81939,41.43621],[41.7124,41.47417],[41.7148,41.4932],[41.54366,41.52185]]]]}},{type:"Feature",properties:{iso1A2:"TT",iso1A3:"TTO",iso1N3:"780",wikidata:"Q754",nameEn:"Trinidad and Tobago",groups:["029","003","419","019"],driveSide:"left",callingCodes:["1 868"]},geometry:{type:"MultiPolygon",coordinates:[[[[-61.62505,11.18974],[-62.08693,10.04435],[-60.89962,9.81445],[-60.07172,11.77667],[-61.62505,11.18974]]]]}},{type:"Feature",properties:{iso1A2:"TV",iso1A3:"TUV",iso1N3:"798",wikidata:"Q672",nameEn:"Tuvalu",groups:["061","009"],driveSide:"left",callingCodes:["688"]},geometry:{type:"MultiPolygon",coordinates:[[[[174,-5],[174,-11.5],[179.99999,-11.5],[179.99999,-5],[174,-5]]]]}},{type:"Feature",properties:{iso1A2:"TW",iso1A3:"TWN",iso1N3:"158",wikidata:"Q865",nameEn:"Taiwan",groups:["030","142"],callingCodes:["886"]},geometry:{type:"MultiPolygon",coordinates:[[[[123.0791,22.07818],[122.26612,25.98197],[120.49232,25.22863],[118.56434,24.49266],[118.42453,24.54644],[118.35291,24.51645],[118.28244,24.51231],[118.11703,24.39734],[120.69238,21.52331],[123.0791,22.07818]]]]}},{type:"Feature",properties:{iso1A2:"TZ",iso1A3:"TZA",iso1N3:"834",wikidata:"Q924",nameEn:"Tanzania",groups:["014","202","002"],driveSide:"left",callingCodes:["255"]},geometry:{type:"MultiPolygon",coordinates:[[[[30.80408,-0.99911],[30.76635,-0.9852],[30.70631,-1.01175],[30.64166,-1.06601],[30.47194,-1.0555],[30.45116,-1.10641],[30.50889,-1.16412],[30.57123,-1.33264],[30.71974,-1.43244],[30.84079,-1.64652],[30.80802,-1.91477],[30.89303,-2.08223],[30.83915,-2.35795],[30.54501,-2.41404],[30.41789,-2.66266],[30.52747,-2.65841],[30.40662,-2.86151],[30.4987,-2.9573],[30.57926,-2.89791],[30.6675,-2.98987],[30.83823,-2.97837],[30.84165,-3.25152],[30.45915,-3.56532],[30.22042,-4.01738],[30.03323,-4.26631],[29.88172,-4.35743],[29.82885,-4.36153],[29.77289,-4.41733],[29.75109,-4.45836],[29.63827,-4.44681],[29.43673,-4.44845],[29.52552,-6.2731],[30.2567,-7.14121],[30.79243,-8.27382],[31.00796,-8.58615],[31.37533,-8.60769],[31.57147,-8.70619],[31.57147,-8.81388],[31.71158,-8.91386],[31.81587,-8.88618],[31.94663,-8.93846],[31.94196,-9.02303],[31.98866,-9.07069],[32.08206,-9.04609],[32.16146,-9.05993],[32.25486,-9.13371],[32.43543,-9.11988],[32.49147,-9.14754],[32.53661,-9.24281],[32.75611,-9.28583],[32.76233,-9.31963],[32.95389,-9.40138],[32.99397,-9.36712],[33.14925,-9.49322],[33.31581,-9.48554],[33.48052,-9.62442],[33.76677,-9.58516],[33.93298,-9.71647],[33.9638,-9.62206],[33.95829,-9.54066],[34.03865,-9.49398],[34.54499,-10.0678],[34.51911,-10.12279],[34.57581,-10.56271],[34.65946,-10.6828],[34.67047,-10.93796],[34.61161,-11.01611],[34.63305,-11.11731],[34.79375,-11.32245],[34.91153,-11.39799],[34.96296,-11.57354],[35.63599,-11.55927],[35.82767,-11.41081],[36.19094,-11.57593],[36.19094,-11.70008],[36.62068,-11.72884],[36.80309,-11.56836],[37.3936,-11.68949],[37.76614,-11.53352],[37.8388,-11.3123],[37.93618,-11.26228],[38.21598,-11.27289],[38.47258,-11.4199],[38.88996,-11.16978],[39.24395,-11.17433],[39.58249,-10.96043],[40.00295,-10.80255],[40.44265,-10.4618],[40.74206,-10.25691],[40.14328,-4.64201],[39.62121,-4.68136],[39.44306,-4.93877],[39.21631,-4.67835],[37.81321,-3.69179],[37.75036,-3.54243],[37.63099,-3.50723],[37.5903,-3.42735],[37.71745,-3.304],[37.67199,-3.06222],[34.0824,-1.02264],[34.03084,-1.05101],[34.02286,-1.00779],[33.93107,-0.99298],[30.80408,-0.99911]]]]}},{type:"Feature",properties:{iso1A2:"UA",iso1A3:"UKR",iso1N3:"804",wikidata:"Q212",nameEn:"Ukraine",groups:["151","150"],callingCodes:["380"]},geometry:{type:"MultiPolygon",coordinates:[[[[33.57318,46.10317],[33.61467,46.13561],[33.63854,46.14147],[33.61517,46.22615],[33.646,46.23028],[33.74047,46.18555],[33.79715,46.20482],[33.85234,46.19863],[33.91549,46.15938],[34.05272,46.10838],[34.07311,46.11769],[34.12929,46.10494],[34.181,46.06804],[34.25111,46.0532],[34.33912,46.06114],[34.41221,46.00245],[34.44155,45.95995],[34.48729,45.94267],[34.52011,45.95097],[34.55889,45.99347],[34.60861,45.99347],[34.66679,45.97136],[34.75479,45.90705],[34.80153,45.90047],[34.79905,45.81009],[34.96015,45.75634],[35.23066,45.79231],[37.62608,46.82615],[38.12112,46.86078],[38.3384,46.98085],[38.22955,47.12069],[38.23049,47.2324],[38.32112,47.2585],[38.33074,47.30508],[38.22225,47.30788],[38.28954,47.39255],[38.28679,47.53552],[38.35062,47.61631],[38.76379,47.69346],[38.79628,47.81109],[38.87979,47.87719],[39.73935,47.82876],[39.82213,47.96396],[39.77544,48.04206],[39.88256,48.04482],[39.83724,48.06501],[39.94847,48.22811],[40.00752,48.22445],[39.99241,48.31768],[39.97325,48.31399],[39.9693,48.29904],[39.95248,48.29972],[39.91465,48.26743],[39.90041,48.3049],[39.84273,48.30947],[39.84136,48.33321],[39.94847,48.35055],[39.88794,48.44226],[39.86196,48.46633],[39.84548,48.57821],[39.79764,48.58668],[39.67226,48.59368],[39.71765,48.68673],[39.73104,48.7325],[39.79466,48.83739],[39.97182,48.79398],[40.08168,48.87443],[40.03636,48.91957],[39.98967,48.86901],[39.78368,48.91596],[39.74874,48.98675],[39.72649,48.9754],[39.71353,48.98959],[39.6683,48.99454],[39.6836,49.05121],[39.93437,49.05709],[40.01988,49.1761],[40.22176,49.25683],[40.18331,49.34996],[40.14912,49.37681],[40.1141,49.38798],[40.03087,49.45452],[40.03636,49.52321],[40.16683,49.56865],[40.13249,49.61672],[39.84548,49.56064],[39.65047,49.61761],[39.59142,49.73758],[39.44496,49.76067],[39.27968,49.75976],[39.1808,49.88911],[38.9391,49.79524],[38.90477,49.86787],[38.73311,49.90238],[38.68677,50.00904],[38.65688,49.97176],[38.35408,50.00664],[38.32524,50.08866],[38.18517,50.08161],[38.21675,49.98104],[38.02999,49.90592],[38.02999,49.94482],[37.90776,50.04194],[37.79515,50.08425],[37.75807,50.07896],[37.61113,50.21976],[37.62879,50.24481],[37.62486,50.29966],[37.47243,50.36277],[37.48204,50.46079],[37.08468,50.34935],[36.91762,50.34963],[36.69377,50.26982],[36.64571,50.218],[36.56655,50.2413],[36.58371,50.28563],[36.47817,50.31457],[36.30101,50.29088],[36.20763,50.3943],[36.06893,50.45205],[35.8926,50.43829],[35.80388,50.41356],[35.73659,50.35489],[35.61711,50.35707],[35.58003,50.45117],[35.47463,50.49247],[35.39464,50.64751],[35.48116,50.66405],[35.47704,50.77274],[35.41367,50.80227],[35.39307,50.92145],[35.32598,50.94524],[35.40837,51.04119],[35.31774,51.08434],[35.20375,51.04723],[35.12685,51.16191],[35.14058,51.23162],[34.97304,51.2342],[34.82472,51.17483],[34.6874,51.18],[34.6613,51.25053],[34.38802,51.2746],[34.31661,51.23936],[34.23009,51.26429],[34.33446,51.363],[34.22048,51.4187],[34.30562,51.5205],[34.17599,51.63253],[34.07765,51.67065],[34.42922,51.72852],[34.41136,51.82793],[34.09413,52.00835],[34.11199,52.14087],[34.05239,52.20132],[33.78789,52.37204],[33.55718,52.30324],[33.48027,52.31499],[33.51323,52.35779],[33.18913,52.3754],[32.89937,52.2461],[32.85405,52.27888],[32.69475,52.25535],[32.54781,52.32423],[32.3528,52.32842],[32.38988,52.24946],[32.33083,52.23685],[32.34044,52.1434],[32.2777,52.10266],[32.23331,52.08085],[32.08813,52.03319],[31.92159,52.05144],[31.96141,52.08015],[31.85018,52.11305],[31.81722,52.09955],[31.7822,52.11406],[31.38326,52.12991],[31.25142,52.04131],[31.13332,52.1004],[30.95589,52.07775],[30.90897,52.00699],[30.76443,51.89739],[30.68804,51.82806],[30.51946,51.59649],[30.64992,51.35014],[30.56203,51.25655],[30.36153,51.33984],[30.34642,51.42555],[30.17888,51.51025],[29.77376,51.4461],[29.7408,51.53417],[29.54372,51.48372],[29.49773,51.39814],[29.42357,51.4187],[29.32881,51.37843],[29.25191,51.49828],[29.25603,51.57089],[29.20659,51.56918],[29.16402,51.64679],[29.1187,51.65872],[28.99098,51.56833],[28.95528,51.59222],[28.81795,51.55552],[28.76027,51.48802],[28.78224,51.45294],[28.75615,51.41442],[28.73143,51.46236],[28.69161,51.44695],[28.64429,51.5664],[28.47051,51.59734],[28.37592,51.54505],[28.23452,51.66988],[28.10658,51.57857],[27.95827,51.56065],[27.91844,51.61952],[27.85253,51.62293],[27.76052,51.47604],[27.67125,51.50854],[27.71932,51.60672],[27.55727,51.63486],[27.51058,51.5854],[27.47212,51.61184],[27.24828,51.60161],[27.26613,51.65957],[27.20948,51.66713],[27.20602,51.77291],[26.99422,51.76933],[26.9489,51.73788],[26.80043,51.75777],[26.69759,51.82284],[26.46962,51.80501],[26.39367,51.87315],[26.19084,51.86781],[26.00408,51.92967],[25.83217,51.92587],[25.80574,51.94556],[25.73673,51.91973],[25.46163,51.92205],[25.20228,51.97143],[24.98784,51.91273],[24.37123,51.88222],[24.29021,51.80841],[24.3163,51.75063],[24.13075,51.66979],[23.99907,51.58369],[23.8741,51.59734],[23.91118,51.63316],[23.7766,51.66809],[23.60906,51.62122],[23.6736,51.50255],[23.62751,51.50512],[23.69905,51.40871],[23.63858,51.32182],[23.80678,51.18405],[23.90376,51.07697],[23.92217,51.00836],[24.04576,50.90196],[24.14524,50.86128],[24.0952,50.83262],[23.99254,50.83847],[23.95925,50.79271],[24.0595,50.71625],[24.0996,50.60752],[24.07048,50.5071],[24.03668,50.44507],[23.99563,50.41289],[23.79445,50.40481],[23.71382,50.38248],[23.67635,50.33385],[23.28221,50.0957],[22.99329,49.84249],[22.83179,49.69875],[22.80261,49.69098],[22.78304,49.65543],[22.64534,49.53094],[22.69444,49.49378],[22.748,49.32759],[22.72009,49.20288],[22.86336,49.10513],[22.89122,49.00725],[22.56155,49.08865],[22.54338,49.01424],[22.48296,48.99172],[22.42934,48.92857],[22.34151,48.68893],[22.21379,48.6218],[22.16023,48.56548],[22.14689,48.4005],[22.2083,48.42534],[22.38133,48.23726],[22.49806,48.25189],[22.59007,48.15121],[22.58733,48.10813],[22.66835,48.09162],[22.73427,48.12005],[22.81804,48.11363],[22.87847,48.04665],[22.84276,47.98602],[22.89849,47.95851],[22.94301,47.96672],[22.92241,48.02002],[23.0158,47.99338],[23.08858,48.00716],[23.1133,48.08061],[23.15999,48.12188],[23.27397,48.08245],[23.33577,48.0237],[23.4979,47.96858],[23.52803,48.01818],[23.5653,48.00499],[23.63894,48.00293],[23.66262,47.98786],[23.75188,47.99705],[23.80904,47.98142],[23.8602,47.9329],[23.89352,47.94512],[23.94192,47.94868],[23.96337,47.96672],[23.98553,47.96076],[24.00801,47.968],[24.02999,47.95087],[24.06466,47.95317],[24.11281,47.91487],[24.22566,47.90231],[24.34926,47.9244],[24.43578,47.97131],[24.61994,47.95062],[24.70632,47.84428],[24.81893,47.82031],[24.88896,47.7234],[25.11144,47.75203],[25.23778,47.89403],[25.63878,47.94924],[25.77723,47.93919],[26.05901,47.9897],[26.17711,47.99246],[26.33504,48.18418],[26.55202,48.22445],[26.62823,48.25804],[26.6839,48.35828],[26.79239,48.29071],[26.82809,48.31629],[26.71274,48.40388],[26.85556,48.41095],[26.93384,48.36558],[27.03821,48.37653],[27.0231,48.42485],[27.08078,48.43214],[27.13434,48.37288],[27.27855,48.37534],[27.32159,48.4434],[27.37604,48.44398],[27.37741,48.41026],[27.44333,48.41209],[27.46942,48.454],[27.5889,48.49224],[27.59027,48.46311],[27.6658,48.44034],[27.74422,48.45926],[27.79225,48.44244],[27.81902,48.41874],[27.87533,48.4037],[27.88391,48.36699],[27.95883,48.32368],[28.04527,48.32661],[28.09873,48.3124],[28.07504,48.23494],[28.17666,48.25963],[28.19314,48.20749],[28.2856,48.23202],[28.32508,48.23384],[28.35519,48.24957],[28.36996,48.20543],[28.34912,48.1787],[28.30586,48.1597],[28.30609,48.14018],[28.34009,48.13147],[28.38712,48.17567],[28.43701,48.15832],[28.42454,48.12047],[28.48428,48.0737],[28.53921,48.17453],[28.69896,48.13106],[28.85232,48.12506],[28.8414,48.03392],[28.9306,47.96255],[29.1723,47.99013],[29.19839,47.89261],[29.27804,47.88893],[29.20663,47.80367],[29.27255,47.79953],[29.22242,47.73607],[29.22414,47.60012],[29.11743,47.55001],[29.18603,47.43387],[29.3261,47.44664],[29.39889,47.30179],[29.47854,47.30366],[29.48678,47.36043],[29.5733,47.36508],[29.59665,47.25521],[29.54996,47.24962],[29.57696,47.13581],[29.49732,47.12878],[29.53044,47.07851],[29.61038,47.09932],[29.62137,47.05069],[29.57056,46.94766],[29.72986,46.92234],[29.75458,46.8604],[29.87405,46.88199],[29.98814,46.82358],[29.94522,46.80055],[29.9743,46.75325],[29.94409,46.56002],[29.88916,46.54302],[30.02511,46.45132],[30.16794,46.40967],[30.09103,46.38694],[29.94114,46.40114],[29.88329,46.35851],[29.74496,46.45605],[29.66359,46.4215],[29.6763,46.36041],[29.5939,46.35472],[29.49914,46.45889],[29.35357,46.49505],[29.24886,46.37912],[29.23547,46.55435],[29.02409,46.49582],[29.01241,46.46177],[28.9306,46.45699],[29.004,46.31495],[28.98478,46.31803],[28.94953,46.25852],[29.06656,46.19716],[28.94643,46.09176],[29.00613,46.04962],[28.98004,46.00385],[28.74383,45.96664],[28.78503,45.83475],[28.69852,45.81753],[28.70401,45.78019],[28.52823,45.73803],[28.47879,45.66994],[28.51587,45.6613],[28.54196,45.58062],[28.49252,45.56716],[28.51449,45.49982],[28.43072,45.48538],[28.41836,45.51715],[28.30201,45.54744],[28.21139,45.46895],[28.28504,45.43907],[28.34554,45.32102],[28.5735,45.24759],[28.71358,45.22631],[28.78911,45.24179],[28.81383,45.3384],[28.94292,45.28045],[28.96077,45.33164],[29.24779,45.43388],[29.42632,45.44545],[29.59798,45.38857],[29.68175,45.26885],[29.65428,45.25629],[29.69272,45.19227],[30.04414,45.08461],[31.62627,45.50633],[33.54017,46.0123],[33.59087,46.06013],[33.57318,46.10317]]]]}},{type:"Feature",properties:{iso1A2:"UG",iso1A3:"UGA",iso1N3:"800",wikidata:"Q1036",nameEn:"Uganda",groups:["014","202","002"],driveSide:"left",callingCodes:["256"]},geometry:{type:"MultiPolygon",coordinates:[[[[33.93107,-0.99298],[33.9264,-0.54188],[33.98449,-0.13079],[33.90936,0.10581],[34.10067,0.36372],[34.08727,0.44713],[34.11408,0.48884],[34.13493,0.58118],[34.20196,0.62289],[34.27345,0.63182],[34.31516,0.75693],[34.40041,0.80266],[34.43349,0.85254],[34.52369,1.10692],[34.57427,1.09868],[34.58029,1.14712],[34.67562,1.21265],[34.80223,1.22754],[34.82606,1.26626],[34.82606,1.30944],[34.7918,1.36752],[34.87819,1.5596],[34.92734,1.56109],[34.9899,1.6668],[34.98692,1.97348],[34.90947,2.42447],[34.95267,2.47209],[34.77244,2.70272],[34.78137,2.76223],[34.73967,2.85447],[34.65774,2.8753],[34.60114,2.93034],[34.56242,3.11478],[34.45815,3.18319],[34.40006,3.37949],[34.41794,3.44342],[34.39112,3.48802],[34.44922,3.51627],[34.45815,3.67385],[34.15429,3.80464],[34.06046,4.15235],[33.9873,4.23316],[33.51264,3.75068],[33.18356,3.77812],[33.02852,3.89296],[32.89746,3.81339],[32.72021,3.77327],[32.41337,3.748],[32.20782,3.6053],[32.19888,3.50867],[32.08866,3.53543],[32.08491,3.56287],[32.05187,3.589],[31.95907,3.57408],[31.96205,3.6499],[31.86821,3.78664],[31.81459,3.82083],[31.72075,3.74354],[31.50776,3.63652],[31.50478,3.67814],[31.29476,3.8015],[31.16666,3.79853],[30.97601,3.693],[30.85153,3.48867],[30.94081,3.50847],[30.93486,3.40737],[30.84251,3.26908],[30.77101,3.04897],[30.8574,2.9508],[30.8857,2.83923],[30.75612,2.5863],[30.74271,2.43601],[30.83059,2.42559],[30.91102,2.33332],[30.96911,2.41071],[31.06593,2.35862],[31.07934,2.30207],[31.12104,2.27676],[31.1985,2.29462],[31.20148,2.2217],[31.28042,2.17853],[31.30127,2.11006],[30.48503,1.21675],[30.24671,1.14974],[30.22139,0.99635],[30.1484,0.89805],[29.98307,0.84295],[29.95477,0.64486],[29.97413,0.52124],[29.87284,0.39166],[29.81922,0.16824],[29.77454,0.16675],[29.7224,0.07291],[29.72687,-0.08051],[29.65091,-0.46777],[29.67474,-0.47969],[29.67176,-0.55714],[29.62708,-0.71055],[29.63006,-0.8997],[29.58388,-0.89821],[29.59061,-1.39016],[29.82657,-1.31187],[29.912,-1.48269],[30.16369,-1.34303],[30.35212,-1.06896],[30.47194,-1.0555],[30.64166,-1.06601],[30.70631,-1.01175],[30.76635,-0.9852],[30.80408,-0.99911],[33.93107,-0.99298]]]]}},{type:"Feature",properties:{iso1A2:"UM",iso1A3:"UMI",iso1N3:"581",wikidata:"Q16645",nameEn:"United States Minor Outlying Islands",country:"US",groups:["057","009"]},geometry:{type:"MultiPolygon",coordinates:[[[[-175.33482,-1.40631],[-175.33167,1.67574],[-177.43928,1.65656],[-177.43039,-1.43294],[-175.33482,-1.40631]]],[[[-161.04969,-1.36251],[-158.62058,-1.35506],[-158.62734,1.1296],[-161.05669,1.11722],[-161.04969,-1.36251]]],[[[-161.06795,5.2462],[-161.0731,7.1291],[-163.24994,7.12322],[-163.24478,5.24198],[-161.06795,5.2462]]],[[[-170.65691,16.57199],[-168.87689,16.01159],[-169.2329,17.4933],[-170.65691,16.57199]]],[[[-176.29741,29.09786],[-177.77531,29.29793],[-177.5224,27.7635],[-176.29741,29.09786]]],[[[-74.7289,18.71009],[-75.71816,18.46438],[-74.76465,18.06252],[-74.7289,18.71009]]],[[[167.34779,18.97692],[166.67967,20.14834],[165.82549,18.97692],[167.34779,18.97692]]]]}},{type:"Feature",properties:{iso1A2:"US",iso1A3:"USA",iso1N3:"840",wikidata:"Q30",nameEn:"United States of America",groups:["021","003","019"],roadSpeedUnit:"mph",callingCodes:["1"]},geometry:{type:"MultiPolygon",coordinates:[[[[-177.8563,29.18961],[-179.49839,27.86265],[-151.6784,9.55515],[-154.05867,45.51124],[-177.5224,27.7635],[-177.8563,29.18961]]],[[[169.34848,52.47228],[180,51.0171],[179.84401,55.10087],[169.34848,52.47228]]],[[[-168.95635,65.98512],[-169.03888,65.48473],[-172.76104,63.77445],[-179.55295,57.62081],[-179.55295,50.81807],[-133.92876,54.62289],[-130.61931,54.70835],[-130.64499,54.76912],[-130.44184,54.85377],[-130.27203,54.97174],[-130.18765,55.07744],[-130.08035,55.21556],[-129.97513,55.28029],[-130.15373,55.74895],[-130.00857,55.91344],[-130.00093,56.00325],[-130.10173,56.12178],[-130.33965,56.10849],[-130.77769,56.36185],[-131.8271,56.62247],[-133.38523,58.42773],[-133.84645,58.73543],[-134.27175,58.8634],[-134.48059,59.13231],[-134.55699,59.1297],[-134.7047,59.2458],[-135.00267,59.28745],[-135.03069,59.56208],[-135.48007,59.79937],[-136.31566,59.59083],[-136.22381,59.55526],[-136.33727,59.44466],[-136.47323,59.46617],[-136.52365,59.16752],[-136.82619,59.16198],[-137.4925,58.89415],[-137.60623,59.24465],[-138.62145,59.76431],[-138.71149,59.90728],[-139.05365,59.99655],[-139.20603,60.08896],[-139.05831,60.35205],[-139.68991,60.33693],[-139.98024,60.18027],[-140.45648,60.30919],[-140.5227,60.22077],[-141.00116,60.30648],[-140.97446,84.39275],[-168.25765,71.99091],[-168.95635,65.98512]]],[[[-97.13927,25.96583],[-96.92418,25.97377],[-82.02215,24.23074],[-79.89631,24.6597],[-79.14818,27.83105],[-61.98255,37.34815],[-67.16117,44.20069],[-66.93432,44.82597],[-66.96824,44.83078],[-66.98249,44.87071],[-66.96824,44.90965],[-67.0216,44.95333],[-67.11316,45.11176],[-67.15965,45.16179],[-67.19603,45.16771],[-67.20349,45.1722],[-67.22751,45.16344],[-67.27039,45.1934],[-67.29748,45.18173],[-67.29754,45.14865],[-67.34927,45.122],[-67.48201,45.27351],[-67.42394,45.37969],[-67.50578,45.48971],[-67.42144,45.50584],[-67.43815,45.59162],[-67.6049,45.60725],[-67.80705,45.69528],[-67.80653,45.80022],[-67.75654,45.82324],[-67.80961,45.87531],[-67.75196,45.91814],[-67.78111,45.9392],[-67.78578,47.06473],[-67.87993,47.10377],[-67.94843,47.1925],[-68.23244,47.35712],[-68.37458,47.35851],[-68.38332,47.28723],[-68.57914,47.28431],[-68.60575,47.24659],[-68.70125,47.24399],[-68.89222,47.1807],[-69.05039,47.2456],[-69.05073,47.30076],[-69.05148,47.42012],[-69.22119,47.46461],[-69.99966,46.69543],[-70.05812,46.41768],[-70.18547,46.35357],[-70.29078,46.18832],[-70.23855,46.1453],[-70.31025,45.96424],[-70.24694,45.95138],[-70.25976,45.89675],[-70.41523,45.79497],[-70.38934,45.73215],[-70.54019,45.67291],[-70.68516,45.56964],[-70.72651,45.49771],[-70.62518,45.42286],[-70.65383,45.37592],[-70.78372,45.43269],[-70.82638,45.39828],[-70.80236,45.37444],[-70.84816,45.22698],[-70.89864,45.2398],[-70.91169,45.29849],[-70.95193,45.33895],[-71.0107,45.34819],[-71.01866,45.31573],[-71.08364,45.30623],[-71.14568,45.24128],[-71.19723,45.25438],[-71.22338,45.25184],[-71.29371,45.29996],[-71.37133,45.24624],[-71.44252,45.2361],[-71.40364,45.21382],[-71.42778,45.12624],[-71.48735,45.07784],[-71.50067,45.01357],[-73.35025,45.00942],[-74.32699,44.99029],[-74.66689,45.00646],[-74.8447,45.00606],[-74.99101,44.98051],[-75.01363,44.95608],[-75.2193,44.87821],[-75.41441,44.76614],[-75.76813,44.51537],[-75.8217,44.43176],[-75.95947,44.34463],[-76.00018,44.34896],[-76.16285,44.28262],[-76.1664,44.23051],[-76.244,44.19643],[-76.31222,44.19894],[-76.35324,44.13493],[-76.43859,44.09393],[-76.79706,43.63099],[-79.25796,43.54052],[-79.06921,43.26183],[-79.05512,43.25375],[-79.05544,43.21224],[-79.05002,43.20133],[-79.05384,43.17418],[-79.04652,43.16396],[-79.0427,43.13934],[-79.06881,43.12029],[-79.05671,43.10937],[-79.07486,43.07845],[-79.01055,43.06659],[-78.99941,43.05612],[-79.02424,43.01983],[-79.02074,42.98444],[-78.98126,42.97],[-78.96312,42.95509],[-78.93224,42.95229],[-78.90905,42.93022],[-78.90712,42.89733],[-78.93684,42.82887],[-82.67862,41.67615],[-83.11184,41.95671],[-83.14962,42.04089],[-83.12724,42.2376],[-83.09837,42.28877],[-83.07837,42.30978],[-83.02253,42.33045],[-82.82964,42.37355],[-82.64242,42.55594],[-82.58873,42.54984],[-82.57583,42.5718],[-82.51858,42.611],[-82.51063,42.66025],[-82.46613,42.76615],[-82.4826,42.8068],[-82.45331,42.93139],[-82.4253,42.95423],[-82.4146,42.97626],[-82.42469,42.992],[-82.48419,45.30225],[-83.59589,45.82131],[-83.43746,45.99749],[-83.57017,46.105],[-83.83329,46.12169],[-83.90453,46.05922],[-83.95399,46.05634],[-84.1096,46.23987],[-84.09756,46.25512],[-84.11615,46.2681],[-84.11254,46.32329],[-84.13451,46.39218],[-84.11196,46.50248],[-84.12885,46.53068],[-84.17723,46.52753],[-84.1945,46.54061],[-84.2264,46.53337],[-84.26351,46.49508],[-84.29893,46.49127],[-84.34174,46.50683],[-84.42101,46.49853],[-84.4481,46.48972],[-84.47607,46.45225],[-84.55635,46.45974],[-84.85871,46.88881],[-88.37033,48.30586],[-89.48837,48.01412],[-89.57972,48.00023],[-89.77248,48.02607],[-89.89974,47.98109],[-90.07418,48.11043],[-90.56312,48.09488],[-90.56444,48.12184],[-90.75045,48.09143],[-90.87588,48.2484],[-91.08016,48.18096],[-91.25025,48.08522],[-91.43248,48.04912],[-91.45829,48.07454],[-91.58025,48.04339],[-91.55649,48.10611],[-91.70451,48.11805],[-91.71231,48.19875],[-91.86125,48.21278],[-91.98929,48.25409],[-92.05339,48.35958],[-92.14732,48.36578],[-92.202,48.35252],[-92.26662,48.35651],[-92.30939,48.31251],[-92.27167,48.25046],[-92.37185,48.22259],[-92.48147,48.36609],[-92.45588,48.40624],[-92.50712,48.44921],[-92.65606,48.43471],[-92.71323,48.46081],[-92.69927,48.49573],[-92.62747,48.50278],[-92.6342,48.54133],[-92.7287,48.54005],[-92.94973,48.60866],[-93.25391,48.64266],[-93.33946,48.62787],[-93.3712,48.60599],[-93.39758,48.60364],[-93.40693,48.60948],[-93.44472,48.59147],[-93.47022,48.54357],[-93.66382,48.51845],[-93.79267,48.51631],[-93.80939,48.52439],[-93.80676,48.58232],[-93.83288,48.62745],[-93.85769,48.63284],[-94.23215,48.65202],[-94.25104,48.65729],[-94.25172,48.68404],[-94.27153,48.70232],[-94.4174,48.71049],[-94.44258,48.69223],[-94.53826,48.70216],[-94.54885,48.71543],[-94.58903,48.71803],[-94.69335,48.77883],[-94.69669,48.80918],[-94.70486,48.82365],[-94.70087,48.8339],[-94.687,48.84077],[-94.75017,49.09931],[-94.77355,49.11998],[-94.82487,49.29483],[-94.8159,49.32299],[-94.85381,49.32492],[-94.95681,49.37035],[-94.99532,49.36579],[-95.01419,49.35647],[-95.05825,49.35311],[-95.12903,49.37056],[-95.15357,49.384],[-95.15355,48.9996],[-97.24024,48.99952],[-101.36198,48.99935],[-104.05004,48.99925],[-110.0051,48.99901],[-114.0683,48.99885],[-116.04938,48.99999],[-117.03266,49.00056],[-123.32163,49.00419],[-123.0093,48.83186],[-123.0093,48.76586],[-123.26565,48.6959],[-123.15614,48.35395],[-123.50039,48.21223],[-125.03842,48.53282],[-133.98258,38.06389],[-118.48109,32.5991],[-117.1243,32.53427],[-115.88053,32.63624],[-114.71871,32.71894],[-114.76736,32.64094],[-114.80584,32.62028],[-114.81141,32.55543],[-114.79524,32.55731],[-114.82011,32.49609],[-112.34553,31.7357],[-111.07523,31.33232],[-109.05235,31.3333],[-108.20979,31.33316],[-108.20899,31.78534],[-106.529,31.784],[-106.52266,31.77509],[-106.51251,31.76922],[-106.50962,31.76155],[-106.50111,31.75714],[-106.48815,31.74769],[-106.47298,31.75054],[-106.46726,31.75998],[-106.45244,31.76523],[-106.43419,31.75478],[-106.41773,31.75196],[-106.38003,31.73151],[-106.3718,31.71165],[-106.34864,31.69663],[-106.33419,31.66303],[-106.30305,31.62154],[-106.28084,31.56173],[-106.24612,31.54193],[-106.23711,31.51262],[-106.20346,31.46305],[-106.09025,31.40569],[-106.00363,31.39181],[-104.77674,30.4236],[-104.5171,29.64671],[-104.3969,29.57105],[-104.39363,29.55396],[-104.37752,29.54255],[-103.15787,28.93865],[-102.60596,29.8192],[-101.47277,29.7744],[-101.05686,29.44738],[-101.01128,29.36947],[-100.96725,29.3477],[-100.94579,29.34523],[-100.94056,29.33371],[-100.87982,29.296],[-100.79696,29.24688],[-100.67294,29.09744],[-100.63689,28.90812],[-100.59809,28.88197],[-100.52313,28.75598],[-100.5075,28.74066],[-100.51222,28.70679],[-100.50029,28.66117],[-99.55409,27.61314],[-99.51478,27.55836],[-99.52955,27.49747],[-99.50208,27.50021],[-99.48045,27.49016],[-99.482,27.47128],[-99.49744,27.43746],[-99.53573,27.30926],[-99.08477,26.39849],[-99.03053,26.41249],[-99.00546,26.3925],[-98.35126,26.15129],[-98.30491,26.10475],[-98.27075,26.09457],[-98.24603,26.07191],[-97.97017,26.05232],[-97.95155,26.0625],[-97.66511,26.01708],[-97.52025,25.88518],[-97.49828,25.89877],[-97.45669,25.86874],[-97.42511,25.83969],[-97.37332,25.83854],[-97.35946,25.92189],[-97.13927,25.96583]]]]}},{type:"Feature",properties:{iso1A2:"UY",iso1A3:"URY",iso1N3:"858",wikidata:"Q77",nameEn:"Uruguay",groups:["005","419","019"],callingCodes:["598"]},geometry:{type:"MultiPolygon",coordinates:[[[[-57.65132,-30.19229],[-57.61478,-30.25165],[-57.64859,-30.35095],[-57.89115,-30.49572],[-57.8024,-30.77193],[-57.89476,-30.95994],[-57.86729,-31.06352],[-57.9908,-31.34924],[-57.98127,-31.3872],[-58.07569,-31.44916],[-58.0023,-31.53084],[-58.00076,-31.65016],[-58.20252,-31.86966],[-58.10036,-32.25338],[-58.22362,-32.52416],[-58.1224,-32.98842],[-58.40475,-33.11777],[-58.44442,-33.84033],[-58.34425,-34.15035],[-57.83001,-34.69099],[-54.78916,-36.21945],[-52.83257,-34.01481],[-53.37138,-33.74313],[-53.39593,-33.75169],[-53.44031,-33.69344],[-53.52794,-33.68908],[-53.53459,-33.16843],[-53.1111,-32.71147],[-53.37671,-32.57005],[-53.39572,-32.58596],[-53.76024,-32.0751],[-54.17384,-31.86168],[-55.50821,-30.91349],[-55.50841,-30.9027],[-55.51862,-30.89828],[-55.52712,-30.89997],[-55.53276,-30.90218],[-55.53431,-30.89714],[-55.54572,-30.89051],[-55.55218,-30.88193],[-55.55373,-30.8732],[-55.5634,-30.8686],[-55.58866,-30.84117],[-55.87388,-31.05053],[-56.4619,-30.38457],[-56.4795,-30.3899],[-56.49267,-30.39471],[-56.90236,-30.02578],[-57.22502,-30.26121],[-57.65132,-30.19229]]]]}},{type:"Feature",properties:{iso1A2:"UZ",iso1A3:"UZB",iso1N3:"860",wikidata:"Q265",nameEn:"Uzbekistan",groups:["143","142"],callingCodes:["998"]},geometry:{type:"MultiPolygon",coordinates:[[[[65.85194,42.85481],[65.53277,43.31856],[65.18666,43.48835],[64.96464,43.74748],[64.53885,43.56941],[63.34656,43.64003],[62.01711,43.51008],[61.01475,44.41383],[58.59711,45.58671],[55.97842,44.99622],[55.97832,44.99622],[55.97822,44.99617],[55.97811,44.99617],[55.97801,44.99612],[55.97801,44.99607],[55.97791,44.99607],[55.9778,44.99607],[55.9777,44.99601],[55.9777,44.99596],[55.9776,44.99591],[55.97749,44.99591],[55.97739,44.99591],[55.97739,44.99586],[55.97729,44.99586],[55.97718,44.99581],[55.97708,44.99576],[55.97698,44.9957],[55.97698,44.99565],[55.97687,44.9956],[55.97677,44.9956],[55.97677,44.99555],[55.97677,44.9955],[55.97667,44.99545],[55.97656,44.99539],[55.97646,44.99534],[55.97646,44.99529],[55.97636,44.99524],[55.97636,44.99519],[55.97625,44.99514],[55.97615,44.99508],[55.97615,44.99503],[55.97615,44.99498],[55.97615,44.99493],[55.97615,44.99483],[55.97615,44.99477],[55.97605,44.99477],[55.97605,44.99467],[55.97605,44.99462],[55.97605,44.99457],[55.97605,44.99452],[55.97594,44.99446],[55.97584,44.99441],[55.97584,44.99436],[55.97584,44.99431],[55.97584,44.99426],[55.97584,44.99421],[55.97584,44.99415],[55.97584,44.99405],[55.97584,44.994],[55.97584,44.9939],[55.97584,44.99384],[55.97584,44.99374],[55.97584,44.99369],[55.97584,44.99359],[55.97584,44.99353],[55.97584,44.99348],[55.97584,44.99343],[55.97584,44.99338],[55.97584,44.99328],[55.97584,44.99322],[56.00314,41.32584],[57.03423,41.25435],[57.13796,41.36625],[57.03359,41.41777],[56.96218,41.80383],[57.03633,41.92043],[57.30275,42.14076],[57.6296,42.16519],[57.84932,42.18555],[57.92897,42.24047],[57.90975,42.4374],[57.99214,42.50021],[58.3492,42.43335],[58.40688,42.29535],[58.51674,42.30348],[58.29427,42.56497],[58.14321,42.62159],[58.27504,42.69632],[58.57991,42.64988],[58.6266,42.79314],[58.93422,42.5407],[59.17317,42.52248],[59.2955,42.37064],[59.4341,42.29738],[59.94633,42.27655],[60.00539,42.212],[59.96419,42.1428],[60.04659,42.08982],[60.0356,42.01028],[59.95046,41.97966],[60.33223,41.75058],[60.08504,41.80997],[60.06032,41.76287],[60.18117,41.60082],[60.06581,41.4363],[60.5078,41.21694],[61.03261,41.25691],[61.22212,41.14946],[61.33199,41.14946],[61.39732,41.19873],[61.4446,41.29407],[61.87856,41.12257],[62.11751,40.58242],[62.34273,40.43206],[62.43337,39.98528],[63.6913,39.27666],[63.70778,39.22349],[64.19086,38.95561],[64.32576,38.98691],[65.55873,38.29052],[65.83913,38.25733],[66.24013,38.16238],[66.41042,38.02403],[66.56697,38.0435],[66.67684,37.96776],[66.53676,37.80084],[66.52852,37.58568],[66.65761,37.45497],[66.52303,37.39827],[66.55743,37.35409],[66.64699,37.32958],[66.95598,37.40162],[67.08232,37.35469],[67.13039,37.27168],[67.2224,37.24545],[67.2581,37.17216],[67.51868,37.26102],[67.78329,37.1834],[67.8474,37.31594],[67.81566,37.43107],[68.12635,37.93],[68.27159,37.91477],[68.40343,38.19484],[68.13289,38.40822],[68.06274,38.39435],[68.11366,38.47169],[68.05873,38.56087],[68.0807,38.64136],[68.05598,38.71641],[68.12877,38.73677],[68.06948,38.82115],[68.19743,38.85985],[68.09704,39.02589],[67.68915,39.00775],[67.67833,39.14479],[67.33226,39.23739],[67.36522,39.31287],[67.45998,39.315],[67.46822,39.46146],[67.39681,39.52505],[67.46547,39.53564],[67.44899,39.57799],[67.62889,39.60234],[67.70992,39.66156],[68.12053,39.56317],[68.54166,39.53929],[68.61972,39.68905],[68.63071,39.85265],[68.88889,39.87163],[68.93695,39.91167],[68.84906,40.04952],[68.96579,40.06949],[69.01935,40.11466],[69.01523,40.15771],[68.62796,40.07789],[68.52771,40.11676],[68.5332,40.14826],[68.77902,40.20492],[68.79276,40.17555],[68.84357,40.18604],[68.85832,40.20885],[69.04544,40.22904],[69.15659,40.2162],[69.2074,40.21488],[69.30448,40.18774],[69.30104,40.24502],[69.25229,40.26362],[69.24817,40.30357],[69.30808,40.2821],[69.32833,40.29794],[69.33794,40.34819],[69.30774,40.36102],[69.28525,40.41894],[69.27066,40.49274],[69.21063,40.54469],[69.2643,40.57506],[69.3455,40.57988],[69.32834,40.70233],[69.38327,40.7918],[69.53021,40.77621],[69.59441,40.70181],[69.69434,40.62615],[70.36655,40.90296],[70.38028,41.02014],[70.45251,41.04438],[70.80009,40.72825],[70.49871,40.52503],[70.32626,40.45174],[70.37511,40.38605],[70.57149,40.3442],[70.56394,40.26421],[70.62342,40.17396],[70.8607,40.217],[70.9818,40.22392],[70.95789,40.28761],[71.05901,40.28765],[71.13042,40.34106],[71.36663,40.31593],[71.4246,40.28619],[71.51215,40.26943],[71.51549,40.22986],[71.61725,40.20615],[71.61931,40.26775],[71.68386,40.26984],[71.70569,40.20391],[71.69621,40.18492],[71.71719,40.17886],[71.73054,40.14818],[71.82646,40.21872],[71.85002,40.25647],[72.05464,40.27586],[71.96401,40.31907],[72.18648,40.49893],[72.24368,40.46091],[72.40346,40.4007],[72.44191,40.48222],[72.41513,40.50856],[72.38384,40.51535],[72.41714,40.55736],[72.34406,40.60144],[72.40517,40.61917],[72.47795,40.5532],[72.66713,40.5219],[72.66713,40.59076],[72.69579,40.59778],[72.73995,40.58409],[72.74768,40.58051],[72.74862,40.57131],[72.75982,40.57273],[72.74894,40.59592],[72.74866,40.60873],[72.80137,40.67856],[72.84754,40.67229],[72.85372,40.7116],[72.8722,40.71111],[72.93296,40.73089],[72.99133,40.76457],[73.0612,40.76678],[73.13412,40.79122],[73.13267,40.83512],[73.01869,40.84681],[72.94454,40.8094],[72.84291,40.85512],[72.68157,40.84942],[72.59136,40.86947],[72.55109,40.96046],[72.48742,40.97136],[72.45206,41.03018],[72.38511,41.02785],[72.36138,41.04384],[72.34757,41.06104],[72.34026,41.04539],[72.324,41.03381],[72.18339,40.99571],[72.17594,41.02377],[72.21061,41.05607],[72.1792,41.10621],[72.14864,41.13363],[72.17594,41.15522],[72.16433,41.16483],[72.10745,41.15483],[72.07249,41.11739],[71.85964,41.19081],[71.91457,41.2982],[71.83914,41.3546],[71.76625,41.4466],[71.71132,41.43012],[71.73054,41.54713],[71.65914,41.49599],[71.6787,41.42111],[71.57227,41.29175],[71.46688,41.31883],[71.43814,41.19644],[71.46148,41.13958],[71.40198,41.09436],[71.34877,41.16807],[71.27187,41.11015],[71.25813,41.18796],[71.11806,41.15359],[71.02193,41.19494],[70.9615,41.16393],[70.86263,41.23833],[70.77885,41.24813],[70.78572,41.36419],[70.67586,41.47953],[70.48909,41.40335],[70.17682,41.5455],[70.69777,41.92554],[71.28719,42.18033],[71.13263,42.28356],[70.94483,42.26238],[69.49545,41.545],[69.45751,41.56863],[69.39485,41.51518],[69.45081,41.46246],[69.37468,41.46555],[69.35554,41.47211],[69.29778,41.43673],[69.25059,41.46693],[69.23332,41.45847],[69.22671,41.46298],[69.20439,41.45391],[69.18528,41.45175],[69.17701,41.43769],[69.15137,41.43078],[69.05006,41.36183],[69.01308,41.22804],[68.7217,41.05025],[68.73945,40.96989],[68.65662,40.93861],[68.62221,41.03019],[68.49983,40.99669],[68.58444,40.91447],[68.63,40.59358],[68.49983,40.56437],[67.96736,40.83798],[68.1271,41.0324],[68.08273,41.08148],[67.98511,41.02794],[67.9644,41.14611],[66.69129,41.1311],[66.53302,41.87388],[66.00546,41.94455],[66.09482,42.93426],[65.85194,42.85481]],[[70.68112,40.90612],[70.6721,40.90555],[70.57501,40.98941],[70.54223,40.98787],[70.56077,41.00642],[70.6158,40.97661],[70.68112,40.90612]]],[[[71.21139,40.03369],[71.12218,40.03052],[71.06305,40.1771],[71.00236,40.18154],[71.01035,40.05481],[71.11037,40.01984],[71.11668,39.99291],[71.09063,39.99],[71.10501,39.95568],[71.04979,39.89808],[71.10531,39.91354],[71.16101,39.88423],[71.23067,39.93581],[71.1427,39.95026],[71.21139,40.03369]]],[[[71.86463,39.98598],[71.78838,40.01404],[71.71511,39.96348],[71.7504,39.93701],[71.84316,39.95582],[71.86463,39.98598]]]]}},{type:"Feature",properties:{iso1A2:"VA",iso1A3:"VAT",iso1N3:"336",wikidata:"Q237",nameEn:"Vatican City",aliases:["Holy See"],groups:["039","150"],callingCodes:["379","39 06"]},geometry:{type:"MultiPolygon",coordinates:[[[[12.45181,41.90056],[12.45446,41.90028],[12.45435,41.90143],[12.45626,41.90172],[12.45691,41.90125],[12.4577,41.90115],[12.45834,41.90174],[12.45826,41.90281],[12.45755,41.9033],[12.45762,41.9058],[12.45561,41.90629],[12.45543,41.90738],[12.45091,41.90625],[12.44984,41.90545],[12.44815,41.90326],[12.44582,41.90194],[12.44834,41.90095],[12.45181,41.90056]]]]}},{type:"Feature",properties:{iso1A2:"VC",iso1A3:"VCT",iso1N3:"670",wikidata:"Q757",nameEn:"St. Vincent and the Grenadines",aliases:["WV"],groups:["029","003","419","019"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["1 784"]},geometry:{type:"MultiPolygon",coordinates:[[[[-61.73897,12.61191],[-61.38256,12.52991],[-61.13395,12.51526],[-60.70539,13.41452],[-61.43129,13.68336],[-61.73897,12.61191]]]]}},{type:"Feature",properties:{iso1A2:"VE",iso1A3:"VEN",iso1N3:"862",wikidata:"Q717",nameEn:"Venezuela",aliases:["YV"],groups:["005","419","019"],callingCodes:["58"]},geometry:{type:"MultiPolygon",coordinates:[[[[-71.22331,13.01387],[-70.92579,11.96275],[-71.3275,11.85],[-71.9675,11.65536],[-72.24983,11.14138],[-72.4767,11.1117],[-72.88002,10.44309],[-72.98085,9.85253],[-73.36905,9.16636],[-73.02119,9.27584],[-72.94052,9.10663],[-72.77415,9.10165],[-72.65474,8.61428],[-72.4042,8.36513],[-72.36987,8.19976],[-72.35163,8.01163],[-72.39137,8.03534],[-72.47213,7.96106],[-72.48801,7.94329],[-72.48183,7.92909],[-72.47042,7.92306],[-72.45806,7.91141],[-72.46183,7.90682],[-72.44454,7.86031],[-72.46763,7.79518],[-72.47827,7.65604],[-72.45321,7.57232],[-72.47415,7.48928],[-72.43132,7.40034],[-72.19437,7.37034],[-72.04895,7.03837],[-71.82441,7.04314],[-71.44118,7.02116],[-71.42212,7.03854],[-71.37234,7.01588],[-71.03941,6.98163],[-70.7596,7.09799],[-70.10716,6.96516],[-69.41843,6.1072],[-67.60654,6.2891],[-67.4625,6.20625],[-67.43513,5.98835],[-67.58558,5.84537],[-67.63914,5.64963],[-67.59141,5.5369],[-67.83341,5.31104],[-67.85358,4.53249],[-67.62671,3.74303],[-67.50067,3.75812],[-67.30945,3.38393],[-67.85862,2.86727],[-67.85862,2.79173],[-67.65696,2.81691],[-67.21967,2.35778],[-66.85795,1.22998],[-66.28507,0.74585],[-65.6727,1.01353],[-65.50158,0.92086],[-65.57288,0.62856],[-65.11657,1.12046],[-64.38932,1.5125],[-64.34654,1.35569],[-64.08274,1.64792],[-64.06135,1.94722],[-63.39827,2.16098],[-63.39114,2.4317],[-64.0257,2.48156],[-64.02908,2.79797],[-64.48379,3.7879],[-64.84028,4.24665],[-64.72977,4.28931],[-64.57648,4.12576],[-64.14512,4.12932],[-63.99183,3.90172],[-63.86082,3.94796],[-63.70218,3.91417],[-63.67099,4.01731],[-63.50611,3.83592],[-63.42233,3.89995],[-63.4464,3.9693],[-63.21111,3.96219],[-62.98296,3.59935],[-62.7655,3.73099],[-62.74411,4.03331],[-62.57656,4.04754],[-62.44822,4.18621],[-62.13094,4.08309],[-61.54629,4.2822],[-61.48569,4.43149],[-61.29675,4.44216],[-61.31457,4.54167],[-61.15703,4.49839],[-60.98303,4.54167],[-60.86539,4.70512],[-60.5802,4.94312],[-60.73204,5.20931],[-61.4041,5.95304],[-61.15058,6.19558],[-61.20762,6.58174],[-61.13632,6.70922],[-60.54873,6.8631],[-60.39419,6.94847],[-60.28074,7.1162],[-60.44116,7.20817],[-60.54098,7.14804],[-60.63367,7.25061],[-60.59802,7.33194],[-60.71923,7.55817],[-60.64793,7.56877],[-60.51959,7.83373],[-60.38056,7.8302],[-60.02407,8.04557],[-59.97059,8.20791],[-59.83156,8.23261],[-59.80661,8.28906],[-59.85562,8.35213],[-59.98508,8.53046],[-59.54058,8.6862],[-60.89962,9.81445],[-62.08693,10.04435],[-61.62505,11.18974],[-63.73917,11.92623],[-63.19938,16.44103],[-67.89186,12.4116],[-68.01417,11.77722],[-68.33524,11.78151],[-68.99639,11.79035],[-71.22331,13.01387]]]]}},{type:"Feature",properties:{iso1A2:"VG",iso1A3:"VGB",iso1N3:"092",wikidata:"Q25305",nameEn:"British Virgin Islands",country:"GB",groups:["029","003","419","019"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["1 284"]},geometry:{type:"MultiPolygon",coordinates:[[[[-64.03057,18.08241],[-63.75633,19.39745],[-65.02435,18.73231],[-64.86027,18.39056],[-64.64067,18.36478],[-64.646,18.10286],[-64.03057,18.08241]]]]}},{type:"Feature",properties:{iso1A2:"VI",iso1A3:"VIR",iso1N3:"850",wikidata:"Q11703",nameEn:"United States Virgin Islands",country:"US",groups:["029","003","419","019"],driveSide:"left",roadSpeedUnit:"mph",callingCodes:["1 340"]},geometry:{type:"MultiPolygon",coordinates:[[[[-65.02435,18.73231],[-65.27974,17.56928],[-64.35558,17.48384],[-64.646,18.10286],[-64.64067,18.36478],[-64.86027,18.39056],[-65.02435,18.73231]]]]}},{type:"Feature",properties:{iso1A2:"VN",iso1A3:"VNM",iso1N3:"704",wikidata:"Q881",nameEn:"Vietnam",groups:["035","142"],callingCodes:["84"]},geometry:{type:"MultiPolygon",coordinates:[[[[108.10003,21.47338],[108.0569,21.53604],[108.02926,21.54997],[107.97932,21.54503],[107.97383,21.53961],[107.97074,21.54072],[107.96774,21.53601],[107.95232,21.5388],[107.92652,21.58906],[107.90006,21.5905],[107.86114,21.65128],[107.80355,21.66141],[107.66967,21.60787],[107.56537,21.61945],[107.54047,21.5934],[107.49065,21.59774],[107.49532,21.62958],[107.47197,21.6672],[107.41593,21.64839],[107.38636,21.59774],[107.35989,21.60063],[107.35834,21.6672],[107.29296,21.74674],[107.24625,21.7077],[107.20734,21.71493],[107.10771,21.79879],[107.02615,21.81981],[107.00964,21.85948],[107.06101,21.88982],[107.05634,21.92303],[106.99252,21.95191],[106.97228,21.92592],[106.92714,21.93459],[106.9178,21.97357],[106.81038,21.97934],[106.74345,22.00965],[106.72551,21.97923],[106.69276,21.96013],[106.68274,21.99811],[106.70142,22.02409],[106.6983,22.15102],[106.67495,22.1885],[106.69986,22.22309],[106.6516,22.33977],[106.55976,22.34841],[106.57221,22.37],[106.55665,22.46498],[106.58395,22.474],[106.61269,22.60301],[106.65316,22.5757],[106.71698,22.58432],[106.72321,22.63606],[106.76293,22.73491],[106.82404,22.7881],[106.83685,22.8098],[106.81271,22.8226],[106.78422,22.81532],[106.71128,22.85982],[106.71387,22.88296],[106.6734,22.89587],[106.6516,22.86862],[106.60179,22.92884],[106.55976,22.92311],[106.51306,22.94891],[106.49749,22.91164],[106.34961,22.86718],[106.27022,22.87722],[106.19705,22.98475],[106.00179,22.99049],[105.99568,22.94178],[105.90119,22.94168],[105.8726,22.92756],[105.72382,23.06641],[105.57594,23.075],[105.56037,23.16806],[105.49966,23.20669],[105.42805,23.30824],[105.40782,23.28107],[105.32376,23.39684],[105.22569,23.27249],[105.17276,23.28679],[105.11672,23.25247],[105.07002,23.26248],[104.98712,23.19176],[104.96532,23.20463],[104.9486,23.17235],[104.91435,23.18666],[104.87992,23.17141],[104.87382,23.12854],[104.79478,23.12934],[104.8334,23.01484],[104.86765,22.95178],[104.84942,22.93631],[104.77114,22.90017],[104.72755,22.81984],[104.65283,22.83419],[104.60457,22.81841],[104.58122,22.85571],[104.47225,22.75813],[104.35593,22.69353],[104.25683,22.76534],[104.27084,22.8457],[104.11384,22.80363],[104.03734,22.72945],[104.01088,22.51823],[103.99247,22.51958],[103.97384,22.50634],[103.96783,22.51173],[103.96352,22.50584],[103.95191,22.5134],[103.94513,22.52553],[103.93286,22.52703],[103.87904,22.56683],[103.64506,22.79979],[103.56255,22.69499],[103.57812,22.65764],[103.52675,22.59155],[103.43646,22.70648],[103.43179,22.75816],[103.32282,22.8127],[103.28079,22.68063],[103.18895,22.64471],[103.15782,22.59873],[103.17961,22.55705],[103.07843,22.50097],[103.0722,22.44775],[102.9321,22.48659],[102.8636,22.60735],[102.60675,22.73376],[102.57095,22.7036],[102.51802,22.77969],[102.46665,22.77108],[102.42618,22.69212],[102.38415,22.67919],[102.41061,22.64184],[102.25339,22.4607],[102.26428,22.41321],[102.16621,22.43336],[102.14099,22.40092],[102.18712,22.30403],[102.51734,22.02676],[102.49092,21.99002],[102.62301,21.91447],[102.67145,21.65894],[102.74189,21.66713],[102.82115,21.73667],[102.81894,21.83888],[102.85637,21.84501],[102.86077,21.71213],[102.97965,21.74076],[102.98846,21.58936],[102.86297,21.4255],[102.94223,21.46034],[102.88939,21.3107],[102.80794,21.25736],[102.89825,21.24707],[102.97745,21.05821],[103.03469,21.05821],[103.12055,20.89994],[103.21497,20.89832],[103.38032,20.79501],[103.45737,20.82382],[103.68633,20.66324],[103.73478,20.6669],[103.82282,20.8732],[103.98024,20.91531],[104.11121,20.96779],[104.27412,20.91433],[104.63957,20.6653],[104.38199,20.47155],[104.40621,20.3849],[104.47886,20.37459],[104.66158,20.47774],[104.72102,20.40554],[104.62195,20.36633],[104.61315,20.24452],[104.86852,20.14121],[104.91695,20.15567],[104.9874,20.09573],[104.8465,19.91783],[104.8355,19.80395],[104.68359,19.72729],[104.64837,19.62365],[104.53169,19.61743],[104.41281,19.70035],[104.23229,19.70242],[104.06498,19.66926],[104.05617,19.61743],[104.10832,19.51575],[104.06058,19.43484],[103.87125,19.31854],[104.5361,18.97747],[104.64617,18.85668],[105.12829,18.70453],[105.19654,18.64196],[105.1327,18.58355],[105.10408,18.43533],[105.15942,18.38691],[105.38366,18.15315],[105.46292,18.22008],[105.64784,17.96687],[105.60381,17.89356],[105.76612,17.67147],[105.85744,17.63221],[106.09019,17.36399],[106.18991,17.28227],[106.24444,17.24714],[106.29287,17.3018],[106.31929,17.20509],[106.43597,17.01362],[106.50862,16.9673],[106.55045,17.0031],[106.54824,16.92729],[106.51963,16.92097],[106.52183,16.87884],[106.55265,16.86831],[106.55485,16.68704],[106.59013,16.62259],[106.58267,16.6012],[106.61477,16.60713],[106.66052,16.56892],[106.65832,16.47816],[106.74418,16.41904],[106.84104,16.55415],[106.88727,16.52671],[106.88067,16.43594],[106.96638,16.34938],[106.97385,16.30204],[107.02597,16.31132],[107.09091,16.3092],[107.15035,16.26271],[107.14595,16.17816],[107.25822,16.13587],[107.33968,16.05549],[107.44975,16.08511],[107.46296,16.01106],[107.39471,15.88829],[107.34188,15.89464],[107.21419,15.83747],[107.21859,15.74638],[107.27143,15.71459],[107.27583,15.62769],[107.34408,15.62345],[107.3815,15.49832],[107.50699,15.48771],[107.53341,15.40496],[107.62367,15.42193],[107.60605,15.37524],[107.62587,15.2266],[107.58844,15.20111],[107.61926,15.13949],[107.61486,15.0566],[107.46516,15.00982],[107.48277,14.93751],[107.59285,14.87795],[107.51579,14.79282],[107.54361,14.69092],[107.55371,14.628],[107.52102,14.59034],[107.52569,14.54665],[107.48521,14.40346],[107.44941,14.41552],[107.39493,14.32655],[107.40427,14.24509],[107.33577,14.11832],[107.37158,14.07906],[107.35757,14.02319],[107.38247,13.99147],[107.44318,13.99751],[107.46498,13.91593],[107.45252,13.78897],[107.53503,13.73908],[107.61909,13.52577],[107.62843,13.3668],[107.49144,13.01215],[107.49611,12.88926],[107.55993,12.7982],[107.5755,12.52177],[107.55059,12.36824],[107.4463,12.29373],[107.42917,12.24657],[107.34511,12.33327],[107.15831,12.27547],[106.99953,12.08983],[106.92325,12.06548],[106.79405,12.0807],[106.70687,11.96956],[106.4111,11.97413],[106.4687,11.86751],[106.44068,11.86294],[106.44535,11.8279],[106.41577,11.76999],[106.45158,11.68616],[106.44691,11.66787],[106.37219,11.69836],[106.30525,11.67549],[106.26478,11.72122],[106.18539,11.75171],[106.13158,11.73283],[106.06708,11.77761],[106.02038,11.77457],[106.00792,11.7197],[105.95188,11.63738],[105.88962,11.67854],[105.8507,11.66635],[105.80867,11.60536],[105.81645,11.56876],[105.87328,11.55953],[105.88962,11.43605],[105.86782,11.28343],[106.10444,11.07879],[106.1527,11.10476],[106.1757,11.07301],[106.20095,10.97795],[106.14301,10.98176],[106.18539,10.79451],[106.06708,10.8098],[105.94535,10.9168],[105.93403,10.83853],[105.84603,10.85873],[105.86376,10.89839],[105.77751,11.03671],[105.50045,10.94586],[105.42884,10.96878],[105.34011,10.86179],[105.11449,10.96332],[105.08326,10.95656],[105.02722,10.89236],[105.09571,10.72722],[104.95094,10.64003],[104.87933,10.52833],[104.59018,10.53073],[104.49869,10.4057],[104.47963,10.43046],[104.43778,10.42386],[103.99198,10.48391],[102.47649,9.66162],[104.81582,8.03101],[109.55486,8.10026],[111.60491,13.57105],[108.00365,17.98159],[108.10003,21.47338]]]]}},{type:"Feature",properties:{iso1A2:"VU",iso1A3:"VUT",iso1N3:"548",wikidata:"Q686",nameEn:"Vanuatu",groups:["054","009"],callingCodes:["678"]},geometry:{type:"MultiPolygon",coordinates:[[[[162.93363,-17.28904],[173.26254,-22.69968],[168.21179,-12.88558],[166.02864,-12.9396],[162.93363,-17.28904]]]]}},{type:"Feature",properties:{iso1A2:"WF",iso1A3:"WLF",iso1N3:"876",wikidata:"Q35555",nameEn:"Wallis and Futuna",country:"FR",groups:["061","009"],callingCodes:["681"]},geometry:{type:"MultiPolygon",coordinates:[[[[-178.60161,-14.95666],[-176.76826,-14.95183],[-174.17905,-14.94502],[-174.18596,-12.48057],[-178.60852,-12.49232],[-178.60161,-14.95666]]]]}},{type:"Feature",properties:{iso1A2:"WS",iso1A3:"WSM",iso1N3:"882",wikidata:"Q683",nameEn:"Samoa",groups:["061","009"],driveSide:"left",callingCodes:["685"]},geometry:{type:"MultiPolygon",coordinates:[[[[-174.17905,-14.94502],[-173.13438,-14.94228],[-171.14262,-14.93704],[-171.14953,-12.4725],[-174.18596,-12.48057],[-174.17905,-14.94502]]]]}},{type:"Feature",properties:{iso1A2:"XK",iso1A3:"XKX",wikidata:"Q1246",nameEn:"Kosovo",aliases:["KV"],groups:["039","150"],isoStatus:"usrAssn",callingCodes:["383"]},geometry:{type:"MultiPolygon",coordinates:[[[[21.39045,42.74888],[21.44047,42.87276],[21.36941,42.87397],[21.32974,42.90424],[21.2719,42.8994],[21.23534,42.95523],[21.23877,43.00848],[21.2041,43.02277],[21.16734,42.99694],[21.14465,43.11089],[21.08952,43.13471],[21.05378,43.10707],[21.00749,43.13984],[20.96287,43.12416],[20.83727,43.17842],[20.88685,43.21697],[20.82145,43.26769],[20.73811,43.25068],[20.68688,43.21335],[20.59929,43.20492],[20.69515,43.09641],[20.64557,43.00826],[20.59929,43.01067],[20.48692,42.93208],[20.53484,42.8885],[20.43734,42.83157],[20.40594,42.84853],[20.35692,42.8335],[20.27869,42.81945],[20.2539,42.76245],[20.04898,42.77701],[20.02088,42.74789],[20.02915,42.71147],[20.0969,42.65559],[20.07761,42.55582],[20.17127,42.50469],[20.21797,42.41237],[20.24399,42.32168],[20.34479,42.32656],[20.3819,42.3029],[20.48857,42.25444],[20.56955,42.12097],[20.55633,42.08173],[20.59434,42.03879],[20.63069,41.94913],[20.57946,41.91593],[20.59524,41.8818],[20.68523,41.85318],[20.76786,41.91839],[20.75464,42.05229],[21.11491,42.20794],[21.16614,42.19815],[21.22728,42.08909],[21.31983,42.10993],[21.29913,42.13954],[21.30496,42.1418],[21.38428,42.24465],[21.43882,42.23609],[21.43882,42.2789],[21.50823,42.27156],[21.52145,42.24465],[21.58992,42.25915],[21.56772,42.30946],[21.5264,42.33634],[21.53467,42.36809],[21.57021,42.3647],[21.59029,42.38042],[21.62887,42.37664],[21.64209,42.41081],[21.62556,42.45106],[21.7035,42.51899],[21.70522,42.54176],[21.7327,42.55041],[21.75672,42.62695],[21.79413,42.65923],[21.75025,42.70125],[21.6626,42.67813],[21.58755,42.70418],[21.59154,42.72643],[21.47498,42.74695],[21.39045,42.74888]]]]}},{type:"Feature",properties:{iso1A2:"YE",iso1A3:"YEM",iso1N3:"887",wikidata:"Q805",nameEn:"Yemen",groups:["145","142"],callingCodes:["967"]},geometry:{type:"MultiPolygon",coordinates:[[[[53.32998,16.16312],[53.09917,16.67084],[52.81185,17.28568],[52.74267,17.29519],[52.78009,17.35124],[52.00311,19.00083],[49.04884,18.59899],[48.19996,18.20584],[47.58351,17.50366],[47.48245,17.10808],[47.00571,16.94765],[46.76494,17.29151],[46.31018,17.20464],[44.50126,17.47475],[43.70631,17.35762],[43.43005,17.56148],[43.29185,17.53224],[43.22533,17.38343],[43.32653,17.31179],[43.20156,17.25901],[43.17787,17.14717],[43.23967,17.03428],[43.18233,17.02673],[43.1813,16.98438],[43.19328,16.94703],[43.1398,16.90696],[43.18338,16.84852],[43.22012,16.83932],[43.22956,16.80613],[43.24801,16.80613],[43.26303,16.79479],[43.25857,16.75304],[43.21325,16.74416],[43.22066,16.65179],[43.15274,16.67248],[43.11601,16.53166],[42.97215,16.51093],[42.94351,16.49467],[42.94625,16.39721],[42.76801,16.40371],[42.15205,16.40211],[41.37609,16.19728],[41.29956,15.565],[42.63806,13.58268],[43.29075,12.79154],[43.32909,12.59711],[43.90659,12.3823],[50.51849,13.0483],[51.12877,12.56479],[52.253,11.68582],[55.69862,12.12478],[53.32998,16.16312]]]]}},{type:"Feature",properties:{iso1A2:"YT",iso1A3:"MYT",iso1N3:"175",wikidata:"Q17063",nameEn:"Mayotte",country:"FR",groups:["EU","014","202","002"],callingCodes:["262"]},geometry:{type:"MultiPolygon",coordinates:[[[[43.83794,-13.66915],[45.54824,-13.22353],[45.50237,-11.90315],[43.83794,-13.66915]]]]}},{type:"Feature",properties:{iso1A2:"ZA",iso1A3:"ZAF",iso1N3:"710",wikidata:"Q258",nameEn:"South Africa",groups:["018","202","002"],driveSide:"left",callingCodes:["27"]},geometry:{type:"MultiPolygon",coordinates:[[[[31.30611,-22.422],[31.16344,-22.32599],[31.08932,-22.34884],[30.86696,-22.28907],[30.6294,-22.32599],[30.48686,-22.31368],[30.38614,-22.34533],[30.28351,-22.35587],[30.2265,-22.2961],[30.13147,-22.30841],[29.92242,-22.19408],[29.76848,-22.14128],[29.64609,-22.12917],[29.37703,-22.19581],[29.21955,-22.17771],[29.18974,-22.18599],[29.15268,-22.21399],[29.10881,-22.21202],[29.0151,-22.22907],[28.91889,-22.44299],[28.63287,-22.55887],[28.34874,-22.5694],[28.04562,-22.8394],[28.04752,-22.90243],[27.93729,-22.96194],[27.93539,-23.04941],[27.74154,-23.2137],[27.6066,-23.21894],[27.52393,-23.37952],[27.33768,-23.40917],[26.99749,-23.65486],[26.84165,-24.24885],[26.51667,-24.47219],[26.46346,-24.60358],[26.39409,-24.63468],[25.8515,-24.75727],[25.84295,-24.78661],[25.88571,-24.87802],[25.72702,-25.25503],[25.69661,-25.29284],[25.6643,-25.4491],[25.58543,-25.6343],[25.33076,-25.76616],[25.12266,-25.75931],[25.01718,-25.72507],[24.8946,-25.80723],[24.67319,-25.81749],[24.44703,-25.73021],[24.36531,-25.773],[24.18287,-25.62916],[23.9244,-25.64286],[23.47588,-25.29971],[23.03497,-25.29971],[22.86012,-25.50572],[22.70808,-25.99186],[22.56365,-26.19668],[22.41921,-26.23078],[22.21206,-26.3773],[22.06192,-26.61882],[21.90703,-26.66808],[21.83291,-26.65959],[21.77114,-26.69015],[21.7854,-26.79199],[21.69322,-26.86152],[21.37869,-26.82083],[21.13353,-26.86661],[20.87031,-26.80047],[20.68596,-26.9039],[20.63275,-26.78181],[20.61754,-26.4692],[20.86081,-26.14892],[20.64795,-25.47827],[20.29826,-24.94869],[20.03678,-24.81004],[20.02809,-24.78725],[19.99817,-24.76768],[19.99882,-28.42622],[18.99885,-28.89165],[17.4579,-28.68718],[17.15405,-28.08573],[16.90446,-28.057],[16.59922,-28.53246],[16.46592,-28.57126],[16.45332,-28.63117],[12.51595,-32.27486],[38.88176,-48.03306],[34.51034,-26.91792],[32.35222,-26.86027],[32.29584,-26.852],[32.22302,-26.84136],[32.19409,-26.84032],[32.13315,-26.84345],[32.09664,-26.80721],[32.00893,-26.8096],[31.97463,-27.11057],[31.97592,-27.31675],[31.49834,-27.31549],[31.15027,-27.20151],[30.96088,-27.0245],[30.97757,-26.92706],[30.88826,-26.79622],[30.81101,-26.84722],[30.78927,-26.48271],[30.95819,-26.26303],[31.13073,-25.91558],[31.31237,-25.7431],[31.4175,-25.71886],[31.86881,-25.99973],[31.974,-25.95387],[31.92649,-25.84216],[32.00631,-25.65044],[31.97875,-25.46356],[32.01676,-25.38117],[32.03196,-25.10785],[31.9835,-24.29983],[31.90368,-24.18892],[31.87707,-23.95293],[31.77445,-23.90082],[31.70223,-23.72695],[31.67942,-23.60858],[31.56539,-23.47268],[31.55779,-23.176],[31.30611,-22.422]],[[29.33204,-29.45598],[29.28545,-29.58456],[29.12553,-29.76266],[29.16548,-29.91706],[28.9338,-30.05072],[28.80222,-30.10579],[28.68627,-30.12885],[28.399,-30.1592],[28.2319,-30.28476],[28.12073,-30.68072],[27.74814,-30.60635],[27.69467,-30.55862],[27.67819,-30.53437],[27.6521,-30.51707],[27.62137,-30.50509],[27.56781,-30.44562],[27.56901,-30.42504],[27.45452,-30.32239],[27.38108,-30.33456],[27.36649,-30.27246],[27.37293,-30.19401],[27.40778,-30.14577],[27.32555,-30.14785],[27.29603,-30.05473],[27.22719,-30.00718],[27.09489,-29.72796],[27.01016,-29.65439],[27.33464,-29.48161],[27.4358,-29.33465],[27.47254,-29.31968],[27.45125,-29.29708],[27.48679,-29.29349],[27.54258,-29.25575],[27.5158,-29.2261],[27.55974,-29.18954],[27.75458,-28.89839],[27.8907,-28.91612],[27.88933,-28.88156],[27.9392,-28.84864],[27.98675,-28.8787],[28.02503,-28.85991],[28.1317,-28.7293],[28.2348,-28.69471],[28.30518,-28.69531],[28.40612,-28.6215],[28.65091,-28.57025],[28.68043,-28.58744],[29.40524,-29.21246],[29.44883,-29.3772],[29.33204,-29.45598]]]]}},{type:"Feature",properties:{iso1A2:"ZM",iso1A3:"ZMB",iso1N3:"894",wikidata:"Q953",nameEn:"Zambia",groups:["014","202","002"],driveSide:"left",callingCodes:["260"]},geometry:{type:"MultiPolygon",coordinates:[[[[32.95389,-9.40138],[32.76233,-9.31963],[32.75611,-9.28583],[32.53661,-9.24281],[32.49147,-9.14754],[32.43543,-9.11988],[32.25486,-9.13371],[32.16146,-9.05993],[32.08206,-9.04609],[31.98866,-9.07069],[31.94196,-9.02303],[31.94663,-8.93846],[31.81587,-8.88618],[31.71158,-8.91386],[31.57147,-8.81388],[31.57147,-8.70619],[31.37533,-8.60769],[31.00796,-8.58615],[30.79243,-8.27382],[28.88917,-8.4831],[28.9711,-8.66935],[28.38526,-9.23393],[28.36562,-9.30091],[28.52636,-9.35379],[28.51627,-9.44726],[28.56208,-9.49122],[28.68532,-9.78],[28.62795,-9.92942],[28.65032,-10.65133],[28.37241,-11.57848],[28.48357,-11.87532],[29.18592,-12.37921],[29.4992,-12.43843],[29.48404,-12.23604],[29.8139,-12.14898],[29.81551,-13.44683],[29.65078,-13.41844],[29.60531,-13.21685],[29.01918,-13.41353],[28.33199,-12.41375],[27.59932,-12.22123],[27.21025,-11.76157],[27.22541,-11.60323],[27.04351,-11.61312],[26.88687,-12.01868],[26.01777,-11.91488],[25.33058,-11.65767],[25.34069,-11.19707],[24.42612,-11.44975],[24.34528,-11.06816],[24.00027,-10.89356],[24.02603,-11.15368],[23.98804,-12.13149],[24.06672,-12.29058],[23.90937,-12.844],[24.03339,-12.99091],[21.97988,-13.00148],[22.00323,-16.18028],[22.17217,-16.50269],[23.20038,-17.47563],[23.47474,-17.62877],[24.23619,-17.47489],[24.32811,-17.49082],[24.38712,-17.46818],[24.5621,-17.52963],[24.70864,-17.49501],[25.00198,-17.58221],[25.26433,-17.79571],[25.51646,-17.86232],[25.6827,-17.81987],[25.85738,-17.91403],[25.85892,-17.97726],[26.08925,-17.98168],[26.0908,-17.93021],[26.21601,-17.88608],[26.55918,-17.99638],[26.68403,-18.07411],[26.74314,-18.0199],[26.89926,-17.98756],[27.14196,-17.81398],[27.30736,-17.60487],[27.61377,-17.34378],[27.62795,-17.24365],[27.83141,-16.96274],[28.73725,-16.5528],[28.76199,-16.51575],[28.81454,-16.48611],[28.8501,-16.04537],[28.9243,-15.93987],[29.01298,-15.93805],[29.21955,-15.76589],[29.4437,-15.68702],[29.8317,-15.6126],[30.35574,-15.6513],[30.41902,-15.62269],[30.22098,-14.99447],[33.24249,-14.00019],[33.16749,-13.93992],[33.07568,-13.98447],[33.02977,-14.05022],[32.99042,-13.95689],[32.88985,-13.82956],[32.79015,-13.80755],[32.76962,-13.77224],[32.84528,-13.71576],[32.7828,-13.64805],[32.68654,-13.64268],[32.66468,-13.60019],[32.68436,-13.55769],[32.73683,-13.57682],[32.84176,-13.52794],[32.86113,-13.47292],[33.0078,-13.19492],[32.98289,-13.12671],[33.02181,-12.88707],[32.96733,-12.88251],[32.94397,-12.76868],[33.05917,-12.59554],[33.18837,-12.61377],[33.28177,-12.54692],[33.37517,-12.54085],[33.54485,-12.35996],[33.47636,-12.32498],[33.3705,-12.34931],[33.25998,-12.14242],[33.33937,-11.91252],[33.32692,-11.59248],[33.24252,-11.59302],[33.23663,-11.40637],[33.29267,-11.43536],[33.29267,-11.3789],[33.39697,-11.15296],[33.25998,-10.88862],[33.28022,-10.84428],[33.47636,-10.78465],[33.70675,-10.56896],[33.54797,-10.36077],[33.53863,-10.20148],[33.31297,-10.05133],[33.37902,-9.9104],[33.36581,-9.81063],[33.31517,-9.82364],[33.2095,-9.61099],[33.12144,-9.58929],[33.10163,-9.66525],[33.05485,-9.61316],[33.00256,-9.63053],[33.00476,-9.5133],[32.95389,-9.40138]]]]}},{type:"Feature",properties:{iso1A2:"ZW",iso1A3:"ZWE",iso1N3:"716",wikidata:"Q954",nameEn:"Zimbabwe",groups:["014","202","002"],driveSide:"left",callingCodes:["263"]},geometry:{type:"MultiPolygon",coordinates:[[[[30.41902,-15.62269],[30.35574,-15.6513],[29.8317,-15.6126],[29.4437,-15.68702],[29.21955,-15.76589],[29.01298,-15.93805],[28.9243,-15.93987],[28.8501,-16.04537],[28.81454,-16.48611],[28.76199,-16.51575],[28.73725,-16.5528],[27.83141,-16.96274],[27.62795,-17.24365],[27.61377,-17.34378],[27.30736,-17.60487],[27.14196,-17.81398],[26.89926,-17.98756],[26.74314,-18.0199],[26.68403,-18.07411],[26.55918,-17.99638],[26.21601,-17.88608],[26.0908,-17.93021],[26.08925,-17.98168],[25.85892,-17.97726],[25.85738,-17.91403],[25.6827,-17.81987],[25.51646,-17.86232],[25.26433,-17.79571],[25.23909,-17.90832],[25.31799,-18.07091],[25.39972,-18.12691],[25.53465,-18.39041],[25.68859,-18.56165],[25.79217,-18.6355],[25.82353,-18.82808],[25.94326,-18.90362],[25.99837,-19.02943],[25.96226,-19.08152],[26.17227,-19.53709],[26.72246,-19.92707],[27.21278,-20.08244],[27.29831,-20.28935],[27.28865,-20.49873],[27.69361,-20.48531],[27.72972,-20.51735],[27.69171,-21.08409],[27.91407,-21.31621],[28.01669,-21.57624],[28.29416,-21.59037],[28.49942,-21.66634],[28.58114,-21.63455],[29.07763,-21.81877],[29.04023,-21.85864],[29.02191,-21.90647],[29.02191,-21.95665],[29.04108,-22.00563],[29.08495,-22.04867],[29.14501,-22.07275],[29.1974,-22.07472],[29.24648,-22.05967],[29.3533,-22.18363],[29.37703,-22.19581],[29.64609,-22.12917],[29.76848,-22.14128],[29.92242,-22.19408],[30.13147,-22.30841],[30.2265,-22.2961],[30.28351,-22.35587],[30.38614,-22.34533],[30.48686,-22.31368],[30.6294,-22.32599],[30.86696,-22.28907],[31.08932,-22.34884],[31.16344,-22.32599],[31.30611,-22.422],[31.38336,-22.36919],[32.41234,-21.31246],[32.48236,-21.32873],[32.37115,-21.133],[32.51644,-20.91929],[32.48122,-20.63319],[32.55167,-20.56312],[32.66174,-20.56106],[32.85987,-20.27841],[32.85987,-20.16686],[32.93032,-20.03868],[33.01178,-20.02007],[33.06461,-19.77787],[32.95013,-19.67219],[32.84666,-19.68462],[32.84446,-19.48343],[32.78282,-19.47513],[32.77966,-19.36098],[32.85107,-19.29238],[32.87088,-19.09279],[32.84006,-19.0262],[32.72118,-19.02204],[32.69917,-18.94293],[32.73439,-18.92628],[32.70137,-18.84712],[32.82465,-18.77419],[32.9017,-18.7992],[32.95013,-18.69079],[32.88629,-18.58023],[32.88629,-18.51344],[33.02278,-18.4696],[33.03159,-18.35054],[32.94133,-17.99705],[33.0492,-17.60298],[32.98536,-17.55891],[32.96554,-17.48964],[33.0426,-17.3468],[33.00517,-17.30477],[32.96554,-17.11971],[32.84113,-16.92259],[32.91051,-16.89446],[32.97655,-16.70689],[32.78943,-16.70267],[32.69917,-16.66893],[32.71017,-16.59932],[32.42838,-16.4727],[32.28529,-16.43892],[32.02772,-16.43892],[31.91324,-16.41569],[31.90223,-16.34388],[31.67988,-16.19595],[31.42451,-16.15154],[31.30563,-16.01193],[31.13171,-15.98019],[30.97761,-16.05848],[30.91597,-15.99924],[30.42568,-15.9962],[30.41902,-15.62269]]]]}}];
-       var rawBorders = {
-       type: type,
-       features: features
-       };
+       var _layerEnabled$1 = false;
 
-       var borders = rawBorders;
-       var whichPolygonGetter = {};
-       var featuresByCode = {};
-       var idFilterRegex = /\bThe\b|\bthe\b|\band\b|\bof\b|[-_ .,()&[\]/]/g;
-       var levels = ['subterritory', 'territory', 'country', 'intermediateRegion', 'subregion', 'region', 'union', 'world'];
-       loadDerivedDataAndCaches(borders);
+       var _qaService$1;
 
-       function loadDerivedDataAndCaches(borders) {
-         var identifierProps = ['iso1A2', 'iso1A3', 'm49', 'wikidata', 'emojiFlag', 'nameEn'];
-         var geometryFeatures = [];
+       function svgImproveOSM(projection, context, dispatch) {
+         var throttledRedraw = throttle(function () {
+           return dispatch.call('change');
+         }, 1000);
 
-         for (var i in borders.features) {
-           var _feature = borders.features[i];
-           _feature.properties.id = _feature.properties.iso1A2 || _feature.properties.m49;
-           loadM49(_feature);
-           loadIsoStatus(_feature);
-           loadLevel(_feature);
-           loadGroups(_feature);
-           loadRoadSpeedUnit(_feature);
-           loadDriveSide(_feature);
-           loadFlag(_feature);
-           cacheFeatureByIDs(_feature);
-           if (_feature.geometry) geometryFeatures.push(_feature);
-         }
+         var minZoom = 12;
+         var touchLayer = select(null);
+         var drawLayer = select(null);
+         var layerVisible = false;
+
+         function markerPath(selection, klass) {
+           selection.attr('class', klass).attr('transform', 'translate(-10, -28)').attr('points', '16,3 4,3 1,6 1,17 4,20 7,20 10,27 13,20 16,20 19,17.033 19,6');
+         } // Loosely-coupled improveOSM service for fetching issues
 
-         for (var _i in borders.features) {
-           var _feature2 = borders.features[_i];
 
-           _feature2.properties.groups.sort(function (groupID1, groupID2) {
-             return levels.indexOf(featuresByCode[groupID1].properties.level) - levels.indexOf(featuresByCode[groupID2].properties.level);
-           });
+         function getService() {
+           if (services.improveOSM && !_qaService$1) {
+             _qaService$1 = services.improveOSM;
 
-           loadMembersForGroupsOf(_feature2);
-         }
+             _qaService$1.on('loaded', throttledRedraw);
+           } else if (!services.improveOSM && _qaService$1) {
+             _qaService$1 = null;
+           }
 
-         var geometryOnlyCollection = {
-           type: 'RegionFeatureCollection',
-           features: geometryFeatures
-         };
-         whichPolygonGetter = whichPolygon_1(geometryOnlyCollection);
+           return _qaService$1;
+         } // Show the markers
 
-         function loadGroups(feature) {
-           var props = feature.properties;
 
-           if (!props.groups) {
-             props.groups = [];
+         function editOn() {
+           if (!layerVisible) {
+             layerVisible = true;
+             drawLayer.style('display', 'block');
            }
+         } // Immediately remove the markers and their touch targets
 
-           if (props.country) {
-             props.groups.push(props.country);
-           }
 
-           if (props.m49 !== '001') {
-             props.groups.push('001');
+         function editOff() {
+           if (layerVisible) {
+             layerVisible = false;
+             drawLayer.style('display', 'none');
+             drawLayer.selectAll('.qaItem.improveOSM').remove();
+             touchLayer.selectAll('.qaItem.improveOSM').remove();
            }
-         }
+         } // Enable the layer.  This shows the markers and transitions them to visible.
 
-         function loadM49(feature) {
-           var props = feature.properties;
 
-           if (!props.m49 && props.iso1N3) {
-             props.m49 = props.iso1N3;
-           }
-         }
+         function layerOn() {
+           editOn();
+           drawLayer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end interrupt', function () {
+             return dispatch.call('change');
+           });
+         } // Disable the layer.  This transitions the layer invisible and then hides the markers.
 
-         function loadIsoStatus(feature) {
-           var props = feature.properties;
 
-           if (!props.isoStatus && props.iso1A2) {
-             props.isoStatus = 'official';
-           }
-         }
+         function layerOff() {
+           throttledRedraw.cancel();
+           drawLayer.interrupt();
+           touchLayer.selectAll('.qaItem.improveOSM').remove();
+           drawLayer.transition().duration(250).style('opacity', 0).on('end interrupt', function () {
+             editOff();
+             dispatch.call('change');
+           });
+         } // Update the issue markers
 
-         function loadLevel(feature) {
-           var props = feature.properties;
-           if (props.level) return;
 
-           if (!props.country) {
-             props.level = 'country';
-           } else if (props.isoStatus === 'official') {
-             props.level = 'territory';
-           } else {
-             props.level = 'subterritory';
-           }
-         }
+         function updateMarkers() {
+           if (!layerVisible || !_layerEnabled$1) return;
+           var service = getService();
+           var selectedID = context.selectedErrorID();
+           var data = service ? service.getItems(projection) : [];
+           var getTransform = svgPointTransform(projection); // Draw markers..
 
-         function loadRoadSpeedUnit(feature) {
-           var props = feature.properties;
+           var markers = drawLayer.selectAll('.qaItem.improveOSM').data(data, function (d) {
+             return d.id;
+           }); // exit
 
-           if (props.roadSpeedUnit === undefined && props.iso1A2 && props.iso1A2 !== 'EU') {
-             props.roadSpeedUnit = 'km/h';
-           }
-         }
+           markers.exit().remove(); // enter
+
+           var markersEnter = markers.enter().append('g').attr('class', function (d) {
+             return "qaItem ".concat(d.service, " itemId-").concat(d.id, " itemType-").concat(d.itemType);
+           });
+           markersEnter.append('polygon').call(markerPath, 'shadow');
+           markersEnter.append('ellipse').attr('cx', 0).attr('cy', 0).attr('rx', 4.5).attr('ry', 2).attr('class', 'stroke');
+           markersEnter.append('polygon').attr('fill', 'currentColor').call(markerPath, 'qaItem-fill');
+           markersEnter.append('use').attr('transform', 'translate(-6.5, -23)').attr('class', 'icon-annotation').attr('width', '13px').attr('height', '13px').attr('xlink:href', function (d) {
+             var picon = d.icon;
+
+             if (!picon) {
+               return '';
+             } else {
+               var isMaki = /^maki-/.test(picon);
+               return "#".concat(picon).concat(isMaki ? '-11' : '');
+             }
+           }); // update
+
+           markers.merge(markersEnter).sort(sortY).classed('selected', function (d) {
+             return d.id === selectedID;
+           }).attr('transform', getTransform); // Draw targets..
+
+           if (touchLayer.empty()) return;
+           var fillClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
+           var targets = touchLayer.selectAll('.qaItem.improveOSM').data(data, function (d) {
+             return d.id;
+           }); // exit
+
+           targets.exit().remove(); // enter/update
 
-         function loadDriveSide(feature) {
-           var props = feature.properties;
+           targets.enter().append('rect').attr('width', '20px').attr('height', '30px').attr('x', '-10px').attr('y', '-28px').merge(targets).sort(sortY).attr('class', function (d) {
+             return "qaItem ".concat(d.service, " target ").concat(fillClass, " itemId-").concat(d.id);
+           }).attr('transform', getTransform);
 
-           if (props.driveSide === undefined && props.iso1A2 && props.iso1A2 !== 'EU') {
-             props.driveSide = 'right';
+           function sortY(a, b) {
+             return a.id === selectedID ? 1 : b.id === selectedID ? -1 : b.loc[1] - a.loc[1];
            }
-         }
+         } // Draw the ImproveOSM layer and schedule loading issues and updating markers.
 
-         function loadFlag(feature) {
-           if (!feature.properties.iso1A2) return;
-           var flag = feature.properties.iso1A2.replace(/./g, function (_char) {
-             return String.fromCodePoint(_char.charCodeAt(0) + 127397);
-           });
-           feature.properties.emojiFlag = flag;
-         }
 
-         function loadMembersForGroupsOf(feature) {
-           var featureID = feature.properties.id;
-           var standardizedGroupIDs = [];
+         function drawImproveOSM(selection) {
+           var service = getService();
+           var surface = context.surface();
 
-           for (var j in feature.properties.groups) {
-             var groupID = feature.properties.groups[j];
-             var groupFeature = featuresByCode[groupID];
-             standardizedGroupIDs.push(groupFeature.properties.id);
+           if (surface && !surface.empty()) {
+             touchLayer = surface.selectAll('.data-layer.touch .layer-touch.markers');
+           }
 
-             if (groupFeature.properties.members) {
-               groupFeature.properties.members.push(featureID);
+           drawLayer = selection.selectAll('.layer-improveOSM').data(service ? [0] : []);
+           drawLayer.exit().remove();
+           drawLayer = drawLayer.enter().append('g').attr('class', 'layer-improveOSM').style('display', _layerEnabled$1 ? 'block' : 'none').merge(drawLayer);
+
+           if (_layerEnabled$1) {
+             if (service && ~~context.map().zoom() >= minZoom) {
+               editOn();
+               service.loadIssues(projection);
+               updateMarkers();
              } else {
-               groupFeature.properties.members = [featureID];
+               editOff();
              }
            }
+         } // Toggles the layer on and off
 
-           feature.properties.groups = standardizedGroupIDs;
-         }
 
-         function cacheFeatureByIDs(feature) {
-           for (var k in identifierProps) {
-             var prop = identifierProps[k];
-             var id = prop && feature.properties[prop];
+         drawImproveOSM.enabled = function (val) {
+           if (!arguments.length) return _layerEnabled$1;
+           _layerEnabled$1 = val;
 
-             if (id) {
-               id = id.replace(idFilterRegex, '').toUpperCase();
-               featuresByCode[id] = feature;
-             }
-           }
+           if (_layerEnabled$1) {
+             layerOn();
+           } else {
+             layerOff();
 
-           if (feature.properties.aliases) {
-             for (var j in feature.properties.aliases) {
-               var alias = feature.properties.aliases[j].replace(idFilterRegex, '').toUpperCase();
-               featuresByCode[alias] = feature;
+             if (context.selectedErrorID()) {
+               context.enter(modeBrowse(context));
              }
            }
-         }
-       }
-
-       function locArray(loc) {
-         if (Array.isArray(loc)) {
-           return loc;
-         } else if (loc.coordinates) {
-           return loc.coordinates;
-         }
 
-         return loc.geometry.coordinates;
-       }
+           dispatch.call('change');
+           return this;
+         };
 
-       function smallestFeature(loc) {
-         var query = locArray(loc);
-         var featureProperties = whichPolygonGetter(query);
-         if (!featureProperties) return null;
-         return featuresByCode[featureProperties.id];
-       }
+         drawImproveOSM.supported = function () {
+           return !!getService();
+         };
 
-       function countryFeature(loc) {
-         var feature = smallestFeature(loc);
-         if (!feature) return null;
-         var countryCode = feature.properties.country || feature.properties.iso1A2;
-         return featuresByCode[countryCode];
+         return drawImproveOSM;
        }
 
-       function featureForLoc(loc, opts) {
-         if (opts && opts.level && opts.level !== 'country') {
-           var features = featuresContaining(loc);
-           var targetLevel = opts.level;
-           var targetLevelIndex = levels.indexOf(targetLevel);
-           if (targetLevelIndex === -1) return null;
+       var _layerEnabled = false;
 
-           for (var i in features) {
-             var _feature3 = features[i];
+       var _qaService;
 
-             if (_feature3.properties.level === targetLevel || levels.indexOf(_feature3.properties.level) > targetLevelIndex) {
-               return _feature3;
-             }
-           }
+       function svgOsmose(projection, context, dispatch) {
+         var throttledRedraw = throttle(function () {
+           return dispatch.call('change');
+         }, 1000);
 
-           return null;
-         }
+         var minZoom = 12;
+         var touchLayer = select(null);
+         var drawLayer = select(null);
+         var layerVisible = false;
 
-         return countryFeature(loc);
-       }
+         function markerPath(selection, klass) {
+           selection.attr('class', klass).attr('transform', 'translate(-10, -28)').attr('points', '16,3 4,3 1,6 1,17 4,20 7,20 10,27 13,20 16,20 19,17.033 19,6');
+         } // Loosely-coupled osmose service for fetching issues
 
-       function featureForID(id) {
-         var stringID;
 
-         if (typeof id === 'number') {
-           stringID = id.toString();
+         function getService() {
+           if (services.osmose && !_qaService) {
+             _qaService = services.osmose;
 
-           if (stringID.length === 1) {
-             stringID = '00' + stringID;
-           } else if (stringID.length === 2) {
-             stringID = '0' + stringID;
+             _qaService.on('loaded', throttledRedraw);
+           } else if (!services.osmose && _qaService) {
+             _qaService = null;
            }
-         } else {
-           stringID = id.replace(idFilterRegex, '').toUpperCase();
-         }
 
-         return featuresByCode[stringID] || null;
-       }
+           return _qaService;
+         } // Show the markers
 
-       function smallestOrMatchingFeature(query) {
-         if (_typeof(query) === 'object') {
-           return smallestFeature(query);
-         }
 
-         return featureForID(query);
-       }
+         function editOn() {
+           if (!layerVisible) {
+             layerVisible = true;
+             drawLayer.style('display', 'block');
+           }
+         } // Immediately remove the markers and their touch targets
 
-       function feature$1(query, opts) {
-         if (_typeof(query) === 'object') {
-           return featureForLoc(query, opts);
-         }
 
-         return featureForID(query);
-       }
-       function iso1A2Code(query, opts) {
-         var match = feature$1(query, opts);
-         if (!match) return null;
-         return match.properties.iso1A2 || null;
-       }
-       function featuresContaining(query, strict) {
-         var feature = smallestOrMatchingFeature(query);
-         if (!feature) return [];
-         var features = [];
+         function editOff() {
+           if (layerVisible) {
+             layerVisible = false;
+             drawLayer.style('display', 'none');
+             drawLayer.selectAll('.qaItem.osmose').remove();
+             touchLayer.selectAll('.qaItem.osmose').remove();
+           }
+         } // Enable the layer.  This shows the markers and transitions them to visible.
 
-         if (!strict || _typeof(query) === 'object') {
-           features.push(feature);
-         }
 
-         var properties = feature.properties;
+         function layerOn() {
+           editOn();
+           drawLayer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end interrupt', function () {
+             return dispatch.call('change');
+           });
+         } // Disable the layer.  This transitions the layer invisible and then hides the markers.
 
-         for (var i in properties.groups) {
-           var groupID = properties.groups[i];
-           features.push(featuresByCode[groupID]);
-         }
 
-         return features;
-       }
-       function roadSpeedUnit(query) {
-         var feature = smallestOrMatchingFeature(query);
-         return feature && feature.properties.roadSpeedUnit || null;
-       }
+         function layerOff() {
+           throttledRedraw.cancel();
+           drawLayer.interrupt();
+           touchLayer.selectAll('.qaItem.osmose').remove();
+           drawLayer.transition().duration(250).style('opacity', 0).on('end interrupt', function () {
+             editOff();
+             dispatch.call('change');
+           });
+         } // Update the issue markers
+
 
-       var _dataDeprecated;
+         function updateMarkers() {
+           if (!layerVisible || !_layerEnabled) return;
+           var service = getService();
+           var selectedID = context.selectedErrorID();
+           var data = service ? service.getItems(projection) : [];
+           var getTransform = svgPointTransform(projection); // Draw markers..
 
-       var _nsi;
+           var markers = drawLayer.selectAll('.qaItem.osmose').data(data, function (d) {
+             return d.id;
+           }); // exit
 
-       function validationOutdatedTags() {
-         var type = 'outdated_tags';
-         var nsiKeys = ['amenity', 'shop', 'tourism', 'leisure', 'office']; // A concern here in switching to async data means that `_dataDeprecated`
-         // and `_nsi` will not be available at first, so the data on early tiles
-         // may not have tags validated fully.
-         // initialize deprecated tags array
+           markers.exit().remove(); // enter
 
-         _mainFileFetcher.get('deprecated').then(function (d) {
-           return _dataDeprecated = d;
-         })["catch"](function () {
-           /* ignore */
-         });
-         _mainFileFetcher.get('nsi_brands').then(function (d) {
-           _nsi = {
-             brands: d.brands,
-             matcher: matcher$1(),
-             wikidata: {},
-             wikipedia: {}
-           }; // initialize name-suggestion-index matcher
+           var markersEnter = markers.enter().append('g').attr('class', function (d) {
+             return "qaItem ".concat(d.service, " itemId-").concat(d.id, " itemType-").concat(d.itemType);
+           });
+           markersEnter.append('polygon').call(markerPath, 'shadow');
+           markersEnter.append('ellipse').attr('cx', 0).attr('cy', 0).attr('rx', 4.5).attr('ry', 2).attr('class', 'stroke');
+           markersEnter.append('polygon').attr('fill', function (d) {
+             return service.getColor(d.item);
+           }).call(markerPath, 'qaItem-fill');
+           markersEnter.append('use').attr('transform', 'translate(-6.5, -23)').attr('class', 'icon-annotation').attr('width', '13px').attr('height', '13px').attr('xlink:href', function (d) {
+             var picon = d.icon;
 
-           _nsi.matcher.buildMatchIndex(d.brands); // index all known wikipedia and wikidata tags
+             if (!picon) {
+               return '';
+             } else {
+               var isMaki = /^maki-/.test(picon);
+               return "#".concat(picon).concat(isMaki ? '-11' : '');
+             }
+           }); // update
 
+           markers.merge(markersEnter).sort(sortY).classed('selected', function (d) {
+             return d.id === selectedID;
+           }).attr('transform', getTransform); // Draw targets..
 
-           Object.keys(d.brands).forEach(function (kvnd) {
-             var brand = d.brands[kvnd];
-             var wd = brand.tags['brand:wikidata'];
-             var wp = brand.tags['brand:wikipedia'];
+           if (touchLayer.empty()) return;
+           var fillClass = context.getDebug('target') ? 'pink' : 'nocolor';
+           var targets = touchLayer.selectAll('.qaItem.osmose').data(data, function (d) {
+             return d.id;
+           }); // exit
 
-             if (wd) {
-               _nsi.wikidata[wd] = kvnd;
-             }
+           targets.exit().remove(); // enter/update
 
-             if (wp) {
-               _nsi.wikipedia[wp] = kvnd;
-             }
-           });
-           return _nsi;
-         })["catch"](function () {
-           /* ignore */
-         });
+           targets.enter().append('rect').attr('width', '20px').attr('height', '30px').attr('x', '-10px').attr('y', '-28px').merge(targets).sort(sortY).attr('class', function (d) {
+             return "qaItem ".concat(d.service, " target ").concat(fillClass, " itemId-").concat(d.id);
+           }).attr('transform', getTransform);
 
-         function oldTagIssues(entity, graph) {
-           var oldTags = Object.assign({}, entity.tags); // shallow copy
+           function sortY(a, b) {
+             return a.id === selectedID ? 1 : b.id === selectedID ? -1 : b.loc[1] - a.loc[1];
+           }
+         } // Draw the Osmose layer and schedule loading issues and updating markers.
 
-           var preset = _mainPresetIndex.match(entity, graph);
-           var subtype = 'deprecated_tags';
-           if (!preset) return []; // upgrade preset..
 
-           if (preset.replacement) {
-             var newPreset = _mainPresetIndex.item(preset.replacement);
-             graph = actionChangePreset(entity.id, preset, newPreset, true
-             /* skip field defaults */
-             )(graph);
-             entity = graph.entity(entity.id);
-             preset = newPreset;
-           } // upgrade tags..
+         function drawOsmose(selection) {
+           var service = getService();
+           var surface = context.surface();
 
+           if (surface && !surface.empty()) {
+             touchLayer = surface.selectAll('.data-layer.touch .layer-touch.markers');
+           }
 
-           if (_dataDeprecated) {
-             var deprecatedTags = entity.deprecatedTags(_dataDeprecated);
+           drawLayer = selection.selectAll('.layer-osmose').data(service ? [0] : []);
+           drawLayer.exit().remove();
+           drawLayer = drawLayer.enter().append('g').attr('class', 'layer-osmose').style('display', _layerEnabled ? 'block' : 'none').merge(drawLayer);
 
-             if (deprecatedTags.length) {
-               deprecatedTags.forEach(function (tag) {
-                 graph = actionUpgradeTags(entity.id, tag.old, tag.replace)(graph);
-               });
-               entity = graph.entity(entity.id);
+           if (_layerEnabled) {
+             if (service && ~~context.map().zoom() >= minZoom) {
+               editOn();
+               service.loadIssues(projection);
+               updateMarkers();
+             } else {
+               editOff();
              }
-           } // add missing addTags..
+           }
+         } // Toggles the layer on and off
 
 
-           var newTags = Object.assign({}, entity.tags); // shallow copy
+         drawOsmose.enabled = function (val) {
+           if (!arguments.length) return _layerEnabled;
+           _layerEnabled = val;
 
-           if (preset.tags !== preset.addTags) {
-             Object.keys(preset.addTags).forEach(function (k) {
-               if (!newTags[k]) {
-                 if (preset.addTags[k] === '*') {
-                   newTags[k] = 'yes';
-                 } else {
-                   newTags[k] = preset.addTags[k];
-                 }
-               }
+           if (_layerEnabled) {
+             // Strings supplied by Osmose fetched before showing layer for first time
+             // NOTE: Currently no way to change locale in iD at runtime, would need to re-call this method if that's ever implemented
+             // Also, If layer is toggled quickly multiple requests are sent
+             getService().loadStrings().then(layerOn)["catch"](function (err) {
+               console.log(err); // eslint-disable-line no-console
              });
+           } else {
+             layerOff();
+
+             if (context.selectedErrorID()) {
+               context.enter(modeBrowse(context));
+             }
            }
 
-           if (_nsi) {
-             // Do `wikidata` or `wikipedia` identify this entity as a brand?  #6416
-             // If so, these tags can be swapped to `brand:wikidata`/`brand:wikipedia`
-             var isBrand;
+           dispatch.call('change');
+           return this;
+         };
 
-             if (newTags.wikidata) {
-               // try matching `wikidata`
-               isBrand = _nsi.wikidata[newTags.wikidata];
-             }
+         drawOsmose.supported = function () {
+           return !!getService();
+         };
 
-             if (!isBrand && newTags.wikipedia) {
-               // fallback to `wikipedia`
-               isBrand = _nsi.wikipedia[newTags.wikipedia];
-             }
+         return drawOsmose;
+       }
 
-             if (isBrand && !newTags.office) {
-               // but avoid doing this for corporate offices
-               if (newTags.wikidata) {
-                 newTags['brand:wikidata'] = newTags.wikidata;
-                 delete newTags.wikidata;
-               }
+       function svgStreetside(projection, context, dispatch) {
+         var throttledRedraw = throttle(function () {
+           dispatch.call('change');
+         }, 1000);
 
-               if (newTags.wikipedia) {
-                 newTags['brand:wikipedia'] = newTags.wikipedia;
-                 delete newTags.wikipedia;
-               } // I considered setting `name` and other tags here, but they aren't unique per wikidata
-               // (Q2759586 -> in USA "Papa John's", in Russia "Папа Джонс")
-               // So users will really need to use a preset or assign `name` themselves.
+         var minZoom = 14;
+         var minMarkerZoom = 16;
+         var minViewfieldZoom = 18;
+         var layer = select(null);
+         var _viewerYaw = 0;
+         var _selectedSequence = null;
 
-             } // try key/value|name match against name-suggestion-index
+         var _streetside;
+         /**
+          * init().
+          */
 
 
-             if (newTags.name) {
-               for (var i = 0; i < nsiKeys.length; i++) {
-                 var k = nsiKeys[i];
-                 if (!newTags[k]) continue;
-                 var center = entity.extent(graph).center();
-                 var countryCode = iso1A2Code(center);
+         function init() {
+           if (svgStreetside.initialized) return; // run once
 
-                 var match = _nsi.matcher.matchKVN(k, newTags[k], newTags.name, countryCode && countryCode.toLowerCase());
+           svgStreetside.enabled = false;
+           svgStreetside.initialized = true;
+         }
+         /**
+          * getService().
+          */
 
-                 if (!match) continue; // for now skip ambiguous matches (like Target~(USA) vs Target~(Australia))
 
-                 if (match.d) continue;
-                 var brand = _nsi.brands[match.kvnd];
+         function getService() {
+           if (services.streetside && !_streetside) {
+             _streetside = services.streetside;
 
-                 if (brand && brand.tags['brand:wikidata'] && brand.tags['brand:wikidata'] !== entity.tags['not:brand:wikidata']) {
-                   subtype = 'noncanonical_brand';
-                   var keepTags = ['takeaway'].reduce(function (acc, k) {
-                     if (newTags[k]) {
-                       acc[k] = newTags[k];
-                     }
+             _streetside.event.on('viewerChanged.svgStreetside', viewerChanged).on('loadedImages.svgStreetside', throttledRedraw);
+           } else if (!services.streetside && _streetside) {
+             _streetside = null;
+           }
 
-                     return acc;
-                   }, {});
-                   nsiKeys.forEach(function (k) {
-                     return delete newTags[k];
-                   });
-                   Object.assign(newTags, brand.tags, keepTags);
-                   break;
-                 }
-               }
-             }
-           } // determine diff
+           return _streetside;
+         }
+         /**
+          * showLayer().
+          */
 
 
-           var tagDiff = utilTagDiff(oldTags, newTags);
-           if (!tagDiff.length) return [];
-           var isOnlyAddingTags = tagDiff.every(function (d) {
-             return d.type === '+';
+         function showLayer() {
+           var service = getService();
+           if (!service) return;
+           editOn();
+           layer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end', function () {
+             dispatch.call('change');
            });
-           var prefix = '';
+         }
+         /**
+          * hideLayer().
+          */
 
-           if (subtype === 'noncanonical_brand') {
-             prefix = 'noncanonical_brand.';
-           } else if (subtype === 'deprecated_tags' && isOnlyAddingTags) {
-             subtype = 'incomplete_tags';
-             prefix = 'incomplete.';
-           } // don't allow autofixing brand tags
 
+         function hideLayer() {
+           throttledRedraw.cancel();
+           layer.transition().duration(250).style('opacity', 0).on('end', editOff);
+         }
+         /**
+          * editOn().
+          */
 
-           var autoArgs = subtype !== 'noncanonical_brand' ? [doUpgrade, _t('issues.fix.upgrade_tags.annotation')] : null;
-           return [new validationIssue({
-             type: type,
-             subtype: subtype,
-             severity: 'warning',
-             message: showMessage,
-             reference: showReference,
-             entityIds: [entity.id],
-             hash: JSON.stringify(tagDiff),
-             dynamicFixes: function dynamicFixes() {
-               return [new validationIssueFix({
-                 autoArgs: autoArgs,
-                 title: _t.html('issues.fix.upgrade_tags.title'),
-                 onClick: function onClick(context) {
-                   context.perform(doUpgrade, _t('issues.fix.upgrade_tags.annotation'));
-                 }
-               })];
-             }
-           })];
 
-           function doUpgrade(graph) {
-             var currEntity = graph.hasEntity(entity.id);
-             if (!currEntity) return graph;
-             var newTags = Object.assign({}, currEntity.tags); // shallow copy
+         function editOn() {
+           layer.style('display', 'block');
+         }
+         /**
+          * editOff().
+          */
 
-             tagDiff.forEach(function (diff) {
-               if (diff.type === '-') {
-                 delete newTags[diff.key];
-               } else if (diff.type === '+') {
-                 newTags[diff.key] = diff.newVal;
-               }
-             });
-             return actionChangeTags(currEntity.id, newTags)(graph);
-           }
 
-           function showMessage(context) {
-             var currEntity = context.hasEntity(entity.id);
-             if (!currEntity) return '';
-             var messageID = "issues.outdated_tags.".concat(prefix, "message");
+         function editOff() {
+           layer.selectAll('.viewfield-group').remove();
+           layer.style('display', 'none');
+         }
+         /**
+          * click() Handles 'bubble' point click event.
+          */
 
-             if (subtype === 'noncanonical_brand' && isOnlyAddingTags) {
-               messageID += '_incomplete';
-             }
 
-             return _t.html(messageID, {
-               feature: utilDisplayLabel(currEntity, context.graph())
-             });
-           }
+         function click(d3_event, d) {
+           var service = getService();
+           if (!service) return; // try to preserve the viewer rotation when staying on the same sequence
 
-           function showReference(selection) {
-             var enter = selection.selectAll('.issue-reference').data([0]).enter();
-             enter.append('div').attr('class', 'issue-reference').html(_t.html("issues.outdated_tags.".concat(prefix, "reference")));
-             enter.append('strong').html(_t.html('issues.suggested'));
-             enter.append('table').attr('class', 'tagDiff-table').selectAll('.tagDiff-row').data(tagDiff).enter().append('tr').attr('class', 'tagDiff-row').append('td').attr('class', function (d) {
-               var klass = d.type === '+' ? 'add' : 'remove';
-               return "tagDiff-cell tagDiff-cell-".concat(klass);
-             }).html(function (d) {
-               return d.display;
-             });
+           if (d.sequenceKey !== _selectedSequence) {
+             _viewerYaw = 0; // reset
            }
-         }
-
-         function oldMultipolygonIssues(entity, graph) {
-           var multipolygon, outerWay;
 
-           if (entity.type === 'relation') {
-             outerWay = osmOldMultipolygonOuterMemberOfRelation(entity, graph);
-             multipolygon = entity;
-           } else if (entity.type === 'way') {
-             multipolygon = osmIsOldMultipolygonOuterMember(entity, graph);
-             outerWay = entity;
-           } else {
-             return [];
-           }
+           _selectedSequence = d.sequenceKey;
+           service.ensureViewerLoaded(context).then(function () {
+             service.selectImage(context, d.key).yaw(_viewerYaw).showViewer(context);
+           });
+           context.map().centerEase(d.loc);
+         }
+         /**
+          * mouseover().
+          */
 
-           if (!multipolygon || !outerWay) return [];
-           return [new validationIssue({
-             type: type,
-             subtype: 'old_multipolygon',
-             severity: 'warning',
-             message: showMessage,
-             reference: showReference,
-             entityIds: [outerWay.id, multipolygon.id],
-             dynamicFixes: function dynamicFixes() {
-               return [new validationIssueFix({
-                 autoArgs: [doUpgrade, _t('issues.fix.move_tags.annotation')],
-                 title: _t.html('issues.fix.move_tags.title'),
-                 onClick: function onClick(context) {
-                   context.perform(doUpgrade, _t('issues.fix.move_tags.annotation'));
-                 }
-               })];
-             }
-           })];
 
-           function doUpgrade(graph) {
-             var currMultipolygon = graph.hasEntity(multipolygon.id);
-             var currOuterWay = graph.hasEntity(outerWay.id);
-             if (!currMultipolygon || !currOuterWay) return graph;
-             currMultipolygon = currMultipolygon.mergeTags(currOuterWay.tags);
-             graph = graph.replace(currMultipolygon);
-             return actionChangeTags(currOuterWay.id, {})(graph);
-           }
+         function mouseover(d3_event, d) {
+           var service = getService();
+           if (service) service.setStyles(context, d);
+         }
+         /**
+          * mouseout().
+          */
 
-           function showMessage(context) {
-             var currMultipolygon = context.hasEntity(multipolygon.id);
-             if (!currMultipolygon) return '';
-             return _t.html('issues.old_multipolygon.message', {
-               multipolygon: utilDisplayLabel(currMultipolygon, context.graph())
-             });
-           }
 
-           function showReference(selection) {
-             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.old_multipolygon.reference'));
-           }
+         function mouseout() {
+           var service = getService();
+           if (service) service.setStyles(context, null);
          }
+         /**
+          * transform().
+          */
 
-         var validation = function checkOutdatedTags(entity, graph) {
-           var issues = oldMultipolygonIssues(entity, graph);
-           if (!issues.length) issues = oldTagIssues(entity, graph);
-           return issues;
-         };
 
-         validation.type = type;
-         return validation;
-       }
+         function transform(d) {
+           var t = svgPointTransform(projection)(d);
+           var rot = d.ca + _viewerYaw;
 
-       function validationPrivateData() {
-         var type = 'private_data'; // assume that some buildings are private
+           if (rot) {
+             t += ' rotate(' + Math.floor(rot) + ',0,0)';
+           }
 
-         var privateBuildingValues = {
-           detached: true,
-           farm: true,
-           house: true,
-           houseboat: true,
-           residential: true,
-           semidetached_house: true,
-           static_caravan: true
-         }; // but they might be public if they have one of these other tags
+           return t;
+         }
 
-         var publicKeys = {
-           amenity: true,
-           craft: true,
-           historic: true,
-           leisure: true,
-           office: true,
-           shop: true,
-           tourism: true
-         }; // these tags may contain personally identifying info
+         function viewerChanged() {
+           var service = getService();
+           if (!service) return;
+           var viewer = service.viewer();
+           if (!viewer) return; // update viewfield rotation
 
-         var personalTags = {
-           'contact:email': true,
-           'contact:fax': true,
-           'contact:phone': true,
-           email: true,
-           fax: true,
-           phone: true
-         };
+           _viewerYaw = viewer.getYaw(); // avoid updating if the map is currently transformed
+           // e.g. during drags or easing.
 
-         var validation = function checkPrivateData(entity) {
-           var tags = entity.tags;
-           if (!tags.building || !privateBuildingValues[tags.building]) return [];
-           var keepTags = {};
+           if (context.map().isTransformed()) return;
+           layer.selectAll('.viewfield-group.currentView').attr('transform', transform);
+         }
 
-           for (var k in tags) {
-             if (publicKeys[k]) return []; // probably a public feature
+         function filterBubbles(bubbles) {
+           var fromDate = context.photos().fromDate();
+           var toDate = context.photos().toDate();
+           var usernames = context.photos().usernames();
 
-             if (!personalTags[k]) {
-               keepTags[k] = tags[k];
-             }
+           if (fromDate) {
+             var fromTimestamp = new Date(fromDate).getTime();
+             bubbles = bubbles.filter(function (bubble) {
+               return new Date(bubble.captured_at).getTime() >= fromTimestamp;
+             });
            }
 
-           var tagDiff = utilTagDiff(tags, keepTags);
-           if (!tagDiff.length) return [];
-           var fixID = tagDiff.length === 1 ? 'remove_tag' : 'remove_tags';
-           return [new validationIssue({
-             type: type,
-             severity: 'warning',
-             message: showMessage,
-             reference: showReference,
-             entityIds: [entity.id],
-             dynamicFixes: function dynamicFixes() {
-               return [new validationIssueFix({
-                 icon: 'iD-operation-delete',
-                 title: _t.html('issues.fix.' + fixID + '.title'),
-                 onClick: function onClick(context) {
-                   context.perform(doUpgrade, _t('issues.fix.upgrade_tags.annotation'));
-                 }
-               })];
-             }
-           })];
+           if (toDate) {
+             var toTimestamp = new Date(toDate).getTime();
+             bubbles = bubbles.filter(function (bubble) {
+               return new Date(bubble.captured_at).getTime() <= toTimestamp;
+             });
+           }
 
-           function doUpgrade(graph) {
-             var currEntity = graph.hasEntity(entity.id);
-             if (!currEntity) return graph;
-             var newTags = Object.assign({}, currEntity.tags); // shallow copy
+           if (usernames) {
+             bubbles = bubbles.filter(function (bubble) {
+               return usernames.indexOf(bubble.captured_by) !== -1;
+             });
+           }
 
-             tagDiff.forEach(function (diff) {
-               if (diff.type === '-') {
-                 delete newTags[diff.key];
-               } else if (diff.type === '+') {
-                 newTags[diff.key] = diff.newVal;
-               }
+           return bubbles;
+         }
+
+         function filterSequences(sequences) {
+           var fromDate = context.photos().fromDate();
+           var toDate = context.photos().toDate();
+           var usernames = context.photos().usernames();
+
+           if (fromDate) {
+             var fromTimestamp = new Date(fromDate).getTime();
+             sequences = sequences.filter(function (sequences) {
+               return new Date(sequences.properties.captured_at).getTime() >= fromTimestamp;
              });
-             return actionChangeTags(currEntity.id, newTags)(graph);
            }
 
-           function showMessage(context) {
-             var currEntity = context.hasEntity(this.entityIds[0]);
-             if (!currEntity) return '';
-             return _t.html('issues.private_data.contact.message', {
-               feature: utilDisplayLabel(currEntity, context.graph())
+           if (toDate) {
+             var toTimestamp = new Date(toDate).getTime();
+             sequences = sequences.filter(function (sequences) {
+               return new Date(sequences.properties.captured_at).getTime() <= toTimestamp;
              });
            }
 
-           function showReference(selection) {
-             var enter = selection.selectAll('.issue-reference').data([0]).enter();
-             enter.append('div').attr('class', 'issue-reference').html(_t.html('issues.private_data.reference'));
-             enter.append('strong').html(_t.html('issues.suggested'));
-             enter.append('table').attr('class', 'tagDiff-table').selectAll('.tagDiff-row').data(tagDiff).enter().append('tr').attr('class', 'tagDiff-row').append('td').attr('class', function (d) {
-               var klass = d.type === '+' ? 'add' : 'remove';
-               return 'tagDiff-cell tagDiff-cell-' + klass;
-             }).html(function (d) {
-               return d.display;
+           if (usernames) {
+             sequences = sequences.filter(function (sequences) {
+               return usernames.indexOf(sequences.properties.captured_by) !== -1;
              });
            }
-         };
 
-         validation.type = type;
-         return validation;
-       }
+           return sequences;
+         }
+         /**
+          * update().
+          */
 
-       var _discardNameRegexes = [];
-       function validationSuspiciousName() {
-         var type = 'suspicious_name';
-         var keysToTestForGenericValues = ['aerialway', 'aeroway', 'amenity', 'building', 'craft', 'highway', 'leisure', 'railway', 'man_made', 'office', 'shop', 'tourism', 'waterway']; // A concern here in switching to async data means that `_nsiFilters` will not
-         // be available at first, so the data on early tiles may not have tags validated fully.
 
-         _mainFileFetcher.get('nsi_filters').then(function (filters) {
-           // known list of generic names (e.g. "bar")
-           _discardNameRegexes = filters.discardNames.map(function (discardName) {
-             return new RegExp(discardName, 'i');
-           });
-         })["catch"](function () {
-           /* ignore */
-         });
+         function update() {
+           var viewer = context.container().select('.photoviewer');
+           var selected = viewer.empty() ? undefined : viewer.datum();
+           var z = ~~context.map().zoom();
+           var showMarkers = z >= minMarkerZoom;
+           var showViewfields = z >= minViewfieldZoom;
+           var service = getService();
+           var sequences = [];
+           var bubbles = [];
 
-         function isDiscardedSuggestionName(lowercaseName) {
-           return _discardNameRegexes.some(function (regex) {
-             return regex.test(lowercaseName);
-           });
-         } // test if the name is just the key or tag value (e.g. "park")
+           if (context.photos().showsPanoramic()) {
+             sequences = service ? service.sequences(projection) : [];
+             bubbles = service && showMarkers ? service.bubbles(projection) : [];
+             sequences = filterSequences(sequences);
+             bubbles = filterBubbles(bubbles);
+           }
 
+           var traces = layer.selectAll('.sequences').selectAll('.sequence').data(sequences, function (d) {
+             return d.properties.key;
+           }); // exit
 
-         function nameMatchesRawTag(lowercaseName, tags) {
-           for (var i = 0; i < keysToTestForGenericValues.length; i++) {
-             var key = keysToTestForGenericValues[i];
-             var val = tags[key];
+           traces.exit().remove(); // enter/update
 
-             if (val) {
-               val = val.toLowerCase();
+           traces = traces.enter().append('path').attr('class', 'sequence').merge(traces).attr('d', svgPath(projection).geojson);
+           var groups = layer.selectAll('.markers').selectAll('.viewfield-group').data(bubbles, function (d) {
+             // force reenter once bubbles are attached to a sequence
+             return d.key + (d.sequenceKey ? 'v1' : 'v0');
+           }); // exit
 
-               if (key === lowercaseName || val === lowercaseName || key.replace(/\_/g, ' ') === lowercaseName || val.replace(/\_/g, ' ') === lowercaseName) {
-                 return true;
-               }
-             }
-           }
+           groups.exit().remove(); // enter
 
-           return false;
-         }
+           var groupsEnter = groups.enter().append('g').attr('class', 'viewfield-group').on('mouseenter', mouseover).on('mouseleave', mouseout).on('click', click);
+           groupsEnter.append('g').attr('class', 'viewfield-scale'); // update
 
-         function isGenericName(name, tags) {
-           name = name.toLowerCase();
-           return nameMatchesRawTag(name, tags) || isDiscardedSuggestionName(name);
-         }
+           var markers = groups.merge(groupsEnter).sort(function (a, b) {
+             return a === selected ? 1 : b === selected ? -1 : b.loc[1] - a.loc[1];
+           }).attr('transform', transform).select('.viewfield-scale');
+           markers.selectAll('circle').data([0]).enter().append('circle').attr('dx', '0').attr('dy', '0').attr('r', '6');
+           var viewfields = markers.selectAll('.viewfield').data(showViewfields ? [0] : []);
+           viewfields.exit().remove(); // viewfields may or may not be drawn...
+           // but if they are, draw below the circles
 
-         function makeGenericNameIssue(entityId, nameKey, genericName, langCode) {
-           return new validationIssue({
-             type: type,
-             subtype: 'generic_name',
-             severity: 'warning',
-             message: function message(context) {
-               var entity = context.hasEntity(this.entityIds[0]);
-               if (!entity) return '';
-               var preset = _mainPresetIndex.match(entity, context.graph());
-               var langName = langCode && _mainLocalizer.languageName(langCode);
-               return _t.html('issues.generic_name.message' + (langName ? '_language' : ''), {
-                 feature: preset.name(),
-                 name: genericName,
-                 language: langName
-               });
-             },
-             reference: showReference,
-             entityIds: [entityId],
-             hash: nameKey + '=' + genericName,
-             dynamicFixes: function dynamicFixes() {
-               return [new validationIssueFix({
-                 icon: 'iD-operation-delete',
-                 title: _t.html('issues.fix.remove_the_name.title'),
-                 onClick: function onClick(context) {
-                   var entityId = this.issue.entityIds[0];
-                   var entity = context.entity(entityId);
-                   var tags = Object.assign({}, entity.tags); // shallow copy
+           viewfields.enter().insert('path', 'circle').attr('class', 'viewfield').attr('transform', 'scale(1.5,1.5),translate(-8, -13)').attr('d', viewfieldPath);
 
-                   delete tags[nameKey];
-                   context.perform(actionChangeTags(entityId, tags), _t('issues.fix.remove_generic_name.annotation'));
-                 }
-               })];
-             }
-           });
+           function viewfieldPath() {
+             var d = this.parentNode.__data__;
 
-           function showReference(selection) {
-             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.generic_name.reference'));
+             if (d.pano) {
+               return 'M 8,13 m -10,0 a 10,10 0 1,0 20,0 a 10,10 0 1,0 -20,0';
+             } else {
+               return 'M 6,9 C 8,8.4 8,8.4 10,9 L 16,-2 C 12,-5 4,-5 0,-2 z';
+             }
            }
          }
+         /**
+          * drawImages()
+          * drawImages is the method that is returned (and that runs) every time 'svgStreetside()' is called.
+          * 'svgStreetside()' is called from index.js
+          */
 
-         function makeIncorrectNameIssue(entityId, nameKey, incorrectName, langCode) {
-           return new validationIssue({
-             type: type,
-             subtype: 'not_name',
-             severity: 'warning',
-             message: function message(context) {
-               var entity = context.hasEntity(this.entityIds[0]);
-               if (!entity) return '';
-               var preset = _mainPresetIndex.match(entity, context.graph());
-               var langName = langCode && _mainLocalizer.languageName(langCode);
-               return _t.html('issues.incorrect_name.message' + (langName ? '_language' : ''), {
-                 feature: preset.name(),
-                 name: incorrectName,
-                 language: langName
-               });
-             },
-             reference: showReference,
-             entityIds: [entityId],
-             hash: nameKey + '=' + incorrectName,
-             dynamicFixes: function dynamicFixes() {
-               return [new validationIssueFix({
-                 icon: 'iD-operation-delete',
-                 title: _t.html('issues.fix.remove_the_name.title'),
-                 onClick: function onClick(context) {
-                   var entityId = this.issue.entityIds[0];
-                   var entity = context.entity(entityId);
-                   var tags = Object.assign({}, entity.tags); // shallow copy
 
-                   delete tags[nameKey];
-                   context.perform(actionChangeTags(entityId, tags), _t('issues.fix.remove_mistaken_name.annotation'));
-                 }
-               })];
-             }
-           });
+         function drawImages(selection) {
+           var enabled = svgStreetside.enabled;
+           var service = getService();
+           layer = selection.selectAll('.layer-streetside-images').data(service ? [0] : []);
+           layer.exit().remove();
+           var layerEnter = layer.enter().append('g').attr('class', 'layer-streetside-images').style('display', enabled ? 'block' : 'none');
+           layerEnter.append('g').attr('class', 'sequences');
+           layerEnter.append('g').attr('class', 'markers');
+           layer = layerEnter.merge(layer);
 
-           function showReference(selection) {
-             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.generic_name.reference'));
+           if (enabled) {
+             if (service && ~~context.map().zoom() >= minZoom) {
+               editOn();
+               update();
+               service.loadBubbles(projection);
+             } else {
+               editOff();
+             }
            }
          }
+         /**
+          * drawImages.enabled().
+          */
 
-         var validation = function checkGenericName(entity) {
-           // a generic name is okay if it's a known brand or entity
-           if (entity.hasWikidata()) return [];
-           var issues = [];
-           var notNames = (entity.tags['not:name'] || '').split(';');
 
-           for (var key in entity.tags) {
-             var m = key.match(/^name(?:(?::)([a-zA-Z_-]+))?$/);
-             if (!m) continue;
-             var langCode = m.length >= 2 ? m[1] : null;
-             var value = entity.tags[key];
+         drawImages.enabled = function (_) {
+           if (!arguments.length) return svgStreetside.enabled;
+           svgStreetside.enabled = _;
 
-             if (notNames.length) {
-               for (var i in notNames) {
-                 var notName = notNames[i];
+           if (svgStreetside.enabled) {
+             showLayer();
+             context.photos().on('change.streetside', update);
+           } else {
+             hideLayer();
+             context.photos().on('change.streetside', null);
+           }
 
-                 if (notName && value === notName) {
-                   issues.push(makeIncorrectNameIssue(entity.id, key, value, langCode));
-                   continue;
-                 }
-               }
-             }
+           dispatch.call('change');
+           return this;
+         };
+         /**
+          * drawImages.supported().
+          */
 
-             if (isGenericName(value, entity.tags)) {
-               issues.push(makeGenericNameIssue(entity.id, key, value, langCode));
-             }
-           }
 
-           return issues;
+         drawImages.supported = function () {
+           return !!getService();
          };
 
-         validation.type = type;
-         return validation;
+         init();
+         return drawImages;
        }
 
-       function validationUnsquareWay(context) {
-         var type = 'unsquare_way';
-         var DEFAULT_DEG_THRESHOLD = 5; // see also issues.js
-         // use looser epsilon for detection to reduce warnings of buildings that are essentially square already
+       function svgMapillaryImages(projection, context, dispatch) {
+         var throttledRedraw = throttle(function () {
+           dispatch.call('change');
+         }, 1000);
 
-         var epsilon = 0.05;
-         var nodeThreshold = 10;
+         var minZoom = 12;
+         var minMarkerZoom = 16;
+         var minViewfieldZoom = 18;
+         var layer = select(null);
 
-         function isBuilding(entity, graph) {
-           if (entity.type !== 'way' || entity.geometry(graph) !== 'area') return false;
-           return entity.tags.building && entity.tags.building !== 'no';
-         }
+         var _mapillary;
 
-         var validation = function checkUnsquareWay(entity, graph) {
-           if (!isBuilding(entity, graph)) return []; // don't flag ways marked as physically unsquare
+         function init() {
+           if (svgMapillaryImages.initialized) return; // run once
 
-           if (entity.tags.nonsquare === 'yes') return [];
-           var isClosed = entity.isClosed();
-           if (!isClosed) return []; // this building has bigger problems
-           // don't flag ways with lots of nodes since they are likely detail-mapped
+           svgMapillaryImages.enabled = false;
+           svgMapillaryImages.initialized = true;
+         }
 
-           var nodes = graph.childNodes(entity).slice(); // shallow copy
+         function getService() {
+           if (services.mapillary && !_mapillary) {
+             _mapillary = services.mapillary;
 
-           if (nodes.length > nodeThreshold + 1) return []; // +1 because closing node appears twice
-           // ignore if not all nodes are fully downloaded
+             _mapillary.event.on('loadedImages', throttledRedraw);
+           } else if (!services.mapillary && _mapillary) {
+             _mapillary = null;
+           }
 
-           var osm = services.osm;
-           if (!osm || nodes.some(function (node) {
-             return !osm.isDataLoaded(node.loc);
-           })) return []; // don't flag connected ways to avoid unresolvable unsquare loops
+           return _mapillary;
+         }
 
-           var hasConnectedSquarableWays = nodes.some(function (node) {
-             return graph.parentWays(node).some(function (way) {
-               if (way.id === entity.id) return false;
-               if (isBuilding(way, graph)) return true;
-               return graph.parentRelations(way).some(function (parentRelation) {
-                 return parentRelation.isMultipolygon() && parentRelation.tags.building && parentRelation.tags.building !== 'no';
-               });
-             });
+         function showLayer() {
+           var service = getService();
+           if (!service) return;
+           editOn();
+           layer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end', function () {
+             dispatch.call('change');
            });
-           if (hasConnectedSquarableWays) return []; // user-configurable square threshold
+         }
 
-           var storedDegreeThreshold = corePreferences('validate-square-degrees');
-           var degreeThreshold = isNaN(storedDegreeThreshold) ? DEFAULT_DEG_THRESHOLD : parseFloat(storedDegreeThreshold);
-           var points = nodes.map(function (node) {
-             return context.projection(node.loc);
+         function hideLayer() {
+           throttledRedraw.cancel();
+           layer.transition().duration(250).style('opacity', 0).on('end', editOff);
+         }
+
+         function editOn() {
+           layer.style('display', 'block');
+         }
+
+         function editOff() {
+           layer.selectAll('.viewfield-group').remove();
+           layer.style('display', 'none');
+         }
+
+         function click(d3_event, image) {
+           var service = getService();
+           if (!service) return;
+           service.ensureViewerLoaded(context).then(function () {
+             service.selectImage(context, image.id).showViewer(context);
            });
-           if (!geoOrthoCanOrthogonalize(points, isClosed, epsilon, degreeThreshold, true)) return [];
-           var autoArgs; // don't allow autosquaring features linked to wikidata
+           context.map().centerEase(image.loc);
+         }
 
-           if (!entity.tags.wikidata) {
-             // use same degree threshold as for detection
-             var autoAction = actionOrthogonalize(entity.id, context.projection, undefined, degreeThreshold);
-             autoAction.transitionable = false; // when autofixing, do it instantly
+         function mouseover(d3_event, image) {
+           var service = getService();
+           if (service) service.setStyles(context, image);
+         }
 
-             autoArgs = [autoAction, _t('operations.orthogonalize.annotation.feature', {
-               n: 1
-             })];
+         function mouseout() {
+           var service = getService();
+           if (service) service.setStyles(context, null);
+         }
+
+         function transform(d) {
+           var t = svgPointTransform(projection)(d);
+
+           if (d.ca) {
+             t += ' rotate(' + Math.floor(d.ca) + ',0,0)';
            }
 
-           return [new validationIssue({
-             type: type,
-             subtype: 'building',
-             severity: 'warning',
-             message: function message(context) {
-               var entity = context.hasEntity(this.entityIds[0]);
-               return entity ? _t.html('issues.unsquare_way.message', {
-                 feature: utilDisplayLabel(entity, context.graph())
-               }) : '';
-             },
-             reference: showReference,
-             entityIds: [entity.id],
-             hash: JSON.stringify(autoArgs !== undefined) + degreeThreshold,
-             dynamicFixes: function dynamicFixes() {
-               return [new validationIssueFix({
-                 icon: 'iD-operation-orthogonalize',
-                 title: _t.html('issues.fix.square_feature.title'),
-                 autoArgs: autoArgs,
-                 onClick: function onClick(context, completionHandler) {
-                   var entityId = this.issue.entityIds[0]; // use same degree threshold as for detection
+           return t;
+         }
 
-                   context.perform(actionOrthogonalize(entityId, context.projection, undefined, degreeThreshold), _t('operations.orthogonalize.annotation.feature', {
-                     n: 1
-                   })); // run after the squaring transition (currently 150ms)
+         function filterImages(images) {
+           var showsPano = context.photos().showsPanoramic();
+           var showsFlat = context.photos().showsFlat();
+           var fromDate = context.photos().fromDate();
+           var toDate = context.photos().toDate();
 
-                   window.setTimeout(function () {
-                     completionHandler();
-                   }, 175);
-                 }
-               })
-               /*
-               new validationIssueFix({
-                   title: t.html('issues.fix.tag_as_unsquare.title'),
-                   onClick: function(context) {
-                       var entityId = this.issue.entityIds[0];
-                       var entity = context.entity(entityId);
-                       var tags = Object.assign({}, entity.tags);  // shallow copy
-                       tags.nonsquare = 'yes';
-                       context.perform(
-                           actionChangeTags(entityId, tags),
-                           t('issues.fix.tag_as_unsquare.annotation')
-                       );
-                   }
-               })
-               */
-               ];
-             }
-           })];
+           if (!showsPano || !showsFlat) {
+             images = images.filter(function (image) {
+               if (image.is_pano) return showsPano;
+               return showsFlat;
+             });
+           }
 
-           function showReference(selection) {
-             selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.unsquare_way.buildings.reference'));
+           if (fromDate) {
+             images = images.filter(function (image) {
+               return new Date(image.captured_at).getTime() >= new Date(fromDate).getTime();
+             });
            }
-         };
 
-         validation.type = type;
-         return validation;
-       }
+           if (toDate) {
+             images = images.filter(function (image) {
+               return new Date(image.captured_at).getTime() <= new Date(toDate).getTime();
+             });
+           }
 
-       var Validations = /*#__PURE__*/Object.freeze({
-               __proto__: null,
-               validationAlmostJunction: validationAlmostJunction,
-               validationCloseNodes: validationCloseNodes,
-               validationCrossingWays: validationCrossingWays,
-               validationDisconnectedWay: validationDisconnectedWay,
-               validationFormatting: validationFormatting,
-               validationHelpRequest: validationHelpRequest,
-               validationImpossibleOneway: validationImpossibleOneway,
-               validationIncompatibleSource: validationIncompatibleSource,
-               validationMaprules: validationMaprules,
-               validationMismatchedGeometry: validationMismatchedGeometry,
-               validationMissingRole: validationMissingRole,
-               validationMissingTag: validationMissingTag,
-               validationOutdatedTags: validationOutdatedTags,
-               validationPrivateData: validationPrivateData,
-               validationSuspiciousName: validationSuspiciousName,
-               validationUnsquareWay: validationUnsquareWay
-       });
+           return images;
+         }
+
+         function filterSequences(sequences) {
+           var showsPano = context.photos().showsPanoramic();
+           var showsFlat = context.photos().showsFlat();
+           var fromDate = context.photos().fromDate();
+           var toDate = context.photos().toDate();
+
+           if (!showsPano || !showsFlat) {
+             sequences = sequences.filter(function (sequence) {
+               if (sequence.properties.hasOwnProperty('is_pano')) {
+                 if (sequence.properties.is_pano) return showsPano;
+                 return showsFlat;
+               }
 
-       function coreValidator(context) {
-         var dispatch$1 = dispatch('validated', 'focusedIssue');
-         var validator = utilRebind({}, dispatch$1, 'on');
-         var _rules = {};
-         var _disabledRules = {};
-         var _ignoredIssueIDs = {}; // issue.id -> true
+               return false;
+             });
+           }
 
-         var _baseCache = validationCache(); // issues before any user edits
+           if (fromDate) {
+             sequences = sequences.filter(function (sequence) {
+               return new Date(sequence.properties.captured_at).getTime() >= new Date(fromDate).getTime().toString();
+             });
+           }
 
+           if (toDate) {
+             sequences = sequences.filter(function (sequence) {
+               return new Date(sequence.properties.captured_at).getTime() <= new Date(toDate).getTime().toString();
+             });
+           }
 
-         var _headCache = validationCache(); // issues after all user edits
+           return sequences;
+         }
 
+         function update() {
+           var z = ~~context.map().zoom();
+           var showMarkers = z >= minMarkerZoom;
+           var showViewfields = z >= minViewfieldZoom;
+           var service = getService();
+           var sequences = service ? service.sequences(projection) : [];
+           var images = service && showMarkers ? service.images(projection) : [];
+           images = filterImages(images);
+           sequences = filterSequences(sequences);
+           service.filterViewer(context);
+           var traces = layer.selectAll('.sequences').selectAll('.sequence').data(sequences, function (d) {
+             return d.properties.id;
+           }); // exit
 
-         var _validatedGraph = null;
+           traces.exit().remove(); // enter/update
 
-         var _deferred = new Set(); //
-         // initialize the validator rulesets
-         //
+           traces = traces.enter().append('path').attr('class', 'sequence').merge(traces).attr('d', svgPath(projection).geojson);
+           var groups = layer.selectAll('.markers').selectAll('.viewfield-group').data(images, function (d) {
+             return d.id;
+           }); // exit
 
+           groups.exit().remove(); // enter
 
-         validator.init = function () {
-           Object.values(Validations).forEach(function (validation) {
-             if (typeof validation !== 'function') return;
-             var fn = validation(context);
-             var key = fn.type;
-             _rules[key] = fn;
-           });
-           var disabledRules = corePreferences('validate-disabledRules');
+           var groupsEnter = groups.enter().append('g').attr('class', 'viewfield-group').on('mouseenter', mouseover).on('mouseleave', mouseout).on('click', click);
+           groupsEnter.append('g').attr('class', 'viewfield-scale'); // update
 
-           if (disabledRules) {
-             disabledRules.split(',').forEach(function (key) {
-               _disabledRules[key] = true;
-             });
+           var markers = groups.merge(groupsEnter).sort(function (a, b) {
+             return b.loc[1] - a.loc[1]; // sort Y
+           }).attr('transform', transform).select('.viewfield-scale');
+           markers.selectAll('circle').data([0]).enter().append('circle').attr('dx', '0').attr('dy', '0').attr('r', '6');
+           var viewfields = markers.selectAll('.viewfield').data(showViewfields ? [0] : []);
+           viewfields.exit().remove();
+           viewfields.enter() // viewfields may or may not be drawn...
+           .insert('path', 'circle') // but if they are, draw below the circles
+           .attr('class', 'viewfield').classed('pano', function () {
+             return this.parentNode.__data__.is_pano;
+           }).attr('transform', 'scale(1.5,1.5),translate(-8, -13)').attr('d', viewfieldPath);
+
+           function viewfieldPath() {
+             if (this.parentNode.__data__.is_pano) {
+               return 'M 8,13 m -10,0 a 10,10 0 1,0 20,0 a 10,10 0 1,0 -20,0';
+             } else {
+               return 'M 6,9 C 8,8.4 8,8.4 10,9 L 16,-2 C 12,-5 4,-5 0,-2 z';
+             }
            }
-         };
+         }
 
-         function reset(resetIgnored) {
-           Array.from(_deferred).forEach(function (handle) {
-             window.cancelIdleCallback(handle);
+         function drawImages(selection) {
+           var enabled = svgMapillaryImages.enabled;
+           var service = getService();
+           layer = selection.selectAll('.layer-mapillary').data(service ? [0] : []);
+           layer.exit().remove();
+           var layerEnter = layer.enter().append('g').attr('class', 'layer-mapillary').style('display', enabled ? 'block' : 'none');
+           layerEnter.append('g').attr('class', 'sequences');
+           layerEnter.append('g').attr('class', 'markers');
+           layer = layerEnter.merge(layer);
 
-             _deferred["delete"](handle);
-           }); // clear caches
+           if (enabled) {
+             if (service && ~~context.map().zoom() >= minZoom) {
+               editOn();
+               update();
+               service.loadImages(projection);
+             } else {
+               editOff();
+             }
+           }
+         }
 
-           if (resetIgnored) _ignoredIssueIDs = {};
-           _baseCache = validationCache();
-           _headCache = validationCache();
-           _validatedGraph = null;
-         } //
-         // clear caches, called whenever iD resets after a save
-         //
+         drawImages.enabled = function (_) {
+           if (!arguments.length) return svgMapillaryImages.enabled;
+           svgMapillaryImages.enabled = _;
 
+           if (svgMapillaryImages.enabled) {
+             showLayer();
+             context.photos().on('change.mapillary_images', update);
+           } else {
+             hideLayer();
+             context.photos().on('change.mapillary_images', null);
+           }
 
-         validator.reset = function () {
-           reset(true);
+           dispatch.call('change');
+           return this;
          };
 
-         validator.resetIgnoredIssues = function () {
-           _ignoredIssueIDs = {}; // reload UI
+         drawImages.supported = function () {
+           return !!getService();
+         };
 
-           dispatch$1.call('validated');
-         }; // must update issues when the user changes the unsquare thereshold
+         init();
+         return drawImages;
+       }
 
+       function svgMapillaryPosition(projection, context) {
+         var throttledRedraw = throttle(function () {
+           update();
+         }, 1000);
 
-         validator.reloadUnsquareIssues = function () {
-           reloadUnsquareIssues(_headCache, context.graph());
-           reloadUnsquareIssues(_baseCache, context.history().base());
-           dispatch$1.call('validated');
-         };
+         var minZoom = 12;
+         var minViewfieldZoom = 18;
+         var layer = select(null);
 
-         function reloadUnsquareIssues(cache, graph) {
-           var checkUnsquareWay = _rules.unsquare_way;
-           if (typeof checkUnsquareWay !== 'function') return; // uncache existing
+         var _mapillary;
 
-           cache.uncacheIssuesOfType('unsquare_way');
-           var buildings = context.history().tree().intersects(geoExtent([-180, -90], [180, 90]), graph) // everywhere
-           .filter(function (entity) {
-             return entity.type === 'way' && entity.tags.building && entity.tags.building !== 'no';
-           }); // rerun for all buildings
+         var viewerCompassAngle;
 
-           buildings.forEach(function (entity) {
-             var detected = checkUnsquareWay(entity, graph);
-             if (detected.length !== 1) return;
-             var issue = detected[0];
+         function init() {
+           if (svgMapillaryPosition.initialized) return; // run once
 
-             if (!cache.issuesByEntityID[entity.id]) {
-               cache.issuesByEntityID[entity.id] = new Set();
-             }
+           svgMapillaryPosition.initialized = true;
+         }
 
-             cache.issuesByEntityID[entity.id].add(issue.id);
-             cache.issuesByIssueID[issue.id] = issue;
-           });
-         } // options = {
-         //     what: 'all',     // 'all' or 'edited'
-         //     where: 'all',   // 'all' or 'visible'
-         //     includeIgnored: false   // true, false, or 'only'
-         //     includeDisabledRules: false   // true, false, or 'only'
-         // };
+         function getService() {
+           if (services.mapillary && !_mapillary) {
+             _mapillary = services.mapillary;
 
+             _mapillary.event.on('imageChanged', throttledRedraw);
 
-         validator.getIssues = function (options) {
-           var opts = Object.assign({
-             what: 'all',
-             where: 'all',
-             includeIgnored: false,
-             includeDisabledRules: false
-           }, options);
-           var issues = Object.values(_headCache.issuesByIssueID);
-           var view = context.map().extent();
-           return issues.filter(function (issue) {
-             if (!issue) return false;
-             if (opts.includeDisabledRules === 'only' && !_disabledRules[issue.type]) return false;
-             if (!opts.includeDisabledRules && _disabledRules[issue.type]) return false;
-             if (opts.includeIgnored === 'only' && !_ignoredIssueIDs[issue.id]) return false;
-             if (!opts.includeIgnored && _ignoredIssueIDs[issue.id]) return false; // Sanity check:  This issue may be for an entity that not longer exists.
-             // If we detect this, uncache and return false so it is not included..
+             _mapillary.event.on('bearingChanged', function (e) {
+               viewerCompassAngle = e.bearing;
+               if (context.map().isTransformed()) return;
+               layer.selectAll('.viewfield-group.currentView').filter(function (d) {
+                 return d.is_pano;
+               }).attr('transform', transform);
+             });
+           } else if (!services.mapillary && _mapillary) {
+             _mapillary = null;
+           }
 
-             var entityIds = issue.entityIds || [];
+           return _mapillary;
+         }
 
-             for (var i = 0; i < entityIds.length; i++) {
-               var entityId = entityIds[i];
+         function editOn() {
+           layer.style('display', 'block');
+         }
 
-               if (!context.hasEntity(entityId)) {
-                 delete _headCache.issuesByEntityID[entityId];
-                 delete _headCache.issuesByIssueID[issue.id];
-                 return false;
-               }
-             }
+         function editOff() {
+           layer.selectAll('.viewfield-group').remove();
+           layer.style('display', 'none');
+         }
 
-             if (opts.what === 'edited' && _baseCache.issuesByIssueID[issue.id]) return false;
+         function transform(d) {
+           var t = svgPointTransform(projection)(d);
 
-             if (opts.where === 'visible') {
-               var extent = issue.extent(context.graph());
-               if (!view.intersects(extent)) return false;
-             }
+           if (d.is_pano && viewerCompassAngle !== null && isFinite(viewerCompassAngle)) {
+             t += ' rotate(' + Math.floor(viewerCompassAngle) + ',0,0)';
+           } else if (d.ca) {
+             t += ' rotate(' + Math.floor(d.ca) + ',0,0)';
+           }
 
-             return true;
-           });
-         };
+           return t;
+         }
 
-         validator.getResolvedIssues = function () {
-           var baseIssues = Object.values(_baseCache.issuesByIssueID);
-           return baseIssues.filter(function (issue) {
-             return !_headCache.issuesByIssueID[issue.id];
-           });
-         };
+         function update() {
+           var z = ~~context.map().zoom();
+           var showViewfields = z >= minViewfieldZoom;
+           var service = getService();
+           var image = service && service.getActiveImage();
+           var groups = layer.selectAll('.markers').selectAll('.viewfield-group').data(image ? [image] : [], function (d) {
+             return d.id;
+           }); // exit
 
-         validator.focusIssue = function (issue) {
-           var extent = issue.extent(context.graph());
+           groups.exit().remove(); // enter
 
-           if (extent) {
-             var setZoom = Math.max(context.map().zoom(), 19);
-             context.map().unobscuredCenterZoomEase(extent.center(), setZoom); // select the first entity
+           var groupsEnter = groups.enter().append('g').attr('class', 'viewfield-group currentView highlighted');
+           groupsEnter.append('g').attr('class', 'viewfield-scale'); // update
 
-             if (issue.entityIds && issue.entityIds.length) {
-               window.setTimeout(function () {
-                 var ids = issue.entityIds;
-                 context.enter(modeSelect(context, [ids[0]]));
-                 dispatch$1.call('focusedIssue', this, issue);
-               }, 250); // after ease
-             }
+           var markers = groups.merge(groupsEnter).attr('transform', transform).select('.viewfield-scale');
+           markers.selectAll('circle').data([0]).enter().append('circle').attr('dx', '0').attr('dy', '0').attr('r', '6');
+           var viewfields = markers.selectAll('.viewfield').data(showViewfields ? [0] : []);
+           viewfields.exit().remove();
+           viewfields.enter().insert('path', 'circle').attr('class', 'viewfield').attr('transform', 'scale(1.5,1.5),translate(-8, -13)').attr('d', 'M 6,9 C 8,8.4 8,8.4 10,9 L 16,-2 C 12,-5 4,-5 0,-2 z');
+         }
+
+         function drawImages(selection) {
+           var service = getService();
+           layer = selection.selectAll('.layer-mapillary-position').data(service ? [0] : []);
+           layer.exit().remove();
+           var layerEnter = layer.enter().append('g').attr('class', 'layer-mapillary-position');
+           layerEnter.append('g').attr('class', 'markers');
+           layer = layerEnter.merge(layer);
+
+           if (service && ~~context.map().zoom() >= minZoom) {
+             editOn();
+             update();
+           } else {
+             editOff();
            }
+         }
+
+         drawImages.enabled = function () {
+           update();
+           return this;
          };
 
-         validator.getIssuesBySeverity = function (options) {
-           var groups = utilArrayGroupBy(validator.getIssues(options), 'severity');
-           groups.error = groups.error || [];
-           groups.warning = groups.warning || [];
-           return groups;
-         }; // show some issue types in a particular order
+         drawImages.supported = function () {
+           return !!getService();
+         };
 
+         init();
+         return drawImages;
+       }
 
-         var orderedIssueTypes = [// flag missing data first
-         'missing_tag', 'missing_role', // then flag identity issues
-         'outdated_tags', 'mismatched_geometry', // flag geometry issues where fixing them might solve connectivity issues
-         'crossing_ways', 'almost_junction', // then flag connectivity issues
-         'disconnected_way', 'impossible_oneway']; // returns the issues that the given entity IDs have in common, matching the given options
+       function svgMapillarySigns(projection, context, dispatch) {
+         var throttledRedraw = throttle(function () {
+           dispatch.call('change');
+         }, 1000);
 
-         validator.getSharedEntityIssues = function (entityIDs, options) {
-           var cache = _headCache; // gather the issues that are common to all the entities
+         var minZoom = 12;
+         var layer = select(null);
 
-           var issueIDs = entityIDs.reduce(function (acc, entityID) {
-             var entityIssueIDs = cache.issuesByEntityID[entityID] || new Set();
+         var _mapillary;
 
-             if (!acc) {
-               return new Set(entityIssueIDs);
-             }
+         function init() {
+           if (svgMapillarySigns.initialized) return; // run once
 
-             return new Set(_toConsumableArray(acc).filter(function (elem) {
-               return entityIssueIDs.has(elem);
-             }));
-           }, null) || [];
-           var opts = options || {};
-           return Array.from(issueIDs).map(function (id) {
-             return cache.issuesByIssueID[id];
-           }).filter(function (issue) {
-             if (!issue) return false;
-             if (opts.includeDisabledRules === 'only' && !_disabledRules[issue.type]) return false;
-             if (!opts.includeDisabledRules && _disabledRules[issue.type]) return false;
-             if (opts.includeIgnored === 'only' && !_ignoredIssueIDs[issue.id]) return false;
-             if (!opts.includeIgnored && _ignoredIssueIDs[issue.id]) return false;
-             return true;
-           }).sort(function (issue1, issue2) {
-             if (issue1.type === issue2.type) {
-               // issues of the same type, sort deterministically
-               return issue1.id < issue2.id ? -1 : 1;
-             }
+           svgMapillarySigns.enabled = false;
+           svgMapillarySigns.initialized = true;
+         }
 
-             var index1 = orderedIssueTypes.indexOf(issue1.type);
-             var index2 = orderedIssueTypes.indexOf(issue2.type);
+         function getService() {
+           if (services.mapillary && !_mapillary) {
+             _mapillary = services.mapillary;
 
-             if (index1 !== -1 && index2 !== -1) {
-               // both issue types have explicit sort orders
-               return index1 - index2;
-             } else if (index1 === -1 && index2 === -1) {
-               // neither issue type has an explicit sort order, sort by type
-               return issue1.type < issue2.type ? -1 : 1;
-             } else {
-               // order explicit types before everything else
-               return index1 !== -1 ? -1 : 1;
-             }
-           });
-         };
+             _mapillary.event.on('loadedSigns', throttledRedraw);
+           } else if (!services.mapillary && _mapillary) {
+             _mapillary = null;
+           }
 
-         validator.getEntityIssues = function (entityID, options) {
-           return validator.getSharedEntityIssues([entityID], options);
-         };
+           return _mapillary;
+         }
 
-         validator.getRuleKeys = function () {
-           return Object.keys(_rules);
-         };
+         function showLayer() {
+           var service = getService();
+           if (!service) return;
+           service.loadSignResources(context);
+           editOn();
+         }
 
-         validator.isRuleEnabled = function (key) {
-           return !_disabledRules[key];
-         };
+         function hideLayer() {
+           throttledRedraw.cancel();
+           editOff();
+         }
 
-         validator.toggleRule = function (key) {
-           if (_disabledRules[key]) {
-             delete _disabledRules[key];
-           } else {
-             _disabledRules[key] = true;
-           }
+         function editOn() {
+           layer.style('display', 'block');
+         }
 
-           corePreferences('validate-disabledRules', Object.keys(_disabledRules).join(','));
-           validator.validate();
-         };
+         function editOff() {
+           layer.selectAll('.icon-sign').remove();
+           layer.style('display', 'none');
+         }
 
-         validator.disableRules = function (keys) {
-           _disabledRules = {};
-           keys.forEach(function (k) {
-             _disabledRules[k] = true;
+         function click(d3_event, d) {
+           var service = getService();
+           if (!service) return;
+           context.map().centerEase(d.loc);
+           var selectedImageId = service.getActiveImage() && service.getActiveImage().id;
+           service.getDetections(d.id).then(function (detections) {
+             if (detections.length) {
+               var imageId = detections[0].image.id;
+
+               if (imageId === selectedImageId) {
+                 service.highlightDetection(detections[0]).selectImage(context, imageId);
+               } else {
+                 service.ensureViewerLoaded(context).then(function () {
+                   service.highlightDetection(detections[0]).selectImage(context, imageId).showViewer(context);
+                 });
+               }
+             }
            });
-           corePreferences('validate-disabledRules', Object.keys(_disabledRules).join(','));
-           validator.validate();
-         };
+         }
 
-         validator.ignoreIssue = function (id) {
-           _ignoredIssueIDs[id] = true;
-         }; //
-         // Run validation on a single entity for the given graph
-         //
+         function filterData(detectedFeatures) {
+           var fromDate = context.photos().fromDate();
+           var toDate = context.photos().toDate();
 
+           if (fromDate) {
+             var fromTimestamp = new Date(fromDate).getTime();
+             detectedFeatures = detectedFeatures.filter(function (feature) {
+               return new Date(feature.last_seen_at).getTime() >= fromTimestamp;
+             });
+           }
 
-         function validateEntity(entity, graph) {
-           var entityIssues = []; // runs validation and appends resulting issues
+           if (toDate) {
+             var toTimestamp = new Date(toDate).getTime();
+             detectedFeatures = detectedFeatures.filter(function (feature) {
+               return new Date(feature.first_seen_at).getTime() <= toTimestamp;
+             });
+           }
 
-           function runValidation(key) {
-             var fn = _rules[key];
+           return detectedFeatures;
+         }
 
-             if (typeof fn !== 'function') {
-               console.error('no such validation rule = ' + key); // eslint-disable-line no-console
+         function update() {
+           var service = getService();
+           var data = service ? service.signs(projection) : [];
+           data = filterData(data);
+           var transform = svgPointTransform(projection);
+           var signs = layer.selectAll('.icon-sign').data(data, function (d) {
+             return d.id;
+           }); // exit
 
-               return;
-             }
+           signs.exit().remove(); // enter
 
-             var detected = fn(entity, graph);
-             entityIssues = entityIssues.concat(detected);
-           } // run all rules
+           var enter = signs.enter().append('g').attr('class', 'icon-sign icon-detected').on('click', click);
+           enter.append('use').attr('width', '24px').attr('height', '24px').attr('x', '-12px').attr('y', '-12px').attr('xlink:href', function (d) {
+             return '#' + d.value;
+           });
+           enter.append('rect').attr('width', '24px').attr('height', '24px').attr('x', '-12px').attr('y', '-12px'); // update
 
+           signs.merge(enter).attr('transform', transform);
+         }
 
-           Object.keys(_rules).forEach(runValidation);
-           return entityIssues;
+         function drawSigns(selection) {
+           var enabled = svgMapillarySigns.enabled;
+           var service = getService();
+           layer = selection.selectAll('.layer-mapillary-signs').data(service ? [0] : []);
+           layer.exit().remove();
+           layer = layer.enter().append('g').attr('class', 'layer-mapillary-signs layer-mapillary-detections').style('display', enabled ? 'block' : 'none').merge(layer);
+
+           if (enabled) {
+             if (service && ~~context.map().zoom() >= minZoom) {
+               editOn();
+               update();
+               service.loadSigns(projection);
+               service.showSignDetections(true);
+             } else {
+               editOff();
+             }
+           } else if (service) {
+             service.showSignDetections(false);
+           }
          }
 
-         function entityIDsToValidate(entityIDs, graph) {
-           var processedIDs = new Set();
-           return entityIDs.reduce(function (acc, entityID) {
-             // keep redundancy check separate from `acc` because an `entityID`
-             // could have been added to `acc` as a related entity through an earlier pass
-             if (processedIDs.has(entityID)) return acc;
-             processedIDs.add(entityID);
-             var entity = graph.hasEntity(entityID);
-             if (!entity) return acc;
-             acc.add(entityID);
-             var checkParentRels = [entity];
+         drawSigns.enabled = function (_) {
+           if (!arguments.length) return svgMapillarySigns.enabled;
+           svgMapillarySigns.enabled = _;
 
-             if (entity.type === 'node') {
-               graph.parentWays(entity).forEach(function (parentWay) {
-                 acc.add(parentWay.id); // include parent ways
+           if (svgMapillarySigns.enabled) {
+             showLayer();
+             context.photos().on('change.mapillary_signs', update);
+           } else {
+             hideLayer();
+             context.photos().on('change.mapillary_signs', null);
+           }
 
-                 checkParentRels.push(parentWay);
-               });
-             } else if (entity.type === 'relation') {
-               entity.members.forEach(function (member) {
-                 acc.add(member.id); // include members
-               });
-             } else if (entity.type === 'way') {
-               entity.nodes.forEach(function (nodeID) {
-                 acc.add(nodeID); // include child nodes
+           dispatch.call('change');
+           return this;
+         };
 
-                 graph._parentWays[nodeID].forEach(function (wayID) {
-                   acc.add(wayID); // include connected ways
-                 });
-               });
-             }
+         drawSigns.supported = function () {
+           return !!getService();
+         };
 
-             checkParentRels.forEach(function (entity) {
-               // include parent relations
-               if (entity.type !== 'relation') {
-                 // but not super-relations
-                 graph.parentRelations(entity).forEach(function (parentRelation) {
-                   acc.add(parentRelation.id);
-                 });
-               }
-             });
-             return acc;
-           }, new Set());
-         } //
-         // Run validation for several entities, supplied `entityIDs`,
-         // against `graph` for the given `cache`
-         //
+         init();
+         return drawSigns;
+       }
 
+       function svgMapillaryMapFeatures(projection, context, dispatch) {
+         var throttledRedraw = throttle(function () {
+           dispatch.call('change');
+         }, 1000);
 
-         function validateEntities(entityIDs, graph, cache) {
-           // clear caches for existing issues related to these entities
-           entityIDs.forEach(cache.uncacheEntityID); // detect new issues and update caches
+         var minZoom = 12;
+         var layer = select(null);
 
-           entityIDs.forEach(function (entityID) {
-             var entity = graph.hasEntity(entityID); // don't validate deleted entities
+         var _mapillary;
 
-             if (!entity) return;
-             var issues = validateEntity(entity, graph);
-             cache.cacheIssues(issues);
-           });
-         } //
-         // Validates anything that has changed since the last time it was run.
-         // Also updates the "validatedGraph" to be the current graph
-         // and dispatches a `validated` event when finished.
-         //
+         function init() {
+           if (svgMapillaryMapFeatures.initialized) return; // run once
 
+           svgMapillaryMapFeatures.enabled = false;
+           svgMapillaryMapFeatures.initialized = true;
+         }
 
-         validator.validate = function () {
-           var currGraph = context.graph();
-           _validatedGraph = _validatedGraph || context.history().base();
+         function getService() {
+           if (services.mapillary && !_mapillary) {
+             _mapillary = services.mapillary;
 
-           if (currGraph === _validatedGraph) {
-             dispatch$1.call('validated');
-             return;
+             _mapillary.event.on('loadedMapFeatures', throttledRedraw);
+           } else if (!services.mapillary && _mapillary) {
+             _mapillary = null;
            }
 
-           var oldGraph = _validatedGraph;
-           var difference = coreDifference(oldGraph, currGraph);
-           _validatedGraph = currGraph;
-           var createdAndModifiedEntityIDs = difference.extantIDs(true); // created/modified (true = w/relation members)
+           return _mapillary;
+         }
 
-           var entityIDsToCheck = entityIDsToValidate(createdAndModifiedEntityIDs, currGraph); // check modified and deleted entities against the old graph in order to update their related entities
-           // (e.g. deleting the only highway connected to a road should create a disconnected highway issue)
+         function showLayer() {
+           var service = getService();
+           if (!service) return;
+           service.loadObjectResources(context);
+           editOn();
+         }
 
-           var modifiedAndDeletedEntityIDs = difference.deleted().concat(difference.modified()).map(function (entity) {
-             return entity.id;
-           });
-           var entityIDsToCheckForOldGraph = entityIDsToValidate(modifiedAndDeletedEntityIDs, oldGraph); // concat the sets
+         function hideLayer() {
+           throttledRedraw.cancel();
+           editOff();
+         }
 
-           entityIDsToCheckForOldGraph.forEach(entityIDsToCheck.add, entityIDsToCheck);
-           validateEntities(entityIDsToCheck, context.graph(), _headCache);
-           dispatch$1.call('validated');
-         };
+         function editOn() {
+           layer.style('display', 'block');
+         }
 
-         context.history().on('reset.validator', function () {
-           // cached issues aren't valid any longer if the history has been reset
-           reset(false);
-           validator.validate();
-         }); // WHEN TO RUN VALIDATION:
-         // When graph changes:
+         function editOff() {
+           layer.selectAll('.icon-map-feature').remove();
+           layer.style('display', 'none');
+         }
 
-         context.history().on('restore.validator', validator.validate) // restore saved history
-         .on('undone.validator', validator.validate) // undo
-         .on('redone.validator', validator.validate); // redo
-         // but not on 'change' (e.g. while drawing)
-         // When user changes editing modes:
+         function click(d3_event, d) {
+           var service = getService();
+           if (!service) return;
+           context.map().centerEase(d.loc);
+           var selectedImageId = service.getActiveImage() && service.getActiveImage().id;
+           service.getDetections(d.id).then(function (detections) {
+             if (detections.length) {
+               var imageId = detections[0].image.id;
 
-         context.on('exit.validator', validator.validate); // When merging fetched data:
+               if (imageId === selectedImageId) {
+                 service.highlightDetection(detections[0]).selectImage(context, imageId);
+               } else {
+                 service.ensureViewerLoaded(context).then(function () {
+                   service.highlightDetection(detections[0]).selectImage(context, imageId).showViewer(context);
+                 });
+               }
+             }
+           });
+         }
 
-         context.history().on('merge.validator', function (entities) {
-           if (!entities) return;
-           var handle = window.requestIdleCallback(function () {
-             var entityIDs = entities.map(function (entity) {
-               return entity.id;
+         function filterData(detectedFeatures) {
+           var fromDate = context.photos().fromDate();
+           var toDate = context.photos().toDate();
+
+           if (fromDate) {
+             detectedFeatures = detectedFeatures.filter(function (feature) {
+               return new Date(feature.last_seen_at).getTime() >= new Date(fromDate).getTime();
              });
-             var headGraph = context.graph();
-             validateEntities(entityIDsToValidate(entityIDs, headGraph), headGraph, _headCache);
-             var baseGraph = context.history().base();
-             validateEntities(entityIDsToValidate(entityIDs, baseGraph), baseGraph, _baseCache);
-             dispatch$1.call('validated');
-           });
+           }
 
-           _deferred.add(handle);
-         });
-         return validator;
-       }
+           if (toDate) {
+             detectedFeatures = detectedFeatures.filter(function (feature) {
+               return new Date(feature.first_seen_at).getTime() <= new Date(toDate).getTime();
+             });
+           }
 
-       function validationCache() {
-         var cache = {
-           issuesByIssueID: {},
-           // issue.id -> issue
-           issuesByEntityID: {} // entity.id -> set(issue.id)
+           return detectedFeatures;
+         }
 
-         };
+         function update() {
+           var service = getService();
+           var data = service ? service.mapFeatures(projection) : [];
+           data = filterData(data);
+           var transform = svgPointTransform(projection);
+           var mapFeatures = layer.selectAll('.icon-map-feature').data(data, function (d) {
+             return d.id;
+           }); // exit
 
-         cache.cacheIssues = function (issues) {
-           issues.forEach(function (issue) {
-             var entityIds = issue.entityIds || [];
-             entityIds.forEach(function (entityId) {
-               if (!cache.issuesByEntityID[entityId]) {
-                 cache.issuesByEntityID[entityId] = new Set();
-               }
+           mapFeatures.exit().remove(); // enter
 
-               cache.issuesByEntityID[entityId].add(issue.id);
-             });
-             cache.issuesByIssueID[issue.id] = issue;
+           var enter = mapFeatures.enter().append('g').attr('class', 'icon-map-feature icon-detected').on('click', click);
+           enter.append('title').text(function (d) {
+             var id = d.value.replace(/--/g, '.').replace(/-/g, '_');
+             return _t('mapillary_map_features.' + id);
            });
-         };
-
-         cache.uncacheIssue = function (issue) {
-           // When multiple entities are involved (e.g. crossing_ways),
-           // remove this issue from the other entity caches too..
-           var entityIds = issue.entityIds || [];
-           entityIds.forEach(function (entityId) {
-             if (cache.issuesByEntityID[entityId]) {
-               cache.issuesByEntityID[entityId]["delete"](issue.id);
+           enter.append('use').attr('width', '24px').attr('height', '24px').attr('x', '-12px').attr('y', '-12px').attr('xlink:href', function (d) {
+             if (d.value === 'object--billboard') {
+               // no billboard icon right now, so use the advertisement icon
+               return '#object--sign--advertisement';
              }
-           });
-           delete cache.issuesByIssueID[issue.id];
-         };
-
-         cache.uncacheIssues = function (issues) {
-           issues.forEach(cache.uncacheIssue);
-         };
 
-         cache.uncacheIssuesOfType = function (type) {
-           var issuesOfType = Object.values(cache.issuesByIssueID).filter(function (issue) {
-             return issue.type === type;
+             return '#' + d.value;
            });
-           cache.uncacheIssues(issuesOfType);
-         }; //
-         // Remove a single entity and all its related issues from the caches
-         //
+           enter.append('rect').attr('width', '24px').attr('height', '24px').attr('x', '-12px').attr('y', '-12px'); // update
 
+           mapFeatures.merge(enter).attr('transform', transform);
+         }
 
-         cache.uncacheEntityID = function (entityID) {
-           var issueIDs = cache.issuesByEntityID[entityID];
-           if (!issueIDs) return;
-           issueIDs.forEach(function (issueID) {
-             var issue = cache.issuesByIssueID[issueID];
+         function drawMapFeatures(selection) {
+           var enabled = svgMapillaryMapFeatures.enabled;
+           var service = getService();
+           layer = selection.selectAll('.layer-mapillary-map-features').data(service ? [0] : []);
+           layer.exit().remove();
+           layer = layer.enter().append('g').attr('class', 'layer-mapillary-map-features layer-mapillary-detections').style('display', enabled ? 'block' : 'none').merge(layer);
 
-             if (issue) {
-               cache.uncacheIssue(issue);
+           if (enabled) {
+             if (service && ~~context.map().zoom() >= minZoom) {
+               editOn();
+               update();
+               service.loadMapFeatures(projection);
+               service.showFeatureDetections(true);
              } else {
-               delete cache.issuesByIssueID[issueID];
+               editOff();
              }
-           });
-           delete cache.issuesByEntityID[entityID];
+           } else if (service) {
+             service.showFeatureDetections(false);
+           }
+         }
+
+         drawMapFeatures.enabled = function (_) {
+           if (!arguments.length) return svgMapillaryMapFeatures.enabled;
+           svgMapillaryMapFeatures.enabled = _;
+
+           if (svgMapillaryMapFeatures.enabled) {
+             showLayer();
+             context.photos().on('change.mapillary_map_features', update);
+           } else {
+             hideLayer();
+             context.photos().on('change.mapillary_map_features', null);
+           }
+
+           dispatch.call('change');
+           return this;
          };
 
-         return cache;
+         drawMapFeatures.supported = function () {
+           return !!getService();
+         };
+
+         init();
+         return drawMapFeatures;
        }
 
-       function coreUploader(context) {
-         var dispatch$1 = dispatch( // Start and end events are dispatched exactly once each per legitimate outside call to `save`
-         'saveStarted', // dispatched as soon as a call to `save` has been deemed legitimate
-         'saveEnded', // dispatched after the result event has been dispatched
-         'willAttemptUpload', // dispatched before the actual upload call occurs, if it will
-         'progressChanged', // Each save results in one of these outcomes:
-         'resultNoChanges', // upload wasn't attempted since there were no edits
-         'resultErrors', // upload failed due to errors
-         'resultConflicts', // upload failed due to data conflicts
-         'resultSuccess' // upload completed without errors
-         );
-         var _isSaving = false;
-         var _conflicts = [];
-         var _errors = [];
+       function svgOpenstreetcamImages(projection, context, dispatch) {
+         var throttledRedraw = throttle(function () {
+           dispatch.call('change');
+         }, 1000);
 
-         var _origChanges;
+         var minZoom = 12;
+         var minMarkerZoom = 16;
+         var minViewfieldZoom = 18;
+         var layer = select(null);
 
-         var _discardTags = {};
-         _mainFileFetcher.get('discarded').then(function (d) {
-           _discardTags = d;
-         })["catch"](function () {
-           /* ignore */
-         });
-         var uploader = utilRebind({}, dispatch$1, 'on');
+         var _openstreetcam;
 
-         uploader.isSaving = function () {
-           return _isSaving;
-         };
+         function init() {
+           if (svgOpenstreetcamImages.initialized) return; // run once
 
-         uploader.save = function (changeset, tryAgain, checkConflicts) {
-           // Guard against accidentally entering save code twice - #4641
-           if (_isSaving && !tryAgain) {
-             return;
-           }
+           svgOpenstreetcamImages.enabled = false;
+           svgOpenstreetcamImages.initialized = true;
+         }
 
-           var osm = context.connection();
-           if (!osm) return; // If user somehow got logged out mid-save, try to reauthenticate..
-           // This can happen if they were logged in from before, but the tokens are no longer valid.
+         function getService() {
+           if (services.openstreetcam && !_openstreetcam) {
+             _openstreetcam = services.openstreetcam;
 
-           if (!osm.authenticated()) {
-             osm.authenticate(function (err) {
-               if (!err) {
-                 uploader.save(changeset, tryAgain, checkConflicts); // continue where we left off..
-               }
-             });
-             return;
+             _openstreetcam.event.on('loadedImages', throttledRedraw);
+           } else if (!services.openstreetcam && _openstreetcam) {
+             _openstreetcam = null;
            }
 
-           if (!_isSaving) {
-             _isSaving = true;
-             dispatch$1.call('saveStarted', this);
-           }
+           return _openstreetcam;
+         }
 
-           var history = context.history();
-           _conflicts = [];
-           _errors = []; // Store original changes, in case user wants to download them as an .osc file
+         function showLayer() {
+           var service = getService();
+           if (!service) return;
+           editOn();
+           layer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end', function () {
+             dispatch.call('change');
+           });
+         }
 
-           _origChanges = history.changes(actionDiscardTags(history.difference(), _discardTags)); // First time, `history.perform` a no-op action.
-           // Any conflict resolutions will be done as `history.replace`
-           // Remember to pop this later if needed
+         function hideLayer() {
+           throttledRedraw.cancel();
+           layer.transition().duration(250).style('opacity', 0).on('end', editOff);
+         }
 
-           if (!tryAgain) {
-             history.perform(actionNoop());
-           } // Attempt a fast upload.. If there are conflicts, re-enter with `checkConflicts = true`
+         function editOn() {
+           layer.style('display', 'block');
+         }
 
+         function editOff() {
+           layer.selectAll('.viewfield-group').remove();
+           layer.style('display', 'none');
+         }
 
-           if (!checkConflicts) {
-             upload(changeset); // Do the full (slow) conflict check..
-           } else {
-             performFullConflictCheck(changeset);
-           }
-         };
+         function click(d3_event, d) {
+           var service = getService();
+           if (!service) return;
+           service.ensureViewerLoaded(context).then(function () {
+             service.selectImage(context, d.key).showViewer(context);
+           });
+           context.map().centerEase(d.loc);
+         }
 
-         function performFullConflictCheck(changeset) {
-           var osm = context.connection();
-           if (!osm) return;
-           var history = context.history();
-           var localGraph = context.graph();
-           var remoteGraph = coreGraph(history.base(), true);
-           var summary = history.difference().summary();
-           var _toCheck = [];
+         function mouseover(d3_event, d) {
+           var service = getService();
+           if (service) service.setStyles(context, d);
+         }
 
-           for (var i = 0; i < summary.length; i++) {
-             var item = summary[i];
+         function mouseout() {
+           var service = getService();
+           if (service) service.setStyles(context, null);
+         }
 
-             if (item.changeType === 'modified') {
-               _toCheck.push(item.entity.id);
-             }
-           }
+         function transform(d) {
+           var t = svgPointTransform(projection)(d);
 
-           var _toLoad = withChildNodes(_toCheck, localGraph);
+           if (d.ca) {
+             t += ' rotate(' + Math.floor(d.ca) + ',0,0)';
+           }
 
-           var _loaded = {};
-           var _toLoadCount = 0;
-           var _toLoadTotal = _toLoad.length;
+           return t;
+         }
 
-           if (_toCheck.length) {
-             dispatch$1.call('progressChanged', this, _toLoadCount, _toLoadTotal);
+         function filterImages(images) {
+           var fromDate = context.photos().fromDate();
+           var toDate = context.photos().toDate();
+           var usernames = context.photos().usernames();
 
-             _toLoad.forEach(function (id) {
-               _loaded[id] = false;
+           if (fromDate) {
+             var fromTimestamp = new Date(fromDate).getTime();
+             images = images.filter(function (item) {
+               return new Date(item.captured_at).getTime() >= fromTimestamp;
              });
-
-             osm.loadMultiple(_toLoad, loaded);
-           } else {
-             upload(changeset);
            }
 
-           return;
-
-           function withChildNodes(ids, graph) {
-             var s = new Set(ids);
-             ids.forEach(function (id) {
-               var entity = graph.entity(id);
-               if (entity.type !== 'way') return;
-               graph.childNodes(entity).forEach(function (child) {
-                 if (child.version !== undefined) {
-                   s.add(child.id);
-                 }
-               });
+           if (toDate) {
+             var toTimestamp = new Date(toDate).getTime();
+             images = images.filter(function (item) {
+               return new Date(item.captured_at).getTime() <= toTimestamp;
              });
-             return Array.from(s);
-           } // Reload modified entities into an alternate graph and check for conflicts..
-
-
-           function loaded(err, result) {
-             if (_errors.length) return;
+           }
 
-             if (err) {
-               _errors.push({
-                 msg: err.message || err.responseText,
-                 details: [_t('save.status_code', {
-                   code: err.status
-                 })]
-               });
+           if (usernames) {
+             images = images.filter(function (item) {
+               return usernames.indexOf(item.captured_by) !== -1;
+             });
+           }
 
-               didResultInErrors();
-             } else {
-               var loadMore = [];
-               result.data.forEach(function (entity) {
-                 remoteGraph.replace(entity);
-                 _loaded[entity.id] = true;
-                 _toLoad = _toLoad.filter(function (val) {
-                   return val !== entity.id;
-                 });
-                 if (!entity.visible) return; // Because loadMultiple doesn't download /full like loadEntity,
-                 // need to also load children that aren't already being checked..
+           return images;
+         }
 
-                 var i, id;
+         function filterSequences(sequences) {
+           var fromDate = context.photos().fromDate();
+           var toDate = context.photos().toDate();
+           var usernames = context.photos().usernames();
 
-                 if (entity.type === 'way') {
-                   for (i = 0; i < entity.nodes.length; i++) {
-                     id = entity.nodes[i];
+           if (fromDate) {
+             var fromTimestamp = new Date(fromDate).getTime();
+             sequences = sequences.filter(function (image) {
+               return new Date(image.properties.captured_at).getTime() >= fromTimestamp;
+             });
+           }
 
-                     if (_loaded[id] === undefined) {
-                       _loaded[id] = false;
-                       loadMore.push(id);
-                     }
-                   }
-                 } else if (entity.type === 'relation' && entity.isMultipolygon()) {
-                   for (i = 0; i < entity.members.length; i++) {
-                     id = entity.members[i].id;
+           if (toDate) {
+             var toTimestamp = new Date(toDate).getTime();
+             sequences = sequences.filter(function (image) {
+               return new Date(image.properties.captured_at).getTime() <= toTimestamp;
+             });
+           }
 
-                     if (_loaded[id] === undefined) {
-                       _loaded[id] = false;
-                       loadMore.push(id);
-                     }
-                   }
-                 }
-               });
-               _toLoadCount += result.data.length;
-               _toLoadTotal += loadMore.length;
-               dispatch$1.call('progressChanged', this, _toLoadCount, _toLoadTotal);
+           if (usernames) {
+             sequences = sequences.filter(function (image) {
+               return usernames.indexOf(image.properties.captured_by) !== -1;
+             });
+           }
 
-               if (loadMore.length) {
-                 _toLoad.push.apply(_toLoad, loadMore);
+           return sequences;
+         }
 
-                 osm.loadMultiple(loadMore, loaded);
-               }
+         function update() {
+           var viewer = context.container().select('.photoviewer');
+           var selected = viewer.empty() ? undefined : viewer.datum();
+           var z = ~~context.map().zoom();
+           var showMarkers = z >= minMarkerZoom;
+           var showViewfields = z >= minViewfieldZoom;
+           var service = getService();
+           var sequences = [];
+           var images = [];
 
-               if (!_toLoad.length) {
-                 detectConflicts();
-                 upload(changeset);
-               }
-             }
+           if (context.photos().showsFlat()) {
+             sequences = service ? service.sequences(projection) : [];
+             images = service && showMarkers ? service.images(projection) : [];
+             sequences = filterSequences(sequences);
+             images = filterImages(images);
            }
 
-           function detectConflicts() {
-             function choice(id, text, _action) {
-               return {
-                 id: id,
-                 text: text,
-                 action: function action() {
-                   history.replace(_action);
-                 }
-               };
-             }
-
-             function formatUser(d) {
-               return '<a href="' + osm.userURL(d) + '" target="_blank">' + d + '</a>';
-             }
-
-             function entityName(entity) {
-               return utilDisplayName(entity) || utilDisplayType(entity.id) + ' ' + entity.id;
-             }
+           var traces = layer.selectAll('.sequences').selectAll('.sequence').data(sequences, function (d) {
+             return d.properties.key;
+           }); // exit
 
-             function sameVersions(local, remote) {
-               if (local.version !== remote.version) return false;
+           traces.exit().remove(); // enter/update
 
-               if (local.type === 'way') {
-                 var children = utilArrayUnion(local.nodes, remote.nodes);
+           traces = traces.enter().append('path').attr('class', 'sequence').merge(traces).attr('d', svgPath(projection).geojson);
+           var groups = layer.selectAll('.markers').selectAll('.viewfield-group').data(images, function (d) {
+             return d.key;
+           }); // exit
 
-                 for (var i = 0; i < children.length; i++) {
-                   var a = localGraph.hasEntity(children[i]);
-                   var b = remoteGraph.hasEntity(children[i]);
-                   if (a && b && a.version !== b.version) return false;
-                 }
-               }
+           groups.exit().remove(); // enter
 
-               return true;
-             }
+           var groupsEnter = groups.enter().append('g').attr('class', 'viewfield-group').on('mouseenter', mouseover).on('mouseleave', mouseout).on('click', click);
+           groupsEnter.append('g').attr('class', 'viewfield-scale'); // update
 
-             _toCheck.forEach(function (id) {
-               var local = localGraph.entity(id);
-               var remote = remoteGraph.entity(id);
-               if (sameVersions(local, remote)) return;
-               var merge = actionMergeRemoteChanges(id, localGraph, remoteGraph, _discardTags, formatUser);
-               history.replace(merge);
-               var mergeConflicts = merge.conflicts();
-               if (!mergeConflicts.length) return; // merged safely
+           var markers = groups.merge(groupsEnter).sort(function (a, b) {
+             return a === selected ? 1 : b === selected ? -1 : b.loc[1] - a.loc[1]; // sort Y
+           }).attr('transform', transform).select('.viewfield-scale');
+           markers.selectAll('circle').data([0]).enter().append('circle').attr('dx', '0').attr('dy', '0').attr('r', '6');
+           var viewfields = markers.selectAll('.viewfield').data(showViewfields ? [0] : []);
+           viewfields.exit().remove();
+           viewfields.enter() // viewfields may or may not be drawn...
+           .insert('path', 'circle') // but if they are, draw below the circles
+           .attr('class', 'viewfield').attr('transform', 'scale(1.5,1.5),translate(-8, -13)').attr('d', 'M 6,9 C 8,8.4 8,8.4 10,9 L 16,-2 C 12,-5 4,-5 0,-2 z');
+         }
 
-               var forceLocal = actionMergeRemoteChanges(id, localGraph, remoteGraph, _discardTags).withOption('force_local');
-               var forceRemote = actionMergeRemoteChanges(id, localGraph, remoteGraph, _discardTags).withOption('force_remote');
-               var keepMine = _t('save.conflict.' + (remote.visible ? 'keep_local' : 'restore'));
-               var keepTheirs = _t('save.conflict.' + (remote.visible ? 'keep_remote' : 'delete'));
+         function drawImages(selection) {
+           var enabled = svgOpenstreetcamImages.enabled,
+               service = getService();
+           layer = selection.selectAll('.layer-openstreetcam').data(service ? [0] : []);
+           layer.exit().remove();
+           var layerEnter = layer.enter().append('g').attr('class', 'layer-openstreetcam').style('display', enabled ? 'block' : 'none');
+           layerEnter.append('g').attr('class', 'sequences');
+           layerEnter.append('g').attr('class', 'markers');
+           layer = layerEnter.merge(layer);
 
-               _conflicts.push({
-                 id: id,
-                 name: entityName(local),
-                 details: mergeConflicts,
-                 chosen: 1,
-                 choices: [choice(id, keepMine, forceLocal), choice(id, keepTheirs, forceRemote)]
-               });
-             });
+           if (enabled) {
+             if (service && ~~context.map().zoom() >= minZoom) {
+               editOn();
+               update();
+               service.loadImages(projection);
+             } else {
+               editOff();
+             }
            }
          }
 
-         function upload(changeset) {
-           var osm = context.connection();
-
-           if (!osm) {
-             _errors.push({
-               msg: 'No OSM Service'
-             });
-           }
+         drawImages.enabled = function (_) {
+           if (!arguments.length) return svgOpenstreetcamImages.enabled;
+           svgOpenstreetcamImages.enabled = _;
 
-           if (_conflicts.length) {
-             didResultInConflicts(changeset);
-           } else if (_errors.length) {
-             didResultInErrors();
+           if (svgOpenstreetcamImages.enabled) {
+             showLayer();
+             context.photos().on('change.openstreetcam_images', update);
            } else {
-             var history = context.history();
-             var changes = history.changes(actionDiscardTags(history.difference(), _discardTags));
-
-             if (changes.modified.length || changes.created.length || changes.deleted.length) {
-               dispatch$1.call('willAttemptUpload', this);
-               osm.putChangeset(changeset, changes, uploadCallback);
-             } else {
-               // changes were insignificant or reverted by user
-               didResultInNoChanges();
-             }
+             hideLayer();
+             context.photos().on('change.openstreetcam_images', null);
            }
-         }
 
-         function uploadCallback(err, changeset) {
-           if (err) {
-             if (err.status === 409) {
-               // 409 Conflict
-               uploader.save(changeset, true, true); // tryAgain = true, checkConflicts = true
-             } else {
-               _errors.push({
-                 msg: err.message || err.responseText,
-                 details: [_t('save.status_code', {
-                   code: err.status
-                 })]
-               });
+           dispatch.call('change');
+           return this;
+         };
+
+         drawImages.supported = function () {
+           return !!getService();
+         };
 
-               didResultInErrors();
-             }
-           } else {
-             didResultInSuccess(changeset);
-           }
-         }
+         init();
+         return drawImages;
+       }
 
-         function didResultInNoChanges() {
-           dispatch$1.call('resultNoChanges', this);
-           endSave();
-           context.flush(); // reset iD
-         }
+       function svgOsm(projection, context, dispatch) {
+         var enabled = true;
 
-         function didResultInErrors() {
-           context.history().pop();
-           dispatch$1.call('resultErrors', this, _errors);
-           endSave();
+         function drawOsm(selection) {
+           selection.selectAll('.layer-osm').data(['covered', 'areas', 'lines', 'points', 'labels']).enter().append('g').attr('class', function (d) {
+             return 'layer-osm ' + d;
+           });
+           selection.selectAll('.layer-osm.points').selectAll('.points-group').data(['points', 'midpoints', 'vertices', 'turns']).enter().append('g').attr('class', function (d) {
+             return 'points-group ' + d;
+           });
          }
 
-         function didResultInConflicts(changeset) {
-           _conflicts.sort(function (a, b) {
-             return b.id.localeCompare(a.id);
+         function showLayer() {
+           var layer = context.surface().selectAll('.data-layer.osm');
+           layer.interrupt();
+           layer.classed('disabled', false).style('opacity', 0).transition().duration(250).style('opacity', 1).on('end interrupt', function () {
+             dispatch.call('change');
            });
+         }
 
-           dispatch$1.call('resultConflicts', this, changeset, _conflicts, _origChanges);
-           endSave();
+         function hideLayer() {
+           var layer = context.surface().selectAll('.data-layer.osm');
+           layer.interrupt();
+           layer.transition().duration(250).style('opacity', 0).on('end interrupt', function () {
+             layer.classed('disabled', true);
+             dispatch.call('change');
+           });
          }
 
-         function didResultInSuccess(changeset) {
-           // delete the edit stack cached to local storage
-           context.history().clearSaved();
-           dispatch$1.call('resultSuccess', this, changeset); // Add delay to allow for postgres replication #1646 #2678
+         drawOsm.enabled = function (val) {
+           if (!arguments.length) return enabled;
+           enabled = val;
 
-           window.setTimeout(function () {
-             endSave();
-             context.flush(); // reset iD
-           }, 2500);
-         }
+           if (enabled) {
+             showLayer();
+           } else {
+             hideLayer();
+           }
 
-         function endSave() {
-           _isSaving = false;
-           dispatch$1.call('saveEnded', this);
+           dispatch.call('change');
+           return this;
+         };
+
+         return drawOsm;
+       }
+
+       var _notesEnabled = false;
+
+       var _osmService;
+
+       function svgNotes(projection, context, dispatch) {
+         if (!dispatch) {
+           dispatch = dispatch$8('change');
          }
 
-         uploader.cancelConflictResolution = function () {
-           context.history().pop();
-         };
+         var throttledRedraw = throttle(function () {
+           dispatch.call('change');
+         }, 1000);
 
-         uploader.processResolvedConflicts = function (changeset) {
-           var history = context.history();
+         var minZoom = 12;
+         var touchLayer = select(null);
+         var drawLayer = select(null);
+         var _notesVisible = false;
 
-           for (var i = 0; i < _conflicts.length; i++) {
-             if (_conflicts[i].chosen === 1) {
-               // user chose "use theirs"
-               var entity = context.hasEntity(_conflicts[i].id);
+         function markerPath(selection, klass) {
+           selection.attr('class', klass).attr('transform', 'translate(-8, -22)').attr('d', 'm17.5,0l-15,0c-1.37,0 -2.5,1.12 -2.5,2.5l0,11.25c0,1.37 1.12,2.5 2.5,2.5l3.75,0l0,3.28c0,0.38 0.43,0.6 0.75,0.37l4.87,-3.65l5.62,0c1.37,0 2.5,-1.12 2.5,-2.5l0,-11.25c0,-1.37 -1.12,-2.5 -2.5,-2.5z');
+         } // Loosely-coupled osm service for fetching notes.
 
-               if (entity && entity.type === 'way') {
-                 var children = utilArrayUniq(entity.nodes);
 
-                 for (var j = 0; j < children.length; j++) {
-                   history.replace(actionRevert(children[j]));
-                 }
-               }
+         function getService() {
+           if (services.osm && !_osmService) {
+             _osmService = services.osm;
 
-               history.replace(actionRevert(_conflicts[i].id));
-             }
+             _osmService.on('loadedNotes', throttledRedraw);
+           } else if (!services.osm && _osmService) {
+             _osmService = null;
            }
 
-           uploader.save(changeset, true, false); // tryAgain = true, checkConflicts = false
-         };
+           return _osmService;
+         } // Show the notes
 
-         uploader.reset = function () {};
 
-         return uploader;
-       }
+         function editOn() {
+           if (!_notesVisible) {
+             _notesVisible = true;
+             drawLayer.style('display', 'block');
+           }
+         } // Immediately remove the notes and their touch targets
 
-       var abs$4 = Math.abs;
-       var exp$2 = Math.exp;
-       var E = Math.E;
 
-       var FORCED$g = fails(function () {
-         return Math.sinh(-2e-17) != -2e-17;
-       });
+         function editOff() {
+           if (_notesVisible) {
+             _notesVisible = false;
+             drawLayer.style('display', 'none');
+             drawLayer.selectAll('.note').remove();
+             touchLayer.selectAll('.note').remove();
+           }
+         } // Enable the layer.  This shows the notes and transitions them to visible.
 
-       // `Math.sinh` method
-       // https://tc39.es/ecma262/#sec-math.sinh
-       // V8 near Chromium 38 has a problem with very small numbers
-       _export({ target: 'Math', stat: true, forced: FORCED$g }, {
-         sinh: function sinh(x) {
-           return abs$4(x = +x) < 1 ? (mathExpm1(x) - mathExpm1(-x)) / 2 : (exp$2(x - 1) - exp$2(-x - 1)) * (E / 2);
-         }
-       });
 
-       var isRetina = window.devicePixelRatio && window.devicePixelRatio >= 2; // listen for DPI change, e.g. when dragging a browser window from a retina to non-retina screen
+         function layerOn() {
+           editOn();
+           drawLayer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end interrupt', function () {
+             dispatch.call('change');
+           });
+         } // Disable the layer.  This transitions the layer invisible and then hides the notes.
 
-       window.matchMedia("\n        (-webkit-min-device-pixel-ratio: 2), /* Safari */\n        (min-resolution: 2dppx),             /* standard */\n        (min-resolution: 192dpi)             /* fallback */\n    ").addListener(function () {
-         isRetina = window.devicePixelRatio && window.devicePixelRatio >= 2;
-       });
 
-       function localeDateString(s) {
-         if (!s) return null;
-         var options = {
-           day: 'numeric',
-           month: 'short',
-           year: 'numeric'
-         };
-         var d = new Date(s);
-         if (isNaN(d.getTime())) return null;
-         return d.toLocaleDateString(_mainLocalizer.localeCode(), options);
-       }
+         function layerOff() {
+           throttledRedraw.cancel();
+           drawLayer.interrupt();
+           touchLayer.selectAll('.note').remove();
+           drawLayer.transition().duration(250).style('opacity', 0).on('end interrupt', function () {
+             editOff();
+             dispatch.call('change');
+           });
+         } // Update the note markers
 
-       function vintageRange(vintage) {
-         var s;
 
-         if (vintage.start || vintage.end) {
-           s = vintage.start || '?';
+         function updateMarkers() {
+           if (!_notesVisible || !_notesEnabled) return;
+           var service = getService();
+           var selectedID = context.selectedNoteID();
+           var data = service ? service.notes(projection) : [];
+           var getTransform = svgPointTransform(projection); // Draw markers..
 
-           if (vintage.start !== vintage.end) {
-             s += ' - ' + (vintage.end || '?');
-           }
-         }
+           var notes = drawLayer.selectAll('.note').data(data, function (d) {
+             return d.status + d.id;
+           }); // exit
 
-         return s;
-       }
+           notes.exit().remove(); // enter
 
-       function rendererBackgroundSource(data) {
-         var source = Object.assign({}, data); // shallow copy
+           var notesEnter = notes.enter().append('g').attr('class', function (d) {
+             return 'note note-' + d.id + ' ' + d.status;
+           }).classed('new', function (d) {
+             return d.id < 0;
+           });
+           notesEnter.append('ellipse').attr('cx', 0.5).attr('cy', 1).attr('rx', 6.5).attr('ry', 3).attr('class', 'stroke');
+           notesEnter.append('path').call(markerPath, 'shadow');
+           notesEnter.append('use').attr('class', 'note-fill').attr('width', '20px').attr('height', '20px').attr('x', '-8px').attr('y', '-22px').attr('xlink:href', '#iD-icon-note');
+           notesEnter.selectAll('.icon-annotation').data(function (d) {
+             return [d];
+           }).enter().append('use').attr('class', 'icon-annotation').attr('width', '10px').attr('height', '10px').attr('x', '-3px').attr('y', '-19px').attr('xlink:href', function (d) {
+             if (d.id < 0) return '#iD-icon-plus';
+             if (d.status === 'open') return '#iD-icon-close';
+             return '#iD-icon-apply';
+           }); // update
 
-         var _offset = [0, 0];
-         var _name = source.name;
-         var _description = source.description;
+           notes.merge(notesEnter).sort(sortY).classed('selected', function (d) {
+             var mode = context.mode();
+             var isMoving = mode && mode.id === 'drag-note'; // no shadows when dragging
 
-         var _best = !!source.best;
+             return !isMoving && d.id === selectedID;
+           }).attr('transform', getTransform); // Draw targets..
 
-         var _template = source.encrypted ? utilAesDecrypt(source.template) : source.template;
+           if (touchLayer.empty()) return;
+           var fillClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
+           var targets = touchLayer.selectAll('.note').data(data, function (d) {
+             return d.id;
+           }); // exit
 
-         source.tileSize = data.tileSize || 256;
-         source.zoomExtent = data.zoomExtent || [0, 22];
-         source.overzoom = data.overzoom !== false;
+           targets.exit().remove(); // enter/update
 
-         source.offset = function (val) {
-           if (!arguments.length) return _offset;
-           _offset = val;
-           return source;
-         };
+           targets.enter().append('rect').attr('width', '20px').attr('height', '20px').attr('x', '-8px').attr('y', '-22px').merge(targets).sort(sortY).attr('class', function (d) {
+             var newClass = d.id < 0 ? 'new' : '';
+             return 'note target note-' + d.id + ' ' + fillClass + newClass;
+           }).attr('transform', getTransform);
 
-         source.nudge = function (val, zoomlevel) {
-           _offset[0] += val[0] / Math.pow(2, zoomlevel);
-           _offset[1] += val[1] / Math.pow(2, zoomlevel);
-           return source;
-         };
+           function sortY(a, b) {
+             if (a.id === selectedID) return 1;
+             if (b.id === selectedID) return -1;
+             return b.loc[1] - a.loc[1];
+           }
+         } // Draw the notes layer and schedule loading notes and updating markers.
 
-         source.name = function () {
-           var id_safe = source.id.replace(/\./g, '<TX_DOT>');
-           return _t('imagery.' + id_safe + '.name', {
-             "default": _name
-           });
-         };
 
-         source.label = function () {
-           var id_safe = source.id.replace(/\./g, '<TX_DOT>');
-           return _t.html('imagery.' + id_safe + '.name', {
-             "default": _name
-           });
-         };
+         function drawNotes(selection) {
+           var service = getService();
+           var surface = context.surface();
 
-         source.description = function () {
-           var id_safe = source.id.replace(/\./g, '<TX_DOT>');
-           return _t.html('imagery.' + id_safe + '.description', {
-             "default": _description
-           });
-         };
+           if (surface && !surface.empty()) {
+             touchLayer = surface.selectAll('.data-layer.touch .layer-touch.markers');
+           }
 
-         source.best = function () {
-           return _best;
-         };
+           drawLayer = selection.selectAll('.layer-notes').data(service ? [0] : []);
+           drawLayer.exit().remove();
+           drawLayer = drawLayer.enter().append('g').attr('class', 'layer-notes').style('display', _notesEnabled ? 'block' : 'none').merge(drawLayer);
 
-         source.area = function () {
-           if (!data.polygon) return Number.MAX_VALUE; // worldwide
+           if (_notesEnabled) {
+             if (service && ~~context.map().zoom() >= minZoom) {
+               editOn();
+               service.loadNotes(projection);
+               updateMarkers();
+             } else {
+               editOff();
+             }
+           }
+         } // Toggles the layer on and off
 
-           var area = d3_geoArea({
-             type: 'MultiPolygon',
-             coordinates: [data.polygon]
-           });
-           return isNaN(area) ? 0 : area;
-         };
 
-         source.imageryUsed = function () {
-           return _name || source.id;
-         };
+         drawNotes.enabled = function (val) {
+           if (!arguments.length) return _notesEnabled;
+           _notesEnabled = val;
 
-         source.template = function (val) {
-           if (!arguments.length) return _template;
+           if (_notesEnabled) {
+             layerOn();
+           } else {
+             layerOff();
 
-           if (source.id === 'custom') {
-             _template = val;
+             if (context.selectedNoteID()) {
+               context.enter(modeBrowse(context));
+             }
            }
 
-           return source;
+           dispatch.call('change');
+           return this;
          };
 
-         source.url = function (coord) {
-           var result = _template;
-           if (result === '') return result; // source 'none'
-           // Guess a type based on the tokens present in the template
-           // (This is for 'custom' source, where we don't know)
+         return drawNotes;
+       }
 
-           if (!source.type) {
-             if (/SERVICE=WMS|\{(proj|wkid|bbox)\}/.test(_template)) {
-               source.type = 'wms';
-               source.projection = 'EPSG:3857'; // guess
-             } else if (/\{(x|y)\}/.test(_template)) {
-               source.type = 'tms';
-             } else if (/\{u\}/.test(_template)) {
-               source.type = 'bing';
-             }
-           }
+       function svgTouch() {
+         function drawTouch(selection) {
+           selection.selectAll('.layer-touch').data(['areas', 'lines', 'points', 'turns', 'markers']).enter().append('g').attr('class', function (d) {
+             return 'layer-touch ' + d;
+           });
+         }
 
-           if (source.type === 'wms') {
-             var tileToProjectedCoords = function tileToProjectedCoords(x, y, z) {
-               //polyfill for IE11, PhantomJS
-               var sinh = Math.sinh || function (x) {
-                 var y = Math.exp(x);
-                 return (y - 1 / y) / 2;
-               };
+         return drawTouch;
+       }
 
-               var zoomSize = Math.pow(2, z);
-               var lon = x / zoomSize * Math.PI * 2 - Math.PI;
-               var lat = Math.atan(sinh(Math.PI * (1 - 2 * y / zoomSize)));
+       function refresh(selection, node) {
+         var cr = node.getBoundingClientRect();
+         var prop = [cr.width, cr.height];
+         selection.property('__dimensions__', prop);
+         return prop;
+       }
 
-               switch (source.projection) {
-                 case 'EPSG:4326':
-                   return {
-                     x: lon * 180 / Math.PI,
-                     y: lat * 180 / Math.PI
-                   };
+       function utilGetDimensions(selection, force) {
+         if (!selection || selection.empty()) {
+           return [0, 0];
+         }
 
-                 default:
-                   // EPSG:3857 and synonyms
-                   var mercCoords = mercatorRaw(lon, lat);
-                   return {
-                     x: 20037508.34 / Math.PI * mercCoords[0],
-                     y: 20037508.34 / Math.PI * mercCoords[1]
-                   };
-               }
-             };
+         var node = selection.node(),
+             cached = selection.property('__dimensions__');
+         return !cached || force ? refresh(selection, node) : cached;
+       }
+       function utilSetDimensions(selection, dimensions) {
+         if (!selection || selection.empty()) {
+           return selection;
+         }
 
-             var tileSize = source.tileSize;
-             var projection = source.projection;
-             var minXmaxY = tileToProjectedCoords(coord[0], coord[1], coord[2]);
-             var maxXminY = tileToProjectedCoords(coord[0] + 1, coord[1] + 1, coord[2]);
-             result = result.replace(/\{(\w+)\}/g, function (token, key) {
-               switch (key) {
-                 case 'width':
-                 case 'height':
-                   return tileSize;
+         var node = selection.node();
 
-                 case 'proj':
-                   return projection;
+         if (dimensions === null) {
+           refresh(selection, node);
+           return selection;
+         }
 
-                 case 'wkid':
-                   return projection.replace(/^EPSG:/, '');
+         return selection.property('__dimensions__', [dimensions[0], dimensions[1]]).attr('width', dimensions[0]).attr('height', dimensions[1]);
+       }
 
-                 case 'bbox':
-                   // WMS 1.3 flips x/y for some coordinate systems including EPSG:4326 - #7557
-                   if (projection === 'EPSG:4326' && // The CRS parameter implies version 1.3 (prior versions use SRS)
-                   /VERSION=1.3|CRS={proj}/.test(source.template())) {
-                     return maxXminY.y + ',' + minXmaxY.x + ',' + minXmaxY.y + ',' + maxXminY.x;
-                   } else {
-                     return minXmaxY.x + ',' + maxXminY.y + ',' + maxXminY.x + ',' + minXmaxY.y;
-                   }
+       function svgLayers(projection, context) {
+         var dispatch = dispatch$8('change');
+         var svg = select(null);
+         var _layers = [{
+           id: 'osm',
+           layer: svgOsm(projection, context, dispatch)
+         }, {
+           id: 'notes',
+           layer: svgNotes(projection, context, dispatch)
+         }, {
+           id: 'data',
+           layer: svgData(projection, context, dispatch)
+         }, {
+           id: 'keepRight',
+           layer: svgKeepRight(projection, context, dispatch)
+         }, {
+           id: 'improveOSM',
+           layer: svgImproveOSM(projection, context, dispatch)
+         }, {
+           id: 'osmose',
+           layer: svgOsmose(projection, context, dispatch)
+         }, {
+           id: 'streetside',
+           layer: svgStreetside(projection, context, dispatch)
+         }, {
+           id: 'mapillary',
+           layer: svgMapillaryImages(projection, context, dispatch)
+         }, {
+           id: 'mapillary-position',
+           layer: svgMapillaryPosition(projection, context)
+         }, {
+           id: 'mapillary-map-features',
+           layer: svgMapillaryMapFeatures(projection, context, dispatch)
+         }, {
+           id: 'mapillary-signs',
+           layer: svgMapillarySigns(projection, context, dispatch)
+         }, {
+           id: 'openstreetcam',
+           layer: svgOpenstreetcamImages(projection, context, dispatch)
+         }, {
+           id: 'debug',
+           layer: svgDebug(projection, context)
+         }, {
+           id: 'geolocate',
+           layer: svgGeolocate(projection)
+         }, {
+           id: 'touch',
+           layer: svgTouch()
+         }];
 
-                 case 'w':
-                   return minXmaxY.x;
+         function drawLayers(selection) {
+           svg = selection.selectAll('.surface').data([0]);
+           svg = svg.enter().append('svg').attr('class', 'surface').merge(svg);
+           var defs = svg.selectAll('.surface-defs').data([0]);
+           defs.enter().append('defs').attr('class', 'surface-defs');
+           var groups = svg.selectAll('.data-layer').data(_layers);
+           groups.exit().remove();
+           groups.enter().append('g').attr('class', function (d) {
+             return 'data-layer ' + d.id;
+           }).merge(groups).each(function (d) {
+             select(this).call(d.layer);
+           });
+         }
 
-                 case 's':
-                   return maxXminY.y;
+         drawLayers.all = function () {
+           return _layers;
+         };
 
-                 case 'n':
-                   return maxXminY.x;
+         drawLayers.layer = function (id) {
+           var obj = _layers.find(function (o) {
+             return o.id === id;
+           });
 
-                 case 'e':
-                   return minXmaxY.y;
+           return obj && obj.layer;
+         };
 
-                 default:
-                   return token;
-               }
-             });
-           } else if (source.type === 'tms') {
-             result = result.replace('{x}', coord[0]).replace('{y}', coord[1]) // TMS-flipped y coordinate
-             .replace(/\{[t-]y\}/, Math.pow(2, coord[2]) - coord[1] - 1).replace(/\{z(oom)?\}/, coord[2]) // only fetch retina tiles for retina screens
-             .replace(/\{@2x\}|\{r\}/, isRetina ? '@2x' : '');
-           } else if (source.type === 'bing') {
-             result = result.replace('{u}', function () {
-               var u = '';
+         drawLayers.only = function (what) {
+           var arr = [].concat(what);
 
-               for (var zoom = coord[2]; zoom > 0; zoom--) {
-                 var b = 0;
-                 var mask = 1 << zoom - 1;
-                 if ((coord[0] & mask) !== 0) b++;
-                 if ((coord[1] & mask) !== 0) b += 2;
-                 u += b.toString();
-               }
+           var all = _layers.map(function (layer) {
+             return layer.id;
+           });
 
-               return u;
-             });
-           } // these apply to any type..
+           return drawLayers.remove(utilArrayDifference(all, arr));
+         };
 
+         drawLayers.remove = function (what) {
+           var arr = [].concat(what);
+           arr.forEach(function (id) {
+             _layers = _layers.filter(function (o) {
+               return o.id !== id;
+             });
+           });
+           dispatch.call('change');
+           return this;
+         };
 
-           result = result.replace(/\{switch:([^}]+)\}/, function (s, r) {
-             var subdomains = r.split(',');
-             return subdomains[(coord[0] + coord[1]) % subdomains.length];
+         drawLayers.add = function (what) {
+           var arr = [].concat(what);
+           arr.forEach(function (obj) {
+             if ('id' in obj && 'layer' in obj) {
+               _layers.push(obj);
+             }
            });
-           return result;
-         };
-
-         source.validZoom = function (z) {
-           return source.zoomExtent[0] <= z && (source.overzoom || source.zoomExtent[1] > z);
+           dispatch.call('change');
+           return this;
          };
 
-         source.isLocatorOverlay = function () {
-           return source.id === 'mapbox_locator_overlay';
+         drawLayers.dimensions = function (val) {
+           if (!arguments.length) return utilGetDimensions(svg);
+           utilSetDimensions(svg, val);
+           return this;
          };
-         /* hides a source from the list, but leaves it available for use */
 
+         return utilRebind(drawLayers, dispatch, 'on');
+       }
 
-         source.isHidden = function () {
-           return source.id === 'DigitalGlobe-Premium-vintage' || source.id === 'DigitalGlobe-Standard-vintage';
+       function svgLines(projection, context) {
+         var detected = utilDetect();
+         var highway_stack = {
+           motorway: 0,
+           motorway_link: 1,
+           trunk: 2,
+           trunk_link: 3,
+           primary: 4,
+           primary_link: 5,
+           secondary: 6,
+           tertiary: 7,
+           unclassified: 8,
+           residential: 9,
+           service: 10,
+           footway: 11
          };
 
-         source.copyrightNotices = function () {};
+         function drawTargets(selection, graph, entities, filter) {
+           var targetClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
+           var nopeClass = context.getDebug('target') ? 'red ' : 'nocolor ';
+           var getPath = svgPath(projection).geojson;
+           var activeID = context.activeID();
+           var base = context.history().base(); // The targets and nopes will be MultiLineString sub-segments of the ways
 
-         source.getMetadata = function (center, tileCoord, callback) {
-           var vintage = {
-             start: localeDateString(source.startDate),
-             end: localeDateString(source.endDate)
-           };
-           vintage.range = vintageRange(vintage);
-           var metadata = {
-             vintage: vintage
+           var data = {
+             targets: [],
+             nopes: []
            };
-           callback(null, metadata);
-         };
+           entities.forEach(function (way) {
+             var features = svgSegmentWay(way, graph, activeID);
+             data.targets.push.apply(data.targets, features.passive);
+             data.nopes.push.apply(data.nopes, features.active);
+           }); // Targets allow hover and vertex snapping
 
-         return source;
-       }
+           var targetData = data.targets.filter(getPath);
+           var targets = selection.selectAll('.line.target-allowed').filter(function (d) {
+             return filter(d.properties.entity);
+           }).data(targetData, function key(d) {
+             return d.id;
+           }); // exit
 
-       rendererBackgroundSource.Bing = function (data, dispatch) {
-         // http://msdn.microsoft.com/en-us/library/ff701716.aspx
-         // http://msdn.microsoft.com/en-us/library/ff701701.aspx
-         data.template = 'https://ecn.t{switch:0,1,2,3}.tiles.virtualearth.net/tiles/a{u}.jpeg?g=587&mkt=en-gb&n=z';
-         var bing = rendererBackgroundSource(data); // var key = 'Arzdiw4nlOJzRwOz__qailc8NiR31Tt51dN2D7cm57NrnceZnCpgOkmJhNpGoppU'; // P2, JOSM, etc
+           targets.exit().remove();
 
-         var key = 'Ak5oTE46TUbjRp08OFVcGpkARErDobfpuyNKa-W2mQ8wbt1K1KL8p1bIRwWwcF-Q'; // iD
+           var segmentWasEdited = function segmentWasEdited(d) {
+             var wayID = d.properties.entity.id; // if the whole line was edited, don't draw segment changes
 
-         var url = 'https://dev.virtualearth.net/REST/v1/Imagery/Metadata/Aerial?include=ImageryProviders&key=' + key;
-         var cache = {};
-         var inflight = {};
-         var providers = [];
-         d3_json(url).then(function (json) {
-           providers = json.resourceSets[0].resources[0].imageryProviders.map(function (provider) {
-             return {
-               attribution: provider.attribution,
-               areas: provider.coverageAreas.map(function (area) {
-                 return {
-                   zoom: [area.zoomMin, area.zoomMax],
-                   extent: geoExtent([area.bbox[1], area.bbox[0]], [area.bbox[3], area.bbox[2]])
-                 };
-               })
-             };
-           });
-           dispatch.call('change');
-         })["catch"](function () {
-           /* ignore */
-         });
+             if (!base.entities[wayID] || !fastDeepEqual(graph.entities[wayID].nodes, base.entities[wayID].nodes)) {
+               return false;
+             }
 
-         bing.copyrightNotices = function (zoom, extent) {
-           zoom = Math.min(zoom, 21);
-           return providers.filter(function (provider) {
-             return provider.areas.some(function (area) {
-               return extent.intersects(area.extent) && area.zoom[0] <= zoom && area.zoom[1] >= zoom;
+             return d.properties.nodes.some(function (n) {
+               return !base.entities[n.id] || !fastDeepEqual(graph.entities[n.id].loc, base.entities[n.id].loc);
              });
-           }).map(function (provider) {
-             return provider.attribution;
-           }).join(', ');
-         };
+           }; // enter/update
 
-         bing.getMetadata = function (center, tileCoord, callback) {
-           var tileID = tileCoord.slice(0, 3).join('/');
-           var zoom = Math.min(tileCoord[2], 21);
-           var centerPoint = center[1] + ',' + center[0]; // lat,lng
 
-           var url = 'https://dev.virtualearth.net/REST/v1/Imagery/Metadata/Aerial/' + centerPoint + '?zl=' + zoom + '&key=' + key;
-           if (inflight[tileID]) return;
+           targets.enter().append('path').merge(targets).attr('d', getPath).attr('class', function (d) {
+             return 'way line target target-allowed ' + targetClass + d.id;
+           }).classed('segment-edited', segmentWasEdited); // NOPE
 
-           if (!cache[tileID]) {
-             cache[tileID] = {};
-           }
+           var nopeData = data.nopes.filter(getPath);
+           var nopes = selection.selectAll('.line.target-nope').filter(function (d) {
+             return filter(d.properties.entity);
+           }).data(nopeData, function key(d) {
+             return d.id;
+           }); // exit
 
-           if (cache[tileID] && cache[tileID].metadata) {
-             return callback(null, cache[tileID].metadata);
-           }
+           nopes.exit().remove(); // enter/update
 
-           inflight[tileID] = true;
-           d3_json(url).then(function (result) {
-             delete inflight[tileID];
+           nopes.enter().append('path').merge(nopes).attr('d', getPath).attr('class', function (d) {
+             return 'way line target target-nope ' + nopeClass + d.id;
+           }).classed('segment-edited', segmentWasEdited);
+         }
 
-             if (!result) {
-               throw new Error('Unknown Error');
-             }
+         function drawLines(selection, graph, entities, filter) {
+           var base = context.history().base();
 
-             var vintage = {
-               start: localeDateString(result.resourceSets[0].resources[0].vintageStart),
-               end: localeDateString(result.resourceSets[0].resources[0].vintageEnd)
-             };
-             vintage.range = vintageRange(vintage);
-             var metadata = {
-               vintage: vintage
-             };
-             cache[tileID].metadata = metadata;
-             if (callback) callback(null, metadata);
-           })["catch"](function (err) {
-             delete inflight[tileID];
-             if (callback) callback(err.message);
-           });
-         };
+           function waystack(a, b) {
+             var selected = context.selectedIDs();
+             var scoreA = selected.indexOf(a.id) !== -1 ? 20 : 0;
+             var scoreB = selected.indexOf(b.id) !== -1 ? 20 : 0;
 
-         bing.terms_url = 'https://blog.openstreetmap.org/2010/11/30/microsoft-imagery-details';
-         return bing;
-       };
+             if (a.tags.highway) {
+               scoreA -= highway_stack[a.tags.highway];
+             }
 
-       rendererBackgroundSource.Esri = function (data) {
-         // in addition to using the tilemap at zoom level 20, overzoom real tiles - #4327 (deprecated technique, but it works)
-         if (data.template.match(/blankTile/) === null) {
-           data.template = data.template + '?blankTile=false';
-         }
+             if (b.tags.highway) {
+               scoreB -= highway_stack[b.tags.highway];
+             }
 
-         var esri = rendererBackgroundSource(data);
-         var cache = {};
-         var inflight = {};
+             return scoreA - scoreB;
+           }
 
-         var _prevCenter; // use a tilemap service to set maximum zoom for esri tiles dynamically
-         // https://developers.arcgis.com/documentation/tiled-elevation-service/
+           function drawLineGroup(selection, klass, isSelected) {
+             // Note: Don't add `.selected` class in draw modes
+             var mode = context.mode();
+             var isDrawing = mode && /^draw/.test(mode.id);
+             var selectedClass = !isDrawing && isSelected ? 'selected ' : '';
+             var lines = selection.selectAll('path').filter(filter).data(getPathData(isSelected), osmEntity.key);
+             lines.exit().remove(); // Optimization: Call expensive TagClasses only on enter selection. This
+             // works because osmEntity.key is defined to include the entity v attribute.
 
+             lines.enter().append('path').attr('class', function (d) {
+               var prefix = 'way line'; // if this line isn't styled by its own tags
 
-         esri.fetchTilemap = function (center) {
-           // skip if we have already fetched a tilemap within 5km
-           if (_prevCenter && geoSphericalDistance(center, _prevCenter) < 5000) return;
-           _prevCenter = center; // tiles are available globally to zoom level 19, afterward they may or may not be present
+               if (!d.hasInterestingTags()) {
+                 var parentRelations = graph.parentRelations(d);
+                 var parentMultipolygons = parentRelations.filter(function (relation) {
+                   return relation.isMultipolygon();
+                 }); // and if it's a member of at least one multipolygon relation
 
-           var z = 20; // first generate a random url using the template
+                 if (parentMultipolygons.length > 0 && // and only multipolygon relations
+                 parentRelations.length === parentMultipolygons.length) {
+                   // then fudge the classes to style this as an area edge
+                   prefix = 'relation area';
+                 }
+               }
 
-           var dummyUrl = esri.url([1, 2, 3]); // calculate url z/y/x from the lat/long of the center of the map
+               var oldMPClass = oldMultiPolygonOuters[d.id] ? 'old-multipolygon ' : '';
+               return prefix + ' ' + klass + ' ' + selectedClass + oldMPClass + d.id;
+             }).classed('added', function (d) {
+               return !base.entities[d.id];
+             }).classed('geometry-edited', function (d) {
+               return graph.entities[d.id] && base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].nodes, base.entities[d.id].nodes);
+             }).classed('retagged', function (d) {
+               return graph.entities[d.id] && base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].tags, base.entities[d.id].tags);
+             }).call(svgTagClasses()).merge(lines).sort(waystack).attr('d', getPath).call(svgTagClasses().tags(svgRelationMemberTags(graph)));
+             return selection;
+           }
 
-           var x = Math.floor((center[0] + 180) / 360 * Math.pow(2, z));
-           var y = Math.floor((1 - Math.log(Math.tan(center[1] * Math.PI / 180) + 1 / Math.cos(center[1] * Math.PI / 180)) / Math.PI) / 2 * Math.pow(2, z)); // fetch an 8x8 grid to leverage cache
+           function getPathData(isSelected) {
+             return function () {
+               var layer = this.parentNode.__data__;
+               var data = pathdata[layer] || [];
+               return data.filter(function (d) {
+                 if (isSelected) {
+                   return context.selectedIDs().indexOf(d.id) !== -1;
+                 } else {
+                   return context.selectedIDs().indexOf(d.id) === -1;
+                 }
+               });
+             };
+           }
 
-           var tilemapUrl = dummyUrl.replace(/tile\/[0-9]+\/[0-9]+\/[0-9]+\?blankTile=false/, 'tilemap') + '/' + z + '/' + y + '/' + x + '/8/8'; // make the request and introspect the response from the tilemap server
+           function addMarkers(layergroup, pathclass, groupclass, groupdata, marker) {
+             var markergroup = layergroup.selectAll('g.' + groupclass).data([pathclass]);
+             markergroup = markergroup.enter().append('g').attr('class', groupclass).merge(markergroup);
+             var markers = markergroup.selectAll('path').filter(filter).data(function data() {
+               return groupdata[this.parentNode.__data__] || [];
+             }, function key(d) {
+               return [d.id, d.index];
+             });
+             markers.exit().remove();
+             markers = markers.enter().append('path').attr('class', pathclass).merge(markers).attr('marker-mid', marker).attr('d', function (d) {
+               return d.d;
+             });
 
-           d3_json(tilemapUrl).then(function (tilemap) {
-             if (!tilemap) {
-               throw new Error('Unknown Error');
+             if (detected.ie) {
+               markers.each(function () {
+                 this.parentNode.insertBefore(this, this);
+               });
              }
+           }
 
-             var hasTiles = true;
+           var getPath = svgPath(projection, graph);
+           var ways = [];
+           var onewaydata = {};
+           var sideddata = {};
+           var oldMultiPolygonOuters = {};
 
-             for (var i = 0; i < tilemap.data.length; i++) {
-               // 0 means an individual tile in the grid doesn't exist
-               if (!tilemap.data[i]) {
-                 hasTiles = false;
-                 break;
-               }
-             } // if any tiles are missing at level 20 we restrict maxZoom to 19
+           for (var i = 0; i < entities.length; i++) {
+             var entity = entities[i];
+             var outer = osmOldMultipolygonOuterMember(entity, graph);
 
+             if (outer) {
+               ways.push(entity.mergeTags(outer.tags));
+               oldMultiPolygonOuters[outer.id] = true;
+             } else if (entity.geometry(graph) === 'line') {
+               ways.push(entity);
+             }
+           }
 
-             esri.zoomExtent[1] = hasTiles ? 22 : 19;
-           })["catch"](function () {
-             /* ignore */
+           ways = ways.filter(getPath);
+           var pathdata = utilArrayGroupBy(ways, function (way) {
+             return way.layer();
            });
-         };
+           Object.keys(pathdata).forEach(function (k) {
+             var v = pathdata[k];
+             var onewayArr = v.filter(function (d) {
+               return d.isOneWay();
+             });
+             var onewaySegments = svgMarkerSegments(projection, graph, 35, function shouldReverse(entity) {
+               return entity.tags.oneway === '-1';
+             }, function bothDirections(entity) {
+               return entity.tags.oneway === 'reversible' || entity.tags.oneway === 'alternating';
+             });
+             onewaydata[k] = utilArrayFlatten(onewayArr.map(onewaySegments));
+             var sidedArr = v.filter(function (d) {
+               return d.isSided();
+             });
+             var sidedSegments = svgMarkerSegments(projection, graph, 30, function shouldReverse() {
+               return false;
+             }, function bothDirections() {
+               return false;
+             });
+             sideddata[k] = utilArrayFlatten(sidedArr.map(sidedSegments));
+           });
+           var covered = selection.selectAll('.layer-osm.covered'); // under areas
 
-         esri.getMetadata = function (center, tileCoord, callback) {
-           var tileID = tileCoord.slice(0, 3).join('/');
-           var zoom = Math.min(tileCoord[2], esri.zoomExtent[1]);
-           var centerPoint = center[0] + ',' + center[1]; // long, lat (as it should be)
+           var uncovered = selection.selectAll('.layer-osm.lines'); // over areas
 
-           var unknown = _t('info_panels.background.unknown');
-           var metadataLayer;
-           var vintage = {};
-           var metadata = {};
-           if (inflight[tileID]) return;
+           var touchLayer = selection.selectAll('.layer-touch.lines'); // Draw lines..
 
-           switch (true) {
-             case zoom >= 20 && esri.id === 'EsriWorldImageryClarity':
-               metadataLayer = 4;
-               break;
+           [covered, uncovered].forEach(function (selection) {
+             var range = selection === covered ? range$1(-10, 0) : range$1(0, 11);
+             var layergroup = selection.selectAll('g.layergroup').data(range);
+             layergroup = layergroup.enter().append('g').attr('class', function (d) {
+               return 'layergroup layer' + String(d);
+             }).merge(layergroup);
+             layergroup.selectAll('g.linegroup').data(['shadow', 'casing', 'stroke', 'shadow-highlighted', 'casing-highlighted', 'stroke-highlighted']).enter().append('g').attr('class', function (d) {
+               return 'linegroup line-' + d;
+             });
+             layergroup.selectAll('g.line-shadow').call(drawLineGroup, 'shadow', false);
+             layergroup.selectAll('g.line-casing').call(drawLineGroup, 'casing', false);
+             layergroup.selectAll('g.line-stroke').call(drawLineGroup, 'stroke', false);
+             layergroup.selectAll('g.line-shadow-highlighted').call(drawLineGroup, 'shadow', true);
+             layergroup.selectAll('g.line-casing-highlighted').call(drawLineGroup, 'casing', true);
+             layergroup.selectAll('g.line-stroke-highlighted').call(drawLineGroup, 'stroke', true);
+             addMarkers(layergroup, 'oneway', 'onewaygroup', onewaydata, 'url(#ideditor-oneway-marker)');
+             addMarkers(layergroup, 'sided', 'sidedgroup', sideddata, function marker(d) {
+               var category = graph.entity(d.id).sidednessIdentifier();
+               return 'url(#ideditor-sided-marker-' + category + ')';
+             });
+           }); // Draw touch targets..
 
-             case zoom >= 19:
-               metadataLayer = 3;
-               break;
+           touchLayer.call(drawTargets, graph, ways, filter);
+         }
 
-             case zoom >= 17:
-               metadataLayer = 2;
-               break;
+         return drawLines;
+       }
 
-             case zoom >= 13:
-               metadataLayer = 0;
-               break;
+       function svgMidpoints(projection, context) {
+         var targetRadius = 8;
 
-             default:
-               metadataLayer = 99;
-           }
+         function drawTargets(selection, graph, entities, filter) {
+           var fillClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
+           var getTransform = svgPointTransform(projection).geojson;
+           var data = entities.map(function (midpoint) {
+             return {
+               type: 'Feature',
+               id: midpoint.id,
+               properties: {
+                 target: true,
+                 entity: midpoint
+               },
+               geometry: {
+                 type: 'Point',
+                 coordinates: midpoint.loc
+               }
+             };
+           });
+           var targets = selection.selectAll('.midpoint.target').filter(function (d) {
+             return filter(d.properties.entity);
+           }).data(data, function key(d) {
+             return d.id;
+           }); // exit
 
-           var url; // build up query using the layer appropriate to the current zoom
+           targets.exit().remove(); // enter/update
 
-           if (esri.id === 'EsriWorldImagery') {
-             url = 'https://services.arcgisonline.com/arcgis/rest/services/World_Imagery/MapServer/';
-           } else if (esri.id === 'EsriWorldImageryClarity') {
-             url = 'https://serviceslab.arcgisonline.com/arcgis/rest/services/Clarity_World_Imagery/MapServer/';
-           }
+           targets.enter().append('circle').attr('r', targetRadius).merge(targets).attr('class', function (d) {
+             return 'node midpoint target ' + fillClass + d.id;
+           }).attr('transform', getTransform);
+         }
 
-           url += metadataLayer + '/query?returnGeometry=false&geometry=' + centerPoint + '&inSR=4326&geometryType=esriGeometryPoint&outFields=*&f=json';
+         function drawMidpoints(selection, graph, entities, filter, extent) {
+           var drawLayer = selection.selectAll('.layer-osm.points .points-group.midpoints');
+           var touchLayer = selection.selectAll('.layer-touch.points');
+           var mode = context.mode();
 
-           if (!cache[tileID]) {
-             cache[tileID] = {};
+           if (mode && mode.id !== 'select' || !context.map().withinEditableZoom()) {
+             drawLayer.selectAll('.midpoint').remove();
+             touchLayer.selectAll('.midpoint.target').remove();
+             return;
            }
 
-           if (cache[tileID] && cache[tileID].metadata) {
-             return callback(null, cache[tileID].metadata);
-           } // accurate metadata is only available >= 13
-
+           var poly = extent.polygon();
+           var midpoints = {};
 
-           if (metadataLayer === 99) {
-             vintage = {
-               start: null,
-               end: null,
-               range: null
-             };
-             metadata = {
-               vintage: null,
-               source: unknown,
-               description: unknown,
-               resolution: unknown,
-               accuracy: unknown
-             };
-             callback(null, metadata);
-           } else {
-             inflight[tileID] = true;
-             d3_json(url).then(function (result) {
-               delete inflight[tileID];
+           for (var i = 0; i < entities.length; i++) {
+             var entity = entities[i];
+             if (entity.type !== 'way') continue;
+             if (!filter(entity)) continue;
+             if (context.selectedIDs().indexOf(entity.id) < 0) continue;
+             var nodes = graph.childNodes(entity);
 
-               if (!result) {
-                 throw new Error('Unknown Error');
-               } else if (result.features && result.features.length < 1) {
-                 throw new Error('No Results');
-               } else if (result.error && result.error.message) {
-                 throw new Error(result.error.message);
-               } // pass through the discrete capture date from metadata
+             for (var j = 0; j < nodes.length - 1; j++) {
+               var a = nodes[j];
+               var b = nodes[j + 1];
+               var id = [a.id, b.id].sort().join('-');
 
+               if (midpoints[id]) {
+                 midpoints[id].parents.push(entity);
+               } else if (geoVecLength(projection(a.loc), projection(b.loc)) > 40) {
+                 var point = geoVecInterp(a.loc, b.loc, 0.5);
+                 var loc = null;
 
-               var captureDate = localeDateString(result.features[0].attributes.SRC_DATE2);
-               vintage = {
-                 start: captureDate,
-                 end: captureDate,
-                 range: captureDate
-               };
-               metadata = {
-                 vintage: vintage,
-                 source: clean(result.features[0].attributes.NICE_NAME),
-                 description: clean(result.features[0].attributes.NICE_DESC),
-                 resolution: clean(+parseFloat(result.features[0].attributes.SRC_RES).toFixed(4)),
-                 accuracy: clean(+parseFloat(result.features[0].attributes.SRC_ACC).toFixed(4))
-               }; // append units - meters
+                 if (extent.intersects(point)) {
+                   loc = point;
+                 } else {
+                   for (var k = 0; k < 4; k++) {
+                     point = geoLineIntersection([a.loc, b.loc], [poly[k], poly[k + 1]]);
 
-               if (isFinite(metadata.resolution)) {
-                 metadata.resolution += ' m';
-               }
+                     if (point && geoVecLength(projection(a.loc), projection(point)) > 20 && geoVecLength(projection(b.loc), projection(point)) > 20) {
+                       loc = point;
+                       break;
+                     }
+                   }
+                 }
 
-               if (isFinite(metadata.accuracy)) {
-                 metadata.accuracy += ' m';
+                 if (loc) {
+                   midpoints[id] = {
+                     type: 'midpoint',
+                     id: id,
+                     loc: loc,
+                     edge: [a.id, b.id],
+                     parents: [entity]
+                   };
+                 }
                }
+             }
+           }
 
-               cache[tileID].metadata = metadata;
-               if (callback) callback(null, metadata);
-             })["catch"](function (err) {
-               delete inflight[tileID];
-               if (callback) callback(err.message);
-             });
-           }
-
-           function clean(val) {
-             return String(val).trim() || unknown;
-           }
-         };
-
-         return esri;
-       };
-
-       rendererBackgroundSource.None = function () {
-         var source = rendererBackgroundSource({
-           id: 'none',
-           template: ''
-         });
-
-         source.name = function () {
-           return _t('background.none');
-         };
-
-         source.label = function () {
-           return _t.html('background.none');
-         };
-
-         source.imageryUsed = function () {
-           return null;
-         };
+           function midpointFilter(d) {
+             if (midpoints[d.id]) return true;
 
-         source.area = function () {
-           return -1; // sources in background pane are sorted by area
-         };
+             for (var i = 0; i < d.parents.length; i++) {
+               if (filter(d.parents[i])) {
+                 return true;
+               }
+             }
 
-         return source;
-       };
+             return false;
+           }
 
-       rendererBackgroundSource.Custom = function (template) {
-         var source = rendererBackgroundSource({
-           id: 'custom',
-           template: template
-         });
+           var groups = drawLayer.selectAll('.midpoint').filter(midpointFilter).data(Object.values(midpoints), function (d) {
+             return d.id;
+           });
+           groups.exit().remove();
+           var enter = groups.enter().insert('g', ':first-child').attr('class', 'midpoint');
+           enter.append('polygon').attr('points', '-6,8 10,0 -6,-8').attr('class', 'shadow');
+           enter.append('polygon').attr('points', '-3,4 5,0 -3,-4').attr('class', 'fill');
+           groups = groups.merge(enter).attr('transform', function (d) {
+             var translate = svgPointTransform(projection);
+             var a = graph.entity(d.edge[0]);
+             var b = graph.entity(d.edge[1]);
+             var angle = geoAngle(a, b, projection) * (180 / Math.PI);
+             return translate(d) + ' rotate(' + angle + ')';
+           }).call(svgTagClasses().tags(function (d) {
+             return d.parents[0].tags;
+           })); // Propagate data bindings.
 
-         source.name = function () {
-           return _t('background.custom');
-         };
+           groups.select('polygon.shadow');
+           groups.select('polygon.fill'); // Draw touch targets..
 
-         source.label = function () {
-           return _t.html('background.custom');
-         };
+           touchLayer.call(drawTargets, graph, Object.values(midpoints), midpointFilter);
+         }
 
-         source.imageryUsed = function () {
-           // sanitize personal connection tokens - #6801
-           var cleaned = source.template(); // from query string parameters
+         return drawMidpoints;
+       }
 
-           if (cleaned.indexOf('?') !== -1) {
-             var parts = cleaned.split('?', 2);
-             var qs = utilStringQs(parts[1]);
-             ['access_token', 'connectId', 'token'].forEach(function (param) {
-               if (qs[param]) {
-                 qs[param] = '{apikey}';
-               }
-             });
-             cleaned = parts[0] + '?' + utilQsString(qs, true); // true = soft encode
-           } // from wms/wmts api path parameters
+       function svgPoints(projection, context) {
+         function markerPath(selection, klass) {
+           selection.attr('class', klass).attr('transform', 'translate(-8, -23)').attr('d', 'M 17,8 C 17,13 11,21 8.5,23.5 C 6,21 0,13 0,8 C 0,4 4,-0.5 8.5,-0.5 C 13,-0.5 17,4 17,8 z');
+         }
 
+         function sortY(a, b) {
+           return b.loc[1] - a.loc[1];
+         } // Avoid exit/enter if we're just moving stuff around.
+         // The node will get a new version but we only need to run the update selection.
 
-           cleaned = cleaned.replace(/token\/(\w+)/, 'token/{apikey}');
-           return 'Custom (' + cleaned + ' )';
-         };
 
-         source.area = function () {
-           return -2; // sources in background pane are sorted by area
-         };
+         function fastEntityKey(d) {
+           var mode = context.mode();
+           var isMoving = mode && /^(add|draw|drag|move|rotate)/.test(mode.id);
+           return isMoving ? d.id : osmEntity.key(d);
+         }
 
-         return source;
-       };
+         function drawTargets(selection, graph, entities, filter) {
+           var fillClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
+           var getTransform = svgPointTransform(projection).geojson;
+           var activeID = context.activeID();
+           var data = [];
+           entities.forEach(function (node) {
+             if (activeID === node.id) return; // draw no target on the activeID
 
-       function rendererTileLayer(context) {
-         var transformProp = utilPrefixCSSProperty('Transform');
-         var tiler = utilTiler();
-         var _tileSize = 256;
+             data.push({
+               type: 'Feature',
+               id: node.id,
+               properties: {
+                 target: true,
+                 entity: node
+               },
+               geometry: node.asGeoJSON()
+             });
+           });
+           var targets = selection.selectAll('.point.target').filter(function (d) {
+             return filter(d.properties.entity);
+           }).data(data, function key(d) {
+             return d.id;
+           }); // exit
 
-         var _projection;
+           targets.exit().remove(); // enter/update
 
-         var _cache = {};
+           targets.enter().append('rect').attr('x', -10).attr('y', -26).attr('width', 20).attr('height', 30).merge(targets).attr('class', function (d) {
+             return 'node point target ' + fillClass + d.id;
+           }).attr('transform', getTransform);
+         }
 
-         var _tileOrigin;
+         function drawPoints(selection, graph, entities, filter) {
+           var wireframe = context.surface().classed('fill-wireframe');
+           var zoom = geoScaleToZoom(projection.scale());
+           var base = context.history().base(); // Points with a direction will render as vertices at higher zooms..
 
-         var _zoom;
+           function renderAsPoint(entity) {
+             return entity.geometry(graph) === 'point' && !(zoom >= 18 && entity.directions(graph, projection).length);
+           } // All points will render as vertices in wireframe mode too..
 
-         var _source;
 
-         function tileSizeAtZoom(d, z) {
-           var EPSILON = 0.002; // close seams
+           var points = wireframe ? [] : entities.filter(renderAsPoint);
+           points.sort(sortY);
+           var drawLayer = selection.selectAll('.layer-osm.points .points-group.points');
+           var touchLayer = selection.selectAll('.layer-touch.points'); // Draw points..
 
-           return _tileSize * Math.pow(2, z - d[2]) / _tileSize + EPSILON;
-         }
+           var groups = drawLayer.selectAll('g.point').filter(filter).data(points, fastEntityKey);
+           groups.exit().remove();
+           var enter = groups.enter().append('g').attr('class', function (d) {
+             return 'node point ' + d.id;
+           }).order();
+           enter.append('path').call(markerPath, 'shadow');
+           enter.append('ellipse').attr('cx', 0.5).attr('cy', 1).attr('rx', 6.5).attr('ry', 3).attr('class', 'stroke');
+           enter.append('path').call(markerPath, 'stroke');
+           enter.append('use').attr('transform', 'translate(-5, -19)').attr('class', 'icon').attr('width', '11px').attr('height', '11px');
+           groups = groups.merge(enter).attr('transform', svgPointTransform(projection)).classed('added', function (d) {
+             return !base.entities[d.id]; // if it doesn't exist in the base graph, it's new
+           }).classed('moved', function (d) {
+             return base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].loc, base.entities[d.id].loc);
+           }).classed('retagged', function (d) {
+             return base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].tags, base.entities[d.id].tags);
+           }).call(svgTagClasses());
+           groups.select('.shadow'); // propagate bound data
 
-         function atZoom(t, distance) {
-           var power = Math.pow(2, distance);
-           return [Math.floor(t[0] * power), Math.floor(t[1] * power), t[2] + distance];
-         }
+           groups.select('.stroke'); // propagate bound data
 
-         function lookUp(d) {
-           for (var up = -1; up > -d[2]; up--) {
-             var tile = atZoom(d, up);
+           groups.select('.icon') // propagate bound data
+           .attr('xlink:href', function (entity) {
+             var preset = _mainPresetIndex.match(entity, graph);
+             var picon = preset && preset.icon;
 
-             if (_cache[_source.url(tile)] !== false) {
-               return tile;
+             if (!picon) {
+               return '';
+             } else {
+               var isMaki = /^maki-/.test(picon);
+               return '#' + picon + (isMaki ? '-11' : '');
              }
-           }
-         }
+           }); // Draw touch targets..
 
-         function uniqueBy(a, n) {
-           var o = [];
-           var seen = {};
+           touchLayer.call(drawTargets, graph, points, filter);
+         }
 
-           for (var i = 0; i < a.length; i++) {
-             if (seen[a[i][n]] === undefined) {
-               o.push(a[i]);
-               seen[a[i][n]] = true;
-             }
-           }
+         return drawPoints;
+       }
 
-           return o;
+       function svgTurns(projection, context) {
+         function icon(turn) {
+           var u = turn.u ? '-u' : '';
+           if (turn.no) return '#iD-turn-no' + u;
+           if (turn.only) return '#iD-turn-only' + u;
+           return '#iD-turn-yes' + u;
          }
 
-         function addSource(d) {
-           d.push(_source.url(d));
-           return d;
-         } // Update tiles based on current state of `projection`.
-
+         function drawTurns(selection, graph, turns) {
+           function turnTransform(d) {
+             var pxRadius = 50;
+             var toWay = graph.entity(d.to.way);
+             var toPoints = graph.childNodes(toWay).map(function (n) {
+               return n.loc;
+             }).map(projection);
+             var toLength = geoPathLength(toPoints);
+             var mid = toLength / 2; // midpoint of destination way
 
-         function background(selection) {
-           _zoom = geoScaleToZoom(_projection.scale(), _tileSize);
-           var pixelOffset;
+             var toNode = graph.entity(d.to.node);
+             var toVertex = graph.entity(d.to.vertex);
+             var a = geoAngle(toVertex, toNode, projection);
+             var o = projection(toVertex.loc);
+             var r = d.u ? 0 // u-turn: no radius
+             : !toWay.__via ? pxRadius // leaf way: put marker at pxRadius
+             : Math.min(mid, pxRadius); // via way: prefer pxRadius, fallback to mid for very short ways
 
-           if (_source) {
-             pixelOffset = [_source.offset()[0] * Math.pow(2, _zoom), _source.offset()[1] * Math.pow(2, _zoom)];
-           } else {
-             pixelOffset = [0, 0];
+             return 'translate(' + (r * Math.cos(a) + o[0]) + ',' + (r * Math.sin(a) + o[1]) + ') ' + 'rotate(' + a * 180 / Math.PI + ')';
            }
 
-           var translate = [_projection.translate()[0] + pixelOffset[0], _projection.translate()[1] + pixelOffset[1]];
-           tiler.scale(_projection.scale() * 2 * Math.PI).translate(translate);
-           _tileOrigin = [_projection.scale() * Math.PI - translate[0], _projection.scale() * Math.PI - translate[1]];
-           render(selection);
-         } // Derive the tiles onscreen, remove those offscreen and position them.
-         // Important that this part not depend on `_projection` because it's
-         // rentered when tiles load/error (see #644).
+           var drawLayer = selection.selectAll('.layer-osm.points .points-group.turns');
+           var touchLayer = selection.selectAll('.layer-touch.turns'); // Draw turns..
 
+           var groups = drawLayer.selectAll('g.turn').data(turns, function (d) {
+             return d.key;
+           }); // exit
 
-         function render(selection) {
-           if (!_source) return;
-           var requests = [];
-           var showDebug = context.getDebug('tile') && !_source.overlay;
+           groups.exit().remove(); // enter
 
-           if (_source.validZoom(_zoom)) {
-             tiler.skipNullIsland(!!_source.overlay);
-             tiler().forEach(function (d) {
-               addSource(d);
-               if (d[3] === '') return;
-               if (typeof d[3] !== 'string') return; // Workaround for #2295
+           var groupsEnter = groups.enter().append('g').attr('class', function (d) {
+             return 'turn ' + d.key;
+           });
+           var turnsEnter = groupsEnter.filter(function (d) {
+             return !d.u;
+           });
+           turnsEnter.append('rect').attr('transform', 'translate(-22, -12)').attr('width', '44').attr('height', '24');
+           turnsEnter.append('use').attr('transform', 'translate(-22, -12)').attr('width', '44').attr('height', '24');
+           var uEnter = groupsEnter.filter(function (d) {
+             return d.u;
+           });
+           uEnter.append('circle').attr('r', '16');
+           uEnter.append('use').attr('transform', 'translate(-16, -16)').attr('width', '32').attr('height', '32'); // update
 
-               requests.push(d);
+           groups = groups.merge(groupsEnter).attr('opacity', function (d) {
+             return d.direct === false ? '0.7' : null;
+           }).attr('transform', turnTransform);
+           groups.select('use').attr('xlink:href', icon);
+           groups.select('rect'); // propagate bound data
 
-               if (_cache[d[3]] === false && lookUp(d)) {
-                 requests.push(addSource(lookUp(d)));
-               }
-             });
-             requests = uniqueBy(requests, 3).filter(function (r) {
-               // don't re-request tiles which have failed in the past
-               return _cache[r[3]] !== false;
-             });
-           }
+           groups.select('circle'); // propagate bound data
+           // Draw touch targets..
 
-           function load(d3_event, d) {
-             _cache[d[3]] = true;
-             select(this).on('error', null).on('load', null).classed('tile-loaded', true);
-             render(selection);
-           }
+           var fillClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
+           groups = touchLayer.selectAll('g.turn').data(turns, function (d) {
+             return d.key;
+           }); // exit
 
-           function error(d3_event, d) {
-             _cache[d[3]] = false;
-             select(this).on('error', null).on('load', null).remove();
-             render(selection);
-           }
+           groups.exit().remove(); // enter
 
-           function imageTransform(d) {
-             var ts = _tileSize * Math.pow(2, _zoom - d[2]);
+           groupsEnter = groups.enter().append('g').attr('class', function (d) {
+             return 'turn ' + d.key;
+           });
+           turnsEnter = groupsEnter.filter(function (d) {
+             return !d.u;
+           });
+           turnsEnter.append('rect').attr('class', 'target ' + fillClass).attr('transform', 'translate(-22, -12)').attr('width', '44').attr('height', '24');
+           uEnter = groupsEnter.filter(function (d) {
+             return d.u;
+           });
+           uEnter.append('circle').attr('class', 'target ' + fillClass).attr('r', '16'); // update
 
-             var scale = tileSizeAtZoom(d, _zoom);
-             return 'translate(' + (d[0] * ts - _tileOrigin[0]) + 'px,' + (d[1] * ts - _tileOrigin[1]) + 'px) ' + 'scale(' + scale + ',' + scale + ')';
-           }
+           groups = groups.merge(groupsEnter).attr('transform', turnTransform);
+           groups.select('rect'); // propagate bound data
 
-           function tileCenter(d) {
-             var ts = _tileSize * Math.pow(2, _zoom - d[2]);
+           groups.select('circle'); // propagate bound data
 
-             return [d[0] * ts - _tileOrigin[0] + ts / 2, d[1] * ts - _tileOrigin[1] + ts / 2];
-           }
+           return this;
+         }
 
-           function debugTransform(d) {
-             var coord = tileCenter(d);
-             return 'translate(' + coord[0] + 'px,' + coord[1] + 'px)';
-           } // Pick a representative tile near the center of the viewport
-           // (This is useful for sampling the imagery vintage)
+         return drawTurns;
+       }
 
+       function svgVertices(projection, context) {
+         var radiuses = {
+           //       z16-, z17,   z18+,  w/icon
+           shadow: [6, 7.5, 7.5, 12],
+           stroke: [2.5, 3.5, 3.5, 8],
+           fill: [1, 1.5, 1.5, 1.5]
+         };
 
-           var dims = tiler.size();
-           var mapCenter = [dims[0] / 2, dims[1] / 2];
-           var minDist = Math.max(dims[0], dims[1]);
-           var nearCenter;
-           requests.forEach(function (d) {
-             var c = tileCenter(d);
-             var dist = geoVecLength(c, mapCenter);
+         var _currHoverTarget;
 
-             if (dist < minDist) {
-               minDist = dist;
-               nearCenter = d;
-             }
-           });
-           var image = selection.selectAll('img').data(requests, function (d) {
-             return d[3];
-           });
-           image.exit().style(transformProp, imageTransform).classed('tile-removing', true).classed('tile-center', false).each(function () {
-             var tile = select(this);
-             window.setTimeout(function () {
-               if (tile.classed('tile-removing')) {
-                 tile.remove();
-               }
-             }, 300);
-           });
-           image.enter().append('img').attr('class', 'tile').attr('draggable', 'false').style('width', _tileSize + 'px').style('height', _tileSize + 'px').attr('src', function (d) {
-             return d[3];
-           }).on('error', error).on('load', load).merge(image).style(transformProp, imageTransform).classed('tile-debug', showDebug).classed('tile-removing', false).classed('tile-center', function (d) {
-             return d === nearCenter;
-           });
-           var debug = selection.selectAll('.tile-label-debug').data(showDebug ? requests : [], function (d) {
-             return d[3];
-           });
-           debug.exit().remove();
+         var _currPersistent = {};
+         var _currHover = {};
+         var _prevHover = {};
+         var _currSelected = {};
+         var _prevSelected = {};
+         var _radii = {};
 
-           if (showDebug) {
-             var debugEnter = debug.enter().append('div').attr('class', 'tile-label-debug');
-             debugEnter.append('div').attr('class', 'tile-label-debug-coord');
-             debugEnter.append('div').attr('class', 'tile-label-debug-vintage');
-             debug = debug.merge(debugEnter);
-             debug.style(transformProp, debugTransform);
-             debug.selectAll('.tile-label-debug-coord').html(function (d) {
-               return d[2] + ' / ' + d[0] + ' / ' + d[1];
-             });
-             debug.selectAll('.tile-label-debug-vintage').each(function (d) {
-               var span = select(this);
-               var center = context.projection.invert(tileCenter(d));
+         function sortY(a, b) {
+           return b.loc[1] - a.loc[1];
+         } // Avoid exit/enter if we're just moving stuff around.
+         // The node will get a new version but we only need to run the update selection.
 
-               _source.getMetadata(center, d, function (err, result) {
-                 span.html(result && result.vintage && result.vintage.range || _t('info_panels.background.vintage') + ': ' + _t('info_panels.background.unknown'));
-               });
-             });
-           }
+
+         function fastEntityKey(d) {
+           var mode = context.mode();
+           var isMoving = mode && /^(add|draw|drag|move|rotate)/.test(mode.id);
+           return isMoving ? d.id : osmEntity.key(d);
          }
 
-         background.projection = function (val) {
-           if (!arguments.length) return _projection;
-           _projection = val;
-           return background;
-         };
+         function draw(selection, graph, vertices, sets, filter) {
+           sets = sets || {
+             selected: {},
+             important: {},
+             hovered: {}
+           };
+           var icons = {};
+           var directions = {};
+           var wireframe = context.surface().classed('fill-wireframe');
+           var zoom = geoScaleToZoom(projection.scale());
+           var z = zoom < 17 ? 0 : zoom < 18 ? 1 : 2;
+           var activeID = context.activeID();
+           var base = context.history().base();
 
-         background.dimensions = function (val) {
-           if (!arguments.length) return tiler.size();
-           tiler.size(val);
-           return background;
-         };
+           function getIcon(d) {
+             // always check latest entity, as fastEntityKey avoids enter/exit now
+             var entity = graph.entity(d.id);
+             if (entity.id in icons) return icons[entity.id];
+             icons[entity.id] = entity.hasInterestingTags() && _mainPresetIndex.match(entity, graph).icon;
+             return icons[entity.id];
+           } // memoize directions results, return false for empty arrays (for use in filter)
 
-         background.source = function (val) {
-           if (!arguments.length) return _source;
-           _source = val;
-           _tileSize = _source.tileSize;
-           _cache = {};
-           tiler.tileSize(_source.tileSize).zoomExtent(_source.zoomExtent);
-           return background;
-         };
 
-         return background;
-       }
+           function getDirections(entity) {
+             if (entity.id in directions) return directions[entity.id];
+             var angles = entity.directions(graph, projection);
+             directions[entity.id] = angles.length ? angles : false;
+             return angles;
+           }
 
-       var _imageryIndex = null;
-       function rendererBackground(context) {
-         var dispatch$1 = dispatch('change');
-         var detected = utilDetect();
-         var baseLayer = rendererTileLayer(context).projection(context.projection);
-         var _isValid = true;
-         var _overlayLayers = [];
-         var _brightness = 1;
-         var _contrast = 1;
-         var _saturation = 1;
-         var _sharpness = 1;
+           function updateAttributes(selection) {
+             ['shadow', 'stroke', 'fill'].forEach(function (klass) {
+               var rads = radiuses[klass];
+               selection.selectAll('.' + klass).each(function (entity) {
+                 var i = z && getIcon(entity);
+                 var r = rads[i ? 3 : z]; // slightly increase the size of unconnected endpoints #3775
 
-         function ensureImageryIndex() {
-           return _mainFileFetcher.get('imagery').then(function (sources) {
-             if (_imageryIndex) return _imageryIndex;
-             _imageryIndex = {
-               imagery: sources,
-               features: {}
-             }; // use which-polygon to support efficient index and querying for imagery
+                 if (entity.id !== activeID && entity.isEndpoint(graph) && !entity.isConnected(graph)) {
+                   r += 1.5;
+                 }
 
-             var features = sources.map(function (source) {
-               if (!source.polygon) return null; // workaround for editor-layer-index weirdness..
-               // Add an extra array nest to each element in `source.polygon`
-               // so the rings are not treated as a bunch of holes:
-               // what we have: [ [[outer],[hole],[hole]] ]
-               // what we want: [ [[outer]],[[outer]],[[outer]] ]
+                 if (klass === 'shadow') {
+                   // remember this value, so we don't need to
+                   _radii[entity.id] = r; // recompute it when we draw the touch targets
+                 }
 
-               var rings = source.polygon.map(function (ring) {
-                 return [ring];
+                 select(this).attr('r', r).attr('visibility', i && klass === 'fill' ? 'hidden' : null);
                });
-               var feature = {
-                 type: 'Feature',
-                 properties: {
-                   id: source.id
-                 },
-                 geometry: {
-                   type: 'MultiPolygon',
-                   coordinates: rings
-                 }
-               };
-               _imageryIndex.features[source.id] = feature;
-               return feature;
-             }).filter(Boolean);
-             _imageryIndex.query = whichPolygon_1({
-               type: 'FeatureCollection',
-               features: features
-             }); // Instantiate `rendererBackgroundSource` objects for each source
+             });
+           }
+
+           vertices.sort(sortY);
+           var groups = selection.selectAll('g.vertex').filter(filter).data(vertices, fastEntityKey); // exit
+
+           groups.exit().remove(); // enter
+
+           var enter = groups.enter().append('g').attr('class', function (d) {
+             return 'node vertex ' + d.id;
+           }).order();
+           enter.append('circle').attr('class', 'shadow');
+           enter.append('circle').attr('class', 'stroke'); // Vertices with tags get a fill.
 
-             _imageryIndex.backgrounds = sources.map(function (source) {
-               if (source.type === 'bing') {
-                 return rendererBackgroundSource.Bing(source, dispatch$1);
-               } else if (/^EsriWorldImagery/.test(source.id)) {
-                 return rendererBackgroundSource.Esri(source);
-               } else {
-                 return rendererBackgroundSource(source);
-               }
-             }); // Add 'None'
+           enter.filter(function (d) {
+             return d.hasInterestingTags();
+           }).append('circle').attr('class', 'fill'); // update
 
-             _imageryIndex.backgrounds.unshift(rendererBackgroundSource.None()); // Add 'Custom'
+           groups = groups.merge(enter).attr('transform', svgPointTransform(projection)).classed('sibling', function (d) {
+             return d.id in sets.selected;
+           }).classed('shared', function (d) {
+             return graph.isShared(d);
+           }).classed('endpoint', function (d) {
+             return d.isEndpoint(graph);
+           }).classed('added', function (d) {
+             return !base.entities[d.id]; // if it doesn't exist in the base graph, it's new
+           }).classed('moved', function (d) {
+             return base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].loc, base.entities[d.id].loc);
+           }).classed('retagged', function (d) {
+             return base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].tags, base.entities[d.id].tags);
+           }).call(updateAttributes); // Vertices with icons get a `use`.
 
+           var iconUse = groups.selectAll('.icon').data(function data(d) {
+             return zoom >= 17 && getIcon(d) ? [d] : [];
+           }, fastEntityKey); // exit
 
-             var template = corePreferences('background-custom-template') || '';
-             var custom = rendererBackgroundSource.Custom(template);
+           iconUse.exit().remove(); // enter
 
-             _imageryIndex.backgrounds.unshift(custom);
+           iconUse.enter().append('use').attr('class', 'icon').attr('width', '11px').attr('height', '11px').attr('transform', 'translate(-5.5, -5.5)').attr('xlink:href', function (d) {
+             var picon = getIcon(d);
+             var isMaki = /^maki-/.test(picon);
+             return '#' + picon + (isMaki ? '-11' : '');
+           }); // Vertices with directions get viewfields
 
-             return _imageryIndex;
-           });
-         }
+           var dgroups = groups.selectAll('.viewfieldgroup').data(function data(d) {
+             return zoom >= 18 && getDirections(d) ? [d] : [];
+           }, fastEntityKey); // exit
 
-         function background(selection) {
-           var currSource = baseLayer.source(); // If we are displaying an Esri basemap at high zoom,
-           // check its tilemap to see how high the zoom can go
+           dgroups.exit().remove(); // enter/update
 
-           if (context.map().zoom() > 18) {
-             if (currSource && /^EsriWorldImagery/.test(currSource.id)) {
-               var center = context.map().center();
-               currSource.fetchTilemap(center);
-             }
-           } // Is the imagery valid here? - #4827
+           dgroups = dgroups.enter().insert('g', '.shadow').attr('class', 'viewfieldgroup').merge(dgroups);
+           var viewfields = dgroups.selectAll('.viewfield').data(getDirections, function key(d) {
+             return osmEntity.key(d);
+           }); // exit
 
+           viewfields.exit().remove(); // enter/update
 
-           var sources = background.sources(context.map().extent());
-           var wasValid = _isValid;
-           _isValid = !!sources.filter(function (d) {
-             return d === currSource;
-           }).length;
+           viewfields.enter().append('path').attr('class', 'viewfield').attr('d', 'M0,0H0').merge(viewfields).attr('marker-start', 'url(#ideditor-viewfield-marker' + (wireframe ? '-wireframe' : '') + ')').attr('transform', function (d) {
+             return 'rotate(' + d + ')';
+           });
+         }
 
-           if (wasValid !== _isValid) {
-             // change in valid status
-             background.updateImagery();
-           }
+         function drawTargets(selection, graph, entities, filter) {
+           var targetClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
+           var nopeClass = context.getDebug('target') ? 'red ' : 'nocolor ';
+           var getTransform = svgPointTransform(projection).geojson;
+           var activeID = context.activeID();
+           var data = {
+             targets: [],
+             nopes: []
+           };
+           entities.forEach(function (node) {
+             if (activeID === node.id) return; // draw no target on the activeID
 
-           var baseFilter = '';
+             var vertexType = svgPassiveVertex(node, graph, activeID);
 
-           if (detected.cssfilters) {
-             if (_brightness !== 1) {
-               baseFilter += " brightness(".concat(_brightness, ")");
+             if (vertexType !== 0) {
+               // passive or adjacent - allow to connect
+               data.targets.push({
+                 type: 'Feature',
+                 id: node.id,
+                 properties: {
+                   target: true,
+                   entity: node
+                 },
+                 geometry: node.asGeoJSON()
+               });
+             } else {
+               data.nopes.push({
+                 type: 'Feature',
+                 id: node.id + '-nope',
+                 properties: {
+                   nope: true,
+                   target: true,
+                   entity: node
+                 },
+                 geometry: node.asGeoJSON()
+               });
              }
+           }); // Targets allow hover and vertex snapping
 
-             if (_contrast !== 1) {
-               baseFilter += " contrast(".concat(_contrast, ")");
-             }
+           var targets = selection.selectAll('.vertex.target-allowed').filter(function (d) {
+             return filter(d.properties.entity);
+           }).data(data.targets, function key(d) {
+             return d.id;
+           }); // exit
 
-             if (_saturation !== 1) {
-               baseFilter += " saturate(".concat(_saturation, ")");
-             }
+           targets.exit().remove(); // enter/update
 
-             if (_sharpness < 1) {
-               // gaussian blur
-               var blur = d3_interpolateNumber(0.5, 5)(1 - _sharpness);
-               baseFilter += " blur(".concat(blur, "px)");
-             }
-           }
+           targets.enter().append('circle').attr('r', function (d) {
+             return _radii[d.id] || radiuses.shadow[3];
+           }).merge(targets).attr('class', function (d) {
+             return 'node vertex target target-allowed ' + targetClass + d.id;
+           }).attr('transform', getTransform); // NOPE
 
-           var base = selection.selectAll('.layer-background').data([0]);
-           base = base.enter().insert('div', '.layer-data').attr('class', 'layer layer-background').merge(base);
+           var nopes = selection.selectAll('.vertex.target-nope').filter(function (d) {
+             return filter(d.properties.entity);
+           }).data(data.nopes, function key(d) {
+             return d.id;
+           }); // exit
 
-           if (detected.cssfilters) {
-             base.style('filter', baseFilter || null);
-           } else {
-             base.style('opacity', _brightness);
-           }
+           nopes.exit().remove(); // enter/update
 
-           var imagery = base.selectAll('.layer-imagery').data([0]);
-           imagery.enter().append('div').attr('class', 'layer layer-imagery').merge(imagery).call(baseLayer);
-           var maskFilter = '';
-           var mixBlendMode = '';
+           nopes.enter().append('circle').attr('r', function (d) {
+             return _radii[d.properties.entity.id] || radiuses.shadow[3];
+           }).merge(nopes).attr('class', function (d) {
+             return 'node vertex target target-nope ' + nopeClass + d.id;
+           }).attr('transform', getTransform);
+         } // Points can also render as vertices:
+         // 1. in wireframe mode or
+         // 2. at higher zooms if they have a direction
 
-           if (detected.cssfilters && _sharpness > 1) {
-             // apply unsharp mask
-             mixBlendMode = 'overlay';
-             maskFilter = 'saturate(0) blur(3px) invert(1)';
-             var contrast = _sharpness - 1;
-             maskFilter += " contrast(".concat(contrast, ")");
-             var brightness = d3_interpolateNumber(1, 0.85)(_sharpness - 1);
-             maskFilter += " brightness(".concat(brightness, ")");
-           }
 
-           var mask = base.selectAll('.layer-unsharp-mask').data(detected.cssfilters && _sharpness > 1 ? [0] : []);
-           mask.exit().remove();
-           mask.enter().append('div').attr('class', 'layer layer-mask layer-unsharp-mask').merge(mask).call(baseLayer).style('filter', maskFilter || null).style('mix-blend-mode', mixBlendMode || null);
-           var overlays = selection.selectAll('.layer-overlay').data(_overlayLayers, function (d) {
-             return d.source().name();
-           });
-           overlays.exit().remove();
-           overlays.enter().insert('div', '.layer-data').attr('class', 'layer layer-overlay').merge(overlays).each(function (layer, i, nodes) {
-             return select(nodes[i]).call(layer);
-           });
+         function renderAsVertex(entity, graph, wireframe, zoom) {
+           var geometry = entity.geometry(graph);
+           return geometry === 'vertex' || geometry === 'point' && (wireframe || zoom >= 18 && entity.directions(graph, projection).length);
          }
 
-         background.updateImagery = function () {
-           var currSource = baseLayer.source();
-           if (context.inIntro() || !currSource) return;
+         function isEditedNode(node, base, head) {
+           var baseNode = base.entities[node.id];
+           var headNode = head.entities[node.id];
+           return !headNode || !baseNode || !fastDeepEqual(headNode.tags, baseNode.tags) || !fastDeepEqual(headNode.loc, baseNode.loc);
+         }
 
-           var o = _overlayLayers.filter(function (d) {
-             return !d.source().isLocatorOverlay() && !d.source().isHidden();
-           }).map(function (d) {
-             return d.source().id;
-           }).join(',');
+         function getSiblingAndChildVertices(ids, graph, wireframe, zoom) {
+           var results = {};
+           var seenIds = {};
 
-           var meters = geoOffsetToMeters(currSource.offset());
-           var EPSILON = 0.01;
-           var x = +meters[0].toFixed(2);
-           var y = +meters[1].toFixed(2);
-           var hash = utilStringQs(window.location.hash);
-           var id = currSource.id;
+           function addChildVertices(entity) {
+             // avoid redundant work and infinite recursion of circular relations
+             if (seenIds[entity.id]) return;
+             seenIds[entity.id] = true;
+             var geometry = entity.geometry(graph);
 
-           if (id === 'custom') {
-             id = "custom:".concat(currSource.template());
-           }
+             if (!context.features().isHiddenFeature(entity, graph, geometry)) {
+               var i;
 
-           if (id) {
-             hash.background = id;
-           } else {
-             delete hash.background;
-           }
+               if (entity.type === 'way') {
+                 for (i = 0; i < entity.nodes.length; i++) {
+                   var child = graph.hasEntity(entity.nodes[i]);
 
-           if (o) {
-             hash.overlays = o;
-           } else {
-             delete hash.overlays;
-           }
+                   if (child) {
+                     addChildVertices(child);
+                   }
+                 }
+               } else if (entity.type === 'relation') {
+                 for (i = 0; i < entity.members.length; i++) {
+                   var member = graph.hasEntity(entity.members[i].id);
 
-           if (Math.abs(x) > EPSILON || Math.abs(y) > EPSILON) {
-             hash.offset = "".concat(x, ",").concat(y);
-           } else {
-             delete hash.offset;
+                   if (member) {
+                     addChildVertices(member);
+                   }
+                 }
+               } else if (renderAsVertex(entity, graph, wireframe, zoom)) {
+                 results[entity.id] = entity;
+               }
+             }
            }
 
-           if (!window.mocha) {
-             window.location.replace('#' + utilQsString(hash, true));
-           }
+           ids.forEach(function (id) {
+             var entity = graph.hasEntity(id);
+             if (!entity) return;
 
-           var imageryUsed = [];
-           var photoOverlaysUsed = [];
-           var currUsed = currSource.imageryUsed();
+             if (entity.type === 'node') {
+               if (renderAsVertex(entity, graph, wireframe, zoom)) {
+                 results[entity.id] = entity;
+                 graph.parentWays(entity).forEach(function (entity) {
+                   addChildVertices(entity);
+                 });
+               }
+             } else {
+               // way, relation
+               addChildVertices(entity);
+             }
+           });
+           return results;
+         }
 
-           if (currUsed && _isValid) {
-             imageryUsed.push(currUsed);
-           }
+         function drawVertices(selection, graph, entities, filter, extent, fullRedraw) {
+           var wireframe = context.surface().classed('fill-wireframe');
+           var visualDiff = context.surface().classed('highlight-edited');
+           var zoom = geoScaleToZoom(projection.scale());
+           var mode = context.mode();
+           var isMoving = mode && /^(add|draw|drag|move|rotate)/.test(mode.id);
+           var base = context.history().base();
+           var drawLayer = selection.selectAll('.layer-osm.points .points-group.vertices');
+           var touchLayer = selection.selectAll('.layer-touch.points');
 
-           _overlayLayers.filter(function (d) {
-             return !d.source().isLocatorOverlay() && !d.source().isHidden();
-           }).forEach(function (d) {
-             return imageryUsed.push(d.source().imageryUsed());
-           });
+           if (fullRedraw) {
+             _currPersistent = {};
+             _radii = {};
+           } // Collect important vertices from the `entities` list..
+           // (during a partial redraw, it will not contain everything)
 
-           var dataLayer = context.layers().layer('data');
 
-           if (dataLayer && dataLayer.enabled() && dataLayer.hasData()) {
-             imageryUsed.push(dataLayer.getSrc());
-           }
+           for (var i = 0; i < entities.length; i++) {
+             var entity = entities[i];
+             var geometry = entity.geometry(graph);
+             var keep = false; // a point that looks like a vertex..
 
-           var photoOverlayLayers = {
-             streetside: 'Bing Streetside',
-             mapillary: 'Mapillary Images',
-             'mapillary-map-features': 'Mapillary Map Features',
-             'mapillary-signs': 'Mapillary Signs',
-             openstreetcam: 'OpenStreetCam Images'
-           };
+             if (geometry === 'point' && renderAsVertex(entity, graph, wireframe, zoom)) {
+               _currPersistent[entity.id] = entity;
+               keep = true; // a vertex of some importance..
+             } else if (geometry === 'vertex' && (entity.hasInterestingTags() || entity.isEndpoint(graph) || entity.isConnected(graph) || visualDiff && isEditedNode(entity, base, graph))) {
+               _currPersistent[entity.id] = entity;
+               keep = true;
+             } // whatever this is, it's not a persistent vertex..
 
-           for (var layerID in photoOverlayLayers) {
-             var layer = context.layers().layer(layerID);
 
-             if (layer && layer.enabled()) {
-               photoOverlaysUsed.push(layerID);
-               imageryUsed.push(photoOverlayLayers[layerID]);
+             if (!keep && !fullRedraw) {
+               delete _currPersistent[entity.id];
              }
-           }
+           } // 3 sets of vertices to consider:
 
-           context.history().imageryUsed(imageryUsed);
-           context.history().photoOverlaysUsed(photoOverlaysUsed);
-         };
 
-         var _checkedBlocklists;
+           var sets = {
+             persistent: _currPersistent,
+             // persistent = important vertices (render always)
+             selected: _currSelected,
+             // selected + siblings of selected (render always)
+             hovered: _currHover // hovered + siblings of hovered (render only in draw modes)
 
-         background.sources = function (extent, zoom, includeCurrent) {
-           if (!_imageryIndex) return []; // called before init()?
+           };
+           var all = Object.assign({}, isMoving ? _currHover : {}, _currSelected, _currPersistent); // Draw the vertices..
+           // The filter function controls the scope of what objects d3 will touch (exit/enter/update)
+           // Adjust the filter function to expand the scope beyond whatever entities were passed in.
 
-           var visible = {};
-           (_imageryIndex.query.bbox(extent.rectangle(), true) || []).forEach(function (d) {
-             return visible[d.id] = true;
-           });
-           var currSource = baseLayer.source();
-           var osm = context.connection();
-           var blocklists = osm && osm.imageryBlocklists();
+           var filterRendered = function filterRendered(d) {
+             return d.id in _currPersistent || d.id in _currSelected || d.id in _currHover || filter(d);
+           };
 
-           if (blocklists && blocklists !== _checkedBlocklists) {
-             _imageryIndex.backgrounds.forEach(function (source) {
-               source.isBlocked = blocklists.some(function (blocklist) {
-                 return blocklist.test(source.template());
-               });
-             });
+           drawLayer.call(draw, graph, currentVisible(all), sets, filterRendered); // Draw touch targets..
+           // When drawing, render all targets (not just those affected by a partial redraw)
 
-             _checkedBlocklists = blocklists;
-           }
+           var filterTouch = function filterTouch(d) {
+             return isMoving ? true : filterRendered(d);
+           };
 
-           return _imageryIndex.backgrounds.filter(function (source) {
-             if (includeCurrent && currSource === source) return true; // optionally always include the current imagery
+           touchLayer.call(drawTargets, graph, currentVisible(all), filterTouch);
 
-             if (source.isBlocked) return false; // even bundled sources may be blocked - #7905
+           function currentVisible(which) {
+             return Object.keys(which).map(graph.hasEntity, graph) // the current version of this entity
+             .filter(function (entity) {
+               return entity && entity.intersects(extent, graph);
+             });
+           }
+         } // partial redraw - only update the selected items..
 
-             if (!source.polygon) return true; // always include imagery with worldwide coverage
 
-             if (zoom && zoom < 6) return false; // optionally exclude local imagery at low zooms
+         drawVertices.drawSelected = function (selection, graph, extent) {
+           var wireframe = context.surface().classed('fill-wireframe');
+           var zoom = geoScaleToZoom(projection.scale());
+           _prevSelected = _currSelected || {};
 
-             return visible[source.id]; // include imagery visible in given extent
-           });
-         };
+           if (context.map().isInWideSelection()) {
+             _currSelected = {};
+             context.selectedIDs().forEach(function (id) {
+               var entity = graph.hasEntity(id);
+               if (!entity) return;
 
-         background.dimensions = function (val) {
-           if (!val) return;
-           baseLayer.dimensions(val);
+               if (entity.type === 'node') {
+                 if (renderAsVertex(entity, graph, wireframe, zoom)) {
+                   _currSelected[entity.id] = entity;
+                 }
+               }
+             });
+           } else {
+             _currSelected = getSiblingAndChildVertices(context.selectedIDs(), graph, wireframe, zoom);
+           } // note that drawVertices will add `_currSelected` automatically if needed..
 
-           _overlayLayers.forEach(function (layer) {
-             return layer.dimensions(val);
-           });
-         };
 
-         background.baseLayerSource = function (d) {
-           if (!arguments.length) return baseLayer.source(); // test source against OSM imagery blocklists..
+           var filter = function filter(d) {
+             return d.id in _prevSelected;
+           };
 
-           var osm = context.connection();
-           if (!osm) return background;
-           var blocklists = osm.imageryBlocklists();
-           var template = d.template();
-           var fail = false;
-           var tested = 0;
-           var regex;
+           drawVertices(selection, graph, Object.values(_prevSelected), filter, extent, false);
+         }; // partial redraw - only update the hovered items..
 
-           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.
 
+         drawVertices.drawHover = function (selection, graph, target, extent) {
+           if (target === _currHoverTarget) return; // continue only if something changed
 
-           if (!tested) {
-             regex = /.*\.google(apis)?\..*\/(vt|kh)[\?\/].*([xyz]=.*){3}.*/;
-             fail = regex.test(template);
-           }
+           var wireframe = context.surface().classed('fill-wireframe');
+           var zoom = geoScaleToZoom(projection.scale());
+           _prevHover = _currHover || {};
+           _currHoverTarget = target;
+           var entity = target && target.properties && target.properties.entity;
 
-           baseLayer.source(!fail ? d : background.findSource('none'));
-           dispatch$1.call('change');
-           background.updateImagery();
-           return background;
-         };
+           if (entity) {
+             _currHover = getSiblingAndChildVertices([entity.id], graph, wireframe, zoom);
+           } else {
+             _currHover = {};
+           } // note that drawVertices will add `_currHover` automatically if needed..
 
-         background.findSource = function (id) {
-           if (!id || !_imageryIndex) return null; // called before init()?
 
-           return _imageryIndex.backgrounds.find(function (d) {
-             return d.id && d.id === id;
-           });
-         };
+           var filter = function filter(d) {
+             return d.id in _prevHover;
+           };
 
-         background.bing = function () {
-           background.baseLayerSource(background.findSource('Bing'));
+           drawVertices(selection, graph, Object.values(_prevHover), filter, extent, false);
          };
 
-         background.showsLayer = function (d) {
-           var currSource = baseLayer.source();
-           if (!d || !currSource) return false;
-           return d.id === currSource.id || _overlayLayers.some(function (layer) {
-             return d.id === layer.source().id;
-           });
-         };
+         return drawVertices;
+       }
 
-         background.overlayLayerSources = function () {
-           return _overlayLayers.map(function (layer) {
-             return layer.source();
-           });
-         };
+       function utilBindOnce(target, type, listener, capture) {
+         var typeOnce = type + '.once';
 
-         background.toggleOverlayLayer = function (d) {
-           var layer;
+         function one() {
+           target.on(typeOnce, null);
+           listener.apply(this, arguments);
+         }
 
-           for (var i = 0; i < _overlayLayers.length; i++) {
-             layer = _overlayLayers[i];
+         target.on(typeOnce, one, capture);
+         return this;
+       }
 
-             if (layer.source() === d) {
-               _overlayLayers.splice(i, 1);
+       function defaultFilter(d3_event) {
+         return !d3_event.ctrlKey && !d3_event.button;
+       }
 
-               dispatch$1.call('change');
-               background.updateImagery();
-               return;
-             }
-           }
+       function defaultExtent() {
+         var e = this;
 
-           layer = rendererTileLayer(context).source(d).projection(context.projection).dimensions(baseLayer.dimensions());
+         if (e instanceof SVGElement) {
+           e = e.ownerSVGElement || e;
 
-           _overlayLayers.push(layer);
+           if (e.hasAttribute('viewBox')) {
+             e = e.viewBox.baseVal;
+             return [[e.x, e.y], [e.x + e.width, e.y + e.height]];
+           }
 
-           dispatch$1.call('change');
-           background.updateImagery();
-         };
+           return [[0, 0], [e.width.baseVal.value, e.height.baseVal.value]];
+         }
 
-         background.nudge = function (d, zoom) {
-           var currSource = baseLayer.source();
+         return [[0, 0], [e.clientWidth, e.clientHeight]];
+       }
 
-           if (currSource) {
-             currSource.nudge(d, zoom);
-             dispatch$1.call('change');
-             background.updateImagery();
-           }
+       function defaultWheelDelta(d3_event) {
+         return -d3_event.deltaY * (d3_event.deltaMode === 1 ? 0.05 : d3_event.deltaMode ? 1 : 0.002);
+       }
+
+       function defaultConstrain(transform, extent, translateExtent) {
+         var dx0 = transform.invertX(extent[0][0]) - translateExtent[0][0],
+             dx1 = transform.invertX(extent[1][0]) - translateExtent[1][0],
+             dy0 = transform.invertY(extent[0][1]) - translateExtent[0][1],
+             dy1 = transform.invertY(extent[1][1]) - translateExtent[1][1];
+         return transform.translate(dx1 > dx0 ? (dx0 + dx1) / 2 : Math.min(0, dx0) || Math.max(0, dx1), dy1 > dy0 ? (dy0 + dy1) / 2 : Math.min(0, dy0) || Math.max(0, dy1));
+       }
 
-           return background;
-         };
+       function utilZoomPan() {
+         var filter = defaultFilter,
+             extent = defaultExtent,
+             constrain = defaultConstrain,
+             wheelDelta = defaultWheelDelta,
+             scaleExtent = [0, Infinity],
+             translateExtent = [[-Infinity, -Infinity], [Infinity, Infinity]],
+             interpolate = interpolateZoom,
+             dispatch = dispatch$8('start', 'zoom', 'end'),
+             _wheelDelay = 150,
+             _transform = identity$2,
+             _activeGesture;
 
-         background.offset = function (d) {
-           var currSource = baseLayer.source();
+         function zoom(selection) {
+           selection.on('pointerdown.zoom', pointerdown).on('wheel.zoom', wheeled).style('touch-action', 'none').style('-webkit-tap-highlight-color', 'rgba(0,0,0,0)');
+           select(window).on('pointermove.zoompan', pointermove).on('pointerup.zoompan pointercancel.zoompan', pointerup);
+         }
 
-           if (!arguments.length) {
-             return currSource && currSource.offset() || [0, 0];
-           }
+         zoom.transform = function (collection, transform, point) {
+           var selection = collection.selection ? collection.selection() : collection;
 
-           if (currSource) {
-             currSource.offset(d);
-             dispatch$1.call('change');
-             background.updateImagery();
+           if (collection !== selection) {
+             schedule(collection, transform, point);
+           } else {
+             selection.interrupt().each(function () {
+               gesture(this, arguments).start(null).zoom(null, null, typeof transform === 'function' ? transform.apply(this, arguments) : transform).end(null);
+             });
            }
-
-           return background;
          };
 
-         background.brightness = function (d) {
-           if (!arguments.length) return _brightness;
-           _brightness = d;
-           if (context.mode()) dispatch$1.call('change');
-           return background;
+         zoom.scaleBy = function (selection, k, p) {
+           zoom.scaleTo(selection, function () {
+             var k0 = _transform.k,
+                 k1 = typeof k === 'function' ? k.apply(this, arguments) : k;
+             return k0 * k1;
+           }, p);
          };
 
-         background.contrast = function (d) {
-           if (!arguments.length) return _contrast;
-           _contrast = d;
-           if (context.mode()) dispatch$1.call('change');
-           return background;
+         zoom.scaleTo = function (selection, k, p) {
+           zoom.transform(selection, function () {
+             var e = extent.apply(this, arguments),
+                 t0 = _transform,
+                 p0 = !p ? centroid(e) : typeof p === 'function' ? p.apply(this, arguments) : p,
+                 p1 = t0.invert(p0),
+                 k1 = typeof k === 'function' ? k.apply(this, arguments) : k;
+             return constrain(translate(scale(t0, k1), p0, p1), e, translateExtent);
+           }, p);
          };
 
-         background.saturation = function (d) {
-           if (!arguments.length) return _saturation;
-           _saturation = d;
-           if (context.mode()) dispatch$1.call('change');
-           return background;
+         zoom.translateBy = function (selection, x, y) {
+           zoom.transform(selection, function () {
+             return constrain(_transform.translate(typeof x === 'function' ? x.apply(this, arguments) : x, typeof y === 'function' ? y.apply(this, arguments) : y), extent.apply(this, arguments), translateExtent);
+           });
          };
 
-         background.sharpness = function (d) {
-           if (!arguments.length) return _sharpness;
-           _sharpness = d;
-           if (context.mode()) dispatch$1.call('change');
-           return background;
+         zoom.translateTo = function (selection, x, y, p) {
+           zoom.transform(selection, function () {
+             var e = extent.apply(this, arguments),
+                 t = _transform,
+                 p0 = !p ? centroid(e) : typeof p === 'function' ? p.apply(this, arguments) : p;
+             return constrain(identity$2.translate(p0[0], p0[1]).scale(t.k).translate(typeof x === 'function' ? -x.apply(this, arguments) : -x, typeof y === 'function' ? -y.apply(this, arguments) : -y), e, translateExtent);
+           }, p);
          };
 
-         var _loadPromise;
-
-         background.ensureLoaded = function () {
-           if (_loadPromise) return _loadPromise;
+         function scale(transform, k) {
+           k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], k));
+           return k === transform.k ? transform : new Transform(k, transform.x, transform.y);
+         }
 
-           function parseMapParams(qmap) {
-             if (!qmap) return false;
-             var params = qmap.split('/').map(Number);
-             if (params.length < 3 || params.some(isNaN)) return false;
-             return geoExtent([params[2], params[1]]); // lon,lat
-           }
+         function translate(transform, p0, p1) {
+           var x = p0[0] - p1[0] * transform.k,
+               y = p0[1] - p1[1] * transform.k;
+           return x === transform.x && y === transform.y ? transform : new Transform(transform.k, x, y);
+         }
 
-           var hash = utilStringQs(window.location.hash);
-           var requested = hash.background || hash.layer;
-           var extent = parseMapParams(hash.map);
-           return _loadPromise = ensureImageryIndex().then(function (imageryIndex) {
-             var first = imageryIndex.backgrounds.length && imageryIndex.backgrounds[0];
-             var best;
+         function centroid(extent) {
+           return [(+extent[0][0] + +extent[1][0]) / 2, (+extent[0][1] + +extent[1][1]) / 2];
+         }
 
-             if (!requested && extent) {
-               best = background.sources(extent).find(function (s) {
-                 return s.best();
-               });
-             } // Decide which background layer to display
+         function schedule(transition, transform, point) {
+           transition.on('start.zoom', function () {
+             gesture(this, arguments).start(null);
+           }).on('interrupt.zoom end.zoom', function () {
+             gesture(this, arguments).end(null);
+           }).tween('zoom', function () {
+             var that = this,
+                 args = arguments,
+                 g = gesture(that, args),
+                 e = extent.apply(that, args),
+                 p = !point ? centroid(e) : typeof point === 'function' ? point.apply(that, args) : point,
+                 w = Math.max(e[1][0] - e[0][0], e[1][1] - e[0][1]),
+                 a = _transform,
+                 b = typeof transform === 'function' ? transform.apply(that, args) : transform,
+                 i = interpolate(a.invert(p).concat(w / a.k), b.invert(p).concat(w / b.k));
+             return function (t) {
+               if (t === 1) {
+                 // Avoid rounding error on end.
+                 t = b;
+               } else {
+                 var l = i(t);
+                 var k = w / l[2];
+                 t = new Transform(k, p[0] - l[0] * k, p[1] - l[1] * k);
+               }
 
+               g.zoom(null, null, t);
+             };
+           });
+         }
 
-             if (requested && requested.indexOf('custom:') === 0) {
-               var template = requested.replace(/^custom:/, '');
-               var custom = background.findSource('custom');
-               background.baseLayerSource(custom.template(template));
-               corePreferences('background-custom-template', template);
-             } else {
-               background.baseLayerSource(background.findSource(requested) || best || background.findSource(corePreferences('background-last-used')) || background.findSource('Bing') || first || background.findSource('none'));
-             }
+         function gesture(that, args, clean) {
+           return !clean && _activeGesture || new Gesture(that, args);
+         }
 
-             var locator = imageryIndex.backgrounds.find(function (d) {
-               return d.overlay && d["default"];
-             });
+         function Gesture(that, args) {
+           this.that = that;
+           this.args = args;
+           this.active = 0;
+           this.extent = extent.apply(that, args);
+         }
 
-             if (locator) {
-               background.toggleOverlayLayer(locator);
+         Gesture.prototype = {
+           start: function start(d3_event) {
+             if (++this.active === 1) {
+               _activeGesture = this;
+               dispatch.call('start', this, d3_event);
              }
 
-             var overlays = (hash.overlays || '').split(',');
-             overlays.forEach(function (overlay) {
-               overlay = background.findSource(overlay);
-
-               if (overlay) {
-                 background.toggleOverlayLayer(overlay);
-               }
-             });
-
-             if (hash.gpx) {
-               var gpx = context.layers().layer('data');
-
-               if (gpx) {
-                 gpx.url(hash.gpx, '.gpx');
-               }
+             return this;
+           },
+           zoom: function zoom(d3_event, key, transform) {
+             if (this.mouse && key !== 'mouse') this.mouse[1] = transform.invert(this.mouse[0]);
+             if (this.pointer0 && key !== 'touch') this.pointer0[1] = transform.invert(this.pointer0[0]);
+             if (this.pointer1 && key !== 'touch') this.pointer1[1] = transform.invert(this.pointer1[0]);
+             _transform = transform;
+             dispatch.call('zoom', this, d3_event, key, transform);
+             return this;
+           },
+           end: function end(d3_event) {
+             if (--this.active === 0) {
+               _activeGesture = null;
+               dispatch.call('end', this, d3_event);
              }
 
-             if (hash.offset) {
-               var offset = hash.offset.replace(/;/g, ',').split(',').map(function (n) {
-                 return !isNaN(n) && n;
-               });
+             return this;
+           }
+         };
 
-               if (offset.length === 2) {
-                 background.offset(geoMetersToOffset(offset));
-               }
+         function wheeled(d3_event) {
+           if (!filter.apply(this, arguments)) return;
+           var g = gesture(this, arguments),
+               t = _transform,
+               k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], t.k * Math.pow(2, wheelDelta.apply(this, arguments)))),
+               p = utilFastMouse(this)(d3_event); // If the mouse is in the same location as before, reuse it.
+           // If there were recent wheel events, reset the wheel idle timeout.
+
+           if (g.wheel) {
+             if (g.mouse[0][0] !== p[0] || g.mouse[0][1] !== p[1]) {
+               g.mouse[1] = t.invert(g.mouse[0] = p);
              }
-           })["catch"](function () {
-             /* ignore */
-           });
-         };
 
-         return utilRebind(background, dispatch$1, 'on');
-       }
+             clearTimeout(g.wheel); // Otherwise, capture the mouse point and location at the start.
+           } else {
+             g.mouse = [p, t.invert(p)];
+             interrupt(this);
+             g.start(d3_event);
+           }
 
-       function rendererFeatures(context) {
-         var dispatch$1 = dispatch('change', 'redraw');
-         var features = utilRebind({}, dispatch$1, 'on');
+           d3_event.preventDefault();
+           d3_event.stopImmediatePropagation();
+           g.wheel = setTimeout(wheelidled, _wheelDelay);
+           g.zoom(d3_event, 'mouse', constrain(translate(scale(t, k), g.mouse[0], g.mouse[1]), g.extent, translateExtent));
 
-         var _deferred = new Set();
+           function wheelidled() {
+             g.wheel = null;
+             g.end(d3_event);
+           }
+         }
 
-         var traffic_roads = {
-           'motorway': true,
-           'motorway_link': true,
-           'trunk': true,
-           'trunk_link': true,
-           'primary': true,
-           'primary_link': true,
-           'secondary': true,
-           'secondary_link': true,
-           'tertiary': true,
-           'tertiary_link': true,
-           'residential': true,
-           'unclassified': true,
-           'living_street': true
-         };
-         var service_roads = {
-           'service': true,
-           'road': true,
-           'track': true
-         };
-         var paths = {
-           'path': true,
-           'footway': true,
-           'cycleway': true,
-           'bridleway': true,
-           'steps': true,
-           'pedestrian': true
-         };
-         var past_futures = {
-           'proposed': true,
-           'construction': true,
-           'abandoned': true,
-           'dismantled': true,
-           'disused': true,
-           'razed': true,
-           'demolished': true,
-           'obliterated': true
-         };
-         var _cullFactor = 1;
-         var _cache = {};
-         var _rules = {};
-         var _stats = {};
-         var _keys = [];
-         var _hidden = [];
-         var _forceVisible = {};
+         var _downPointerIDs = new Set();
 
-         function update() {
-           if (!window.mocha) {
-             var hash = utilStringQs(window.location.hash);
-             var disabled = features.disabled();
+         var _pointerLocGetter;
 
-             if (disabled.length) {
-               hash.disable_features = disabled.join(',');
-             } else {
-               delete hash.disable_features;
-             }
+         function pointerdown(d3_event) {
+           _downPointerIDs.add(d3_event.pointerId);
 
-             window.location.replace('#' + utilQsString(hash, true));
-             corePreferences('disabled-features', disabled.join(','));
-           }
+           if (!filter.apply(this, arguments)) return;
+           var g = gesture(this, arguments, _downPointerIDs.size === 1);
+           var started;
+           d3_event.stopImmediatePropagation();
+           _pointerLocGetter = utilFastMouse(this);
 
-           _hidden = features.hidden();
-           dispatch$1.call('change');
-           dispatch$1.call('redraw');
-         }
+           var loc = _pointerLocGetter(d3_event);
 
-         function defineRule(k, filter, max) {
-           var isEnabled = true;
+           var p = [loc, _transform.invert(loc), d3_event.pointerId];
 
-           _keys.push(k);
+           if (!g.pointer0) {
+             g.pointer0 = p;
+             started = true;
+           } else if (!g.pointer1 && g.pointer0[2] !== p[2]) {
+             g.pointer1 = p;
+           }
 
-           _rules[k] = {
-             filter: filter,
-             enabled: isEnabled,
-             // whether the user wants it enabled..
-             count: 0,
-             currentMax: max || Infinity,
-             defaultMax: max || Infinity,
-             enable: function enable() {
-               this.enabled = true;
-               this.currentMax = this.defaultMax;
-             },
-             disable: function disable() {
-               this.enabled = false;
-               this.currentMax = 0;
-             },
-             hidden: function hidden() {
-               return this.count === 0 && !this.enabled || this.count > this.currentMax * _cullFactor;
-             },
-             autoHidden: function autoHidden() {
-               return this.hidden() && this.currentMax > 0;
-             }
-           };
+           if (started) {
+             interrupt(this);
+             g.start(d3_event);
+           }
          }
 
-         defineRule('points', function isPoint(tags, geometry) {
-           return geometry === 'point';
-         }, 200);
-         defineRule('traffic_roads', function isTrafficRoad(tags) {
-           return traffic_roads[tags.highway];
-         });
-         defineRule('service_roads', function isServiceRoad(tags) {
-           return service_roads[tags.highway];
-         });
-         defineRule('paths', function isPath(tags) {
-           return paths[tags.highway];
-         });
-         defineRule('buildings', function isBuilding(tags) {
-           return !!tags.building && tags.building !== 'no' || tags.parking === 'multi-storey' || tags.parking === 'sheds' || tags.parking === 'carports' || tags.parking === 'garage_boxes';
-         }, 250);
-         defineRule('building_parts', function isBuildingPart(tags) {
-           return tags['building:part'];
-         });
-         defineRule('indoor', function isIndoor(tags) {
-           return tags.indoor;
-         });
-         defineRule('landuse', function isLanduse(tags, geometry) {
-           return geometry === 'area' && !_rules.buildings.filter(tags) && !_rules.building_parts.filter(tags) && !_rules.indoor.filter(tags) && !_rules.water.filter(tags) && !_rules.pistes.filter(tags);
-         });
-         defineRule('boundaries', function isBoundary(tags) {
-           return !!tags.boundary && !(traffic_roads[tags.highway] || service_roads[tags.highway] || paths[tags.highway] || tags.waterway || tags.railway || tags.landuse || tags.natural || tags.building || tags.power);
-         });
-         defineRule('water', function isWater(tags) {
-           return !!tags.waterway || tags.natural === 'water' || tags.natural === 'coastline' || tags.natural === 'bay' || tags.landuse === 'pond' || tags.landuse === 'basin' || tags.landuse === 'reservoir' || tags.landuse === 'salt_pond';
-         });
-         defineRule('rail', function isRail(tags) {
-           return (!!tags.railway || tags.landuse === 'railway') && !(traffic_roads[tags.highway] || service_roads[tags.highway] || paths[tags.highway]);
-         });
-         defineRule('pistes', function isPiste(tags) {
-           return tags['piste:type'];
-         });
-         defineRule('aerialways', function isPiste(tags) {
-           return tags.aerialway && tags.aerialway !== 'yes' && tags.aerialway !== 'station';
-         });
-         defineRule('power', function isPower(tags) {
-           return !!tags.power;
-         }); // contains a past/future tag, but not in active use as a road/path/cycleway/etc..
+         function pointermove(d3_event) {
+           if (!_downPointerIDs.has(d3_event.pointerId)) return;
+           if (!_activeGesture || !_pointerLocGetter) return;
+           var g = gesture(this, arguments);
+           var isPointer0 = g.pointer0 && g.pointer0[2] === d3_event.pointerId;
+           var isPointer1 = !isPointer0 && g.pointer1 && g.pointer1[2] === d3_event.pointerId;
 
-         defineRule('past_future', function isPastFuture(tags) {
-           if (traffic_roads[tags.highway] || service_roads[tags.highway] || paths[tags.highway]) {
-             return false;
+           if ((isPointer0 || isPointer1) && 'buttons' in d3_event && !d3_event.buttons) {
+             // The pointer went up without ending the gesture somehow, e.g.
+             // a down mouse was moved off the map and released. End it here.
+             if (g.pointer0) _downPointerIDs["delete"](g.pointer0[2]);
+             if (g.pointer1) _downPointerIDs["delete"](g.pointer1[2]);
+             g.end(d3_event);
+             return;
            }
 
-           var strings = Object.keys(tags);
+           d3_event.preventDefault();
+           d3_event.stopImmediatePropagation();
 
-           for (var i = 0; i < strings.length; i++) {
-             var s = strings[i];
+           var loc = _pointerLocGetter(d3_event);
 
-             if (past_futures[s] || past_futures[tags[s]]) {
-               return true;
-             }
+           var t, p, l;
+           if (isPointer0) g.pointer0[0] = loc;else if (isPointer1) g.pointer1[0] = loc;
+           t = _transform;
+
+           if (g.pointer1) {
+             var p0 = g.pointer0[0],
+                 l0 = g.pointer0[1],
+                 p1 = g.pointer1[0],
+                 l1 = g.pointer1[1],
+                 dp = (dp = p1[0] - p0[0]) * dp + (dp = p1[1] - p0[1]) * dp,
+                 dl = (dl = l1[0] - l0[0]) * dl + (dl = l1[1] - l0[1]) * dl;
+             t = scale(t, Math.sqrt(dp / dl));
+             p = [(p0[0] + p1[0]) / 2, (p0[1] + p1[1]) / 2];
+             l = [(l0[0] + l1[0]) / 2, (l0[1] + l1[1]) / 2];
+           } else if (g.pointer0) {
+             p = g.pointer0[0];
+             l = g.pointer0[1];
+           } else {
+             return;
            }
 
-           return false;
-         }); // Lines or areas that don't match another feature filter.
-         // IMPORTANT: The 'others' feature must be the last one defined,
-         //   so that code in getMatches can skip this test if `hasMatch = true`
+           g.zoom(d3_event, 'touch', constrain(translate(t, p, l), g.extent, translateExtent));
+         }
 
-         defineRule('others', function isOther(tags, geometry) {
-           return geometry === 'line' || geometry === 'area';
-         });
+         function pointerup(d3_event) {
+           if (!_downPointerIDs.has(d3_event.pointerId)) return;
 
-         features.features = function () {
-           return _rules;
-         };
+           _downPointerIDs["delete"](d3_event.pointerId);
 
-         features.keys = function () {
-           return _keys;
-         };
+           if (!_activeGesture) return;
+           var g = gesture(this, arguments);
+           d3_event.stopImmediatePropagation();
+           if (g.pointer0 && g.pointer0[2] === d3_event.pointerId) delete g.pointer0;else if (g.pointer1 && g.pointer1[2] === d3_event.pointerId) delete g.pointer1;
 
-         features.enabled = function (k) {
-           if (!arguments.length) {
-             return _keys.filter(function (k) {
-               return _rules[k].enabled;
-             });
+           if (g.pointer1 && !g.pointer0) {
+             g.pointer0 = g.pointer1;
+             delete g.pointer1;
            }
 
-           return _rules[k] && _rules[k].enabled;
-         };
-
-         features.disabled = function (k) {
-           if (!arguments.length) {
-             return _keys.filter(function (k) {
-               return !_rules[k].enabled;
-             });
+           if (g.pointer0) {
+             g.pointer0[1] = _transform.invert(g.pointer0[0]);
+           } else {
+             g.end(d3_event);
            }
+         }
 
-           return _rules[k] && !_rules[k].enabled;
+         zoom.wheelDelta = function (_) {
+           return arguments.length ? (wheelDelta = utilFunctor(+_), zoom) : wheelDelta;
          };
 
-         features.hidden = function (k) {
-           if (!arguments.length) {
-             return _keys.filter(function (k) {
-               return _rules[k].hidden();
-             });
-           }
-
-           return _rules[k] && _rules[k].hidden();
+         zoom.filter = function (_) {
+           return arguments.length ? (filter = utilFunctor(!!_), zoom) : filter;
          };
 
-         features.autoHidden = function (k) {
-           if (!arguments.length) {
-             return _keys.filter(function (k) {
-               return _rules[k].autoHidden();
-             });
-           }
-
-           return _rules[k] && _rules[k].autoHidden();
+         zoom.extent = function (_) {
+           return arguments.length ? (extent = utilFunctor([[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]]), zoom) : extent;
          };
 
-         features.enable = function (k) {
-           if (_rules[k] && !_rules[k].enabled) {
-             _rules[k].enable();
-
-             update();
-           }
+         zoom.scaleExtent = function (_) {
+           return arguments.length ? (scaleExtent[0] = +_[0], scaleExtent[1] = +_[1], zoom) : [scaleExtent[0], scaleExtent[1]];
          };
 
-         features.enableAll = function () {
-           var didEnable = false;
-
-           for (var k in _rules) {
-             if (!_rules[k].enabled) {
-               didEnable = true;
-
-               _rules[k].enable();
-             }
-           }
-
-           if (didEnable) update();
+         zoom.translateExtent = function (_) {
+           return arguments.length ? (translateExtent[0][0] = +_[0][0], translateExtent[1][0] = +_[1][0], translateExtent[0][1] = +_[0][1], translateExtent[1][1] = +_[1][1], zoom) : [[translateExtent[0][0], translateExtent[0][1]], [translateExtent[1][0], translateExtent[1][1]]];
          };
 
-         features.disable = function (k) {
-           if (_rules[k] && _rules[k].enabled) {
-             _rules[k].disable();
+         zoom.constrain = function (_) {
+           return arguments.length ? (constrain = _, zoom) : constrain;
+         };
 
-             update();
-           }
+         zoom.interpolate = function (_) {
+           return arguments.length ? (interpolate = _, zoom) : interpolate;
          };
 
-         features.disableAll = function () {
-           var didDisable = false;
+         zoom._transform = function (_) {
+           return arguments.length ? (_transform = _, zoom) : _transform;
+         };
 
-           for (var k in _rules) {
-             if (_rules[k].enabled) {
-               didDisable = true;
+         return utilRebind(zoom, dispatch, 'on');
+       }
 
-               _rules[k].disable();
-             }
-           }
+       // if pointer events are supported. Falls back to default `dblclick` event.
 
-           if (didDisable) update();
-         };
+       function utilDoubleUp() {
+         var dispatch = dispatch$8('doubleUp');
+         var _maxTimespan = 500; // milliseconds
 
-         features.toggle = function (k) {
-           if (_rules[k]) {
-             (function (f) {
-               return f.enabled ? f.disable() : f.enable();
-             })(_rules[k]);
+         var _maxDistance = 20; // web pixels; be somewhat generous to account for touch devices
 
-             update();
-           }
-         };
+         var _pointer; // object representing the pointer that could trigger double up
 
-         features.resetStats = function () {
-           for (var i = 0; i < _keys.length; i++) {
-             _rules[_keys[i]].count = 0;
-           }
 
-           dispatch$1.call('change');
-         };
+         function pointerIsValidFor(loc) {
+           // second pointerup must occur within a small timeframe after the first pointerdown
+           return new Date().getTime() - _pointer.startTime <= _maxTimespan && // all pointer events must occur within a small distance of the first pointerdown
+           geoVecLength(_pointer.startLoc, loc) <= _maxDistance;
+         }
 
-         features.gatherStats = function (d, resolver, dimensions) {
-           var needsRedraw = false;
-           var types = utilArrayGroupBy(d, 'type');
-           var entities = [].concat(types.relation || [], types.way || [], types.node || []);
-           var currHidden, geometry, matches, i, j;
+         function pointerdown(d3_event) {
+           // ignore right-click
+           if (d3_event.ctrlKey || d3_event.button === 2) return;
+           var loc = [d3_event.clientX, d3_event.clientY]; // Don't rely on pointerId here since it can change between pointerdown
+           // events on touch devices
 
-           for (i = 0; i < _keys.length; i++) {
-             _rules[_keys[i]].count = 0;
-           } // adjust the threshold for point/building culling based on viewport size..
-           // a _cullFactor of 1 corresponds to a 1000x1000px viewport..
+           if (_pointer && !pointerIsValidFor(loc)) {
+             // if this pointer is no longer valid, clear it so another can be started
+             _pointer = undefined;
+           }
 
+           if (!_pointer) {
+             _pointer = {
+               startLoc: loc,
+               startTime: new Date().getTime(),
+               upCount: 0,
+               pointerId: d3_event.pointerId
+             };
+           } else {
+             // double down
+             _pointer.pointerId = d3_event.pointerId;
+           }
+         }
 
-           _cullFactor = dimensions[0] * dimensions[1] / 1000000;
+         function pointerup(d3_event) {
+           // ignore right-click
+           if (d3_event.ctrlKey || d3_event.button === 2) return;
+           if (!_pointer || _pointer.pointerId !== d3_event.pointerId) return;
+           _pointer.upCount += 1;
 
-           for (i = 0; i < entities.length; i++) {
-             geometry = entities[i].geometry(resolver);
-             matches = Object.keys(features.getMatches(entities[i], resolver, geometry));
+           if (_pointer.upCount === 2) {
+             // double up!
+             var loc = [d3_event.clientX, d3_event.clientY];
 
-             for (j = 0; j < matches.length; j++) {
-               _rules[matches[j]].count++;
-             }
-           }
+             if (pointerIsValidFor(loc)) {
+               var locInThis = utilFastMouse(this)(d3_event);
+               dispatch.call('doubleUp', this, d3_event, locInThis);
+             } // clear the pointer info in any case
 
-           currHidden = features.hidden();
 
-           if (currHidden !== _hidden) {
-             _hidden = currHidden;
-             needsRedraw = true;
-             dispatch$1.call('change');
+             _pointer = undefined;
            }
+         }
 
-           return needsRedraw;
-         };
-
-         features.stats = function () {
-           for (var i = 0; i < _keys.length; i++) {
-             _stats[_keys[i]] = _rules[_keys[i]].count;
+         function doubleUp(selection) {
+           if ('PointerEvent' in window) {
+             // dblclick isn't well supported on touch devices so manually use
+             // pointer events if they're available
+             selection.on('pointerdown.doubleUp', pointerdown).on('pointerup.doubleUp', pointerup);
+           } else {
+             // fallback to dblclick
+             selection.on('dblclick.doubleUp', function (d3_event) {
+               dispatch.call('doubleUp', this, d3_event, utilFastMouse(this)(d3_event));
+             });
            }
+         }
 
-           return _stats;
+         doubleUp.off = function (selection) {
+           selection.on('pointerdown.doubleUp', null).on('pointerup.doubleUp', null).on('dblclick.doubleUp', null);
          };
 
-         features.clear = function (d) {
-           for (var i = 0; i < d.length; i++) {
-             features.clearEntity(d[i]);
-           }
-         };
+         return utilRebind(doubleUp, dispatch, 'on');
+       }
 
-         features.clearEntity = function (entity) {
-           delete _cache[osmEntity.key(entity)];
-         };
+       var TILESIZE = 256;
+       var minZoom = 2;
+       var maxZoom = 24;
+       var kMin = geoZoomToScale(minZoom, TILESIZE);
+       var kMax = geoZoomToScale(maxZoom, TILESIZE);
 
-         features.reset = function () {
-           Array.from(_deferred).forEach(function (handle) {
-             window.cancelIdleCallback(handle);
+       function clamp$1(num, min, max) {
+         return Math.max(min, Math.min(num, max));
+       }
 
-             _deferred["delete"](handle);
-           });
-           _cache = {};
-         }; // only certain relations are worth checking
+       function rendererMap(context) {
+         var dispatch = dispatch$8('move', 'drawn', 'crossEditableZoom', 'hitMinZoom', 'changeHighlighting', 'changeAreaFill');
+         var projection = context.projection;
+         var curtainProjection = context.curtainProjection;
+         var drawLayers;
+         var drawPoints;
+         var drawVertices;
+         var drawLines;
+         var drawAreas;
+         var drawMidpoints;
+         var drawLabels;
 
+         var _selection = select(null);
 
-         function relationShouldBeChecked(relation) {
-           // multipolygon features have `area` geometry and aren't checked here
-           return relation.tags.type === 'boundary';
-         }
+         var supersurface = select(null);
+         var wrapper = select(null);
+         var surface = select(null);
+         var _dimensions = [1, 1];
+         var _dblClickZoomEnabled = true;
+         var _redrawEnabled = true;
 
-         features.getMatches = function (entity, resolver, geometry) {
-           if (geometry === 'vertex' || geometry === 'relation' && !relationShouldBeChecked(entity)) return {};
-           var ent = osmEntity.key(entity);
+         var _gestureTransformStart;
 
-           if (!_cache[ent]) {
-             _cache[ent] = {};
-           }
+         var _transformStart = projection.transform();
 
-           if (!_cache[ent].matches) {
-             var matches = {};
-             var hasMatch = false;
+         var _transformLast;
 
-             for (var i = 0; i < _keys.length; i++) {
-               if (_keys[i] === 'others') {
-                 if (hasMatch) continue; // If an entity...
-                 //   1. is a way that hasn't matched other 'interesting' feature rules,
+         var _isTransformed = false;
+         var _minzoom = 0;
 
-                 if (entity.type === 'way') {
-                   var parents = features.getParents(entity, resolver, geometry); //   2a. belongs only to a single multipolygon relation
+         var _getMouseCoords;
 
-                   if (parents.length === 1 && parents[0].isMultipolygon() || // 2b. or belongs only to boundary relations
-                   parents.length > 0 && parents.every(function (parent) {
-                     return parent.tags.type === 'boundary';
-                   })) {
-                     // ...then match whatever feature rules the parent relation has matched.
-                     // see #2548, #2887
-                     //
-                     // IMPORTANT:
-                     // For this to work, getMatches must be called on relations before ways.
-                     //
-                     var pkey = osmEntity.key(parents[0]);
+         var _lastPointerEvent;
 
-                     if (_cache[pkey] && _cache[pkey].matches) {
-                       matches = Object.assign({}, _cache[pkey].matches); // shallow copy
+         var _lastWithinEditableZoom; // whether a pointerdown event started the zoom
 
-                       continue;
-                     }
-                   }
-                 }
-               }
 
-               if (_rules[_keys[i]].filter(entity.tags, geometry)) {
-                 matches[_keys[i]] = hasMatch = true;
-               }
-             }
+         var _pointerDown = false; // use pointer events on supported platforms; fallback to mouse events
 
-             _cache[ent].matches = matches;
-           }
+         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse'; // use pointer event interaction if supported; fallback to touch/mouse events in d3-zoom
 
-           return _cache[ent].matches;
-         };
 
-         features.getParents = function (entity, resolver, geometry) {
-           if (geometry === 'point') return [];
-           var ent = osmEntity.key(entity);
+         var _zoomerPannerFunction = 'PointerEvent' in window ? utilZoomPan : d3_zoom;
 
-           if (!_cache[ent]) {
-             _cache[ent] = {};
-           }
+         var _zoomerPanner = _zoomerPannerFunction().scaleExtent([kMin, kMax]).interpolate(interpolate$1).filter(zoomEventFilter).on('zoom.map', zoomPan).on('start.map', function (d3_event) {
+           _pointerDown = d3_event && (d3_event.type === 'pointerdown' || d3_event.sourceEvent && d3_event.sourceEvent.type === 'pointerdown');
+         }).on('end.map', function () {
+           _pointerDown = false;
+         });
 
-           if (!_cache[ent].parents) {
-             var parents = [];
+         var _doubleUpHandler = utilDoubleUp();
 
-             if (geometry === 'vertex') {
-               parents = resolver.parentWays(entity);
-             } else {
-               // 'line', 'area', 'relation'
-               parents = resolver.parentRelations(entity);
-             }
+         var scheduleRedraw = throttle(redraw, 750); // var isRedrawScheduled = false;
+         // var pendingRedrawCall;
+         // function scheduleRedraw() {
+         //     // Only schedule the redraw if one has not already been set.
+         //     if (isRedrawScheduled) return;
+         //     isRedrawScheduled = true;
+         //     var that = this;
+         //     var args = arguments;
+         //     pendingRedrawCall = window.requestIdleCallback(function () {
+         //         // Reset the boolean so future redraws can be set.
+         //         isRedrawScheduled = false;
+         //         redraw.apply(that, args);
+         //     }, { timeout: 1400 });
+         // }
 
-             _cache[ent].parents = parents;
-           }
 
-           return _cache[ent].parents;
-         };
+         function cancelPendingRedraw() {
+           scheduleRedraw.cancel(); // isRedrawScheduled = false;
+           // window.cancelIdleCallback(pendingRedrawCall);
+         }
 
-         features.isHiddenPreset = function (preset, geometry) {
-           if (!_hidden.length) return false;
-           if (!preset.tags) return false;
-           var test = preset.setTags({}, geometry);
+         function map(selection) {
+           _selection = selection;
+           context.on('change.map', immediateRedraw);
+           var osm = context.connection();
 
-           for (var key in _rules) {
-             if (_rules[key].filter(test, geometry)) {
-               if (_hidden.indexOf(key) !== -1) {
-                 return key;
-               }
+           if (osm) {
+             osm.on('change.map', immediateRedraw);
+           }
 
-               return false;
+           function didUndoOrRedo(targetTransform) {
+             var mode = context.mode().id;
+             if (mode !== 'browse' && mode !== 'select') return;
+
+             if (targetTransform) {
+               map.transformEase(targetTransform);
              }
            }
 
-           return false;
-         };
-
-         features.isHiddenFeature = function (entity, resolver, geometry) {
-           if (!_hidden.length) return false;
-           if (!entity.version) return false;
-           if (_forceVisible[entity.id]) return false;
-           var matches = Object.keys(features.getMatches(entity, resolver, geometry));
-           return matches.length && matches.every(function (k) {
-             return features.hidden(k);
+           context.history().on('merge.map', function () {
+             scheduleRedraw();
+           }).on('change.map', immediateRedraw).on('undone.map', function (stack, fromStack) {
+             didUndoOrRedo(fromStack.transform);
+           }).on('redone.map', function (stack) {
+             didUndoOrRedo(stack.transform);
            });
-         };
+           context.background().on('change.map', immediateRedraw);
+           context.features().on('redraw.map', immediateRedraw);
+           drawLayers.on('change.map', function () {
+             context.background().updateImagery();
+             immediateRedraw();
+           });
+           selection.on('wheel.map mousewheel.map', function (d3_event) {
+             // disable swipe-to-navigate browser pages on trackpad/magic mouse – #5552
+             d3_event.preventDefault();
+           }).call(_zoomerPanner).call(_zoomerPanner.transform, projection.transform()).on('dblclick.zoom', null); // override d3-zoom dblclick handling
 
-         features.isHiddenChild = function (entity, resolver, geometry) {
-           if (!_hidden.length) return false;
-           if (!entity.version || geometry === 'point') return false;
-           if (_forceVisible[entity.id]) return false;
-           var parents = features.getParents(entity, resolver, geometry);
-           if (!parents.length) return false;
+           map.supersurface = supersurface = selection.append('div').attr('class', 'supersurface').call(utilSetTransform, 0, 0); // Need a wrapper div because Opera can't cope with an absolutely positioned
+           // SVG element: http://bl.ocks.org/jfirebaugh/6fbfbd922552bf776c16
 
-           for (var i = 0; i < parents.length; i++) {
-             if (!features.isHidden(parents[i], resolver, parents[i].geometry(resolver))) {
-               return false;
+           wrapper = supersurface.append('div').attr('class', 'layer layer-data');
+           map.surface = surface = wrapper.call(drawLayers).selectAll('.surface');
+           surface.call(drawLabels.observe).call(_doubleUpHandler).on(_pointerPrefix + 'down.zoom', function (d3_event) {
+             _lastPointerEvent = d3_event;
+
+             if (d3_event.button === 2) {
+               d3_event.stopPropagation();
              }
-           }
+           }, true).on(_pointerPrefix + 'up.zoom', function (d3_event) {
+             _lastPointerEvent = d3_event;
 
-           return true;
-         };
+             if (resetTransform()) {
+               immediateRedraw();
+             }
+           }).on(_pointerPrefix + 'move.map', function (d3_event) {
+             _lastPointerEvent = d3_event;
+           }).on(_pointerPrefix + 'over.vertices', function (d3_event) {
+             if (map.editableDataEnabled() && !_isTransformed) {
+               var hover = d3_event.target.__data__;
+               surface.call(drawVertices.drawHover, context.graph(), hover, map.extent());
+               dispatch.call('drawn', this, {
+                 full: false
+               });
+             }
+           }).on(_pointerPrefix + 'out.vertices', function (d3_event) {
+             if (map.editableDataEnabled() && !_isTransformed) {
+               var hover = d3_event.relatedTarget && d3_event.relatedTarget.__data__;
+               surface.call(drawVertices.drawHover, context.graph(), hover, map.extent());
+               dispatch.call('drawn', this, {
+                 full: false
+               });
+             }
+           });
+           var detected = utilDetect(); // only WebKit supports gesture events
 
-         features.hasHiddenConnections = function (entity, resolver) {
-           if (!_hidden.length) return false;
-           var childNodes, connections;
+           if ('GestureEvent' in window && // Listening for gesture events on iOS 13.4+ breaks double-tapping,
+           // but we only need to do this on desktop Safari anyway. – #7694
+           !detected.isMobileWebKit) {
+             // Desktop Safari sends gesture events for multitouch trackpad pinches.
+             // We can listen for these and translate them into map zooms.
+             surface.on('gesturestart.surface', function (d3_event) {
+               d3_event.preventDefault();
+               _gestureTransformStart = projection.transform();
+             }).on('gesturechange.surface', gestureChange);
+           } // must call after surface init
 
-           if (entity.type === 'midpoint') {
-             childNodes = [resolver.entity(entity.edge[0]), resolver.entity(entity.edge[1])];
-             connections = [];
-           } else {
-             childNodes = entity.nodes ? resolver.childNodes(entity) : [];
-             connections = features.getParents(entity, resolver, entity.geometry(resolver));
-           } // gather ways connected to child nodes..
 
+           updateAreaFill();
 
-           connections = childNodes.reduce(function (result, e) {
-             return resolver.isShared(e) ? utilArrayUnion(result, resolver.parentWays(e)) : result;
-           }, connections);
-           return connections.some(function (e) {
-             return features.isHidden(e, resolver, e.geometry(resolver));
+           _doubleUpHandler.on('doubleUp.map', function (d3_event, p0) {
+             if (!_dblClickZoomEnabled) return; // don't zoom if targeting something other than the map itself
+
+             if (_typeof(d3_event.target.__data__) === 'object' && // or area fills
+             !select(d3_event.target).classed('fill')) return;
+             var zoomOut = d3_event.shiftKey;
+             var t = projection.transform();
+             var p1 = t.invert(p0);
+             t = t.scale(zoomOut ? 0.5 : 2);
+             t.x = p0[0] - p1[0] * t.k;
+             t.y = p0[1] - p1[1] * t.k;
+             map.transformEase(t);
            });
-         };
 
-         features.isHidden = function (entity, resolver, geometry) {
-           if (!_hidden.length) return false;
-           if (!entity.version) return false;
-           var fn = geometry === 'vertex' ? features.isHiddenChild : features.isHiddenFeature;
-           return fn(entity, resolver, geometry);
-         };
+           context.on('enter.map', function () {
+             if (!map.editableDataEnabled(true
+             /* skip zoom check */
+             )) return; // redraw immediately any objects affected by a change in selectedIDs.
 
-         features.filter = function (d, resolver) {
-           if (!_hidden.length) return d;
-           var result = [];
+             var graph = context.graph();
+             var selectedAndParents = {};
+             context.selectedIDs().forEach(function (id) {
+               var entity = graph.hasEntity(id);
 
-           for (var i = 0; i < d.length; i++) {
-             var entity = d[i];
+               if (entity) {
+                 selectedAndParents[entity.id] = entity;
 
-             if (!features.isHidden(entity, resolver, entity.geometry(resolver))) {
-               result.push(entity);
-             }
-           }
+                 if (entity.type === 'node') {
+                   graph.parentWays(entity).forEach(function (parent) {
+                     selectedAndParents[parent.id] = parent;
+                   });
+                 }
+               }
+             });
+             var data = Object.values(selectedAndParents);
 
-           return result;
-         };
+             var filter = function filter(d) {
+               return d.id in selectedAndParents;
+             };
 
-         features.forceVisible = function (entityIDs) {
-           if (!arguments.length) return Object.keys(_forceVisible);
-           _forceVisible = {};
+             data = context.features().filter(data, graph);
+             surface.call(drawVertices.drawSelected, graph, map.extent()).call(drawLines, graph, data, filter).call(drawAreas, graph, data, filter).call(drawMidpoints, graph, data, filter, map.trimmedExtent());
+             dispatch.call('drawn', this, {
+               full: false
+             }); // redraw everything else later
 
-           for (var i = 0; i < entityIDs.length; i++) {
-             _forceVisible[entityIDs[i]] = true;
-             var entity = context.hasEntity(entityIDs[i]);
+             scheduleRedraw();
+           });
+           map.dimensions(utilGetDimensions(selection));
+         }
 
-             if (entity && entity.type === 'relation') {
-               // also show relation members (one level deep)
-               for (var j in entity.members) {
-                 _forceVisible[entity.members[j].id] = true;
+         function zoomEventFilter(d3_event) {
+           // Fix for #2151, (see also d3/d3-zoom#60, d3/d3-brush#18)
+           // Intercept `mousedown` and check if there is an orphaned zoom gesture.
+           // This can happen if a previous `mousedown` occurred without a `mouseup`.
+           // If we detect this, dispatch `mouseup` to complete the orphaned gesture,
+           // so that d3-zoom won't stop propagation of new `mousedown` events.
+           if (d3_event.type === 'mousedown') {
+             var hasOrphan = false;
+             var listeners = window.__on;
+
+             for (var i = 0; i < listeners.length; i++) {
+               var listener = listeners[i];
+
+               if (listener.name === 'zoom' && listener.type === 'mouseup') {
+                 hasOrphan = true;
+                 break;
                }
              }
-           }
 
-           return features;
-         };
+             if (hasOrphan) {
+               var event = window.CustomEvent;
 
-         features.init = function () {
-           var storage = corePreferences('disabled-features');
+               if (event) {
+                 event = new event('mouseup');
+               } else {
+                 event = window.document.createEvent('Event');
+                 event.initEvent('mouseup', false, false);
+               } // Event needs to be dispatched with an event.view property.
 
-           if (storage) {
-             var storageDisabled = storage.replace(/;/g, ',').split(',');
-             storageDisabled.forEach(features.disable);
+
+               event.view = window;
+               window.dispatchEvent(event);
+             }
            }
 
-           var hash = utilStringQs(window.location.hash);
+           return d3_event.button !== 2; // ignore right clicks
+         }
 
-           if (hash.disable_features) {
-             var hashDisabled = hash.disable_features.replace(/;/g, ',').split(',');
-             hashDisabled.forEach(features.disable);
-           }
-         }; // warm up the feature matching cache upon merging fetched data
+         function pxCenter() {
+           return [_dimensions[0] / 2, _dimensions[1] / 2];
+         }
+
+         function drawEditable(difference, extent) {
+           var mode = context.mode();
+           var graph = context.graph();
+           var features = context.features();
+           var all = context.history().intersects(map.extent());
+           var fullRedraw = false;
+           var data;
+           var set;
+           var filter;
+           var applyFeatureLayerFilters = true;
 
+           if (map.isInWideSelection()) {
+             data = [];
+             utilEntityAndDeepMemberIDs(mode.selectedIDs(), context.graph()).forEach(function (id) {
+               var entity = context.hasEntity(id);
+               if (entity) data.push(entity);
+             });
+             fullRedraw = true;
+             filter = utilFunctor(true); // selected features should always be visible, so we can skip filtering
 
-         context.history().on('merge.features', function (newEntities) {
-           if (!newEntities) return;
-           var handle = window.requestIdleCallback(function () {
-             var graph = context.graph();
-             var types = utilArrayGroupBy(newEntities, 'type'); // ensure that getMatches is called on relations before ways
+             applyFeatureLayerFilters = false;
+           } else if (difference) {
+             var complete = difference.complete(map.extent());
+             data = Object.values(complete).filter(Boolean);
+             set = new Set(Object.keys(complete));
 
-             var entities = [].concat(types.relation || [], types.way || [], types.node || []);
+             filter = function filter(d) {
+               return set.has(d.id);
+             };
 
-             for (var i = 0; i < entities.length; i++) {
-               var geometry = entities[i].geometry(graph);
-               features.getMatches(entities[i], graph, geometry);
+             features.clear(data);
+           } else {
+             // force a full redraw if gatherStats detects that a feature
+             // should be auto-hidden (e.g. points or buildings)..
+             if (features.gatherStats(all, graph, _dimensions)) {
+               extent = undefined;
              }
-           });
 
-           _deferred.add(handle);
-         });
-         return features;
-       }
+             if (extent) {
+               data = context.history().intersects(map.extent().intersection(extent));
+               set = new Set(data.map(function (entity) {
+                 return entity.id;
+               }));
 
-       //
-       // - the activeID - nope
-       // - 1 away (adjacent) to the activeID - yes (vertices will be merged)
-       // - 2 away from the activeID - nope (would create a self intersecting segment)
-       // - all others on a linear way - yes
-       // - all others on a closed way - nope (would create a self intersecting polygon)
-       //
-       // returns
-       // 0 = active vertex - no touch/connect
-       // 1 = passive vertex - yes touch/connect
-       // 2 = adjacent vertex - yes but pay attention segmenting a line here
-       //
+               filter = function filter(d) {
+                 return set.has(d.id);
+               };
+             } else {
+               data = all;
+               fullRedraw = true;
+               filter = utilFunctor(true);
+             }
+           }
 
-       function svgPassiveVertex(node, graph, activeID) {
-         if (!activeID) return 1;
-         if (activeID === node.id) return 0;
-         var parents = graph.parentWays(node);
-         var i, j, nodes, isClosed, ix1, ix2, ix3, ix4, max;
+           if (applyFeatureLayerFilters) {
+             data = features.filter(data, graph);
+           } else {
+             context.features().resetStats();
+           }
 
-         for (i = 0; i < parents.length; i++) {
-           nodes = parents[i].nodes;
-           isClosed = parents[i].isClosed();
+           if (mode && mode.id === 'select') {
+             // update selected vertices - the user might have just double-clicked a way,
+             // creating a new vertex, triggering a partial redraw without a mode change
+             surface.call(drawVertices.drawSelected, graph, map.extent());
+           }
 
-           for (j = 0; j < nodes.length; j++) {
-             // find this vertex, look nearby
-             if (nodes[j] === node.id) {
-               ix1 = j - 2;
-               ix2 = j - 1;
-               ix3 = j + 1;
-               ix4 = j + 2;
+           surface.call(drawVertices, graph, data, filter, map.extent(), fullRedraw).call(drawLines, graph, data, filter).call(drawAreas, graph, data, filter).call(drawMidpoints, graph, data, filter, map.trimmedExtent()).call(drawLabels, graph, data, filter, _dimensions, fullRedraw).call(drawPoints, graph, data, filter);
+           dispatch.call('drawn', this, {
+             full: true
+           });
+         }
 
-               if (isClosed) {
-                 // wraparound if needed
-                 max = nodes.length - 1;
-                 if (ix1 < 0) ix1 = max + ix1;
-                 if (ix2 < 0) ix2 = max + ix2;
-                 if (ix3 > max) ix3 = ix3 - max;
-                 if (ix4 > max) ix4 = ix4 - max;
-               }
+         map.init = function () {
+           drawLayers = svgLayers(projection, context);
+           drawPoints = svgPoints(projection, context);
+           drawVertices = svgVertices(projection, context);
+           drawLines = svgLines(projection, context);
+           drawAreas = svgAreas(projection, context);
+           drawMidpoints = svgMidpoints(projection, context);
+           drawLabels = svgLabels(projection, context);
+         };
 
-               if (nodes[ix1] === activeID) return 0; // no - prevent self intersect
-               else if (nodes[ix2] === activeID) return 2; // ok - adjacent
-                 else if (nodes[ix3] === activeID) return 2; // ok - adjacent
-                   else if (nodes[ix4] === activeID) return 0; // no - prevent self intersect
-                     else if (isClosed && nodes.indexOf(activeID) !== -1) return 0; // no - prevent self intersect
-             }
+         function editOff() {
+           context.features().resetStats();
+           surface.selectAll('.layer-osm *').remove();
+           surface.selectAll('.layer-touch:not(.markers) *').remove();
+           var allowed = {
+             'browse': true,
+             'save': true,
+             'select-note': true,
+             'select-data': true,
+             'select-error': true
+           };
+           var mode = context.mode();
+
+           if (mode && !allowed[mode.id]) {
+             context.enter(modeBrowse(context));
            }
-         }
 
-         return 1; // ok
-       }
-       function svgMarkerSegments(projection, graph, dt, shouldReverse, bothDirections) {
-         return function (entity) {
-           var i = 0;
-           var offset = dt;
-           var segments = [];
-           var clip = d3_geoIdentity().clipExtent(projection.clipExtent()).stream;
-           var coordinates = graph.childNodes(entity).map(function (n) {
-             return n.loc;
+           dispatch.call('drawn', this, {
+             full: true
            });
-           var a, b;
-
-           if (shouldReverse(entity)) {
-             coordinates.reverse();
-           }
+         }
 
-           d3_geoStream({
-             type: 'LineString',
-             coordinates: coordinates
-           }, projection.stream(clip({
-             lineStart: function lineStart() {},
-             lineEnd: function lineEnd() {
-               a = null;
-             },
-             point: function point(x, y) {
-               b = [x, y];
+         function gestureChange(d3_event) {
+           // Remap Safari gesture events to wheel events - #5492
+           // We want these disabled most places, but enabled for zoom/unzoom on map surface
+           // https://developer.mozilla.org/en-US/docs/Web/API/GestureEvent
+           var e = d3_event;
+           e.preventDefault();
+           var props = {
+             deltaMode: 0,
+             // dummy values to ignore in zoomPan
+             deltaY: 1,
+             // dummy values to ignore in zoomPan
+             clientX: e.clientX,
+             clientY: e.clientY,
+             screenX: e.screenX,
+             screenY: e.screenY,
+             x: e.x,
+             y: e.y
+           };
+           var e2 = new WheelEvent('wheel', props);
+           e2._scale = e.scale; // preserve the original scale
 
-               if (a) {
-                 var span = geoVecLength(a, b) - offset;
+           e2._rotation = e.rotation; // preserve the original rotation
 
-                 if (span >= 0) {
-                   var heading = geoVecAngle(a, b);
-                   var dx = dt * Math.cos(heading);
-                   var dy = dt * Math.sin(heading);
-                   var p = [a[0] + offset * Math.cos(heading), a[1] + offset * Math.sin(heading)]; // gather coordinates
+           _selection.node().dispatchEvent(e2);
+         }
 
-                   var coord = [a, p];
+         function zoomPan(event, key, transform) {
+           var source = event && event.sourceEvent || event;
+           var eventTransform = transform || event && event.transform;
+           var x = eventTransform.x;
+           var y = eventTransform.y;
+           var k = eventTransform.k; // Special handling of 'wheel' events:
+           // They might be triggered by the user scrolling the mouse wheel,
+           // or 2-finger pinch/zoom gestures, the transform may need adjustment.
 
-                   for (span -= dt; span >= 0; span -= dt) {
-                     p = geoVecAdd(p, [dx, dy]);
-                     coord.push(p);
-                   }
+           if (source && source.type === 'wheel') {
+             // assume that the gesture is already handled by pointer events
+             if (_pointerDown) return;
+             var detected = utilDetect();
+             var dX = source.deltaX;
+             var dY = source.deltaY;
+             var x2 = x;
+             var y2 = y;
+             var k2 = k;
+             var t0, p0, p1; // Normalize mousewheel scroll speed (Firefox) - #3029
+             // If wheel delta is provided in LINE units, recalculate it in PIXEL units
+             // We are essentially redoing the calculations that occur here:
+             //   https://github.com/d3/d3-zoom/blob/78563a8348aa4133b07cac92e2595c2227ca7cd7/src/zoom.js#L203
+             // See this for more info:
+             //   https://github.com/basilfx/normalize-wheel/blob/master/src/normalizeWheel.js
 
-                   coord.push(b); // generate svg paths
+             if (source.deltaMode === 1
+             /* LINE */
+             ) {
+                 // Convert from lines to pixels, more if the user is scrolling fast.
+                 // (I made up the exp function to roughly match Firefox to what Chrome does)
+                 // These numbers should be floats, because integers are treated as pan gesture below.
+                 var lines = Math.abs(source.deltaY);
+                 var sign = source.deltaY > 0 ? 1 : -1;
+                 dY = sign * clamp$1(Math.exp((lines - 1) * 0.75) * 4.000244140625, 4.000244140625, // min
+                 350.000244140625 // max
+                 ); // On Firefox Windows and Linux we always get +/- the scroll line amount (default 3)
+                 // There doesn't seem to be any scroll acceleration.
+                 // This multiplier increases the speed a little bit - #5512
 
-                   var segment = '';
-                   var j;
+                 if (detected.os !== 'mac') {
+                   dY *= 5;
+                 } // recalculate x2,y2,k2
 
-                   for (j = 0; j < coord.length; j++) {
-                     segment += (j === 0 ? 'M' : 'L') + coord[j][0] + ',' + coord[j][1];
-                   }
 
-                   segments.push({
-                     id: entity.id,
-                     index: i++,
-                     d: segment
-                   });
+                 t0 = _isTransformed ? _transformLast : _transformStart;
+                 p0 = _getMouseCoords(source);
+                 p1 = t0.invert(p0);
+                 k2 = t0.k * Math.pow(2, -dY / 500);
+                 k2 = clamp$1(k2, kMin, kMax);
+                 x2 = p0[0] - p1[0] * k2;
+                 y2 = p0[1] - p1[1] * k2; // 2 finger map pinch zooming (Safari) - #5492
+                 // These are fake `wheel` events we made from Safari `gesturechange` events..
+               } else if (source._scale) {
+               // recalculate x2,y2,k2
+               t0 = _gestureTransformStart;
+               p0 = _getMouseCoords(source);
+               p1 = t0.invert(p0);
+               k2 = t0.k * source._scale;
+               k2 = clamp$1(k2, kMin, kMax);
+               x2 = p0[0] - p1[0] * k2;
+               y2 = p0[1] - p1[1] * k2; // 2 finger map pinch zooming (all browsers except Safari) - #5492
+               // Pinch zooming via the `wheel` event will always have:
+               // - `ctrlKey = true`
+               // - `deltaY` is not round integer pixels (ignore `deltaX`)
+             } else if (source.ctrlKey && !isInteger(dY)) {
+               dY *= 6; // slightly scale up whatever the browser gave us
+               // recalculate x2,y2,k2
 
-                   if (bothDirections(entity)) {
-                     segment = '';
+               t0 = _isTransformed ? _transformLast : _transformStart;
+               p0 = _getMouseCoords(source);
+               p1 = t0.invert(p0);
+               k2 = t0.k * Math.pow(2, -dY / 500);
+               k2 = clamp$1(k2, kMin, kMax);
+               x2 = p0[0] - p1[0] * k2;
+               y2 = p0[1] - p1[1] * k2; // Trackpad scroll zooming with shift or alt/option key down
+             } else if ((source.altKey || source.shiftKey) && isInteger(dY)) {
+               // recalculate x2,y2,k2
+               t0 = _isTransformed ? _transformLast : _transformStart;
+               p0 = _getMouseCoords(source);
+               p1 = t0.invert(p0);
+               k2 = t0.k * Math.pow(2, -dY / 500);
+               k2 = clamp$1(k2, kMin, kMax);
+               x2 = p0[0] - p1[0] * k2;
+               y2 = p0[1] - p1[1] * k2; // 2 finger map panning (Mac only, all browsers) - #5492, #5512
+               // Panning via the `wheel` event will always have:
+               // - `ctrlKey = false`
+               // - `deltaX`,`deltaY` are round integer pixels
+             } else if (detected.os === 'mac' && !source.ctrlKey && isInteger(dX) && isInteger(dY)) {
+               p1 = projection.translate();
+               x2 = p1[0] - dX;
+               y2 = p1[1] - dY;
+               k2 = projection.scale();
+               k2 = clamp$1(k2, kMin, kMax);
+             } // something changed - replace the event transform
 
-                     for (j = coord.length - 1; j >= 0; j--) {
-                       segment += (j === coord.length - 1 ? 'M' : 'L') + coord[j][0] + ',' + coord[j][1];
-                     }
 
-                     segments.push({
-                       id: entity.id,
-                       index: i++,
-                       d: segment
-                     });
-                   }
-                 }
+             if (x2 !== x || y2 !== y || k2 !== k) {
+               x = x2;
+               y = y2;
+               k = k2;
+               eventTransform = identity$2.translate(x2, y2).scale(k2);
 
-                 offset = -span;
+               if (_zoomerPanner._transform) {
+                 // utilZoomPan interface
+                 _zoomerPanner._transform(eventTransform);
+               } else {
+                 // d3_zoom interface
+                 _selection.node().__zoom = eventTransform;
                }
-
-               a = b;
              }
-           })));
-           return segments;
-         };
-       }
-       function svgPath(projection, graph, isArea) {
-         // Explanation of magic numbers:
-         // "padding" here allows space for strokes to extend beyond the viewport,
-         // so that the stroke isn't drawn along the edge of the viewport when
-         // the shape is clipped.
-         //
-         // When drawing lines, pad viewport by 5px.
-         // When drawing areas, pad viewport by 65px in each direction to allow
-         // for 60px area fill stroke (see ".fill-partial path.fill" css rule)
-         var cache = {};
-         var padding = isArea ? 65 : 5;
-         var viewport = projection.clipExtent();
-         var paddedExtent = [[viewport[0][0] - padding, viewport[0][1] - padding], [viewport[1][0] + padding, viewport[1][1] + padding]];
-         var clip = d3_geoIdentity().clipExtent(paddedExtent).stream;
-         var project = projection.stream;
-         var path = d3_geoPath().projection({
-           stream: function stream(output) {
-             return project(clip(output));
            }
-         });
 
-         var svgpath = function svgpath(entity) {
-           if (entity.id in cache) {
-             return cache[entity.id];
-           } else {
-             return cache[entity.id] = path(entity.asGeoJSON(graph));
+           if (_transformStart.x === x && _transformStart.y === y && _transformStart.k === k) {
+             return; // no change
            }
-         };
 
-         svgpath.geojson = function (d) {
-           if (d.__featurehash__ !== undefined) {
-             if (d.__featurehash__ in cache) {
-               return cache[d.__featurehash__];
-             } else {
-               return cache[d.__featurehash__] = path(d);
-             }
-           } else {
-             return path(d);
+           if (geoScaleToZoom(k, TILESIZE) < _minzoom) {
+             surface.interrupt();
+             dispatch.call('hitMinZoom', this, map);
+             setCenterZoom(map.center(), context.minEditableZoom(), 0, true);
+             scheduleRedraw();
+             dispatch.call('move', this, map);
+             return;
            }
-         };
-
-         return svgpath;
-       }
-       function svgPointTransform(projection) {
-         var svgpoint = function svgpoint(entity) {
-           // http://jsperf.com/short-array-join
-           var pt = projection(entity.loc);
-           return 'translate(' + pt[0] + ',' + pt[1] + ')';
-         };
-
-         svgpoint.geojson = function (d) {
-           return svgpoint(d.properties.entity);
-         };
 
-         return svgpoint;
-       }
-       function svgRelationMemberTags(graph) {
-         return function (entity) {
-           var tags = entity.tags;
-           var shouldCopyMultipolygonTags = !entity.hasInterestingTags();
-           graph.parentRelations(entity).forEach(function (relation) {
-             var type = relation.tags.type;
+           projection.transform(eventTransform);
+           var withinEditableZoom = map.withinEditableZoom();
 
-             if (type === 'multipolygon' && shouldCopyMultipolygonTags || type === 'boundary') {
-               tags = Object.assign({}, relation.tags, tags);
+           if (_lastWithinEditableZoom !== withinEditableZoom) {
+             if (_lastWithinEditableZoom !== undefined) {
+               // notify that the map zoomed in or out over the editable zoom threshold
+               dispatch.call('crossEditableZoom', this, withinEditableZoom);
              }
-           });
-           return tags;
-         };
-       }
-       function svgSegmentWay(way, graph, activeID) {
-         // When there is no activeID, we can memoize this expensive computation
-         if (activeID === undefined) {
-           return graph["transient"](way, 'waySegments', getWaySegments);
-         } else {
-           return getWaySegments();
-         }
 
-         function getWaySegments() {
-           var isActiveWay = way.nodes.indexOf(activeID) !== -1;
-           var features = {
-             passive: [],
-             active: []
-           };
-           var start = {};
-           var end = {};
-           var node, type;
+             _lastWithinEditableZoom = withinEditableZoom;
+           }
 
-           for (var i = 0; i < way.nodes.length; i++) {
-             node = graph.entity(way.nodes[i]);
-             type = svgPassiveVertex(node, graph, activeID);
-             end = {
-               node: node,
-               type: type
-             };
+           var scale = k / _transformStart.k;
+           var tX = (x / scale - _transformStart.x) * scale;
+           var tY = (y / scale - _transformStart.y) * scale;
 
-             if (start.type !== undefined) {
-               if (start.node.id === activeID || end.node.id === activeID) ; else if (isActiveWay && (start.type === 2 || end.type === 2)) {
-                 // one adjacent vertex
-                 pushActive(start, end, i);
-               } else if (start.type === 0 && end.type === 0) {
-                 // both active vertices
-                 pushActive(start, end, i);
-               } else {
-                 pushPassive(start, end, i);
-               }
-             }
+           if (context.inIntro()) {
+             curtainProjection.transform({
+               x: x - tX,
+               y: y - tY,
+               k: k
+             });
+           }
 
-             start = end;
+           if (source) {
+             _lastPointerEvent = event;
            }
 
-           return features;
+           _isTransformed = true;
+           _transformLast = eventTransform;
+           utilSetTransform(supersurface, tX, tY, scale);
+           scheduleRedraw();
+           dispatch.call('move', this, map);
 
-           function pushActive(start, end, index) {
-             features.active.push({
-               type: 'Feature',
-               id: way.id + '-' + index + '-nope',
-               properties: {
-                 nope: true,
-                 target: true,
-                 entity: way,
-                 nodes: [start.node, end.node],
-                 index: index
-               },
-               geometry: {
-                 type: 'LineString',
-                 coordinates: [start.node.loc, end.node.loc]
-               }
-             });
+           function isInteger(val) {
+             return typeof val === 'number' && isFinite(val) && Math.floor(val) === val;
            }
+         }
 
-           function pushPassive(start, end, index) {
-             features.passive.push({
-               type: 'Feature',
-               id: way.id + '-' + index,
-               properties: {
-                 target: true,
-                 entity: way,
-                 nodes: [start.node, end.node],
-                 index: index
-               },
-               geometry: {
-                 type: 'LineString',
-                 coordinates: [start.node.loc, end.node.loc]
-               }
-             });
+         function resetTransform() {
+           if (!_isTransformed) return false;
+           utilSetTransform(supersurface, 0, 0);
+           _isTransformed = false;
+
+           if (context.inIntro()) {
+             curtainProjection.transform(projection.transform());
            }
+
+           return true;
          }
-       }
 
-       function svgTagClasses() {
-         var primaries = ['building', 'highway', 'railway', 'waterway', 'aeroway', 'aerialway', 'piste:type', 'boundary', 'power', 'amenity', 'natural', 'landuse', 'leisure', 'military', 'place', 'man_made', 'route', 'attraction', 'building:part', 'indoor'];
-         var statuses = [// nonexistent, might be built
-         'proposed', 'planned', // under maintentance or between groundbreaking and opening
-         'construction', // existent but not functional
-         'disused', // dilapidated to nonexistent
-         'abandoned', // nonexistent, still may appear in imagery
-         'dismantled', 'razed', 'demolished', 'obliterated', // existent occasionally, e.g. stormwater drainage basin
-         'intermittent'];
-         var secondaries = ['oneway', 'bridge', 'tunnel', 'embankment', 'cutting', 'barrier', 'surface', 'tracktype', 'footway', 'crossing', 'service', 'sport', 'public_transport', 'location', 'parking', 'golf', 'type', 'leisure', 'man_made', 'indoor'];
+         function redraw(difference, extent) {
+           if (surface.empty() || !_redrawEnabled) return; // If we are in the middle of a zoom/pan, we can't do differenced redraws.
+           // It would result in artifacts where differenced entities are redrawn with
+           // one transform and unchanged entities with another.
+
+           if (resetTransform()) {
+             difference = extent = undefined;
+           }
+
+           var zoom = map.zoom();
+           var z = String(~~zoom);
+
+           if (surface.attr('data-zoom') !== z) {
+             surface.attr('data-zoom', z);
+           } // class surface as `lowzoom` around z17-z18.5 (based on latitude)
+
+
+           var lat = map.center()[1];
+           var lowzoom = linear().domain([-60, 0, 60]).range([17, 18.5, 17]).clamp(true);
+           surface.classed('low-zoom', zoom <= lowzoom(lat));
+
+           if (!difference) {
+             supersurface.call(context.background());
+             wrapper.call(drawLayers);
+           } // OSM
+
+
+           if (map.editableDataEnabled() || map.isInWideSelection()) {
+             context.loadTiles(projection);
+             drawEditable(difference, extent);
+           } else {
+             editOff();
+           }
+
+           _transformStart = projection.transform();
+           return map;
+         }
 
-         var _tags = function _tags(entity) {
-           return entity.tags;
+         var immediateRedraw = function immediateRedraw(difference, extent) {
+           if (!difference && !extent) cancelPendingRedraw();
+           redraw(difference, extent);
          };
 
-         var tagClasses = function tagClasses(selection) {
-           selection.each(function tagClassesEach(entity) {
-             var value = this.className;
-
-             if (value.baseVal !== undefined) {
-               value = value.baseVal;
-             }
+         map.lastPointerEvent = function () {
+           return _lastPointerEvent;
+         };
 
-             var t = _tags(entity);
+         map.mouse = function (d3_event) {
+           var event = d3_event || _lastPointerEvent;
 
-             var computed = tagClasses.getClassesString(t, value);
+           if (event) {
+             var s;
 
-             if (computed !== value) {
-               select(this).attr('class', computed);
+             while (s = event.sourceEvent) {
+               event = s;
              }
-           });
-         };
 
-         tagClasses.getClassesString = function (t, value) {
-           var primary, status;
-           var i, j, k, v; // in some situations we want to render perimeter strokes a certain way
+             return _getMouseCoords(event);
+           }
 
-           var overrideGeometry;
+           return null;
+         }; // returns Lng/Lat
 
-           if (/\bstroke\b/.test(value)) {
-             if (!!t.barrier && t.barrier !== 'no') {
-               overrideGeometry = 'line';
-             }
-           } // preserve base classes (nothing with `tag-`)
 
+         map.mouseCoordinates = function () {
+           var coord = map.mouse() || pxCenter();
+           return projection.invert(coord);
+         };
 
-           var classes = value.trim().split(/\s+/).filter(function (klass) {
-             return klass.length && !/^tag-/.test(klass);
-           }).map(function (klass) {
-             // special overrides for some perimeter strokes
-             return klass === 'line' || klass === 'area' ? overrideGeometry || klass : klass;
-           }); // pick at most one primary classification tag..
+         map.dblclickZoomEnable = function (val) {
+           if (!arguments.length) return _dblClickZoomEnabled;
+           _dblClickZoomEnabled = val;
+           return map;
+         };
 
-           for (i = 0; i < primaries.length; i++) {
-             k = primaries[i];
-             v = t[k];
-             if (!v || v === 'no') continue;
+         map.redrawEnable = function (val) {
+           if (!arguments.length) return _redrawEnabled;
+           _redrawEnabled = val;
+           return map;
+         };
 
-             if (k === 'piste:type') {
-               // avoid a ':' in the class name
-               k = 'piste';
-             } else if (k === 'building:part') {
-               // avoid a ':' in the class name
-               k = 'building_part';
-             }
+         map.isTransformed = function () {
+           return _isTransformed;
+         };
 
-             primary = k;
+         function setTransform(t2, duration, force) {
+           var t = projection.transform();
+           if (!force && t2.k === t.k && t2.x === t.x && t2.y === t.y) return false;
 
-             if (statuses.indexOf(v) !== -1) {
-               // e.g. `railway=abandoned`
-               status = v;
-               classes.push('tag-' + k);
-             } else {
-               classes.push('tag-' + k);
-               classes.push('tag-' + k + '-' + v);
-             }
+           if (duration) {
+             _selection.transition().duration(duration).on('start', function () {
+               map.startEase();
+             }).call(_zoomerPanner.transform, identity$2.translate(t2.x, t2.y).scale(t2.k));
+           } else {
+             projection.transform(t2);
+             _transformStart = t2;
 
-             break;
+             _selection.call(_zoomerPanner.transform, _transformStart);
            }
 
-           if (!primary) {
-             for (i = 0; i < statuses.length; i++) {
-               for (j = 0; j < primaries.length; j++) {
-                 k = statuses[i] + ':' + primaries[j]; // e.g. `demolished:building=yes`
+           return true;
+         }
 
-                 v = t[k];
-                 if (!v || v === 'no') continue;
-                 status = statuses[i];
-                 break;
-               }
-             }
-           } // add at most one status tag, only if relates to primary tag..
+         function setCenterZoom(loc2, z2, duration, force) {
+           var c = map.center();
+           var z = map.zoom();
+           if (loc2[0] === c[0] && loc2[1] === c[1] && z2 === z && !force) return false;
+           var proj = geoRawMercator().transform(projection.transform()); // copy projection
 
+           var k2 = clamp$1(geoZoomToScale(z2, TILESIZE), kMin, kMax);
+           proj.scale(k2);
+           var t = proj.translate();
+           var point = proj(loc2);
+           var center = pxCenter();
+           t[0] += center[0] - point[0];
+           t[1] += center[1] - point[1];
+           return setTransform(identity$2.translate(t[0], t[1]).scale(k2), duration, force);
+         }
 
-           if (!status) {
-             for (i = 0; i < statuses.length; i++) {
-               k = statuses[i];
-               v = t[k];
-               if (!v || v === 'no') continue;
+         map.pan = function (delta, duration) {
+           var t = projection.translate();
+           var k = projection.scale();
+           t[0] += delta[0];
+           t[1] += delta[1];
 
-               if (v === 'yes') {
-                 // e.g. `railway=rail + abandoned=yes`
-                 status = k;
-               } else if (primary && primary === v) {
-                 // e.g. `railway=rail + abandoned=railway`
-                 status = k;
-               } else if (!primary && primaries.indexOf(v) !== -1) {
-                 // e.g. `abandoned=railway`
-                 status = k;
-                 primary = v;
-                 classes.push('tag-' + v);
-               } // else ignore e.g.  `highway=path + abandoned=railway`
+           if (duration) {
+             _selection.transition().duration(duration).on('start', function () {
+               map.startEase();
+             }).call(_zoomerPanner.transform, identity$2.translate(t[0], t[1]).scale(k));
+           } else {
+             projection.translate(t);
+             _transformStart = projection.transform();
 
+             _selection.call(_zoomerPanner.transform, _transformStart);
 
-               if (status) break;
-             }
+             dispatch.call('move', this, map);
+             immediateRedraw();
            }
 
-           if (status) {
-             classes.push('tag-status');
-             classes.push('tag-status-' + status);
-           } // add any secondary tags
+           return map;
+         };
 
+         map.dimensions = function (val) {
+           if (!arguments.length) return _dimensions;
+           _dimensions = val;
+           drawLayers.dimensions(_dimensions);
+           context.background().dimensions(_dimensions);
+           projection.clipExtent([[0, 0], _dimensions]);
+           _getMouseCoords = utilFastMouse(supersurface.node());
+           scheduleRedraw();
+           return map;
+         };
 
-           for (i = 0; i < secondaries.length; i++) {
-             k = secondaries[i];
-             v = t[k];
-             if (!v || v === 'no' || k === primary) continue;
-             classes.push('tag-' + k);
-             classes.push('tag-' + k + '-' + v);
-           } // For highways, look for surface tagging..
+         function zoomIn(delta) {
+           setCenterZoom(map.center(), ~~map.zoom() + delta, 250, true);
+         }
 
+         function zoomOut(delta) {
+           setCenterZoom(map.center(), ~~map.zoom() - delta, 250, true);
+         }
 
-           if (primary === 'highway' && !osmPathHighwayTagValues[t.highway] || primary === 'aeroway') {
-             var surface = t.highway === 'track' ? 'unpaved' : 'paved';
+         map.zoomIn = function () {
+           zoomIn(1);
+         };
 
-             for (k in t) {
-               v = t[k];
+         map.zoomInFurther = function () {
+           zoomIn(4);
+         };
 
-               if (k in osmPavedTags) {
-                 surface = osmPavedTags[k][v] ? 'paved' : 'unpaved';
-               }
+         map.canZoomIn = function () {
+           return map.zoom() < maxZoom;
+         };
 
-               if (k in osmSemipavedTags && !!osmSemipavedTags[k][v]) {
-                 surface = 'semipaved';
-               }
-             }
+         map.zoomOut = function () {
+           zoomOut(1);
+         };
 
-             classes.push('tag-' + surface);
-           } // If this is a wikidata-tagged item, add a class for that..
+         map.zoomOutFurther = function () {
+           zoomOut(4);
+         };
 
+         map.canZoomOut = function () {
+           return map.zoom() > minZoom;
+         };
 
-           if (t.wikidata || t['brand:wikidata']) {
-             classes.push('tag-wikidata');
+         map.center = function (loc2) {
+           if (!arguments.length) {
+             return projection.invert(pxCenter());
            }
 
-           return classes.join(' ').trim();
+           if (setCenterZoom(loc2, map.zoom())) {
+             dispatch.call('move', this, map);
+           }
+
+           scheduleRedraw();
+           return map;
          };
 
-         tagClasses.tags = function (val) {
-           if (!arguments.length) return _tags;
-           _tags = val;
-           return tagClasses;
+         map.unobscuredCenterZoomEase = function (loc, zoom) {
+           var offset = map.unobscuredOffsetPx();
+           var proj = geoRawMercator().transform(projection.transform()); // copy projection
+           // use the target zoom to calculate the offset center
+
+           proj.scale(geoZoomToScale(zoom, TILESIZE));
+           var locPx = proj(loc);
+           var offsetLocPx = [locPx[0] + offset[0], locPx[1] + offset[1]];
+           var offsetLoc = proj.invert(offsetLocPx);
+           map.centerZoomEase(offsetLoc, zoom);
          };
 
-         return tagClasses;
-       }
+         map.unobscuredOffsetPx = function () {
+           var openPane = context.container().select('.map-panes .map-pane.shown');
 
-       // Patterns only work in Firefox when set directly on element.
-       // (This is not a bug: https://bugzilla.mozilla.org/show_bug.cgi?id=750632)
-       var patterns = {
-         // tag - pattern name
-         // -or-
-         // tag - value - pattern name
-         // -or-
-         // tag - value - rules (optional tag-values, pattern name)
-         // (matches earlier rules first, so fallback should be last entry)
-         amenity: {
-           grave_yard: 'cemetery',
-           fountain: 'water_standing'
-         },
-         landuse: {
-           cemetery: [{
-             religion: 'christian',
-             pattern: 'cemetery_christian'
-           }, {
-             religion: 'buddhist',
-             pattern: 'cemetery_buddhist'
-           }, {
-             religion: 'muslim',
-             pattern: 'cemetery_muslim'
-           }, {
-             religion: 'jewish',
-             pattern: 'cemetery_jewish'
-           }, {
-             pattern: 'cemetery'
-           }],
-           construction: 'construction',
-           farmland: 'farmland',
-           farmyard: 'farmyard',
-           forest: [{
-             leaf_type: 'broadleaved',
-             pattern: 'forest_broadleaved'
-           }, {
-             leaf_type: 'needleleaved',
-             pattern: 'forest_needleleaved'
-           }, {
-             leaf_type: 'leafless',
-             pattern: 'forest_leafless'
-           }, {
-             pattern: 'forest'
-           } // same as 'leaf_type:mixed'
-           ],
-           grave_yard: 'cemetery',
-           grass: [{
-             golf: 'green',
-             pattern: 'golf_green'
-           }, {
-             pattern: 'grass'
-           }],
-           landfill: 'landfill',
-           meadow: 'meadow',
-           military: 'construction',
-           orchard: 'orchard',
-           quarry: 'quarry',
-           vineyard: 'vineyard'
-         },
-         natural: {
-           beach: 'beach',
-           grassland: 'grass',
-           sand: 'beach',
-           scrub: 'scrub',
-           water: [{
-             water: 'pond',
-             pattern: 'pond'
-           }, {
-             water: 'reservoir',
-             pattern: 'water_standing'
-           }, {
-             pattern: 'waves'
-           }],
-           wetland: [{
-             wetland: 'marsh',
-             pattern: 'wetland_marsh'
-           }, {
-             wetland: 'swamp',
-             pattern: 'wetland_swamp'
-           }, {
-             wetland: 'bog',
-             pattern: 'wetland_bog'
-           }, {
-             wetland: 'reedbed',
-             pattern: 'wetland_reedbed'
-           }, {
-             pattern: 'wetland'
-           }],
-           wood: [{
-             leaf_type: 'broadleaved',
-             pattern: 'forest_broadleaved'
-           }, {
-             leaf_type: 'needleleaved',
-             pattern: 'forest_needleleaved'
-           }, {
-             leaf_type: 'leafless',
-             pattern: 'forest_leafless'
-           }, {
-             pattern: 'forest'
-           } // same as 'leaf_type:mixed'
-           ]
-         },
-         traffic_calming: {
-           island: [{
-             surface: 'grass',
-             pattern: 'grass'
-           }],
-           chicane: [{
-             surface: 'grass',
-             pattern: 'grass'
-           }],
-           choker: [{
-             surface: 'grass',
-             pattern: 'grass'
-           }]
-         }
-       };
-       function svgTagPattern(tags) {
-         // Skip pattern filling if this is a building (buildings don't get patterns applied)
-         if (tags.building && tags.building !== 'no') {
-           return null;
-         }
+           if (!openPane.empty()) {
+             return [openPane.node().offsetWidth / 2, 0];
+           }
 
-         for (var tag in patterns) {
-           var entityValue = tags[tag];
-           if (!entityValue) continue;
+           return [0, 0];
+         };
 
-           if (typeof patterns[tag] === 'string') {
-             // extra short syntax (just tag) - pattern name
-             return 'pattern-' + patterns[tag];
-           } else {
-             var values = patterns[tag];
+         map.zoom = function (z2) {
+           if (!arguments.length) {
+             return Math.max(geoScaleToZoom(projection.scale(), TILESIZE), 0);
+           }
 
-             for (var value in values) {
-               if (entityValue !== value) continue;
-               var rules = values[value];
+           if (z2 < _minzoom) {
+             surface.interrupt();
+             dispatch.call('hitMinZoom', this, map);
+             z2 = context.minEditableZoom();
+           }
 
-               if (typeof rules === 'string') {
-                 // short syntax - pattern name
-                 return 'pattern-' + rules;
-               } // long syntax - rule array
+           if (setCenterZoom(map.center(), z2)) {
+             dispatch.call('move', this, map);
+           }
 
+           scheduleRedraw();
+           return map;
+         };
 
-               for (var ruleKey in rules) {
-                 var rule = rules[ruleKey];
-                 var pass = true;
+         map.centerZoom = function (loc2, z2) {
+           if (setCenterZoom(loc2, z2)) {
+             dispatch.call('move', this, map);
+           }
 
-                 for (var criterion in rule) {
-                   if (criterion !== 'pattern') {
-                     // reserved for pattern name
-                     // The only rule is a required tag-value pair
-                     var v = tags[criterion];
+           scheduleRedraw();
+           return map;
+         };
 
-                     if (!v || v !== rule[criterion]) {
-                       pass = false;
-                       break;
-                     }
-                   }
-                 }
+         map.zoomTo = function (entity) {
+           var extent = entity.extent(context.graph());
+           if (!isFinite(extent.area())) return map;
+           var z2 = clamp$1(map.trimmedExtentZoom(extent), 0, 20);
+           return map.centerZoom(extent.center(), z2);
+         };
 
-                 if (pass) {
-                   return 'pattern-' + rule.pattern;
-                 }
-               }
-             }
-           }
-         }
+         map.centerEase = function (loc2, duration) {
+           duration = duration || 250;
+           setCenterZoom(loc2, map.zoom(), duration);
+           return map;
+         };
 
-         return null;
-       }
+         map.zoomEase = function (z2, duration) {
+           duration = duration || 250;
+           setCenterZoom(map.center(), z2, duration, false);
+           return map;
+         };
 
-       function svgAreas(projection, context) {
-         function getPatternStyle(tags) {
-           var imageID = svgTagPattern(tags);
+         map.centerZoomEase = function (loc2, z2, duration) {
+           duration = duration || 250;
+           setCenterZoom(loc2, z2, duration, false);
+           return map;
+         };
 
-           if (imageID) {
-             return 'url("#ideditor-' + imageID + '")';
-           }
+         map.transformEase = function (t2, duration) {
+           duration = duration || 250;
+           setTransform(t2, duration, false
+           /* don't force */
+           );
+           return map;
+         };
 
-           return '';
-         }
+         map.zoomToEase = function (obj, duration) {
+           var extent;
 
-         function drawTargets(selection, graph, entities, filter) {
-           var targetClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
-           var nopeClass = context.getDebug('target') ? 'red ' : 'nocolor ';
-           var getPath = svgPath(projection).geojson;
-           var activeID = context.activeID();
-           var base = context.history().base(); // The targets and nopes will be MultiLineString sub-segments of the ways
+           if (Array.isArray(obj)) {
+             obj.forEach(function (entity) {
+               var entityExtent = entity.extent(context.graph());
 
-           var data = {
-             targets: [],
-             nopes: []
-           };
-           entities.forEach(function (way) {
-             var features = svgSegmentWay(way, graph, activeID);
-             data.targets.push.apply(data.targets, features.passive);
-             data.nopes.push.apply(data.nopes, features.active);
-           }); // Targets allow hover and vertex snapping
+               if (!extent) {
+                 extent = entityExtent;
+               } else {
+                 extent = extent.extend(entityExtent);
+               }
+             });
+           } else {
+             extent = obj.extent(context.graph());
+           }
 
-           var targetData = data.targets.filter(getPath);
-           var targets = selection.selectAll('.area.target-allowed').filter(function (d) {
-             return filter(d.properties.entity);
-           }).data(targetData, function key(d) {
-             return d.id;
-           }); // exit
+           if (!isFinite(extent.area())) return map;
+           var z2 = clamp$1(map.trimmedExtentZoom(extent), 0, 20);
+           return map.centerZoomEase(extent.center(), z2, duration);
+         };
 
-           targets.exit().remove();
+         map.startEase = function () {
+           utilBindOnce(surface, _pointerPrefix + 'down.ease', function () {
+             map.cancelEase();
+           });
+           return map;
+         };
 
-           var segmentWasEdited = function segmentWasEdited(d) {
-             var wayID = d.properties.entity.id; // if the whole line was edited, don't draw segment changes
+         map.cancelEase = function () {
+           _selection.interrupt();
 
-             if (!base.entities[wayID] || !fastDeepEqual(graph.entities[wayID].nodes, base.entities[wayID].nodes)) {
-               return false;
-             }
+           return map;
+         };
 
-             return d.properties.nodes.some(function (n) {
-               return !base.entities[n.id] || !fastDeepEqual(graph.entities[n.id].loc, base.entities[n.id].loc);
-             });
-           }; // enter/update
+         map.extent = function (val) {
+           if (!arguments.length) {
+             return new geoExtent(projection.invert([0, _dimensions[1]]), projection.invert([_dimensions[0], 0]));
+           } else {
+             var extent = geoExtent(val);
+             map.centerZoom(extent.center(), map.extentZoom(extent));
+           }
+         };
 
+         map.trimmedExtent = function (val) {
+           if (!arguments.length) {
+             var headerY = 71;
+             var footerY = 30;
+             var pad = 10;
+             return new geoExtent(projection.invert([pad, _dimensions[1] - footerY - pad]), projection.invert([_dimensions[0] - pad, headerY + pad]));
+           } else {
+             var extent = geoExtent(val);
+             map.centerZoom(extent.center(), map.trimmedExtentZoom(extent));
+           }
+         };
 
-           targets.enter().append('path').merge(targets).attr('d', getPath).attr('class', function (d) {
-             return 'way area target target-allowed ' + targetClass + d.id;
-           }).classed('segment-edited', segmentWasEdited); // NOPE
+         function calcExtentZoom(extent, dim) {
+           var tl = projection([extent[0][0], extent[1][1]]);
+           var br = projection([extent[1][0], extent[0][1]]); // Calculate maximum zoom that fits extent
 
-           var nopeData = data.nopes.filter(getPath);
-           var nopes = selection.selectAll('.area.target-nope').filter(function (d) {
-             return filter(d.properties.entity);
-           }).data(nopeData, function key(d) {
-             return d.id;
-           }); // exit
+           var hFactor = (br[0] - tl[0]) / dim[0];
+           var vFactor = (br[1] - tl[1]) / dim[1];
+           var hZoomDiff = Math.log(Math.abs(hFactor)) / Math.LN2;
+           var vZoomDiff = Math.log(Math.abs(vFactor)) / Math.LN2;
+           var newZoom = map.zoom() - Math.max(hZoomDiff, vZoomDiff);
+           return newZoom;
+         }
 
-           nopes.exit().remove(); // enter/update
+         map.extentZoom = function (val) {
+           return calcExtentZoom(geoExtent(val), _dimensions);
+         };
 
-           nopes.enter().append('path').merge(nopes).attr('d', getPath).attr('class', function (d) {
-             return 'way area target target-nope ' + nopeClass + d.id;
-           }).classed('segment-edited', segmentWasEdited);
-         }
+         map.trimmedExtentZoom = function (val) {
+           var trimY = 120;
+           var trimX = 40;
+           var trimmed = [_dimensions[0] - trimX, _dimensions[1] - trimY];
+           return calcExtentZoom(geoExtent(val), trimmed);
+         };
 
-         function drawAreas(selection, graph, entities, filter) {
-           var path = svgPath(projection, graph, true);
-           var areas = {};
-           var multipolygon;
-           var base = context.history().base();
+         map.withinEditableZoom = function () {
+           return map.zoom() >= context.minEditableZoom();
+         };
 
-           for (var i = 0; i < entities.length; i++) {
-             var entity = entities[i];
-             if (entity.geometry(graph) !== 'area') continue;
-             multipolygon = osmIsOldMultipolygonOuterMember(entity, graph);
+         map.isInWideSelection = function () {
+           return !map.withinEditableZoom() && context.selectedIDs().length;
+         };
 
-             if (multipolygon) {
-               areas[multipolygon.id] = {
-                 entity: multipolygon.mergeTags(entity.tags),
-                 area: Math.abs(entity.area(graph))
-               };
-             } else if (!areas[entity.id]) {
-               areas[entity.id] = {
-                 entity: entity,
-                 area: Math.abs(entity.area(graph))
-               };
-             }
-           }
+         map.editableDataEnabled = function (skipZoomCheck) {
+           var layer = context.layers().layer('osm');
+           if (!layer || !layer.enabled()) return false;
+           return skipZoomCheck || map.withinEditableZoom();
+         };
 
-           var fills = Object.values(areas).filter(function hasPath(a) {
-             return path(a.entity);
-           });
-           fills.sort(function areaSort(a, b) {
-             return b.area - a.area;
-           });
-           fills = fills.map(function (a) {
-             return a.entity;
-           });
-           var strokes = fills.filter(function (area) {
-             return area.type === 'way';
-           });
-           var data = {
-             clip: fills,
-             shadow: strokes,
-             stroke: strokes,
-             fill: fills
-           };
-           var clipPaths = context.surface().selectAll('defs').selectAll('.clipPath-osm').filter(filter).data(data.clip, osmEntity.key);
-           clipPaths.exit().remove();
-           var clipPathsEnter = clipPaths.enter().append('clipPath').attr('class', 'clipPath-osm').attr('id', function (entity) {
-             return 'ideditor-' + entity.id + '-clippath';
-           });
-           clipPathsEnter.append('path');
-           clipPaths.merge(clipPathsEnter).selectAll('path').attr('d', path);
-           var drawLayer = selection.selectAll('.layer-osm.areas');
-           var touchLayer = selection.selectAll('.layer-touch.areas'); // Draw areas..
+         map.notesEditable = function () {
+           var layer = context.layers().layer('notes');
+           if (!layer || !layer.enabled()) return false;
+           return map.withinEditableZoom();
+         };
 
-           var areagroup = drawLayer.selectAll('g.areagroup').data(['fill', 'shadow', 'stroke']);
-           areagroup = areagroup.enter().append('g').attr('class', function (d) {
-             return 'areagroup area-' + d;
-           }).merge(areagroup);
-           var paths = areagroup.selectAll('path').filter(filter).data(function (layer) {
-             return data[layer];
-           }, osmEntity.key);
-           paths.exit().remove();
-           var fillpaths = selection.selectAll('.area-fill path.area').nodes();
-           var bisect = d3_bisector(function (node) {
-             return -node.__data__.area(graph);
-           }).left;
+         map.minzoom = function (val) {
+           if (!arguments.length) return _minzoom;
+           _minzoom = val;
+           return map;
+         };
 
-           function sortedByArea(entity) {
-             if (this._parent.__data__ === 'fill') {
-               return fillpaths[bisect(fillpaths, -entity.area(graph))];
-             }
-           }
+         map.toggleHighlightEdited = function () {
+           surface.classed('highlight-edited', !surface.classed('highlight-edited'));
+           map.pan([0, 0]); // trigger a redraw
 
-           paths = paths.enter().insert('path', sortedByArea).merge(paths).each(function (entity) {
-             var layer = this.parentNode.__data__;
-             this.setAttribute('class', entity.type + ' area ' + layer + ' ' + entity.id);
+           dispatch.call('changeHighlighting', this);
+         };
 
-             if (layer === 'fill') {
-               this.setAttribute('clip-path', 'url(#ideditor-' + entity.id + '-clippath)');
-               this.style.fill = this.style.stroke = getPatternStyle(entity.tags);
-             }
-           }).classed('added', function (d) {
-             return !base.entities[d.id];
-           }).classed('geometry-edited', function (d) {
-             return graph.entities[d.id] && base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].nodes, base.entities[d.id].nodes);
-           }).classed('retagged', function (d) {
-             return graph.entities[d.id] && base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].tags, base.entities[d.id].tags);
-           }).call(svgTagClasses()).attr('d', path); // Draw touch targets..
+         map.areaFillOptions = ['wireframe', 'partial', 'full'];
 
-           touchLayer.call(drawTargets, graph, data.stroke, filter);
-         }
+         map.activeAreaFill = function (val) {
+           if (!arguments.length) return corePreferences('area-fill') || 'partial';
+           corePreferences('area-fill', val);
 
-         return drawAreas;
-       }
+           if (val !== 'wireframe') {
+             corePreferences('area-fill-toggle', val);
+           }
 
-       //[4]           NameStartChar      ::=          ":" | [A-Z] | "_" | [a-z] | [#xC0-#xD6] | [#xD8-#xF6] | [#xF8-#x2FF] | [#x370-#x37D] | [#x37F-#x1FFF] | [#x200C-#x200D] | [#x2070-#x218F] | [#x2C00-#x2FEF] | [#x3001-#xD7FF] | [#xF900-#xFDCF] | [#xFDF0-#xFFFD] | [#x10000-#xEFFFF]
-       //[4a]          NameChar           ::=          NameStartChar | "-" | "." | [0-9] | #xB7 | [#x0300-#x036F] | [#x203F-#x2040]
-       //[5]           Name       ::=          NameStartChar (NameChar)*
-       var nameStartChar = /[A-Z_a-z\xC0-\xD6\xD8-\xF6\u00F8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD]/; //\u10000-\uEFFFF
+           updateAreaFill();
+           map.pan([0, 0]); // trigger a redraw
 
-       var nameChar = new RegExp("[\\-\\.0-9" + nameStartChar.source.slice(1, -1) + "\\u00B7\\u0300-\\u036F\\u203F-\\u2040]");
-       var tagNamePattern = new RegExp('^' + nameStartChar.source + nameChar.source + '*(?:\:' + nameStartChar.source + nameChar.source + '*)?$'); //var tagNamePattern = /^[a-zA-Z_][\w\-\.]*(?:\:[a-zA-Z_][\w\-\.]*)?$/
-       //var handlers = 'resolveEntity,getExternalSubset,characters,endDocument,endElement,endPrefixMapping,ignorableWhitespace,processingInstruction,setDocumentLocator,skippedEntity,startDocument,startElement,startPrefixMapping,notationDecl,unparsedEntityDecl,error,fatalError,warning,attributeDecl,elementDecl,externalEntityDecl,internalEntityDecl,comment,endCDATA,endDTD,endEntity,startCDATA,startDTD,startEntity'.split(',')
-       //S_TAG,        S_ATTR, S_EQ,   S_ATTR_NOQUOT_VALUE
-       //S_ATTR_SPACE, S_ATTR_END,     S_TAG_SPACE, S_TAG_CLOSE
+           dispatch.call('changeAreaFill', this);
+           return map;
+         };
 
-       var S_TAG = 0; //tag name offerring
+         map.toggleWireframe = function () {
+           var activeFill = map.activeAreaFill();
 
-       var S_ATTR = 1; //attr name offerring 
+           if (activeFill === 'wireframe') {
+             activeFill = corePreferences('area-fill-toggle') || 'partial';
+           } else {
+             activeFill = 'wireframe';
+           }
 
-       var S_ATTR_SPACE = 2; //attr name end and space offer
+           map.activeAreaFill(activeFill);
+         };
 
-       var S_EQ = 3; //=space?
+         function updateAreaFill() {
+           var activeFill = map.activeAreaFill();
+           map.areaFillOptions.forEach(function (opt) {
+             surface.classed('fill-' + opt, Boolean(opt === activeFill));
+           });
+         }
 
-       var S_ATTR_NOQUOT_VALUE = 4; //attr value(no quot value only)
+         map.layers = function () {
+           return drawLayers;
+         };
 
-       var S_ATTR_END = 5; //attr value end and no space(quot end)
+         map.doubleUpHandler = function () {
+           return _doubleUpHandler;
+         };
 
-       var S_TAG_SPACE = 6; //(attr value end || tag end ) && (space offer)
+         return utilRebind(map, dispatch, 'on');
+       }
 
-       var S_TAG_CLOSE = 7; //closed el<el />
+       function rendererPhotos(context) {
+         var dispatch = dispatch$8('change');
+         var _layerIDs = ['streetside', 'mapillary', 'mapillary-map-features', 'mapillary-signs', 'openstreetcam'];
+         var _allPhotoTypes = ['flat', 'panoramic'];
 
-       function XMLReader() {}
+         var _shownPhotoTypes = _allPhotoTypes.slice(); // shallow copy
 
-       XMLReader.prototype = {
-         parse: function parse(source, defaultNSMap, entityMap) {
-           var domBuilder = this.domBuilder;
-           domBuilder.startDocument();
 
-           _copy(defaultNSMap, defaultNSMap = {});
+         var _dateFilters = ['fromDate', 'toDate'];
 
-           _parse(source, defaultNSMap, entityMap, domBuilder, this.errorHandler);
+         var _fromDate;
 
-           domBuilder.endDocument();
-         }
-       };
+         var _toDate;
 
-       function _parse(source, defaultNSMapCopy, entityMap, domBuilder, errorHandler) {
-         function fixedFromCharCode(code) {
-           // String.prototype.fromCharCode does not supports
-           // > 2 bytes unicode chars directly
-           if (code > 0xffff) {
-             code -= 0x10000;
-             var surrogate1 = 0xd800 + (code >> 10),
-                 surrogate2 = 0xdc00 + (code & 0x3ff);
-             return String.fromCharCode(surrogate1, surrogate2);
-           } else {
-             return String.fromCharCode(code);
-           }
-         }
+         var _usernames;
 
-         function entityReplacer(a) {
-           var k = a.slice(1, -1);
+         function photos() {}
 
-           if (k in entityMap) {
-             return entityMap[k];
-           } else if (k.charAt(0) === '#') {
-             return fixedFromCharCode(parseInt(k.substr(1).replace('x', '0x')));
+         function updateStorage() {
+           if (window.mocha) return;
+           var hash = utilStringQs(window.location.hash);
+           var enabled = context.layers().all().filter(function (d) {
+             return _layerIDs.indexOf(d.id) !== -1 && d.layer && d.layer.supported() && d.layer.enabled();
+           }).map(function (d) {
+             return d.id;
+           });
+
+           if (enabled.length) {
+             hash.photo_overlay = enabled.join(',');
            } else {
-             errorHandler.error('entity not found:' + a);
-             return a;
+             delete hash.photo_overlay;
            }
-         }
 
-         function appendText(end) {
-           //has some bugs
-           if (end > start) {
-             var xt = source.substring(start, end).replace(/&#?\w+;/g, entityReplacer);
-             locator && position(start);
-             domBuilder.characters(xt, 0, end - start);
-             start = end;
-           }
+           window.location.replace('#' + utilQsString(hash, true));
          }
 
-         function position(p, m) {
-           while (p >= lineEnd && (m = linePattern.exec(source))) {
-             lineStart = m.index;
-             lineEnd = lineStart + m[0].length;
-             locator.lineNumber++; //console.log('line++:',locator,startPos,endPos)
-           }
+         photos.overlayLayerIDs = function () {
+           return _layerIDs;
+         };
 
-           locator.columnNumber = p - lineStart + 1;
-         }
+         photos.allPhotoTypes = function () {
+           return _allPhotoTypes;
+         };
 
-         var lineStart = 0;
-         var lineEnd = 0;
-         var linePattern = /.*(?:\r\n?|\n)|.*$/g;
-         var locator = domBuilder.locator;
-         var parseStack = [{
-           currentNSMap: defaultNSMapCopy
-         }];
-         var closeMap = {};
-         var start = 0;
+         photos.dateFilters = function () {
+           return _dateFilters;
+         };
 
-         while (true) {
-           try {
-             var tagStart = source.indexOf('<', start);
+         photos.dateFilterValue = function (val) {
+           return val === _dateFilters[0] ? _fromDate : _toDate;
+         };
 
-             if (tagStart < 0) {
-               if (!source.substr(start).match(/^\s*$/)) {
-                 var doc = domBuilder.doc;
-                 var text = doc.createTextNode(source.substr(start));
-                 doc.appendChild(text);
-                 domBuilder.currentElement = text;
-               }
+         photos.setDateFilter = function (type, val, updateUrl) {
+           // validate the date
+           var date = val && new Date(val);
 
-               return;
+           if (date && !isNaN(date)) {
+             val = date.toISOString().substr(0, 10);
+           } else {
+             val = null;
+           }
+
+           if (type === _dateFilters[0]) {
+             _fromDate = val;
+
+             if (_fromDate && _toDate && new Date(_toDate) < new Date(_fromDate)) {
+               _toDate = _fromDate;
              }
+           }
 
-             if (tagStart > start) {
-               appendText(tagStart);
+           if (type === _dateFilters[1]) {
+             _toDate = val;
+
+             if (_fromDate && _toDate && new Date(_toDate) < new Date(_fromDate)) {
+               _fromDate = _toDate;
              }
+           }
 
-             switch (source.charAt(tagStart + 1)) {
-               case '/':
-                 var end = source.indexOf('>', tagStart + 3);
-                 var tagName = source.substring(tagStart + 2, end);
-                 var config = parseStack.pop();
+           dispatch.call('change', this);
 
-                 if (end < 0) {
-                   tagName = source.substring(tagStart + 2).replace(/[\s<].*/, ''); //console.error('#@@@@@@'+tagName)
+           if (updateUrl) {
+             var rangeString;
 
-                   errorHandler.error("end tag name: " + tagName + ' is not complete:' + config.tagName);
-                   end = tagStart + 1 + tagName.length;
-                 } else if (tagName.match(/\s</)) {
-                   tagName = tagName.replace(/[\s<].*/, '');
-                   errorHandler.error("end tag name: " + tagName + ' maybe not complete');
-                   end = tagStart + 1 + tagName.length;
-                 } //console.error(parseStack.length,parseStack)
-                 //console.error(config);
+             if (_fromDate || _toDate) {
+               rangeString = (_fromDate || '') + '_' + (_toDate || '');
+             }
 
+             setUrlFilterValue('photo_dates', rangeString);
+           }
+         };
 
-                 var localNSMap = config.localNSMap;
-                 var endMatch = config.tagName == tagName;
-                 var endIgnoreCaseMach = endMatch || config.tagName && config.tagName.toLowerCase() == tagName.toLowerCase();
+         photos.setUsernameFilter = function (val, updateUrl) {
+           if (val && typeof val === 'string') val = val.replace(/;/g, ',').split(',');
 
-                 if (endIgnoreCaseMach) {
-                   domBuilder.endElement(config.uri, config.localName, tagName);
+           if (val) {
+             val = val.map(function (d) {
+               return d.trim();
+             }).filter(Boolean);
 
-                   if (localNSMap) {
-                     for (var prefix in localNSMap) {
-                       domBuilder.endPrefixMapping(prefix);
-                     }
-                   }
+             if (!val.length) {
+               val = null;
+             }
+           }
 
-                   if (!endMatch) {
-                     errorHandler.fatalError("end tag name: " + tagName + ' is not match the current start tagName:' + config.tagName);
-                   }
-                 } else {
-                   parseStack.push(config);
-                 }
+           _usernames = val;
+           dispatch.call('change', this);
 
-                 end++;
-                 break;
-               // end elment
+           if (updateUrl) {
+             var hashString;
 
-               case '?':
-                 // <?...?>
-                 locator && position(tagStart);
-                 end = parseInstruction(source, tagStart, domBuilder);
-                 break;
+             if (_usernames) {
+               hashString = _usernames.join(',');
+             }
 
-               case '!':
-                 // <!doctype,<![CDATA,<!--
-                 locator && position(tagStart);
-                 end = parseDCC(source, tagStart, domBuilder, errorHandler);
-                 break;
+             setUrlFilterValue('photo_username', hashString);
+           }
+         };
 
-               default:
-                 locator && position(tagStart);
-                 var el = new ElementAttributes();
-                 var currentNSMap = parseStack[parseStack.length - 1].currentNSMap; //elStartEnd
+         function setUrlFilterValue(property, val) {
+           if (!window.mocha) {
+             var hash = utilStringQs(window.location.hash);
 
-                 var end = parseElementStartPart(source, tagStart, el, currentNSMap, entityReplacer, errorHandler);
-                 var len = el.length;
+             if (val) {
+               if (hash[property] === val) return;
+               hash[property] = val;
+             } else {
+               if (!(property in hash)) return;
+               delete hash[property];
+             }
 
-                 if (!el.closed && fixSelfClosed(source, end, el.tagName, closeMap)) {
-                   el.closed = true;
+             window.location.replace('#' + utilQsString(hash, true));
+           }
+         }
 
-                   if (!entityMap.nbsp) {
-                     errorHandler.warning('unclosed xml attribute');
-                   }
-                 }
+         function showsLayer(id) {
+           var layer = context.layers().layer(id);
+           return layer && layer.supported() && layer.enabled();
+         }
 
-                 if (locator && len) {
-                   var locator2 = copyLocator(locator, {}); //try{//attribute position fixed
+         photos.shouldFilterByDate = function () {
+           return showsLayer('mapillary') || showsLayer('openstreetcam') || showsLayer('streetside');
+         };
 
-                   for (var i = 0; i < len; i++) {
-                     var a = el[i];
-                     position(a.offset);
-                     a.locator = copyLocator(locator, {});
-                   } //}catch(e){console.error('@@@@@'+e)}
+         photos.shouldFilterByPhotoType = function () {
+           return showsLayer('mapillary') || showsLayer('streetside') && showsLayer('openstreetcam');
+         };
 
+         photos.shouldFilterByUsername = function () {
+           return !showsLayer('mapillary') && showsLayer('openstreetcam') && !showsLayer('streetside');
+         };
 
-                   domBuilder.locator = locator2;
+         photos.showsPhotoType = function (val) {
+           if (!photos.shouldFilterByPhotoType()) return true;
+           return _shownPhotoTypes.indexOf(val) !== -1;
+         };
 
-                   if (appendElement(el, domBuilder, currentNSMap)) {
-                     parseStack.push(el);
-                   }
+         photos.showsFlat = function () {
+           return photos.showsPhotoType('flat');
+         };
 
-                   domBuilder.locator = locator;
-                 } else {
-                   if (appendElement(el, domBuilder, currentNSMap)) {
-                     parseStack.push(el);
-                   }
-                 }
+         photos.showsPanoramic = function () {
+           return photos.showsPhotoType('panoramic');
+         };
 
-                 if (el.uri === 'http://www.w3.org/1999/xhtml' && !el.closed) {
-                   end = parseHtmlSpecialContent(source, end, el.tagName, entityReplacer, domBuilder);
-                 } else {
-                   end++;
-                 }
+         photos.fromDate = function () {
+           return _fromDate;
+         };
 
-             }
-           } catch (e) {
-             errorHandler.error('element parse error: ' + e); //errorHandler.error('element parse error: '+e);
+         photos.toDate = function () {
+           return _toDate;
+         };
 
-             end = -1; //throw e;
-           }
+         photos.togglePhotoType = function (val) {
+           var index = _shownPhotoTypes.indexOf(val);
 
-           if (end > start) {
-             start = end;
+           if (index !== -1) {
+             _shownPhotoTypes.splice(index, 1);
            } else {
-             //TODO: 这里有可能sax回退,有位置错误风险
-             appendText(Math.max(tagStart, start) + 1);
+             _shownPhotoTypes.push(val);
            }
-         }
-       }
 
-       function copyLocator(f, t) {
-         t.lineNumber = f.lineNumber;
-         t.columnNumber = f.columnNumber;
-         return t;
-       }
-       /**
-        * @see #appendElement(source,elStartEnd,el,selfClosed,entityReplacer,domBuilder,parseStack);
-        * @return end of the elementStartPart(end of elementEndPart for selfClosed el)
-        */
+           dispatch.call('change', this);
+           return photos;
+         };
 
+         photos.usernames = function () {
+           return _usernames;
+         };
 
-       function parseElementStartPart(source, start, el, currentNSMap, entityReplacer, errorHandler) {
-         var attrName;
-         var value;
-         var p = ++start;
-         var s = S_TAG; //status
+         photos.init = function () {
+           var hash = utilStringQs(window.location.hash);
 
-         while (true) {
-           var c = source.charAt(p);
+           if (hash.photo_dates) {
+             // expect format like `photo_dates=2019-01-01_2020-12-31`, but allow a couple different separators
+             var parts = /^(.*)[–_](.*)$/g.exec(hash.photo_dates.trim());
+             this.setDateFilter('fromDate', parts && parts.length >= 2 && parts[1], false);
+             this.setDateFilter('toDate', parts && parts.length >= 3 && parts[2], false);
+           }
 
-           switch (c) {
-             case '=':
-               if (s === S_ATTR) {
-                 //attrName
-                 attrName = source.slice(start, p);
-                 s = S_EQ;
-               } else if (s === S_ATTR_SPACE) {
-                 s = S_EQ;
-               } else {
-                 //fatalError: equal must after attrName or space after attrName
-                 throw new Error('attribute equal must after attrName');
-               }
+           if (hash.photo_username) {
+             this.setUsernameFilter(hash.photo_username, false);
+           }
 
-               break;
+           if (hash.photo_overlay) {
+             // support enabling photo layers by default via a URL parameter, e.g. `photo_overlay=openstreetcam;mapillary;streetside`
+             var hashOverlayIDs = hash.photo_overlay.replace(/;/g, ',').split(',');
+             hashOverlayIDs.forEach(function (id) {
+               var layer = _layerIDs.indexOf(id) !== -1 && context.layers().layer(id);
+               if (layer && !layer.enabled()) layer.enabled(true);
+             });
+           }
 
-             case '\'':
-             case '"':
-               if (s === S_EQ || s === S_ATTR //|| s == S_ATTR_SPACE
-               ) {
-                   //equal
-                   if (s === S_ATTR) {
-                     errorHandler.warning('attribute value must after "="');
-                     attrName = source.slice(start, p);
-                   }
+           if (hash.photo) {
+             // support opening a photo via a URL parameter, e.g. `photo=mapillary-fztgSDtLpa08ohPZFZjeRQ`
+             var photoIds = hash.photo.replace(/;/g, ',').split(',');
+             var photoId = photoIds.length && photoIds[0].trim();
+             var results = /(.*)\/(.*)/g.exec(photoId);
 
-                   start = p + 1;
-                   p = source.indexOf(c, start);
+             if (results && results.length >= 3) {
+               var serviceId = results[1];
+               var photoKey = results[2];
+               var service = services[serviceId];
 
-                   if (p > 0) {
-                     value = source.slice(start, p).replace(/&#?\w+;/g, entityReplacer);
-                     el.add(attrName, value, start - 1);
-                     s = S_ATTR_END;
-                   } else {
-                     //fatalError: no end quot match
-                     throw new Error('attribute value no end \'' + c + '\' match');
+               if (service && service.ensureViewerLoaded) {
+                 // if we're showing a photo then make sure its layer is enabled too
+                 var layer = _layerIDs.indexOf(serviceId) !== -1 && context.layers().layer(serviceId);
+                 if (layer && !layer.enabled()) layer.enabled(true);
+                 var baselineTime = Date.now();
+                 service.on('loadedImages.rendererPhotos', function () {
+                   // don't open the viewer if too much time has elapsed
+                   if (Date.now() - baselineTime > 45000) {
+                     service.on('loadedImages.rendererPhotos', null);
+                     return;
                    }
-                 } else if (s == S_ATTR_NOQUOT_VALUE) {
-                 value = source.slice(start, p).replace(/&#?\w+;/g, entityReplacer); //console.log(attrName,value,start,p)
-
-                 el.add(attrName, value, start); //console.dir(el)
 
-                 errorHandler.warning('attribute "' + attrName + '" missed start quot(' + c + ')!!');
-                 start = p + 1;
-                 s = S_ATTR_END;
-               } else {
-                 //fatalError: no equal before
-                 throw new Error('attribute value must after "="');
+                   if (!service.cachedImage(photoKey)) return;
+                   service.on('loadedImages.rendererPhotos', null);
+                   service.ensureViewerLoaded(context).then(function () {
+                     service.selectImage(context, photoKey).showViewer(context);
+                   });
+                 });
                }
+             }
+           }
 
-               break;
+           context.layers().on('change.rendererPhotos', updateStorage);
+         };
 
-             case '/':
-               switch (s) {
-                 case S_TAG:
-                   el.setTagName(source.slice(start, p));
+         return utilRebind(photos, dispatch, 'on');
+       }
 
-                 case S_ATTR_END:
-                 case S_TAG_SPACE:
-                 case S_TAG_CLOSE:
-                   s = S_TAG_CLOSE;
-                   el.closed = true;
+       function uiAccount(context) {
+         var osm = context.connection();
 
-                 case S_ATTR_NOQUOT_VALUE:
-                 case S_ATTR:
-                 case S_ATTR_SPACE:
-                   break;
-                 //case S_EQ:
+         function update(selection) {
+           if (!osm) return;
 
-                 default:
-                   throw new Error("attribute invalid close char('/')");
-               }
+           if (!osm.authenticated()) {
+             selection.selectAll('.userLink, .logoutLink').classed('hide', true);
+             return;
+           }
 
-               break;
+           osm.userDetails(function (err, details) {
+             var userLink = selection.select('.userLink'),
+                 logoutLink = selection.select('.logoutLink');
+             userLink.html('');
+             logoutLink.html('');
+             if (err || !details) return;
+             selection.selectAll('.userLink, .logoutLink').classed('hide', false); // Link
 
-             case '':
-               //end document
-               //throw new Error('unexpected end of input')
-               errorHandler.error('unexpected end of input');
+             var userLinkA = userLink.append('a').attr('href', osm.userURL(details.display_name)).attr('target', '_blank'); // Add thumbnail or dont
 
-               if (s == S_TAG) {
-                 el.setTagName(source.slice(start, p));
-               }
+             if (details.image_url) {
+               userLinkA.append('img').attr('class', 'icon pre-text user-icon').attr('src', details.image_url);
+             } else {
+               userLinkA.call(svgIcon('#iD-icon-avatar', 'pre-text light'));
+             } // Add user name
 
-               return p;
 
-             case '>':
-               switch (s) {
-                 case S_TAG:
-                   el.setTagName(source.slice(start, p));
+             userLinkA.append('span').attr('class', 'label').html(details.display_name);
+             logoutLink.append('a').attr('class', 'logout').attr('href', '#').html(_t.html('logout')).on('click.logout', function (d3_event) {
+               d3_event.preventDefault();
+               osm.logout();
+             });
+           });
+         }
 
-                 case S_ATTR_END:
-                 case S_TAG_SPACE:
-                 case S_TAG_CLOSE:
-                   break;
-                 //normal
+         return function (selection) {
+           selection.append('li').attr('class', 'userLink').classed('hide', true);
+           selection.append('li').attr('class', 'logoutLink').classed('hide', true);
 
-                 case S_ATTR_NOQUOT_VALUE: //Compatible state
+           if (osm) {
+             osm.on('change.account', function () {
+               update(selection);
+             });
+             update(selection);
+           }
+         };
+       }
 
-                 case S_ATTR:
-                   value = source.slice(start, p);
+       function uiAttribution(context) {
+         var _selection = select(null);
 
-                   if (value.slice(-1) === '/') {
-                     el.closed = true;
-                     value = value.slice(0, -1);
-                   }
+         function render(selection, data, klass) {
+           var div = selection.selectAll(".".concat(klass)).data([0]);
+           div = div.enter().append('div').attr('class', klass).merge(div);
+           var attributions = div.selectAll('.attribution').data(data, function (d) {
+             return d.id;
+           });
+           attributions.exit().remove();
+           attributions = attributions.enter().append('span').attr('class', 'attribution').each(function (d, i, nodes) {
+             var attribution = select(nodes[i]);
 
-                 case S_ATTR_SPACE:
-                   if (s === S_ATTR_SPACE) {
-                     value = attrName;
-                   }
+             if (d.terms_html) {
+               attribution.html(d.terms_html);
+               return;
+             }
 
-                   if (s == S_ATTR_NOQUOT_VALUE) {
-                     errorHandler.warning('attribute "' + value + '" missed quot(")!!');
-                     el.add(attrName, value.replace(/&#?\w+;/g, entityReplacer), start);
-                   } else {
-                     if (currentNSMap[''] !== 'http://www.w3.org/1999/xhtml' || !value.match(/^(?:disabled|checked|selected)$/i)) {
-                       errorHandler.warning('attribute "' + value + '" missed value!! "' + value + '" instead!!');
-                     }
+             if (d.terms_url) {
+               attribution = attribution.append('a').attr('href', d.terms_url).attr('target', '_blank');
+             }
 
-                     el.add(value, value, start);
-                   }
+             var sourceID = d.id.replace(/\./g, '<TX_DOT>');
+             var terms_text = _t("imagery.".concat(sourceID, ".attribution.text"), {
+               "default": d.terms_text || d.id || d.name()
+             });
 
-                   break;
+             if (d.icon && !d.overlay) {
+               attribution.append('img').attr('class', 'source-image').attr('src', d.icon);
+             }
 
-                 case S_EQ:
-                   throw new Error('attribute value missed!!');
-               } //                    console.log(tagName,tagNamePattern,tagNamePattern.test(tagName))
+             attribution.append('span').attr('class', 'attribution-text').html(terms_text);
+           }).merge(attributions);
+           var copyright = attributions.selectAll('.copyright-notice').data(function (d) {
+             var notice = d.copyrightNotices(context.map().zoom(), context.map().extent());
+             return notice ? [notice] : [];
+           });
+           copyright.exit().remove();
+           copyright = copyright.enter().append('span').attr('class', 'copyright-notice').merge(copyright);
+           copyright.html(String);
+         }
 
+         function update() {
+           var baselayer = context.background().baseLayerSource();
 
-               return p;
+           _selection.call(render, baselayer ? [baselayer] : [], 'base-layer-attribution');
 
-             /*xml space '\x20' | #x9 | #xD | #xA; */
+           var z = context.map().zoom();
+           var overlays = context.background().overlayLayerSources() || [];
 
-             case "\x80":
-               c = ' ';
+           _selection.call(render, overlays.filter(function (s) {
+             return s.validZoom(z);
+           }), 'overlay-layer-attribution');
+         }
 
-             default:
-               if (c <= ' ') {
-                 //space
-                 switch (s) {
-                   case S_TAG:
-                     el.setTagName(source.slice(start, p)); //tagName
+         return function (selection) {
+           _selection = selection;
+           context.background().on('change.attribution', update);
+           context.map().on('move.attribution', throttle(update, 400, {
+             leading: false
+           }));
+           update();
+         };
+       }
 
-                     s = S_TAG_SPACE;
-                     break;
+       function uiContributors(context) {
+         var osm = context.connection(),
+             debouncedUpdate = debounce(function () {
+           update();
+         }, 1000),
+             limit = 4,
+             hidden = false,
+             wrap = select(null);
 
-                   case S_ATTR:
-                     attrName = source.slice(start, p);
-                     s = S_ATTR_SPACE;
-                     break;
+         function update() {
+           if (!osm) return;
+           var users = {},
+               entities = context.history().intersects(context.map().extent());
+           entities.forEach(function (entity) {
+             if (entity && entity.user) users[entity.user] = true;
+           });
+           var u = Object.keys(users),
+               subset = u.slice(0, u.length > limit ? limit - 1 : limit);
+           wrap.html('').call(svgIcon('#iD-icon-nearby', 'pre-text light'));
+           var userList = select(document.createElement('span'));
+           userList.selectAll().data(subset).enter().append('a').attr('class', 'user-link').attr('href', function (d) {
+             return osm.userURL(d);
+           }).attr('target', '_blank').html(String);
 
-                   case S_ATTR_NOQUOT_VALUE:
-                     var value = source.slice(start, p).replace(/&#?\w+;/g, entityReplacer);
-                     errorHandler.warning('attribute "' + value + '" missed quot(")!!');
-                     el.add(attrName, value, start);
+           if (u.length > limit) {
+             var count = select(document.createElement('span'));
+             var othersNum = u.length - limit + 1;
+             count.append('a').attr('target', '_blank').attr('href', function () {
+               return osm.changesetsURL(context.map().center(), context.map().zoom());
+             }).html(othersNum);
+             wrap.append('span').html(_t.html('contributors.truncated_list', {
+               n: othersNum,
+               users: userList.html(),
+               count: count.html()
+             }));
+           } else {
+             wrap.append('span').html(_t.html('contributors.list', {
+               users: userList.html()
+             }));
+           }
 
-                   case S_ATTR_END:
-                     s = S_TAG_SPACE;
-                     break;
-                   //case S_TAG_SPACE:
-                   //case S_EQ:
-                   //case S_ATTR_SPACE:
-                   //  void();break;
-                   //case S_TAG_CLOSE:
-                   //ignore warning
-                 }
-               } else {
-                 //not space
-                 //S_TAG,      S_ATTR, S_EQ,   S_ATTR_NOQUOT_VALUE
-                 //S_ATTR_SPACE,       S_ATTR_END,     S_TAG_SPACE, S_TAG_CLOSE
-                 switch (s) {
-                   //case S_TAG:void();break;
-                   //case S_ATTR:void();break;
-                   //case S_ATTR_NOQUOT_VALUE:void();break;
-                   case S_ATTR_SPACE:
-                     var tagName = el.tagName;
+           if (!u.length) {
+             hidden = true;
+             wrap.transition().style('opacity', 0);
+           } else if (hidden) {
+             wrap.transition().style('opacity', 1);
+           }
+         }
 
-                     if (currentNSMap[''] !== 'http://www.w3.org/1999/xhtml' || !attrName.match(/^(?:disabled|checked|selected)$/i)) {
-                       errorHandler.warning('attribute "' + attrName + '" missed value!! "' + attrName + '" instead2!!');
-                     }
+         return function (selection) {
+           if (!osm) return;
+           wrap = selection;
+           update();
+           osm.on('loaded.contributors', debouncedUpdate);
+           context.map().on('move.contributors', debouncedUpdate);
+         };
+       }
 
-                     el.add(attrName, attrName, start);
-                     start = p;
-                     s = S_ATTR;
-                     break;
+       var _popoverID = 0;
+       function uiPopover(klass) {
+         var _id = _popoverID++;
 
-                   case S_ATTR_END:
-                     errorHandler.warning('attribute space is required"' + attrName + '"!!');
+         var _anchorSelection = select(null);
 
-                   case S_TAG_SPACE:
-                     s = S_ATTR;
-                     start = p;
-                     break;
+         var popover = function popover(selection) {
+           _anchorSelection = selection;
+           selection.each(setup);
+         };
 
-                   case S_EQ:
-                     s = S_ATTR_NOQUOT_VALUE;
-                     start = p;
-                     break;
+         var _animation = utilFunctor(false);
 
-                   case S_TAG_CLOSE:
-                     throw new Error("elements closed character '/' and '>' must be connected to");
-                 }
-               }
+         var _placement = utilFunctor('top'); // top, bottom, left, right
 
-           } //end outer switch
-           //console.log('p++',p)
 
+         var _alignment = utilFunctor('center'); // leading, center, trailing
 
-           p++;
-         }
-       }
-       /**
-        * @return true if has new namespace define
-        */
 
+         var _scrollContainer = utilFunctor(select(null));
 
-       function appendElement(el, domBuilder, currentNSMap) {
-         var tagName = el.tagName;
-         var localNSMap = null; //var currentNSMap = parseStack[parseStack.length-1].currentNSMap;
+         var _content;
 
-         var i = el.length;
+         var _displayType = utilFunctor('');
 
-         while (i--) {
-           var a = el[i];
-           var qName = a.qName;
-           var value = a.value;
-           var nsp = qName.indexOf(':');
+         var _hasArrow = utilFunctor(true); // use pointer events on supported platforms; fallback to mouse events
 
-           if (nsp > 0) {
-             var prefix = a.prefix = qName.slice(0, nsp);
-             var localName = qName.slice(nsp + 1);
-             var nsPrefix = prefix === 'xmlns' && localName;
-           } else {
-             localName = qName;
-             prefix = null;
-             nsPrefix = qName === 'xmlns' && '';
-           } //can not set prefix,because prefix !== ''
 
+         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
 
-           a.localName = localName; //prefix == null for no ns prefix attribute 
+         popover.displayType = function (val) {
+           if (arguments.length) {
+             _displayType = utilFunctor(val);
+             return popover;
+           } else {
+             return _displayType;
+           }
+         };
 
-           if (nsPrefix !== false) {
-             //hack!!
-             if (localNSMap == null) {
-               localNSMap = {}; //console.log(currentNSMap,0)
+         popover.hasArrow = function (val) {
+           if (arguments.length) {
+             _hasArrow = utilFunctor(val);
+             return popover;
+           } else {
+             return _hasArrow;
+           }
+         };
 
-               _copy(currentNSMap, currentNSMap = {}); //console.log(currentNSMap,1)
+         popover.placement = function (val) {
+           if (arguments.length) {
+             _placement = utilFunctor(val);
+             return popover;
+           } else {
+             return _placement;
+           }
+         };
 
-             }
+         popover.alignment = function (val) {
+           if (arguments.length) {
+             _alignment = utilFunctor(val);
+             return popover;
+           } else {
+             return _alignment;
+           }
+         };
 
-             currentNSMap[nsPrefix] = localNSMap[nsPrefix] = value;
-             a.uri = 'http://www.w3.org/2000/xmlns/';
-             domBuilder.startPrefixMapping(nsPrefix, value);
+         popover.scrollContainer = function (val) {
+           if (arguments.length) {
+             _scrollContainer = utilFunctor(val);
+             return popover;
+           } else {
+             return _scrollContainer;
            }
-         }
+         };
 
-         var i = el.length;
+         popover.content = function (val) {
+           if (arguments.length) {
+             _content = val;
+             return popover;
+           } else {
+             return _content;
+           }
+         };
 
-         while (i--) {
-           a = el[i];
-           var prefix = a.prefix;
+         popover.isShown = function () {
+           var popoverSelection = _anchorSelection.select('.popover-' + _id);
 
-           if (prefix) {
-             //no prefix attribute has no namespace
-             if (prefix === 'xml') {
-               a.uri = 'http://www.w3.org/XML/1998/namespace';
-             }
+           return !popoverSelection.empty() && popoverSelection.classed('in');
+         };
 
-             if (prefix !== 'xmlns') {
-               a.uri = currentNSMap[prefix || '']; //{console.log('###'+a.qName,domBuilder.locator.systemId+'',currentNSMap,a.uri)}
-             }
-           }
-         }
+         popover.show = function () {
+           _anchorSelection.each(show);
+         };
 
-         var nsp = tagName.indexOf(':');
+         popover.updateContent = function () {
+           _anchorSelection.each(updateContent);
+         };
+
+         popover.hide = function () {
+           _anchorSelection.each(hide);
+         };
+
+         popover.toggle = function () {
+           _anchorSelection.each(toggle);
+         };
 
-         if (nsp > 0) {
-           prefix = el.prefix = tagName.slice(0, nsp);
-           localName = el.localName = tagName.slice(nsp + 1);
-         } else {
-           prefix = null; //important!!
+         popover.destroy = function (selection, selector) {
+           // by default, just destroy the current popover
+           selector = selector || '.popover-' + _id;
+           selection.on(_pointerPrefix + 'enter.popover', null).on(_pointerPrefix + 'leave.popover', null).on(_pointerPrefix + 'up.popover', null).on(_pointerPrefix + 'down.popover', null).on('click.popover', null).attr('title', function () {
+             return this.getAttribute('data-original-title') || this.getAttribute('title');
+           }).attr('data-original-title', null).selectAll(selector).remove();
+         };
 
-           localName = el.localName = tagName;
-         } //no prefix element has default namespace
+         popover.destroyAny = function (selection) {
+           selection.call(popover.destroy, '.popover');
+         };
 
+         function setup() {
+           var anchor = select(this);
 
-         var ns = el.uri = currentNSMap[prefix || ''];
-         domBuilder.startElement(ns, localName, tagName, el); //endPrefixMapping and startPrefixMapping have not any help for dom builder
-         //localNSMap = null
+           var animate = _animation.apply(this, arguments);
 
-         if (el.closed) {
-           domBuilder.endElement(ns, localName, tagName);
+           var popoverSelection = anchor.selectAll('.popover-' + _id).data([0]);
+           var enter = popoverSelection.enter().append('div').attr('class', 'popover popover-' + _id + ' ' + (klass ? klass : '')).classed('arrowed', _hasArrow.apply(this, arguments));
+           enter.append('div').attr('class', 'popover-arrow');
+           enter.append('div').attr('class', 'popover-inner');
+           popoverSelection = enter.merge(popoverSelection);
 
-           if (localNSMap) {
-             for (prefix in localNSMap) {
-               domBuilder.endPrefixMapping(prefix);
-             }
+           if (animate) {
+             popoverSelection.classed('fade', true);
            }
-         } else {
-           el.currentNSMap = currentNSMap;
-           el.localNSMap = localNSMap; //parseStack.push(el);
 
-           return true;
-         }
-       }
+           var display = _displayType.apply(this, arguments);
 
-       function parseHtmlSpecialContent(source, elStartEnd, tagName, entityReplacer, domBuilder) {
-         if (/^(?:script|textarea)$/i.test(tagName)) {
-           var elEndStart = source.indexOf('</' + tagName + '>', elStartEnd);
-           var text = source.substring(elStartEnd + 1, elEndStart);
+           if (display === 'hover') {
+             var _lastNonMouseEnterTime;
 
-           if (/[&<]/.test(text)) {
-             if (/^script$/i.test(tagName)) {
-               //if(!/\]\]>/.test(text)){
-               //lexHandler.startCDATA();
-               domBuilder.characters(text, 0, text.length); //lexHandler.endCDATA();
+             anchor.on(_pointerPrefix + 'enter.popover', function (d3_event) {
+               if (d3_event.pointerType) {
+                 if (d3_event.pointerType !== 'mouse') {
+                   _lastNonMouseEnterTime = d3_event.timeStamp; // only allow hover behavior for mouse input
 
-               return elEndStart; //}
-             } //}else{//text area
+                   return;
+                 } else if (_lastNonMouseEnterTime && d3_event.timeStamp - _lastNonMouseEnterTime < 1500) {
+                   // HACK: iOS 13.4 sends an erroneous `mouse` type pointerenter
+                   // event for non-mouse interactions right after sending
+                   // the correct type pointerenter event. Workaround by discarding
+                   // any mouse event that occurs immediately after a non-mouse event.
+                   return;
+                 }
+               } // don't show if buttons are pressed, e.g. during click and drag of map
 
 
-             text = text.replace(/&#?\w+;/g, entityReplacer);
-             domBuilder.characters(text, 0, text.length);
-             return elEndStart; //}
+               if (d3_event.buttons !== 0) return;
+               show.apply(this, arguments);
+             }).on(_pointerPrefix + 'leave.popover', function () {
+               hide.apply(this, arguments);
+             }) // show on focus too for better keyboard navigation support
+             .on('focus.popover', function () {
+               show.apply(this, arguments);
+             }).on('blur.popover', function () {
+               hide.apply(this, arguments);
+             });
+           } else if (display === 'clickFocus') {
+             anchor.on(_pointerPrefix + 'down.popover', function (d3_event) {
+               d3_event.preventDefault();
+               d3_event.stopPropagation();
+             }).on(_pointerPrefix + 'up.popover', function (d3_event) {
+               d3_event.preventDefault();
+               d3_event.stopPropagation();
+             }).on('click.popover', toggle);
+             popoverSelection // This attribute lets the popover take focus
+             .attr('tabindex', 0).on('blur.popover', function () {
+               anchor.each(function () {
+                 hide.apply(this, arguments);
+               });
+             });
            }
          }
 
-         return elStartEnd + 1;
-       }
-
-       function fixSelfClosed(source, elStartEnd, tagName, closeMap) {
-         //if(tagName in closeMap){
-         var pos = closeMap[tagName];
-
-         if (pos == null) {
-           //console.log(tagName)
-           pos = source.lastIndexOf('</' + tagName + '>');
+         function show() {
+           var anchor = select(this);
+           var popoverSelection = anchor.selectAll('.popover-' + _id);
 
-           if (pos < elStartEnd) {
-             //忘记闭合
-             pos = source.lastIndexOf('</' + tagName);
+           if (popoverSelection.empty()) {
+             // popover was removed somehow, put it back
+             anchor.call(popover.destroy);
+             anchor.each(setup);
+             popoverSelection = anchor.selectAll('.popover-' + _id);
            }
 
-           closeMap[tagName] = pos;
-         }
+           popoverSelection.classed('in', true);
 
-         return pos < elStartEnd; //} 
-       }
+           var displayType = _displayType.apply(this, arguments);
 
-       function _copy(source, target) {
-         for (var n in source) {
-           target[n] = source[n];
+           if (displayType === 'clickFocus') {
+             anchor.classed('active', true);
+             popoverSelection.node().focus();
+           }
+
+           anchor.each(updateContent);
          }
-       }
 
-       function parseDCC(source, start, domBuilder, errorHandler) {
-         //sure start with '<!'
-         var next = source.charAt(start + 2);
+         function updateContent() {
+           var anchor = select(this);
 
-         switch (next) {
-           case '-':
-             if (source.charAt(start + 3) === '-') {
-               var end = source.indexOf('-->', start + 4); //append comment source.substring(4,end)//<!--
+           if (_content) {
+             anchor.selectAll('.popover-' + _id + ' > .popover-inner').call(_content.apply(this, arguments));
+           }
 
-               if (end > start) {
-                 domBuilder.comment(source, start + 4, end - start - 4);
-                 return end + 3;
-               } else {
-                 errorHandler.error("Unclosed comment");
-                 return -1;
-               }
-             } else {
-               //error
-               return -1;
-             }
+           updatePosition.apply(this, arguments); // hack: update multiple times to fix instances where the absolute offset is
+           // set before the dynamic popover size is calculated by the browser
 
-           default:
-             if (source.substr(start + 3, 6) == 'CDATA[') {
-               var end = source.indexOf(']]>', start + 9);
-               domBuilder.startCDATA();
-               domBuilder.characters(source, start + 9, end - start - 9);
-               domBuilder.endCDATA();
-               return end + 3;
-             } //<!DOCTYPE
-             //startDTD(java.lang.String name, java.lang.String publicId, java.lang.String systemId) 
+           updatePosition.apply(this, arguments);
+           updatePosition.apply(this, arguments);
+         }
 
+         function updatePosition() {
+           var anchor = select(this);
+           var popoverSelection = anchor.selectAll('.popover-' + _id);
 
-             var matchs = split$1(source, start);
-             var len = matchs.length;
+           var scrollContainer = _scrollContainer && _scrollContainer.apply(this, arguments);
 
-             if (len > 1 && /!doctype/i.test(matchs[0][0])) {
-               var name = matchs[1][0];
-               var pubid = len > 3 && /^public$/i.test(matchs[2][0]) && matchs[3][0];
-               var sysid = len > 4 && matchs[4][0];
-               var lastMatch = matchs[len - 1];
-               domBuilder.startDTD(name, pubid && pubid.replace(/^(['"])(.*?)\1$/, '$2'), sysid && sysid.replace(/^(['"])(.*?)\1$/, '$2'));
-               domBuilder.endDTD();
-               return lastMatch.index + lastMatch[0].length;
-             }
+           var scrollNode = scrollContainer && !scrollContainer.empty() && scrollContainer.node();
+           var scrollLeft = scrollNode ? scrollNode.scrollLeft : 0;
+           var scrollTop = scrollNode ? scrollNode.scrollTop : 0;
 
-         }
+           var placement = _placement.apply(this, arguments);
 
-         return -1;
-       }
+           popoverSelection.classed('left', false).classed('right', false).classed('top', false).classed('bottom', false).classed(placement, true);
 
-       function parseInstruction(source, start, domBuilder) {
-         var end = source.indexOf('?>', start);
+           var alignment = _alignment.apply(this, arguments);
 
-         if (end) {
-           var match = source.substring(start, end).match(/^<\?(\S*)\s*([\s\S]*?)\s*$/);
+           var alignFactor = 0.5;
 
-           if (match) {
-             var len = match[0].length;
-             domBuilder.processingInstruction(match[1], match[2]);
-             return end + 2;
-           } else {
-             //error
-             return -1;
+           if (alignment === 'leading') {
+             alignFactor = 0;
+           } else if (alignment === 'trailing') {
+             alignFactor = 1;
            }
-         }
 
-         return -1;
-       }
-       /**
-        * @param source
-        */
+           var anchorFrame = getFrame(anchor.node());
+           var popoverFrame = getFrame(popoverSelection.node());
+           var position;
 
+           switch (placement) {
+             case 'top':
+               position = {
+                 x: anchorFrame.x + (anchorFrame.w - popoverFrame.w) * alignFactor,
+                 y: anchorFrame.y - popoverFrame.h
+               };
+               break;
 
-       function ElementAttributes(source) {}
+             case 'bottom':
+               position = {
+                 x: anchorFrame.x + (anchorFrame.w - popoverFrame.w) * alignFactor,
+                 y: anchorFrame.y + anchorFrame.h
+               };
+               break;
 
-       ElementAttributes.prototype = {
-         setTagName: function setTagName(tagName) {
-           if (!tagNamePattern.test(tagName)) {
-             throw new Error('invalid tagName:' + tagName);
-           }
+             case 'left':
+               position = {
+                 x: anchorFrame.x - popoverFrame.w,
+                 y: anchorFrame.y + (anchorFrame.h - popoverFrame.h) * alignFactor
+               };
+               break;
 
-           this.tagName = tagName;
-         },
-         add: function add(qName, value, offset) {
-           if (!tagNamePattern.test(qName)) {
-             throw new Error('invalid attribute:' + qName);
+             case 'right':
+               position = {
+                 x: anchorFrame.x + anchorFrame.w,
+                 y: anchorFrame.y + (anchorFrame.h - popoverFrame.h) * alignFactor
+               };
+               break;
            }
 
-           this[this.length++] = {
-             qName: qName,
-             value: value,
-             offset: offset
-           };
-         },
-         length: 0,
-         getLocalName: function getLocalName(i) {
-           return this[i].localName;
-         },
-         getLocator: function getLocator(i) {
-           return this[i].locator;
-         },
-         getQName: function getQName(i) {
-           return this[i].qName;
-         },
-         getURI: function getURI(i) {
-           return this[i].uri;
-         },
-         getValue: function getValue(i) {
-           return this[i].value;
-         } //  ,getIndex:function(uri, localName)){
-         //            if(localName){
-         //                    
-         //            }else{
-         //                    var qName = uri
-         //            }
-         //    },
-         //    getValue:function(){return this.getValue(this.getIndex.apply(this,arguments))},
-         //    getType:function(uri,localName){}
-         //    getType:function(i){},
+           if (position) {
+             if (scrollNode && (placement === 'top' || placement === 'bottom')) {
+               var initialPosX = position.x;
 
-       };
+               if (position.x + popoverFrame.w > scrollNode.offsetWidth - 10) {
+                 position.x = scrollNode.offsetWidth - 10 - popoverFrame.w;
+               } else if (position.x < 10) {
+                 position.x = 10;
+               }
 
-       function _set_proto_(thiz, parent) {
-         thiz.__proto__ = parent;
-         return thiz;
-       }
+               var arrow = anchor.selectAll('.popover-' + _id + ' > .popover-arrow'); // keep the arrow centered on the button, or as close as possible
 
-       if (!(_set_proto_({}, _set_proto_.prototype) instanceof _set_proto_)) {
-         _set_proto_ = function _set_proto_(thiz, parent) {
-           function p() {}
-           p.prototype = parent;
-           p = new p();
+               var arrowPosX = Math.min(Math.max(popoverFrame.w / 2 - (position.x - initialPosX), 10), popoverFrame.w - 10);
+               arrow.style('left', ~~arrowPosX + 'px');
+             }
 
-           for (parent in thiz) {
-             p[parent] = thiz[parent];
+             popoverSelection.style('left', ~~position.x + 'px').style('top', ~~position.y + 'px');
+           } else {
+             popoverSelection.style('left', null).style('top', null);
            }
 
-           return p;
-         };
-       }
-
-       function split$1(source, start) {
-         var match;
-         var buf = [];
-         var reg = /'[^']+'|"[^"]+"|[^\s<>\/=]+=?|(\/?\s*>|<)/g;
-         reg.lastIndex = start;
-         reg.exec(source); //skip <
+           function getFrame(node) {
+             var positionStyle = select(node).style('position');
 
-         while (match = reg.exec(source)) {
-           buf.push(match);
-           if (match[1]) return buf;
+             if (positionStyle === 'absolute' || positionStyle === 'static') {
+               return {
+                 x: node.offsetLeft - scrollLeft,
+                 y: node.offsetTop - scrollTop,
+                 w: node.offsetWidth,
+                 h: node.offsetHeight
+               };
+             } else {
+               return {
+                 x: 0,
+                 y: 0,
+                 w: node.offsetWidth,
+                 h: node.offsetHeight
+               };
+             }
+           }
          }
-       }
 
-       var XMLReader_1 = XMLReader;
-       var sax = {
-         XMLReader: XMLReader_1
-       };
+         function hide() {
+           var anchor = select(this);
 
-       /*
-        * DOM Level 2
-        * Object DOMException
-        * @see http://www.w3.org/TR/REC-DOM-Level-1/ecma-script-language-binding.html
-        * @see http://www.w3.org/TR/2000/REC-DOM-Level-2-Core-20001113/ecma-script-binding.html
-        */
-       function copy$1(src, dest) {
-         for (var p in src) {
-           dest[p] = src[p];
+           if (_displayType.apply(this, arguments) === 'clickFocus') {
+             anchor.classed('active', false);
+           }
+
+           anchor.selectAll('.popover-' + _id).classed('in', false);
          }
-       }
-       /**
-       ^\w+\.prototype\.([_\w]+)\s*=\s*((?:.*\{\s*?[\r\n][\s\S]*?^})|\S.*?(?=[;\r\n]));?
-       ^\w+\.prototype\.([_\w]+)\s*=\s*(\S.*?(?=[;\r\n]));?
-        */
 
+         function toggle() {
+           if (select(this).select('.popover-' + _id).classed('in')) {
+             hide.apply(this, arguments);
+           } else {
+             show.apply(this, arguments);
+           }
+         }
 
-       function _extends(Class, Super) {
-         var pt = Class.prototype;
+         return popover;
+       }
 
-         if (Object.create) {
-           var ppt = Object.create(Super.prototype);
-           pt.__proto__ = ppt;
-         }
+       function uiTooltip(klass) {
+         var tooltip = uiPopover((klass || '') + ' tooltip').displayType('hover');
 
-         if (!(pt instanceof Super)) {
-           var t = function t() {};
-           t.prototype = Super.prototype;
-           t = new t();
-           copy$1(pt, t);
-           Class.prototype = pt = t;
-         }
+         var _title = function _title() {
+           var title = this.getAttribute('data-original-title');
 
-         if (pt.constructor != Class) {
-           if (typeof Class != 'function') {
-             console.error("unknow Class:" + Class);
+           if (title) {
+             return title;
+           } else {
+             title = this.getAttribute('title');
+             this.removeAttribute('title');
+             this.setAttribute('data-original-title', title);
            }
 
-           pt.constructor = Class;
-         }
-       }
+           return title;
+         };
 
-       var htmlns = 'http://www.w3.org/1999/xhtml'; // Node Types
+         var _heading = utilFunctor(null);
 
-       var NodeType = {};
-       var ELEMENT_NODE = NodeType.ELEMENT_NODE = 1;
-       var ATTRIBUTE_NODE = NodeType.ATTRIBUTE_NODE = 2;
-       var TEXT_NODE = NodeType.TEXT_NODE = 3;
-       var CDATA_SECTION_NODE = NodeType.CDATA_SECTION_NODE = 4;
-       var ENTITY_REFERENCE_NODE = NodeType.ENTITY_REFERENCE_NODE = 5;
-       var ENTITY_NODE = NodeType.ENTITY_NODE = 6;
-       var PROCESSING_INSTRUCTION_NODE = NodeType.PROCESSING_INSTRUCTION_NODE = 7;
-       var COMMENT_NODE = NodeType.COMMENT_NODE = 8;
-       var DOCUMENT_NODE = NodeType.DOCUMENT_NODE = 9;
-       var DOCUMENT_TYPE_NODE = NodeType.DOCUMENT_TYPE_NODE = 10;
-       var DOCUMENT_FRAGMENT_NODE = NodeType.DOCUMENT_FRAGMENT_NODE = 11;
-       var NOTATION_NODE = NodeType.NOTATION_NODE = 12; // ExceptionCode
+         var _keys = utilFunctor(null);
 
-       var ExceptionCode = {};
-       var ExceptionMessage = {};
-       var INDEX_SIZE_ERR = ExceptionCode.INDEX_SIZE_ERR = (ExceptionMessage[1] = "Index size error", 1);
-       var DOMSTRING_SIZE_ERR = ExceptionCode.DOMSTRING_SIZE_ERR = (ExceptionMessage[2] = "DOMString size error", 2);
-       var HIERARCHY_REQUEST_ERR = ExceptionCode.HIERARCHY_REQUEST_ERR = (ExceptionMessage[3] = "Hierarchy request error", 3);
-       var WRONG_DOCUMENT_ERR = ExceptionCode.WRONG_DOCUMENT_ERR = (ExceptionMessage[4] = "Wrong document", 4);
-       var INVALID_CHARACTER_ERR = ExceptionCode.INVALID_CHARACTER_ERR = (ExceptionMessage[5] = "Invalid character", 5);
-       var NO_DATA_ALLOWED_ERR = ExceptionCode.NO_DATA_ALLOWED_ERR = (ExceptionMessage[6] = "No data allowed", 6);
-       var NO_MODIFICATION_ALLOWED_ERR = ExceptionCode.NO_MODIFICATION_ALLOWED_ERR = (ExceptionMessage[7] = "No modification allowed", 7);
-       var NOT_FOUND_ERR = ExceptionCode.NOT_FOUND_ERR = (ExceptionMessage[8] = "Not found", 8);
-       var NOT_SUPPORTED_ERR = ExceptionCode.NOT_SUPPORTED_ERR = (ExceptionMessage[9] = "Not supported", 9);
-       var INUSE_ATTRIBUTE_ERR = ExceptionCode.INUSE_ATTRIBUTE_ERR = (ExceptionMessage[10] = "Attribute in use", 10); //level2
+         tooltip.title = function (val) {
+           if (!arguments.length) return _title;
+           _title = utilFunctor(val);
+           return tooltip;
+         };
 
-       var INVALID_STATE_ERR = ExceptionCode.INVALID_STATE_ERR = (ExceptionMessage[11] = "Invalid state", 11);
-       var SYNTAX_ERR = ExceptionCode.SYNTAX_ERR = (ExceptionMessage[12] = "Syntax error", 12);
-       var INVALID_MODIFICATION_ERR = ExceptionCode.INVALID_MODIFICATION_ERR = (ExceptionMessage[13] = "Invalid modification", 13);
-       var NAMESPACE_ERR = ExceptionCode.NAMESPACE_ERR = (ExceptionMessage[14] = "Invalid namespace", 14);
-       var INVALID_ACCESS_ERR = ExceptionCode.INVALID_ACCESS_ERR = (ExceptionMessage[15] = "Invalid access", 15);
+         tooltip.heading = function (val) {
+           if (!arguments.length) return _heading;
+           _heading = utilFunctor(val);
+           return tooltip;
+         };
 
-       function DOMException$2(code, message) {
-         if (message instanceof Error) {
-           var error = message;
-         } else {
-           error = this;
-           Error.call(this, ExceptionMessage[code]);
-           this.message = ExceptionMessage[code];
-           if (Error.captureStackTrace) Error.captureStackTrace(this, DOMException$2);
-         }
+         tooltip.keys = function (val) {
+           if (!arguments.length) return _keys;
+           _keys = utilFunctor(val);
+           return tooltip;
+         };
 
-         error.code = code;
-         if (message) this.message = this.message + ": " + message;
-         return error;
+         tooltip.content(function () {
+           var heading = _heading.apply(this, arguments);
+
+           var text = _title.apply(this, arguments);
+
+           var keys = _keys.apply(this, arguments);
+
+           return function (selection) {
+             var headingSelect = selection.selectAll('.tooltip-heading').data(heading ? [heading] : []);
+             headingSelect.exit().remove();
+             headingSelect.enter().append('div').attr('class', 'tooltip-heading').merge(headingSelect).html(heading);
+             var textSelect = selection.selectAll('.tooltip-text').data(text ? [text] : []);
+             textSelect.exit().remove();
+             textSelect.enter().append('div').attr('class', 'tooltip-text').merge(textSelect).html(text);
+             var keyhintWrap = selection.selectAll('.keyhint-wrap').data(keys && keys.length ? [0] : []);
+             keyhintWrap.exit().remove();
+             var keyhintWrapEnter = keyhintWrap.enter().append('div').attr('class', 'keyhint-wrap');
+             keyhintWrapEnter.append('span').html(_t.html('tooltip_keyhint'));
+             keyhintWrap = keyhintWrapEnter.merge(keyhintWrap);
+             keyhintWrap.selectAll('kbd.shortcut').data(keys && keys.length ? keys : []).enter().append('kbd').attr('class', 'shortcut').html(function (d) {
+               return d;
+             });
+           };
+         });
+         return tooltip;
        }
-       DOMException$2.prototype = Error.prototype;
-       copy$1(ExceptionCode, DOMException$2);
-       /**
-        * @see http://www.w3.org/TR/2000/REC-DOM-Level-2-Core-20001113/core.html#ID-536297177
-        * The NodeList interface provides the abstraction of an ordered collection of nodes, without defining or constraining how this collection is implemented. NodeList objects in the DOM are live.
-        * The items in the NodeList are accessible via an integral index, starting from 0.
-        */
 
-       function NodeList() {}
-       NodeList.prototype = {
-         /**
-          * The number of nodes in the list. The range of valid child node indices is 0 to length-1 inclusive.
-          * @standard level1
-          */
-         length: 0,
+       function uiEditMenu(context) {
+         var dispatch = dispatch$8('toggled');
 
-         /**
-          * Returns the indexth item in the collection. If index is greater than or equal to the number of nodes in the list, this returns null.
-          * @standard level1
-          * @param index  unsigned long 
-          *   Index into the collection.
-          * @return Node
-          *    The node at the indexth position in the NodeList, or null if that is not a valid index. 
-          */
-         item: function item(index) {
-           return this[index] || null;
-         },
-         toString: function toString(isHTML, nodeFilter) {
-           for (var buf = [], i = 0; i < this.length; i++) {
-             serializeToString(this[i], buf, isHTML, nodeFilter);
-           }
+         var _menu = select(null);
+
+         var _operations = []; // the position the menu should be displayed relative to
+
+         var _anchorLoc = [0, 0];
+         var _anchorLocLonLat = [0, 0]; // a string indicating how the menu was opened
 
-           return buf.join('');
-         }
-       };
+         var _triggerType = '';
+         var _vpTopMargin = 85; // viewport top margin
 
-       function LiveNodeList(node, refresh) {
-         this._node = node;
-         this._refresh = refresh;
+         var _vpBottomMargin = 45; // viewport bottom margin
 
-         _updateLiveList(this);
-       }
+         var _vpSideMargin = 35; // viewport side margin
 
-       function _updateLiveList(list) {
-         var inc = list._node._inc || list._node.ownerDocument._inc;
+         var _menuTop = false;
 
-         if (list._inc != inc) {
-           var ls = list._refresh(list._node); //console.log(ls.length)
+         var _menuHeight;
 
+         var _menuWidth; // hardcode these values to make menu positioning easier
 
-           __set__(list, 'length', ls.length);
 
-           copy$1(ls, list);
-           list._inc = inc;
-         }
-       }
+         var _verticalPadding = 4; // see also `.edit-menu .tooltip` CSS; include margin
 
-       LiveNodeList.prototype.item = function (i) {
-         _updateLiveList(this);
+         var _tooltipWidth = 210; // offset the menu slightly from the target location
 
-         return this[i];
-       };
+         var _menuSideMargin = 10;
+         var _tooltips = [];
 
-       _extends(LiveNodeList, NodeList);
-       /**
-        * 
-        * Objects implementing the NamedNodeMap interface are used to represent collections of nodes that can be accessed by name. Note that NamedNodeMap does not inherit from NodeList; NamedNodeMaps are not maintained in any particular order. Objects contained in an object implementing NamedNodeMap may also be accessed by an ordinal index, but this is simply to allow convenient enumeration of the contents of a NamedNodeMap, and does not imply that the DOM specifies an order to these Nodes.
-        * NamedNodeMap objects in the DOM are live.
-        * used for attributes or DocumentType entities 
-        */
+         var editMenu = function editMenu(selection) {
+           var isTouchMenu = _triggerType.includes('touch') || _triggerType.includes('pen');
 
+           var ops = _operations.filter(function (op) {
+             return !isTouchMenu || !op.mouseOnly;
+           });
 
-       function NamedNodeMap() {}
+           if (!ops.length) return;
+           _tooltips = []; // Position the menu above the anchor for stylus and finger input
+           // since the mapper's hand likely obscures the screen below the anchor
 
-       function _findNodeIndex(list, node) {
-         var i = list.length;
+           _menuTop = isTouchMenu; // Show labels for touch input since there aren't hover tooltips
 
-         while (i--) {
-           if (list[i] === node) {
-             return i;
+           var showLabels = isTouchMenu;
+           var buttonHeight = showLabels ? 32 : 34;
+
+           if (showLabels) {
+             // Get a general idea of the width based on the length of the label
+             _menuWidth = 52 + Math.min(120, 6 * Math.max.apply(Math, ops.map(function (op) {
+               return op.title.length;
+             })));
+           } else {
+             _menuWidth = 44;
            }
-         }
-       }
 
-       function _addNamedNode(el, list, newAttr, oldAttr) {
-         if (oldAttr) {
-           list[_findNodeIndex(list, oldAttr)] = newAttr;
-         } else {
-           list[list.length++] = newAttr;
-         }
+           _menuHeight = _verticalPadding * 2 + ops.length * buttonHeight;
+           _menu = selection.append('div').attr('class', 'edit-menu').classed('touch-menu', isTouchMenu).style('padding', _verticalPadding + 'px 0');
 
-         if (el) {
-           newAttr.ownerElement = el;
-           var doc = el.ownerDocument;
+           var buttons = _menu.selectAll('.edit-menu-item').data(ops); // enter
 
-           if (doc) {
-             oldAttr && _onRemoveAttribute(doc, el, oldAttr);
 
-             _onAddAttribute(doc, el, newAttr);
-           }
-         }
-       }
+           var buttonsEnter = buttons.enter().append('button').attr('class', function (d) {
+             return 'edit-menu-item edit-menu-item-' + d.id;
+           }).style('height', buttonHeight + 'px').on('click', click) // don't listen for `mouseup` because we only care about non-mouse pointer types
+           .on('pointerup', pointerup).on('pointerdown mousedown', function pointerdown(d3_event) {
+             // don't let button presses also act as map input - #1869
+             d3_event.stopPropagation();
+           }).on('mouseenter.highlight', function (d3_event, d) {
+             if (!d.relatedEntityIds || select(this).classed('disabled')) return;
+             utilHighlightEntities(d.relatedEntityIds(), true, context);
+           }).on('mouseleave.highlight', function (d3_event, d) {
+             if (!d.relatedEntityIds) return;
+             utilHighlightEntities(d.relatedEntityIds(), false, context);
+           });
+           buttonsEnter.each(function (d) {
+             var tooltip = uiTooltip().heading(d.title).title(d.tooltip()).keys([d.keys[0]]);
 
-       function _removeNamedNode(el, list, attr) {
-         //console.log('remove attr:'+attr)
-         var i = _findNodeIndex(list, attr);
+             _tooltips.push(tooltip);
 
-         if (i >= 0) {
-           var lastIndex = list.length - 1;
+             select(this).call(tooltip).append('div').attr('class', 'icon-wrap').call(svgIcon('#iD-operation-' + d.id, 'operation'));
+           });
 
-           while (i < lastIndex) {
-             list[i] = list[++i];
+           if (showLabels) {
+             buttonsEnter.append('span').attr('class', 'label').html(function (d) {
+               return d.title;
+             });
+           } // update
+
+
+           buttonsEnter.merge(buttons).classed('disabled', function (d) {
+             return d.disabled();
+           });
+           updatePosition();
+           var initialScale = context.projection.scale();
+           context.map().on('move.edit-menu', function () {
+             if (initialScale !== context.projection.scale()) {
+               editMenu.close();
+             }
+           }).on('drawn.edit-menu', function (info) {
+             if (info.full) updatePosition();
+           });
+           var lastPointerUpType; // `pointerup` is always called before `click`
+
+           function pointerup(d3_event) {
+             lastPointerUpType = d3_event.pointerType;
            }
 
-           list.length = lastIndex;
+           function click(d3_event, operation) {
+             d3_event.stopPropagation();
 
-           if (el) {
-             var doc = el.ownerDocument;
+             if (operation.relatedEntityIds) {
+               utilHighlightEntities(operation.relatedEntityIds(), false, context);
+             }
 
-             if (doc) {
-               _onRemoveAttribute(doc, el, attr);
+             if (operation.disabled()) {
+               if (lastPointerUpType === 'touch' || lastPointerUpType === 'pen') {
+                 // there are no tooltips for touch interactions so flash feedback instead
+                 context.ui().flash.duration(4000).iconName('#iD-operation-' + operation.id).iconClass('operation disabled').label(operation.tooltip)();
+               }
+             } else {
+               if (lastPointerUpType === 'touch' || lastPointerUpType === 'pen') {
+                 context.ui().flash.duration(2000).iconName('#iD-operation-' + operation.id).iconClass('operation').label(operation.annotation() || operation.title)();
+               }
 
-               attr.ownerElement = null;
+               operation();
+               editMenu.close();
              }
+
+             lastPointerUpType = null;
            }
-         } else {
-           throw DOMException$2(NOT_FOUND_ERR, new Error(el.tagName + '@' + attr));
-         }
-       }
 
-       NamedNodeMap.prototype = {
-         length: 0,
-         item: NodeList.prototype.item,
-         getNamedItem: function getNamedItem(key) {
-           //          if(key.indexOf(':')>0 || key == 'xmlns'){
-           //                  return null;
-           //          }
-           //console.log()
-           var i = this.length;
+           dispatch.call('toggled', this, true);
+         };
 
-           while (i--) {
-             var attr = this[i]; //console.log(attr.nodeName,key)
+         function updatePosition() {
+           if (!_menu || _menu.empty()) return;
+           var anchorLoc = context.projection(_anchorLocLonLat);
+           var viewport = context.surfaceRect();
 
-             if (attr.nodeName == key) {
-               return attr;
-             }
+           if (anchorLoc[0] < 0 || anchorLoc[0] > viewport.width || anchorLoc[1] < 0 || anchorLoc[1] > viewport.height) {
+             // close the menu if it's gone offscreen
+             editMenu.close();
+             return;
            }
-         },
-         setNamedItem: function setNamedItem(attr) {
-           var el = attr.ownerElement;
 
-           if (el && el != this._ownerElement) {
-             throw new DOMException$2(INUSE_ATTRIBUTE_ERR);
+           var menuLeft = displayOnLeft(viewport);
+           var offset = [0, 0];
+           offset[0] = menuLeft ? -1 * (_menuSideMargin + _menuWidth) : _menuSideMargin;
+
+           if (_menuTop) {
+             if (anchorLoc[1] - _menuHeight < _vpTopMargin) {
+               // menu is near top viewport edge, shift downward
+               offset[1] = -anchorLoc[1] + _vpTopMargin;
+             } else {
+               offset[1] = -_menuHeight;
+             }
+           } else {
+             if (anchorLoc[1] + _menuHeight > viewport.height - _vpBottomMargin) {
+               // menu is near bottom viewport edge, shift upwards
+               offset[1] = -anchorLoc[1] - _menuHeight + viewport.height - _vpBottomMargin;
+             } else {
+               offset[1] = 0;
+             }
            }
 
-           var oldAttr = this.getNamedItem(attr.nodeName);
+           var origin = geoVecAdd(anchorLoc, offset);
 
-           _addNamedNode(this._ownerElement, this, attr, oldAttr);
+           _menu.style('left', origin[0] + 'px').style('top', origin[1] + 'px');
 
-           return oldAttr;
-         },
+           var tooltipSide = tooltipPosition(viewport, menuLeft);
 
-         /* returns Node */
-         setNamedItemNS: function setNamedItemNS(attr) {
-           // raises: WRONG_DOCUMENT_ERR,NO_MODIFICATION_ALLOWED_ERR,INUSE_ATTRIBUTE_ERR
-           var el = attr.ownerElement,
-               oldAttr;
+           _tooltips.forEach(function (tooltip) {
+             tooltip.placement(tooltipSide);
+           });
 
-           if (el && el != this._ownerElement) {
-             throw new DOMException$2(INUSE_ATTRIBUTE_ERR);
-           }
+           function displayOnLeft(viewport) {
+             if (_mainLocalizer.textDirection() === 'ltr') {
+               if (anchorLoc[0] + _menuSideMargin + _menuWidth > viewport.width - _vpSideMargin) {
+                 // right menu would be too close to the right viewport edge, go left
+                 return true;
+               } // prefer right menu
 
-           oldAttr = this.getNamedItemNS(attr.namespaceURI, attr.localName);
 
-           _addNamedNode(this._ownerElement, this, attr, oldAttr);
+               return false;
+             } else {
+               // rtl
+               if (anchorLoc[0] - _menuSideMargin - _menuWidth < _vpSideMargin) {
+                 // left menu would be too close to the left viewport edge, go right
+                 return false;
+               } // prefer left menu
 
-           return oldAttr;
-         },
 
-         /* returns Node */
-         removeNamedItem: function removeNamedItem(key) {
-           var attr = this.getNamedItem(key);
+               return true;
+             }
+           }
 
-           _removeNamedNode(this._ownerElement, this, attr);
+           function tooltipPosition(viewport, menuLeft) {
+             if (_mainLocalizer.textDirection() === 'ltr') {
+               if (menuLeft) {
+                 // if there's not room for a right-side menu then there definitely
+                 // isn't room for right-side tooltips
+                 return 'left';
+               }
 
-           return attr;
-         },
-         // raises: NOT_FOUND_ERR,NO_MODIFICATION_ALLOWED_ERR
-         //for level2
-         removeNamedItemNS: function removeNamedItemNS(namespaceURI, localName) {
-           var attr = this.getNamedItemNS(namespaceURI, localName);
+               if (anchorLoc[0] + _menuSideMargin + _menuWidth + _tooltipWidth > viewport.width - _vpSideMargin) {
+                 // right tooltips would be too close to the right viewport edge, go left
+                 return 'left';
+               } // prefer right tooltips
 
-           _removeNamedNode(this._ownerElement, this, attr);
 
-           return attr;
-         },
-         getNamedItemNS: function getNamedItemNS(namespaceURI, localName) {
-           var i = this.length;
+               return 'right';
+             } else {
+               // rtl
+               if (!menuLeft) {
+                 return 'right';
+               }
 
-           while (i--) {
-             var node = this[i];
+               if (anchorLoc[0] - _menuSideMargin - _menuWidth - _tooltipWidth < _vpSideMargin) {
+                 // left tooltips would be too close to the left viewport edge, go right
+                 return 'right';
+               } // prefer left tooltips
 
-             if (node.localName == localName && node.namespaceURI == namespaceURI) {
-               return node;
+
+               return 'left';
              }
            }
-
-           return null;
          }
-       };
-       /**
-        * @see http://www.w3.org/TR/REC-DOM-Level-1/level-one-core.html#ID-102161490
-        */
 
-       function DOMImplementation(
-       /* Object */
-       features) {
-         this._features = {};
+         editMenu.close = function () {
+           context.map().on('move.edit-menu', null).on('drawn.edit-menu', null);
 
-         if (features) {
-           for (var feature in features) {
-             this._features = features[feature];
+           _menu.remove();
+
+           _tooltips = [];
+           dispatch.call('toggled', this, false);
+         };
+
+         editMenu.anchorLoc = function (val) {
+           if (!arguments.length) return _anchorLoc;
+           _anchorLoc = val;
+           _anchorLocLonLat = context.projection.invert(_anchorLoc);
+           return editMenu;
+         };
+
+         editMenu.triggerType = function (val) {
+           if (!arguments.length) return _triggerType;
+           _triggerType = val;
+           return editMenu;
+         };
+
+         editMenu.operations = function (val) {
+           if (!arguments.length) return _operations;
+           _operations = val;
+           return editMenu;
+         };
+
+         return utilRebind(editMenu, dispatch, 'on');
+       }
+
+       function uiFeatureInfo(context) {
+         function update(selection) {
+           var features = context.features();
+           var stats = features.stats();
+           var count = 0;
+           var hiddenList = features.hidden().map(function (k) {
+             if (stats[k]) {
+               count += stats[k];
+               return _t('inspector.title_count', {
+                 title: _t.html('feature.' + k + '.description'),
+                 count: stats[k]
+               });
+             }
+
+             return null;
+           }).filter(Boolean);
+           selection.html('');
+
+           if (hiddenList.length) {
+             var tooltipBehavior = uiTooltip().placement('top').title(function () {
+               return hiddenList.join('<br/>');
+             });
+             selection.append('a').attr('class', 'chip').attr('href', '#').html(_t.html('feature_info.hidden_warning', {
+               count: count
+             })).call(tooltipBehavior).on('click', function (d3_event) {
+               tooltipBehavior.hide();
+               d3_event.preventDefault(); // open the Map Data pane
+
+               context.ui().togglePanes(context.container().select('.map-panes .map-data-pane'));
+             });
            }
+
+           selection.classed('hide', !hiddenList.length);
          }
+
+         return function (selection) {
+           update(selection);
+           context.features().on('change.feature_info', function () {
+             update(selection);
+           });
+         };
        }
-       DOMImplementation.prototype = {
-         hasFeature: function hasFeature(
-         /* string */
-         feature,
-         /* string */
-         version) {
-           var versions = this._features[feature.toLowerCase()];
 
-           if (versions && (!version || version in versions)) {
-             return true;
-           } else {
-             return false;
-           }
-         },
-         // Introduced in DOM Level 2:
-         createDocument: function createDocument(namespaceURI, qualifiedName, doctype) {
-           // raises:INVALID_CHARACTER_ERR,NAMESPACE_ERR,WRONG_DOCUMENT_ERR
-           var doc = new Document();
-           doc.implementation = this;
-           doc.childNodes = new NodeList();
-           doc.doctype = doctype;
+       function uiFlash(context) {
+         var _flashTimer;
 
-           if (doctype) {
-             doc.appendChild(doctype);
-           }
+         var _duration = 2000;
+         var _iconName = '#iD-icon-no';
+         var _iconClass = 'disabled';
+         var _label = '';
 
-           if (qualifiedName) {
-             var root = doc.createElementNS(namespaceURI, qualifiedName);
-             doc.appendChild(root);
+         function flash() {
+           if (_flashTimer) {
+             _flashTimer.stop();
            }
 
-           return doc;
-         },
-         // Introduced in DOM Level 2:
-         createDocumentType: function createDocumentType(qualifiedName, publicId, systemId) {
-           // raises:INVALID_CHARACTER_ERR,NAMESPACE_ERR
-           var node = new DocumentType();
-           node.name = qualifiedName;
-           node.nodeName = qualifiedName;
-           node.publicId = publicId;
-           node.systemId = systemId; // Introduced in DOM Level 2:
-           //readonly attribute DOMString        internalSubset;
-           //TODO:..
-           //  readonly attribute NamedNodeMap     entities;
-           //  readonly attribute NamedNodeMap     notations;
+           context.container().select('.main-footer-wrap').classed('footer-hide', true).classed('footer-show', false);
+           context.container().select('.flash-wrap').classed('footer-hide', false).classed('footer-show', true);
+           var content = context.container().select('.flash-wrap').selectAll('.flash-content').data([0]); // Enter
 
-           return node;
-         }
-       };
-       /**
-        * @see http://www.w3.org/TR/2000/REC-DOM-Level-2-Core-20001113/core.html#ID-1950641247
-        */
+           var contentEnter = content.enter().append('div').attr('class', 'flash-content');
+           var iconEnter = contentEnter.append('svg').attr('class', 'flash-icon icon').append('g').attr('transform', 'translate(10,10)');
+           iconEnter.append('circle').attr('r', 9);
+           iconEnter.append('use').attr('transform', 'translate(-7,-7)').attr('width', '14').attr('height', '14');
+           contentEnter.append('div').attr('class', 'flash-text'); // Update
 
-       function Node() {}
-       Node.prototype = {
-         firstChild: null,
-         lastChild: null,
-         previousSibling: null,
-         nextSibling: null,
-         attributes: null,
-         parentNode: null,
-         childNodes: null,
-         ownerDocument: null,
-         nodeValue: null,
-         namespaceURI: null,
-         prefix: null,
-         localName: null,
-         // Modified in DOM Level 2:
-         insertBefore: function insertBefore(newChild, refChild) {
-           //raises 
-           return _insertBefore(this, newChild, refChild);
-         },
-         replaceChild: function replaceChild(newChild, oldChild) {
-           //raises 
-           this.insertBefore(newChild, oldChild);
+           content = content.merge(contentEnter);
+           content.selectAll('.flash-icon').attr('class', 'icon flash-icon ' + (_iconClass || ''));
+           content.selectAll('.flash-icon use').attr('xlink:href', _iconName);
+           content.selectAll('.flash-text').attr('class', 'flash-text').html(_label);
+           _flashTimer = d3_timeout(function () {
+             _flashTimer = null;
+             context.container().select('.main-footer-wrap').classed('footer-hide', false).classed('footer-show', true);
+             context.container().select('.flash-wrap').classed('footer-hide', true).classed('footer-show', false);
+           }, _duration);
+           return content;
+         }
 
-           if (oldChild) {
-             this.removeChild(oldChild);
-           }
-         },
-         removeChild: function removeChild(oldChild) {
-           return _removeChild(this, oldChild);
-         },
-         appendChild: function appendChild(newChild) {
-           return this.insertBefore(newChild, null);
-         },
-         hasChildNodes: function hasChildNodes() {
-           return this.firstChild != null;
-         },
-         cloneNode: function cloneNode(deep) {
-           return _cloneNode(this.ownerDocument || this, this, deep);
-         },
-         // Modified in DOM Level 2:
-         normalize: function normalize() {
-           var child = this.firstChild;
+         flash.duration = function (_) {
+           if (!arguments.length) return _duration;
+           _duration = _;
+           return flash;
+         };
 
-           while (child) {
-             var next = child.nextSibling;
+         flash.label = function (_) {
+           if (!arguments.length) return _label;
+           _label = _;
+           return flash;
+         };
 
-             if (next && next.nodeType == TEXT_NODE && child.nodeType == TEXT_NODE) {
-               this.removeChild(next);
-               child.appendData(next.data);
-             } else {
-               child.normalize();
-               child = next;
-             }
-           }
-         },
-         // Introduced in DOM Level 2:
-         isSupported: function isSupported(feature, version) {
-           return this.ownerDocument.implementation.hasFeature(feature, version);
-         },
-         // Introduced in DOM Level 2:
-         hasAttributes: function hasAttributes() {
-           return this.attributes.length > 0;
-         },
-         lookupPrefix: function lookupPrefix(namespaceURI) {
-           var el = this;
+         flash.iconName = function (_) {
+           if (!arguments.length) return _iconName;
+           _iconName = _;
+           return flash;
+         };
 
-           while (el) {
-             var map = el._nsMap; //console.dir(map)
+         flash.iconClass = function (_) {
+           if (!arguments.length) return _iconClass;
+           _iconClass = _;
+           return flash;
+         };
 
-             if (map) {
-               for (var n in map) {
-                 if (map[n] == namespaceURI) {
-                   return n;
-                 }
-               }
-             }
+         return flash;
+       }
 
-             el = el.nodeType == ATTRIBUTE_NODE ? el.ownerDocument : el.parentNode;
+       function uiFullScreen(context) {
+         var element = context.container().node(); // var button = d3_select(null);
+
+         function getFullScreenFn() {
+           if (element.requestFullscreen) {
+             return element.requestFullscreen;
+           } else if (element.msRequestFullscreen) {
+             return element.msRequestFullscreen;
+           } else if (element.mozRequestFullScreen) {
+             return element.mozRequestFullScreen;
+           } else if (element.webkitRequestFullscreen) {
+             return element.webkitRequestFullscreen;
            }
+         }
 
-           return null;
-         },
-         // Introduced in DOM Level 3:
-         lookupNamespaceURI: function lookupNamespaceURI(prefix) {
-           var el = this;
+         function getExitFullScreenFn() {
+           if (document.exitFullscreen) {
+             return document.exitFullscreen;
+           } else if (document.msExitFullscreen) {
+             return document.msExitFullscreen;
+           } else if (document.mozCancelFullScreen) {
+             return document.mozCancelFullScreen;
+           } else if (document.webkitExitFullscreen) {
+             return document.webkitExitFullscreen;
+           }
+         }
 
-           while (el) {
-             var map = el._nsMap; //console.dir(map)
+         function isFullScreen() {
+           return document.fullscreenElement || document.mozFullScreenElement || document.webkitFullscreenElement || document.msFullscreenElement;
+         }
 
-             if (map) {
-               if (prefix in map) {
-                 return map[prefix];
-               }
-             }
+         function isSupported() {
+           return !!getFullScreenFn();
+         }
 
-             el = el.nodeType == ATTRIBUTE_NODE ? el.ownerDocument : el.parentNode;
-           }
+         function fullScreen(d3_event) {
+           d3_event.preventDefault();
 
-           return null;
-         },
-         // Introduced in DOM Level 3:
-         isDefaultNamespace: function isDefaultNamespace(namespaceURI) {
-           var prefix = this.lookupPrefix(namespaceURI);
-           return prefix == null;
+           if (!isFullScreen()) {
+             // button.classed('active', true);
+             getFullScreenFn().apply(element);
+           } else {
+             // button.classed('active', false);
+             getExitFullScreenFn().apply(document);
+           }
          }
-       };
 
-       function _xmlEncoder(c) {
-         return c == '<' && '&lt;' || c == '>' && '&gt;' || c == '&' && '&amp;' || c == '"' && '&quot;' || '&#' + c.charCodeAt() + ';';
+         return function () {
+           // selection) {
+           if (!isSupported()) return; // button = selection.append('button')
+           //     .attr('title', t('full_screen'))
+           //     .on('click', fullScreen)
+           //     .call(tooltip);
+           // button.append('span')
+           //     .attr('class', 'icon full-screen');
+
+           var detected = utilDetect();
+           var keys = detected.os === 'mac' ? [uiCmd('⌃⌘F'), 'f11'] : ['f11'];
+           context.keybinding().on(keys, fullScreen);
+         };
        }
 
-       copy$1(NodeType, Node);
-       copy$1(NodeType, Node.prototype);
-       /**
-        * @param callback return true for continue,false for break
-        * @return boolean true: break visit;
-        */
+       function uiGeolocate(context) {
+         var _geolocationOptions = {
+           // prioritize speed and power usage over precision
+           enableHighAccuracy: false,
+           // don't hang indefinitely getting the location
+           timeout: 6000 // 6sec
 
-       function _visitNode(node, callback) {
-         if (callback(node)) {
-           return true;
-         }
+         };
 
-         if (node = node.firstChild) {
-           do {
-             if (_visitNode(node, callback)) {
-               return true;
-             }
-           } while (node = node.nextSibling);
-         }
-       }
+         var _locating = uiLoading(context).message(_t.html('geolocate.locating')).blocking(true);
 
-       function Document() {}
+         var _layer = context.layers().layer('geolocate');
 
-       function _onAddAttribute(doc, el, newAttr) {
-         doc && doc._inc++;
-         var ns = newAttr.namespaceURI;
+         var _position;
 
-         if (ns == 'http://www.w3.org/2000/xmlns/') {
-           //update namespace
-           el._nsMap[newAttr.prefix ? newAttr.localName : ''] = newAttr.value;
-         }
-       }
+         var _extent;
 
-       function _onRemoveAttribute(doc, el, newAttr, remove) {
-         doc && doc._inc++;
-         var ns = newAttr.namespaceURI;
+         var _timeoutID;
 
-         if (ns == 'http://www.w3.org/2000/xmlns/') {
-           //update namespace
-           delete el._nsMap[newAttr.prefix ? newAttr.localName : ''];
-         }
-       }
+         var _button = select(null);
 
-       function _onUpdateChild(doc, el, newChild) {
-         if (doc && doc._inc) {
-           doc._inc++; //update childNodes
+         function click() {
+           if (context.inIntro()) return;
 
-           var cs = el.childNodes;
+           if (!_layer.enabled() && !_locating.isShown()) {
+             // This timeout ensures that we still call finish() even if
+             // the user declines to share their location in Firefox
+             _timeoutID = setTimeout(error, 10000
+             /* 10sec */
+             );
+             context.container().call(_locating); // get the latest position even if we already have one
 
-           if (newChild) {
-             cs[cs.length++] = newChild;
+             navigator.geolocation.getCurrentPosition(success, error, _geolocationOptions);
            } else {
-             //console.log(1)
-             var child = el.firstChild;
-             var i = 0;
+             _locating.close();
 
-             while (child) {
-               cs[i++] = child;
-               child = child.nextSibling;
-             }
+             _layer.enabled(null, false);
 
-             cs.length = i;
+             updateButtonState();
            }
          }
-       }
-       /**
-        * attributes;
-        * children;
-        * 
-        * writeable properties:
-        * nodeValue,Attr:value,CharacterData:data
-        * prefix
-        */
 
+         function zoomTo() {
+           context.enter(modeBrowse(context));
+           var map = context.map();
 
-       function _removeChild(parentNode, child) {
-         var previous = child.previousSibling;
-         var next = child.nextSibling;
+           _layer.enabled(_position, true);
 
-         if (previous) {
-           previous.nextSibling = next;
-         } else {
-           parentNode.firstChild = next;
+           updateButtonState();
+           map.centerZoomEase(_extent.center(), Math.min(20, map.extentZoom(_extent)));
          }
 
-         if (next) {
-           next.previousSibling = previous;
-         } else {
-           parentNode.lastChild = previous;
+         function success(geolocation) {
+           _position = geolocation;
+           var coords = _position.coords;
+           _extent = geoExtent([coords.longitude, coords.latitude]).padByMeters(coords.accuracy);
+           zoomTo();
+           finish();
          }
 
-         _onUpdateChild(parentNode.ownerDocument, parentNode);
+         function error() {
+           if (_position) {
+             // use the position from a previous call if we have one
+             zoomTo();
+           } else {
+             context.ui().flash.label(_t.html('geolocate.location_unavailable')).iconName('#iD-icon-geolocate')();
+           }
 
-         return child;
-       }
-       /**
-        * preformance key(refChild == null)
-        */
+           finish();
+         }
 
+         function finish() {
+           _locating.close(); // unblock ui
 
-       function _insertBefore(parentNode, newChild, nextChild) {
-         var cp = newChild.parentNode;
 
-         if (cp) {
-           cp.removeChild(newChild); //remove and update
+           if (_timeoutID) {
+             clearTimeout(_timeoutID);
+           }
+
+           _timeoutID = undefined;
          }
 
-         if (newChild.nodeType === DOCUMENT_FRAGMENT_NODE) {
-           var newFirst = newChild.firstChild;
+         function updateButtonState() {
+           _button.classed('active', _layer.enabled());
+         }
 
-           if (newFirst == null) {
-             return newChild;
-           }
+         return function (selection) {
+           if (!navigator.geolocation || !navigator.geolocation.getCurrentPosition) return;
+           _button = selection.append('button').on('click', click).call(svgIcon('#iD-icon-geolocate', 'light')).call(uiTooltip().placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left').title(_t.html('geolocate.title')).keys([_t('geolocate.key')]));
+           context.keybinding().on(_t('geolocate.key'), click);
+         };
+       }
 
-           var newLast = newChild.lastChild;
-         } else {
-           newFirst = newLast = newChild;
-         }
+       function uiPanelBackground(context) {
+         var background = context.background();
+         var _currSourceName = null;
+         var _metadata = {};
+         var _metadataKeys = ['zoom', 'vintage', 'source', 'description', 'resolution', 'accuracy'];
 
-         var pre = nextChild ? nextChild.previousSibling : parentNode.lastChild;
-         newFirst.previousSibling = pre;
-         newLast.nextSibling = nextChild;
+         var debouncedRedraw = debounce(redraw, 250);
 
-         if (pre) {
-           pre.nextSibling = newFirst;
-         } else {
-           parentNode.firstChild = newFirst;
-         }
+         function redraw(selection) {
+           var source = background.baseLayerSource();
+           if (!source) return;
+           var isDG = source.id.match(/^DigitalGlobe/i) !== null;
+           var sourceLabel = source.label();
 
-         if (nextChild == null) {
-           parentNode.lastChild = newLast;
-         } else {
-           nextChild.previousSibling = newLast;
-         }
+           if (_currSourceName !== sourceLabel) {
+             _currSourceName = sourceLabel;
+             _metadata = {};
+           }
 
-         do {
-           newFirst.parentNode = parentNode;
-         } while (newFirst !== newLast && (newFirst = newFirst.nextSibling));
+           selection.html('');
+           var list = selection.append('ul').attr('class', 'background-info');
+           list.append('li').html(_currSourceName);
 
-         _onUpdateChild(parentNode.ownerDocument || parentNode, parentNode); //console.log(parentNode.lastChild.nextSibling == null)
+           _metadataKeys.forEach(function (k) {
+             // DigitalGlobe vintage is available in raster layers for now.
+             if (isDG && k === 'vintage') return;
+             list.append('li').attr('class', 'background-info-list-' + k).classed('hide', !_metadata[k]).html(_t.html('info_panels.background.' + k) + ':').append('span').attr('class', 'background-info-span-' + k).html(_metadata[k]);
+           });
+
+           debouncedGetMetadata(selection);
+           var toggleTiles = context.getDebug('tile') ? 'hide_tiles' : 'show_tiles';
+           selection.append('a').html(_t.html('info_panels.background.' + toggleTiles)).attr('href', '#').attr('class', 'button button-toggle-tiles').on('click', function (d3_event) {
+             d3_event.preventDefault();
+             context.setDebug('tile', !context.getDebug('tile'));
+             selection.call(redraw);
+           });
+
+           if (isDG) {
+             var key = source.id + '-vintage';
+             var sourceVintage = context.background().findSource(key);
+             var showsVintage = context.background().showsLayer(sourceVintage);
+             var toggleVintage = showsVintage ? 'hide_vintage' : 'show_vintage';
+             selection.append('a').html(_t.html('info_panels.background.' + toggleVintage)).attr('href', '#').attr('class', 'button button-toggle-vintage').on('click', function (d3_event) {
+               d3_event.preventDefault();
+               context.background().toggleOverlayLayer(sourceVintage);
+               selection.call(redraw);
+             });
+           } // disable if necessary
 
 
-         if (newChild.nodeType == DOCUMENT_FRAGMENT_NODE) {
-           newChild.firstChild = newChild.lastChild = null;
+           ['DigitalGlobe-Premium', 'DigitalGlobe-Standard'].forEach(function (layerId) {
+             if (source.id !== layerId) {
+               var key = layerId + '-vintage';
+               var sourceVintage = context.background().findSource(key);
+
+               if (context.background().showsLayer(sourceVintage)) {
+                 context.background().toggleOverlayLayer(sourceVintage);
+               }
+             }
+           });
          }
 
-         return newChild;
-       }
+         var debouncedGetMetadata = debounce(getMetadata, 250);
 
-       function _appendSingleChild(parentNode, newChild) {
-         var cp = newChild.parentNode;
+         function getMetadata(selection) {
+           var tile = context.container().select('.layer-background img.tile-center'); // tile near viewport center
 
-         if (cp) {
-           var pre = parentNode.lastChild;
-           cp.removeChild(newChild); //remove and update
+           if (tile.empty()) return;
+           var sourceName = _currSourceName;
+           var d = tile.datum();
+           var zoom = d && d.length >= 3 && d[2] || Math.floor(context.map().zoom());
+           var center = context.map().center(); // update zoom
 
-           var pre = parentNode.lastChild;
-         }
+           _metadata.zoom = String(zoom);
+           selection.selectAll('.background-info-list-zoom').classed('hide', false).selectAll('.background-info-span-zoom').html(_metadata.zoom);
+           if (!d || !d.length >= 3) return;
+           background.baseLayerSource().getMetadata(center, d, function (err, result) {
+             if (err || _currSourceName !== sourceName) return; // update vintage
 
-         var pre = parentNode.lastChild;
-         newChild.parentNode = parentNode;
-         newChild.previousSibling = pre;
-         newChild.nextSibling = null;
+             var vintage = result.vintage;
+             _metadata.vintage = vintage && vintage.range || _t('info_panels.background.unknown');
+             selection.selectAll('.background-info-list-vintage').classed('hide', false).selectAll('.background-info-span-vintage').html(_metadata.vintage); // update other _metadata
 
-         if (pre) {
-           pre.nextSibling = newChild;
-         } else {
-           parentNode.firstChild = newChild;
+             _metadataKeys.forEach(function (k) {
+               if (k === 'zoom' || k === 'vintage') return; // done already
+
+               var val = result[k];
+               _metadata[k] = val;
+               selection.selectAll('.background-info-list-' + k).classed('hide', !val).selectAll('.background-info-span-' + k).html(val);
+             });
+           });
          }
 
-         parentNode.lastChild = newChild;
+         var panel = function panel(selection) {
+           selection.call(redraw);
+           context.map().on('drawn.info-background', function () {
+             selection.call(debouncedRedraw);
+           }).on('move.info-background', function () {
+             selection.call(debouncedGetMetadata);
+           });
+         };
 
-         _onUpdateChild(parentNode.ownerDocument, parentNode, newChild);
+         panel.off = function () {
+           context.map().on('drawn.info-background', null).on('move.info-background', null);
+         };
 
-         return newChild; //console.log("__aa",parentNode.lastChild.nextSibling == null)
+         panel.id = 'background';
+         panel.label = _t.html('info_panels.background.title');
+         panel.key = _t('info_panels.background.key');
+         return panel;
        }
 
-       Document.prototype = {
-         //implementation : null,
-         nodeName: '#document',
-         nodeType: DOCUMENT_NODE,
-         doctype: null,
-         documentElement: null,
-         _inc: 1,
-         insertBefore: function insertBefore(newChild, refChild) {
-           //raises 
-           if (newChild.nodeType == DOCUMENT_FRAGMENT_NODE) {
-             var child = newChild.firstChild;
+       function uiPanelHistory(context) {
+         var osm;
 
-             while (child) {
-               var next = child.nextSibling;
-               this.insertBefore(child, refChild);
-               child = next;
-             }
+         function displayTimestamp(timestamp) {
+           if (!timestamp) return _t('info_panels.history.unknown');
+           var options = {
+             day: 'numeric',
+             month: 'short',
+             year: 'numeric',
+             hour: 'numeric',
+             minute: 'numeric',
+             second: 'numeric'
+           };
+           var d = new Date(timestamp);
+           if (isNaN(d.getTime())) return _t('info_panels.history.unknown');
+           return d.toLocaleString(_mainLocalizer.localeCode(), options);
+         }
 
-             return newChild;
+         function displayUser(selection, userName) {
+           if (!userName) {
+             selection.append('span').html(_t.html('info_panels.history.unknown'));
+             return;
            }
 
-           if (this.documentElement == null && newChild.nodeType == ELEMENT_NODE) {
-             this.documentElement = newChild;
-           }
+           selection.append('span').attr('class', 'user-name').html(userName);
+           var links = selection.append('div').attr('class', 'links');
 
-           return _insertBefore(this, newChild, refChild), newChild.ownerDocument = this, newChild;
-         },
-         removeChild: function removeChild(oldChild) {
-           if (this.documentElement == oldChild) {
-             this.documentElement = null;
+           if (osm) {
+             links.append('a').attr('class', 'user-osm-link').attr('href', osm.userURL(userName)).attr('target', '_blank').html('OSM');
            }
 
-           return _removeChild(this, oldChild);
-         },
-         // Introduced in DOM Level 2:
-         importNode: function importNode(importedNode, deep) {
-           return _importNode(this, importedNode, deep);
-         },
-         // Introduced in DOM Level 2:
-         getElementById: function getElementById(id) {
-           var rtv = null;
-
-           _visitNode(this.documentElement, function (node) {
-             if (node.nodeType == ELEMENT_NODE) {
-               if (node.getAttribute('id') == id) {
-                 rtv = node;
-                 return true;
-               }
-             }
-           });
-
-           return rtv;
-         },
-         //document factory method:
-         createElement: function createElement(tagName) {
-           var node = new Element();
-           node.ownerDocument = this;
-           node.nodeName = tagName;
-           node.tagName = tagName;
-           node.childNodes = new NodeList();
-           var attrs = node.attributes = new NamedNodeMap();
-           attrs._ownerElement = node;
-           return node;
-         },
-         createDocumentFragment: function createDocumentFragment() {
-           var node = new DocumentFragment();
-           node.ownerDocument = this;
-           node.childNodes = new NodeList();
-           return node;
-         },
-         createTextNode: function createTextNode(data) {
-           var node = new Text();
-           node.ownerDocument = this;
-           node.appendData(data);
-           return node;
-         },
-         createComment: function createComment(data) {
-           var node = new Comment();
-           node.ownerDocument = this;
-           node.appendData(data);
-           return node;
-         },
-         createCDATASection: function createCDATASection(data) {
-           var node = new CDATASection();
-           node.ownerDocument = this;
-           node.appendData(data);
-           return node;
-         },
-         createProcessingInstruction: function createProcessingInstruction(target, data) {
-           var node = new ProcessingInstruction();
-           node.ownerDocument = this;
-           node.tagName = node.target = target;
-           node.nodeValue = node.data = data;
-           return node;
-         },
-         createAttribute: function createAttribute(name) {
-           var node = new Attr();
-           node.ownerDocument = this;
-           node.name = name;
-           node.nodeName = name;
-           node.localName = name;
-           node.specified = true;
-           return node;
-         },
-         createEntityReference: function createEntityReference(name) {
-           var node = new EntityReference();
-           node.ownerDocument = this;
-           node.nodeName = name;
-           return node;
-         },
-         // Introduced in DOM Level 2:
-         createElementNS: function createElementNS(namespaceURI, qualifiedName) {
-           var node = new Element();
-           var pl = qualifiedName.split(':');
-           var attrs = node.attributes = new NamedNodeMap();
-           node.childNodes = new NodeList();
-           node.ownerDocument = this;
-           node.nodeName = qualifiedName;
-           node.tagName = qualifiedName;
-           node.namespaceURI = namespaceURI;
+           links.append('a').attr('class', 'user-hdyc-link').attr('href', 'https://hdyc.neis-one.org/?' + userName).attr('target', '_blank').attr('tabindex', -1).html('HDYC');
+         }
 
-           if (pl.length == 2) {
-             node.prefix = pl[0];
-             node.localName = pl[1];
-           } else {
-             //el.prefix = null;
-             node.localName = qualifiedName;
+         function displayChangeset(selection, changeset) {
+           if (!changeset) {
+             selection.append('span').html(_t.html('info_panels.history.unknown'));
+             return;
            }
 
-           attrs._ownerElement = node;
-           return node;
-         },
-         // Introduced in DOM Level 2:
-         createAttributeNS: function createAttributeNS(namespaceURI, qualifiedName) {
-           var node = new Attr();
-           var pl = qualifiedName.split(':');
-           node.ownerDocument = this;
-           node.nodeName = qualifiedName;
-           node.name = qualifiedName;
-           node.namespaceURI = namespaceURI;
-           node.specified = true;
+           selection.append('span').attr('class', 'changeset-id').html(changeset);
+           var links = selection.append('div').attr('class', 'links');
 
-           if (pl.length == 2) {
-             node.prefix = pl[0];
-             node.localName = pl[1];
-           } else {
-             //el.prefix = null;
-             node.localName = qualifiedName;
+           if (osm) {
+             links.append('a').attr('class', 'changeset-osm-link').attr('href', osm.changesetURL(changeset)).attr('target', '_blank').html('OSM');
            }
 
-           return node;
+           links.append('a').attr('class', 'changeset-osmcha-link').attr('href', 'https://osmcha.org/changesets/' + changeset).attr('target', '_blank').html('OSMCha');
+           links.append('a').attr('class', 'changeset-achavi-link').attr('href', 'https://overpass-api.de/achavi/?changeset=' + changeset).attr('target', '_blank').html('Achavi');
          }
-       };
 
-       _extends(Document, Node);
+         function redraw(selection) {
+           var selectedNoteID = context.selectedNoteID();
+           osm = context.connection();
+           var selected, note, entity;
 
-       function Element() {
-         this._nsMap = {};
-       }
-       Element.prototype = {
-         nodeType: ELEMENT_NODE,
-         hasAttribute: function hasAttribute(name) {
-           return this.getAttributeNode(name) != null;
-         },
-         getAttribute: function getAttribute(name) {
-           var attr = this.getAttributeNode(name);
-           return attr && attr.value || '';
-         },
-         getAttributeNode: function getAttributeNode(name) {
-           return this.attributes.getNamedItem(name);
-         },
-         setAttribute: function setAttribute(name, value) {
-           var attr = this.ownerDocument.createAttribute(name);
-           attr.value = attr.nodeValue = "" + value;
-           this.setAttributeNode(attr);
-         },
-         removeAttribute: function removeAttribute(name) {
-           var attr = this.getAttributeNode(name);
-           attr && this.removeAttributeNode(attr);
-         },
-         //four real opeartion method
-         appendChild: function appendChild(newChild) {
-           if (newChild.nodeType === DOCUMENT_FRAGMENT_NODE) {
-             return this.insertBefore(newChild, null);
+           if (selectedNoteID && osm) {
+             // selected 1 note
+             selected = [_t('note.note') + ' ' + selectedNoteID];
+             note = osm.getNote(selectedNoteID);
            } else {
-             return _appendSingleChild(this, newChild);
-           }
-         },
-         setAttributeNode: function setAttributeNode(newAttr) {
-           return this.attributes.setNamedItem(newAttr);
-         },
-         setAttributeNodeNS: function setAttributeNodeNS(newAttr) {
-           return this.attributes.setNamedItemNS(newAttr);
-         },
-         removeAttributeNode: function removeAttributeNode(oldAttr) {
-           //console.log(this == oldAttr.ownerElement)
-           return this.attributes.removeNamedItem(oldAttr.nodeName);
-         },
-         //get real attribute name,and remove it by removeAttributeNode
-         removeAttributeNS: function removeAttributeNS(namespaceURI, localName) {
-           var old = this.getAttributeNodeNS(namespaceURI, localName);
-           old && this.removeAttributeNode(old);
-         },
-         hasAttributeNS: function hasAttributeNS(namespaceURI, localName) {
-           return this.getAttributeNodeNS(namespaceURI, localName) != null;
-         },
-         getAttributeNS: function getAttributeNS(namespaceURI, localName) {
-           var attr = this.getAttributeNodeNS(namespaceURI, localName);
-           return attr && attr.value || '';
-         },
-         setAttributeNS: function setAttributeNS(namespaceURI, qualifiedName, value) {
-           var attr = this.ownerDocument.createAttributeNS(namespaceURI, qualifiedName);
-           attr.value = attr.nodeValue = "" + value;
-           this.setAttributeNode(attr);
-         },
-         getAttributeNodeNS: function getAttributeNodeNS(namespaceURI, localName) {
-           return this.attributes.getNamedItemNS(namespaceURI, localName);
-         },
-         getElementsByTagName: function getElementsByTagName(tagName) {
-           return new LiveNodeList(this, function (base) {
-             var ls = [];
-
-             _visitNode(base, function (node) {
-               if (node !== base && node.nodeType == ELEMENT_NODE && (tagName === '*' || node.tagName == tagName)) {
-                 ls.push(node);
-               }
+             // selected 1..n entities
+             selected = context.selectedIDs().filter(function (e) {
+               return context.hasEntity(e);
              });
 
-             return ls;
-           });
-         },
-         getElementsByTagNameNS: function getElementsByTagNameNS(namespaceURI, localName) {
-           return new LiveNodeList(this, function (base) {
-             var ls = [];
+             if (selected.length) {
+               entity = context.entity(selected[0]);
+             }
+           }
 
-             _visitNode(base, function (node) {
-               if (node !== base && node.nodeType === ELEMENT_NODE && (namespaceURI === '*' || node.namespaceURI === namespaceURI) && (localName === '*' || node.localName == localName)) {
-                 ls.push(node);
-               }
-             });
+           var singular = selected.length === 1 ? selected[0] : null;
+           selection.html('');
+           selection.append('h4').attr('class', 'history-heading').html(singular || _t.html('info_panels.selected', {
+             n: selected.length
+           }));
+           if (!singular) return;
 
-             return ls;
-           });
+           if (entity) {
+             selection.call(redrawEntity, entity);
+           } else if (note) {
+             selection.call(redrawNote, note);
+           }
          }
-       };
-       Document.prototype.getElementsByTagName = Element.prototype.getElementsByTagName;
-       Document.prototype.getElementsByTagNameNS = Element.prototype.getElementsByTagNameNS;
 
-       _extends(Element, Node);
+         function redrawNote(selection, note) {
+           if (!note || note.isNew()) {
+             selection.append('div').html(_t.html('info_panels.history.note_no_history'));
+             return;
+           }
 
-       function Attr() {}
-       Attr.prototype.nodeType = ATTRIBUTE_NODE;
+           var list = selection.append('ul');
+           list.append('li').html(_t.html('info_panels.history.note_comments') + ':').append('span').html(note.comments.length);
 
-       _extends(Attr, Node);
+           if (note.comments.length) {
+             list.append('li').html(_t.html('info_panels.history.note_created_date') + ':').append('span').html(displayTimestamp(note.comments[0].date));
+             list.append('li').html(_t.html('info_panels.history.note_created_user') + ':').call(displayUser, note.comments[0].user);
+           }
 
-       function CharacterData() {}
-       CharacterData.prototype = {
-         data: '',
-         substringData: function substringData(offset, count) {
-           return this.data.substring(offset, offset + count);
-         },
-         appendData: function appendData(text) {
-           text = this.data + text;
-           this.nodeValue = this.data = text;
-           this.length = text.length;
-         },
-         insertData: function insertData(offset, text) {
-           this.replaceData(offset, 0, text);
-         },
-         appendChild: function appendChild(newChild) {
-           throw new Error(ExceptionMessage[HIERARCHY_REQUEST_ERR]);
-         },
-         deleteData: function deleteData(offset, count) {
-           this.replaceData(offset, count, "");
-         },
-         replaceData: function replaceData(offset, count, text) {
-           var start = this.data.substring(0, offset);
-           var end = this.data.substring(offset + count);
-           text = start + text + end;
-           this.nodeValue = this.data = text;
-           this.length = text.length;
+           if (osm) {
+             selection.append('a').attr('class', 'view-history-on-osm').attr('target', '_blank').attr('href', osm.noteURL(note)).call(svgIcon('#iD-icon-out-link', 'inline')).append('span').html(_t.html('info_panels.history.note_link_text'));
+           }
          }
-       };
 
-       _extends(CharacterData, Node);
+         function redrawEntity(selection, entity) {
+           if (!entity || entity.isNew()) {
+             selection.append('div').html(_t.html('info_panels.history.no_history'));
+             return;
+           }
 
-       function Text() {}
-       Text.prototype = {
-         nodeName: "#text",
-         nodeType: TEXT_NODE,
-         splitText: function splitText(offset) {
-           var text = this.data;
-           var newText = text.substring(offset);
-           text = text.substring(0, offset);
-           this.data = this.nodeValue = text;
-           this.length = text.length;
-           var newNode = this.ownerDocument.createTextNode(newText);
+           var links = selection.append('div').attr('class', 'links');
 
-           if (this.parentNode) {
-             this.parentNode.insertBefore(newNode, this.nextSibling);
+           if (osm) {
+             links.append('a').attr('class', 'view-history-on-osm').attr('href', osm.historyURL(entity)).attr('target', '_blank').attr('title', _t('info_panels.history.link_text')).html('OSM');
            }
 
-           return newNode;
+           links.append('a').attr('class', 'pewu-history-viewer-link').attr('href', 'https://pewu.github.io/osm-history/#/' + entity.type + '/' + entity.osmId()).attr('target', '_blank').attr('tabindex', -1).html('PeWu');
+           var list = selection.append('ul');
+           list.append('li').html(_t.html('info_panels.history.version') + ':').append('span').html(entity.version);
+           list.append('li').html(_t.html('info_panels.history.last_edit') + ':').append('span').html(displayTimestamp(entity.timestamp));
+           list.append('li').html(_t.html('info_panels.history.edited_by') + ':').call(displayUser, entity.user);
+           list.append('li').html(_t.html('info_panels.history.changeset') + ':').call(displayChangeset, entity.changeset);
          }
-       };
-
-       _extends(Text, CharacterData);
 
-       function Comment() {}
-       Comment.prototype = {
-         nodeName: "#comment",
-         nodeType: COMMENT_NODE
-       };
+         var panel = function panel(selection) {
+           selection.call(redraw);
+           context.map().on('drawn.info-history', function () {
+             selection.call(redraw);
+           });
+           context.on('enter.info-history', function () {
+             selection.call(redraw);
+           });
+         };
 
-       _extends(Comment, CharacterData);
+         panel.off = function () {
+           context.map().on('drawn.info-history', null);
+           context.on('enter.info-history', null);
+         };
 
-       function CDATASection() {}
-       CDATASection.prototype = {
-         nodeName: "#cdata-section",
-         nodeType: CDATA_SECTION_NODE
-       };
+         panel.id = 'history';
+         panel.label = _t.html('info_panels.history.title');
+         panel.key = _t('info_panels.history.key');
+         return panel;
+       }
 
-       _extends(CDATASection, CharacterData);
+       var OSM_PRECISION = 7;
+       /**
+        * Returns a localized representation of the given length measurement.
+        *
+        * @param {Number} m area in meters
+        * @param {Boolean} isImperial true for U.S. customary units; false for metric
+        */
 
-       function DocumentType() {}
-       DocumentType.prototype.nodeType = DOCUMENT_TYPE_NODE;
+       function displayLength(m, isImperial) {
+         var d = m * (isImperial ? 3.28084 : 1);
+         var unit;
 
-       _extends(DocumentType, Node);
+         if (isImperial) {
+           if (d >= 5280) {
+             d /= 5280;
+             unit = 'miles';
+           } else {
+             unit = 'feet';
+           }
+         } else {
+           if (d >= 1000) {
+             d /= 1000;
+             unit = 'kilometers';
+           } else {
+             unit = 'meters';
+           }
+         }
 
-       function Notation() {}
-       Notation.prototype.nodeType = NOTATION_NODE;
+         return _t('units.' + unit, {
+           quantity: d.toLocaleString(_mainLocalizer.localeCode(), {
+             maximumSignificantDigits: 4
+           })
+         });
+       }
+       /**
+        * Returns a localized representation of the given area measurement.
+        *
+        * @param {Number} m2 area in square meters
+        * @param {Boolean} isImperial true for U.S. customary units; false for metric
+        */
 
-       _extends(Notation, Node);
+       function displayArea(m2, isImperial) {
+         var locale = _mainLocalizer.localeCode();
+         var d = m2 * (isImperial ? 10.7639111056 : 1);
+         var d1, d2, area;
+         var unit1 = '';
+         var unit2 = '';
 
-       function Entity() {}
-       Entity.prototype.nodeType = ENTITY_NODE;
+         if (isImperial) {
+           if (d >= 6969600) {
+             // > 0.25mi² show mi²
+             d1 = d / 27878400;
+             unit1 = 'square_miles';
+           } else {
+             d1 = d;
+             unit1 = 'square_feet';
+           }
 
-       _extends(Entity, Node);
+           if (d > 4356 && d < 43560000) {
+             // 0.1 - 1000 acres
+             d2 = d / 43560;
+             unit2 = 'acres';
+           }
+         } else {
+           if (d >= 250000) {
+             // > 0.25km² show km²
+             d1 = d / 1000000;
+             unit1 = 'square_kilometers';
+           } else {
+             d1 = d;
+             unit1 = 'square_meters';
+           }
 
-       function EntityReference() {}
-       EntityReference.prototype.nodeType = ENTITY_REFERENCE_NODE;
+           if (d > 1000 && d < 10000000) {
+             // 0.1 - 1000 hectares
+             d2 = d / 10000;
+             unit2 = 'hectares';
+           }
+         }
 
-       _extends(EntityReference, Node);
+         area = _t('units.' + unit1, {
+           quantity: d1.toLocaleString(locale, {
+             maximumSignificantDigits: 4
+           })
+         });
 
-       function DocumentFragment() {}
-       DocumentFragment.prototype.nodeName = "#document-fragment";
-       DocumentFragment.prototype.nodeType = DOCUMENT_FRAGMENT_NODE;
+         if (d2) {
+           return _t('units.area_pair', {
+             area1: area,
+             area2: _t('units.' + unit2, {
+               quantity: d2.toLocaleString(locale, {
+                 maximumSignificantDigits: 2
+               })
+             })
+           });
+         } else {
+           return area;
+         }
+       }
 
-       _extends(DocumentFragment, Node);
+       function wrap(x, min, max) {
+         var d = max - min;
+         return ((x - min) % d + d) % d + min;
+       }
 
-       function ProcessingInstruction() {}
+       function clamp(x, min, max) {
+         return Math.max(min, Math.min(x, max));
+       }
 
-       ProcessingInstruction.prototype.nodeType = PROCESSING_INSTRUCTION_NODE;
+       function displayCoordinate(deg, pos, neg) {
+         var locale = _mainLocalizer.localeCode();
+         var min = (Math.abs(deg) - Math.floor(Math.abs(deg))) * 60;
+         var sec = (min - Math.floor(min)) * 60;
+         var displayDegrees = _t('units.arcdegrees', {
+           quantity: Math.floor(Math.abs(deg)).toLocaleString(locale)
+         });
+         var displayCoordinate;
 
-       _extends(ProcessingInstruction, Node);
+         if (Math.floor(sec) > 0) {
+           displayCoordinate = displayDegrees + _t('units.arcminutes', {
+             quantity: Math.floor(min).toLocaleString(locale)
+           }) + _t('units.arcseconds', {
+             quantity: Math.round(sec).toLocaleString(locale)
+           });
+         } else if (Math.floor(min) > 0) {
+           displayCoordinate = displayDegrees + _t('units.arcminutes', {
+             quantity: Math.round(min).toLocaleString(locale)
+           });
+         } else {
+           displayCoordinate = _t('units.arcdegrees', {
+             quantity: Math.round(Math.abs(deg)).toLocaleString(locale)
+           });
+         }
 
-       function XMLSerializer$1() {}
+         if (deg === 0) {
+           return displayCoordinate;
+         } else {
+           return _t('units.coordinate', {
+             coordinate: displayCoordinate,
+             direction: _t('units.' + (deg > 0 ? pos : neg))
+           });
+         }
+       }
+       /**
+        * Returns given coordinate pair in degree-minute-second format.
+        *
+        * @param {Array<Number>} coord longitude and latitude
+        */
 
-       XMLSerializer$1.prototype.serializeToString = function (node, isHtml, nodeFilter) {
-         return nodeSerializeToString.call(node, isHtml, nodeFilter);
-       };
 
-       Node.prototype.toString = nodeSerializeToString;
+       function dmsCoordinatePair(coord) {
+         return _t('units.coordinate_pair', {
+           latitude: displayCoordinate(clamp(coord[1], -90, 90), 'north', 'south'),
+           longitude: displayCoordinate(wrap(coord[0], -180, 180), 'east', 'west')
+         });
+       }
+       /**
+        * Returns the given coordinate pair in decimal format.
+        * note: unlocalized to avoid comma ambiguity - see #4765
+        *
+        * @param {Array<Number>} coord longitude and latitude
+        */
 
-       function nodeSerializeToString(isHtml, nodeFilter) {
-         var buf = [];
-         var refNode = this.nodeType == 9 ? this.documentElement : this;
-         var prefix = refNode.prefix;
-         var uri = refNode.namespaceURI;
+       function decimalCoordinatePair(coord) {
+         return _t('units.coordinate_pair', {
+           latitude: clamp(coord[1], -90, 90).toFixed(OSM_PRECISION),
+           longitude: wrap(coord[0], -180, 180).toFixed(OSM_PRECISION)
+         });
+       }
 
-         if (uri && prefix == null) {
-           //console.log(prefix)
-           var prefix = refNode.lookupPrefix(uri);
+       function uiPanelLocation(context) {
+         var currLocation = '';
 
-           if (prefix == null) {
-             //isHTML = true;
-             var visibleNamespaces = [{
-               namespace: uri,
-               prefix: null
-             } //{namespace:uri,prefix:''}
-             ];
-           }
-         }
+         function redraw(selection) {
+           selection.html('');
+           var list = selection.append('ul'); // Mouse coordinates
 
-         serializeToString(this, buf, isHtml, nodeFilter, visibleNamespaces); //console.log('###',this.nodeType,uri,prefix,buf.join(''))
+           var coord = context.map().mouseCoordinates();
 
-         return buf.join('');
-       }
+           if (coord.some(isNaN)) {
+             coord = context.map().center();
+           }
 
-       function needNamespaceDefine(node, isHTML, visibleNamespaces) {
-         var prefix = node.prefix || '';
-         var uri = node.namespaceURI;
+           list.append('li').html(dmsCoordinatePair(coord)).append('li').html(decimalCoordinatePair(coord)); // Location Info
 
-         if (!prefix && !uri) {
-           return false;
+           selection.append('div').attr('class', 'location-info').html(currLocation || ' ');
+           debouncedGetLocation(selection, coord);
          }
 
-         if (prefix === "xml" && uri === "http://www.w3.org/XML/1998/namespace" || uri == 'http://www.w3.org/2000/xmlns/') {
-           return false;
+         var debouncedGetLocation = debounce(getLocation, 250);
+
+         function getLocation(selection, coord) {
+           if (!services.geocoder) {
+             currLocation = _t('info_panels.location.unknown_location');
+             selection.selectAll('.location-info').html(currLocation);
+           } else {
+             services.geocoder.reverse(coord, function (err, result) {
+               currLocation = result ? result.display_name : _t('info_panels.location.unknown_location');
+               selection.selectAll('.location-info').html(currLocation);
+             });
+           }
          }
 
-         var i = visibleNamespaces.length; //console.log('@@@@',node.tagName,prefix,uri,visibleNamespaces)
+         var panel = function panel(selection) {
+           selection.call(redraw);
+           context.surface().on(('PointerEvent' in window ? 'pointer' : 'mouse') + 'move.info-location', function () {
+             selection.call(redraw);
+           });
+         };
 
-         while (i--) {
-           var ns = visibleNamespaces[i]; // get namespace prefix
-           //console.log(node.nodeType,node.tagName,ns.prefix,prefix)
+         panel.off = function () {
+           context.surface().on('.info-location', null);
+         };
 
-           if (ns.prefix == prefix) {
-             return ns.namespace != uri;
-           }
-         } //console.log(isHTML,uri,prefix=='')
-         //if(isHTML && prefix ==null && uri == 'http://www.w3.org/1999/xhtml'){
-         //    return false;
-         //}
-         //node.flag = '11111'
-         //console.error(3,true,node.flag,node.prefix,node.namespaceURI)
+         panel.id = 'location';
+         panel.label = _t.html('info_panels.location.title');
+         panel.key = _t('info_panels.location.key');
+         return panel;
+       }
 
+       function uiPanelMeasurement(context) {
+         function radiansToMeters(r) {
+           // using WGS84 authalic radius (6371007.1809 m)
+           return r * 6371007.1809;
+         }
 
-         return true;
-       }
+         function steradiansToSqmeters(r) {
+           // http://gis.stackexchange.com/a/124857/40446
+           return r / (4 * Math.PI) * 510065621724000;
+         }
 
-       function serializeToString(node, buf, isHTML, nodeFilter, visibleNamespaces) {
-         if (nodeFilter) {
-           node = nodeFilter(node);
+         function toLineString(feature) {
+           if (feature.type === 'LineString') return feature;
+           var result = {
+             type: 'LineString',
+             coordinates: []
+           };
 
-           if (node) {
-             if (typeof node == 'string') {
-               buf.push(node);
-               return;
-             }
-           } else {
-             return;
-           } //buf.sort.apply(attrs, attributeSorter);
+           if (feature.type === 'Polygon') {
+             result.coordinates = feature.coordinates[0];
+           } else if (feature.type === 'MultiPolygon') {
+             result.coordinates = feature.coordinates[0][0];
+           }
 
+           return result;
          }
 
-         switch (node.nodeType) {
-           case ELEMENT_NODE:
-             if (!visibleNamespaces) visibleNamespaces = [];
-             var startVisibleNamespaces = visibleNamespaces.length;
-             var attrs = node.attributes;
-             var len = attrs.length;
-             var child = node.firstChild;
-             var nodeName = node.tagName;
-             isHTML = htmlns === node.namespaceURI || isHTML;
-             buf.push('<', nodeName);
-
-             for (var i = 0; i < len; i++) {
-               // add namespaces for attributes
-               var attr = attrs.item(i);
+         var _isImperial = !_mainLocalizer.usesMetric();
 
-               if (attr.prefix == 'xmlns') {
-                 visibleNamespaces.push({
-                   prefix: attr.localName,
-                   namespace: attr.value
-                 });
-               } else if (attr.nodeName == 'xmlns') {
-                 visibleNamespaces.push({
-                   prefix: '',
-                   namespace: attr.value
-                 });
-               }
-             }
+         function redraw(selection) {
+           var graph = context.graph();
+           var selectedNoteID = context.selectedNoteID();
+           var osm = services.osm;
+           var localeCode = _mainLocalizer.localeCode();
+           var heading;
+           var center, location, centroid;
+           var closed, geometry;
+           var totalNodeCount,
+               length = 0,
+               area = 0,
+               distance;
 
-             for (var i = 0; i < len; i++) {
-               var attr = attrs.item(i);
+           if (selectedNoteID && osm) {
+             // selected 1 note
+             var note = osm.getNote(selectedNoteID);
+             heading = _t('note.note') + ' ' + selectedNoteID;
+             location = note.loc;
+             geometry = 'note';
+           } else {
+             // selected 1..n entities
+             var selectedIDs = context.selectedIDs().filter(function (id) {
+               return context.hasEntity(id);
+             });
+             var selected = selectedIDs.map(function (id) {
+               return context.entity(id);
+             });
+             heading = selected.length === 1 ? selected[0].id : _t('info_panels.selected', {
+               n: selected.length
+             });
 
-               if (needNamespaceDefine(attr, isHTML, visibleNamespaces)) {
-                 var prefix = attr.prefix || '';
-                 var uri = attr.namespaceURI;
-                 var ns = prefix ? ' xmlns:' + prefix : " xmlns";
-                 buf.push(ns, '="', uri, '"');
-                 visibleNamespaces.push({
-                   prefix: prefix,
-                   namespace: uri
-                 });
-               }
+             if (selected.length) {
+               var extent = geoExtent();
 
-               serializeToString(attr, buf, isHTML, nodeFilter, visibleNamespaces);
-             } // add namespace for current node               
+               for (var i in selected) {
+                 var entity = selected[i];
 
+                 extent._extend(entity.extent(graph));
 
-             if (needNamespaceDefine(node, isHTML, visibleNamespaces)) {
-               var prefix = node.prefix || '';
-               var uri = node.namespaceURI;
-               var ns = prefix ? ' xmlns:' + prefix : " xmlns";
-               buf.push(ns, '="', uri, '"');
-               visibleNamespaces.push({
-                 prefix: prefix,
-                 namespace: uri
-               });
-             }
+                 geometry = entity.geometry(graph);
 
-             if (child || isHTML && !/^(?:meta|link|img|br|hr|input)$/i.test(nodeName)) {
-               buf.push('>'); //if is cdata child node
+                 if (geometry === 'line' || geometry === 'area') {
+                   closed = entity.type === 'relation' || entity.isClosed() && !entity.isDegenerate();
+                   var feature = entity.asGeoJSON(graph);
+                   length += radiansToMeters(d3_geoLength(toLineString(feature)));
+                   centroid = d3_geoPath(context.projection).centroid(entity.asGeoJSON(graph));
+                   centroid = centroid && context.projection.invert(centroid);
 
-               if (isHTML && /^script$/i.test(nodeName)) {
-                 while (child) {
-                   if (child.data) {
-                     buf.push(child.data);
-                   } else {
-                     serializeToString(child, buf, isHTML, nodeFilter, visibleNamespaces);
+                   if (!centroid || !isFinite(centroid[0]) || !isFinite(centroid[1])) {
+                     centroid = entity.extent(graph).center();
                    }
 
-                   child = child.nextSibling;
-                 }
-               } else {
-                 while (child) {
-                   serializeToString(child, buf, isHTML, nodeFilter, visibleNamespaces);
-                   child = child.nextSibling;
+                   if (closed) {
+                     area += steradiansToSqmeters(entity.area(graph));
+                   }
                  }
                }
 
-               buf.push('</', nodeName, '>');
-             } else {
-               buf.push('/>');
-             } // remove added visible namespaces
-             //visibleNamespaces.length = startVisibleNamespaces;
-
+               if (selected.length > 1) {
+                 geometry = null;
+                 closed = null;
+                 centroid = null;
+               }
 
-             return;
+               if (selected.length === 2 && selected[0].type === 'node' && selected[1].type === 'node') {
+                 distance = geoSphericalDistance(selected[0].loc, selected[1].loc);
+               }
 
-           case DOCUMENT_NODE:
-           case DOCUMENT_FRAGMENT_NODE:
-             var child = node.firstChild;
+               if (selected.length === 1 && selected[0].type === 'node') {
+                 location = selected[0].loc;
+               } else {
+                 totalNodeCount = utilGetAllNodes(selectedIDs, context.graph()).length;
+               }
 
-             while (child) {
-               serializeToString(child, buf, isHTML, nodeFilter, visibleNamespaces);
-               child = child.nextSibling;
+               if (!location && !centroid) {
+                 center = extent.center();
+               }
              }
+           }
 
-             return;
-
-           case ATTRIBUTE_NODE:
-             return buf.push(' ', node.name, '="', node.value.replace(/[<&"]/g, _xmlEncoder), '"');
-
-           case TEXT_NODE:
-             return buf.push(node.data.replace(/[<&]/g, _xmlEncoder));
-
-           case CDATA_SECTION_NODE:
-             return buf.push('<![CDATA[', node.data, ']]>');
+           selection.html('');
 
-           case COMMENT_NODE:
-             return buf.push("<!--", node.data, "-->");
+           if (heading) {
+             selection.append('h4').attr('class', 'measurement-heading').html(heading);
+           }
 
-           case DOCUMENT_TYPE_NODE:
-             var pubid = node.publicId;
-             var sysid = node.systemId;
-             buf.push('<!DOCTYPE ', node.name);
+           var list = selection.append('ul');
+           var coordItem;
 
-             if (pubid) {
-               buf.push(' PUBLIC "', pubid);
+           if (geometry) {
+             list.append('li').html(_t.html('info_panels.measurement.geometry') + ':').append('span').html(closed ? _t('info_panels.measurement.closed_' + geometry) : _t('geometry.' + geometry));
+           }
 
-               if (sysid && sysid != '.') {
-                 buf.push('" "', sysid);
-               }
+           if (totalNodeCount) {
+             list.append('li').html(_t.html('info_panels.measurement.node_count') + ':').append('span').html(totalNodeCount.toLocaleString(localeCode));
+           }
 
-               buf.push('">');
-             } else if (sysid && sysid != '.') {
-               buf.push(' SYSTEM "', sysid, '">');
-             } else {
-               var sub = node.internalSubset;
+           if (area) {
+             list.append('li').html(_t.html('info_panels.measurement.area') + ':').append('span').html(displayArea(area, _isImperial));
+           }
 
-               if (sub) {
-                 buf.push(" [", sub, "]");
-               }
+           if (length) {
+             list.append('li').html(_t.html('info_panels.measurement.' + (closed ? 'perimeter' : 'length')) + ':').append('span').html(displayLength(length, _isImperial));
+           }
 
-               buf.push(">");
-             }
+           if (typeof distance === 'number') {
+             list.append('li').html(_t.html('info_panels.measurement.distance') + ':').append('span').html(displayLength(distance, _isImperial));
+           }
 
-             return;
+           if (location) {
+             coordItem = list.append('li').html(_t.html('info_panels.measurement.location') + ':');
+             coordItem.append('span').html(dmsCoordinatePair(location));
+             coordItem.append('span').html(decimalCoordinatePair(location));
+           }
 
-           case PROCESSING_INSTRUCTION_NODE:
-             return buf.push("<?", node.target, " ", node.data, "?>");
+           if (centroid) {
+             coordItem = list.append('li').html(_t.html('info_panels.measurement.centroid') + ':');
+             coordItem.append('span').html(dmsCoordinatePair(centroid));
+             coordItem.append('span').html(decimalCoordinatePair(centroid));
+           }
 
-           case ENTITY_REFERENCE_NODE:
-             return buf.push('&', node.nodeName, ';');
-           //case ENTITY_NODE:
-           //case NOTATION_NODE:
+           if (center) {
+             coordItem = list.append('li').html(_t.html('info_panels.measurement.center') + ':');
+             coordItem.append('span').html(dmsCoordinatePair(center));
+             coordItem.append('span').html(decimalCoordinatePair(center));
+           }
 
-           default:
-             buf.push('??', node.nodeName);
+           if (length || area || typeof distance === 'number') {
+             var toggle = _isImperial ? 'imperial' : 'metric';
+             selection.append('a').html(_t.html('info_panels.measurement.' + toggle)).attr('href', '#').attr('class', 'button button-toggle-units').on('click', function (d3_event) {
+               d3_event.preventDefault();
+               _isImperial = !_isImperial;
+               selection.call(redraw);
+             });
+           }
          }
-       }
 
-       function _importNode(doc, node, deep) {
-         var node2;
+         var panel = function panel(selection) {
+           selection.call(redraw);
+           context.map().on('drawn.info-measurement', function () {
+             selection.call(redraw);
+           });
+           context.on('enter.info-measurement', function () {
+             selection.call(redraw);
+           });
+         };
 
-         switch (node.nodeType) {
-           case ELEMENT_NODE:
-             node2 = node.cloneNode(false);
-             node2.ownerDocument = doc;
-           //var attrs = node2.attributes;
-           //var len = attrs.length;
-           //for(var i=0;i<len;i++){
-           //node2.setAttributeNodeNS(importNode(doc,attrs.item(i),deep));
-           //}
+         panel.off = function () {
+           context.map().on('drawn.info-measurement', null);
+           context.on('enter.info-measurement', null);
+         };
 
-           case DOCUMENT_FRAGMENT_NODE:
-             break;
+         panel.id = 'measurement';
+         panel.label = _t.html('info_panels.measurement.title');
+         panel.key = _t('info_panels.measurement.key');
+         return panel;
+       }
 
-           case ATTRIBUTE_NODE:
-             deep = true;
-             break;
-           //case ENTITY_REFERENCE_NODE:
-           //case PROCESSING_INSTRUCTION_NODE:
-           ////case TEXT_NODE:
-           //case CDATA_SECTION_NODE:
-           //case COMMENT_NODE:
-           //  deep = false;
-           //  break;
-           //case DOCUMENT_NODE:
-           //case DOCUMENT_TYPE_NODE:
-           //cannot be imported.
-           //case ENTITY_NODE:
-           //case NOTATION_NODE:
-           //can not hit in level3
-           //default:throw e;
-         }
+       var uiInfoPanels = {
+         background: uiPanelBackground,
+         history: uiPanelHistory,
+         location: uiPanelLocation,
+         measurement: uiPanelMeasurement
+       };
 
-         if (!node2) {
-           node2 = node.cloneNode(false); //false
-         }
+       function uiInfo(context) {
+         var ids = Object.keys(uiInfoPanels);
+         var wasActive = ['measurement'];
+         var panels = {};
+         var active = {}; // create panels
 
-         node2.ownerDocument = doc;
-         node2.parentNode = null;
+         ids.forEach(function (k) {
+           if (!panels[k]) {
+             panels[k] = uiInfoPanels[k](context);
+             active[k] = false;
+           }
+         });
 
-         if (deep) {
-           var child = node.firstChild;
+         function info(selection) {
+           function redraw() {
+             var activeids = ids.filter(function (k) {
+               return active[k];
+             }).sort();
+             var containers = infoPanels.selectAll('.panel-container').data(activeids, function (k) {
+               return k;
+             });
+             containers.exit().style('opacity', 1).transition().duration(200).style('opacity', 0).on('end', function (d) {
+               select(this).call(panels[d].off).remove();
+             });
+             var enter = containers.enter().append('div').attr('class', function (d) {
+               return 'fillD2 panel-container panel-container-' + d;
+             });
+             enter.style('opacity', 0).transition().duration(200).style('opacity', 1);
+             var title = enter.append('div').attr('class', 'panel-title fillD2');
+             title.append('h3').html(function (d) {
+               return panels[d].label;
+             });
+             title.append('button').attr('class', 'close').on('click', function (d3_event, d) {
+               d3_event.stopImmediatePropagation();
+               d3_event.preventDefault();
+               info.toggle(d);
+             }).call(svgIcon('#iD-icon-close'));
+             enter.append('div').attr('class', function (d) {
+               return 'panel-content panel-content-' + d;
+             }); // redraw the panels
 
-           while (child) {
-             node2.appendChild(_importNode(doc, child, deep));
-             child = child.nextSibling;
+             infoPanels.selectAll('.panel-content').each(function (d) {
+               select(this).call(panels[d]);
+             });
            }
-         }
-
-         return node2;
-       } //
-       //var _relationMap = {firstChild:1,lastChild:1,previousSibling:1,nextSibling:1,
-       //                                      attributes:1,childNodes:1,parentNode:1,documentElement:1,doctype,};
 
+           info.toggle = function (which) {
+             var activeids = ids.filter(function (k) {
+               return active[k];
+             });
 
-       function _cloneNode(doc, node, deep) {
-         var node2 = new node.constructor();
+             if (which) {
+               // toggle one
+               active[which] = !active[which];
 
-         for (var n in node) {
-           var v = node[n];
+               if (activeids.length === 1 && activeids[0] === which) {
+                 // none active anymore
+                 wasActive = [which];
+               }
 
-           if (_typeof(v) != 'object') {
-             if (v != node2[n]) {
-               node2[n] = v;
+               context.container().select('.' + which + '-panel-toggle-item').classed('active', active[which]).select('input').property('checked', active[which]);
+             } else {
+               // toggle all
+               if (activeids.length) {
+                 wasActive = activeids;
+                 activeids.forEach(function (k) {
+                   active[k] = false;
+                 });
+               } else {
+                 wasActive.forEach(function (k) {
+                   active[k] = true;
+                 });
+               }
              }
-           }
-         }
 
-         if (node.childNodes) {
-           node2.childNodes = new NodeList();
+             redraw();
+           };
+
+           var infoPanels = selection.selectAll('.info-panels').data([0]);
+           infoPanels = infoPanels.enter().append('div').attr('class', 'info-panels').merge(infoPanels);
+           redraw();
+           context.keybinding().on(uiCmd('⌘' + _t('info_panels.key')), function (d3_event) {
+             d3_event.stopImmediatePropagation();
+             d3_event.preventDefault();
+             info.toggle();
+           });
+           ids.forEach(function (k) {
+             var key = _t('info_panels.' + k + '.key', {
+               "default": null
+             });
+             if (!key) return;
+             context.keybinding().on(uiCmd('⌘⇧' + key), function (d3_event) {
+               d3_event.stopImmediatePropagation();
+               d3_event.preventDefault();
+               info.toggle(k);
+             });
+           });
          }
 
-         node2.ownerDocument = doc;
+         return info;
+       }
 
-         switch (node2.nodeType) {
-           case ELEMENT_NODE:
-             var attrs = node.attributes;
-             var attrs2 = node2.attributes = new NamedNodeMap();
-             var len = attrs.length;
-             attrs2._ownerElement = node2;
+       function pointBox(loc, context) {
+         var rect = context.surfaceRect();
+         var point = context.curtainProjection(loc);
+         return {
+           left: point[0] + rect.left - 40,
+           top: point[1] + rect.top - 60,
+           width: 80,
+           height: 90
+         };
+       }
+       function pad(locOrBox, padding, context) {
+         var box;
 
-             for (var i = 0; i < len; i++) {
-               node2.setAttributeNode(_cloneNode(doc, attrs.item(i), true));
-             }
+         if (locOrBox instanceof Array) {
+           var rect = context.surfaceRect();
+           var point = context.curtainProjection(locOrBox);
+           box = {
+             left: point[0] + rect.left,
+             top: point[1] + rect.top
+           };
+         } else {
+           box = locOrBox;
+         }
 
-             break;
+         return {
+           left: box.left - padding,
+           top: box.top - padding,
+           width: (box.width || 0) + 2 * padding,
+           height: (box.width || 0) + 2 * padding
+         };
+       }
+       function icon(name, svgklass, useklass) {
+         return '<svg class="icon ' + (svgklass || '') + '">' + '<use xlink:href="' + name + '"' + (useklass ? ' class="' + useklass + '"' : '') + '></use></svg>';
+       }
+       var helpStringReplacements; // Returns the localized HTML element for `id` with a standardized set of icon, key, and
+       // label replacements suitable for tutorials and documentation. Optionally supplemented
+       // with custom `replacements`
 
-           case ATTRIBUTE_NODE:
-             deep = true;
+       function helpHtml(id, replacements) {
+         // only load these the first time
+         if (!helpStringReplacements) {
+           helpStringReplacements = {
+             // insert icons corresponding to various UI elements
+             point_icon: icon('#iD-icon-point', 'inline'),
+             line_icon: icon('#iD-icon-line', 'inline'),
+             area_icon: icon('#iD-icon-area', 'inline'),
+             note_icon: icon('#iD-icon-note', 'inline add-note'),
+             plus: icon('#iD-icon-plus', 'inline'),
+             minus: icon('#iD-icon-minus', 'inline'),
+             layers_icon: icon('#iD-icon-layers', 'inline'),
+             data_icon: icon('#iD-icon-data', 'inline'),
+             inspect: icon('#iD-icon-inspect', 'inline'),
+             help_icon: icon('#iD-icon-help', 'inline'),
+             undo_icon: icon(_mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-redo' : '#iD-icon-undo', 'inline'),
+             redo_icon: icon(_mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-undo' : '#iD-icon-redo', 'inline'),
+             save_icon: icon('#iD-icon-save', 'inline'),
+             // operation icons
+             circularize_icon: icon('#iD-operation-circularize', 'inline operation'),
+             continue_icon: icon('#iD-operation-continue', 'inline operation'),
+             copy_icon: icon('#iD-operation-copy', 'inline operation'),
+             delete_icon: icon('#iD-operation-delete', 'inline operation'),
+             disconnect_icon: icon('#iD-operation-disconnect', 'inline operation'),
+             downgrade_icon: icon('#iD-operation-downgrade', 'inline operation'),
+             extract_icon: icon('#iD-operation-extract', 'inline operation'),
+             merge_icon: icon('#iD-operation-merge', 'inline operation'),
+             move_icon: icon('#iD-operation-move', 'inline operation'),
+             orthogonalize_icon: icon('#iD-operation-orthogonalize', 'inline operation'),
+             paste_icon: icon('#iD-operation-paste', 'inline operation'),
+             reflect_long_icon: icon('#iD-operation-reflect-long', 'inline operation'),
+             reflect_short_icon: icon('#iD-operation-reflect-short', 'inline operation'),
+             reverse_icon: icon('#iD-operation-reverse', 'inline operation'),
+             rotate_icon: icon('#iD-operation-rotate', 'inline operation'),
+             split_icon: icon('#iD-operation-split', 'inline operation'),
+             straighten_icon: icon('#iD-operation-straighten', 'inline operation'),
+             // interaction icons
+             leftclick: icon('#iD-walkthrough-mouse-left', 'inline operation'),
+             rightclick: icon('#iD-walkthrough-mouse-right', 'inline operation'),
+             mousewheel_icon: icon('#iD-walkthrough-mousewheel', 'inline operation'),
+             tap_icon: icon('#iD-walkthrough-tap', 'inline operation'),
+             doubletap_icon: icon('#iD-walkthrough-doubletap', 'inline operation'),
+             longpress_icon: icon('#iD-walkthrough-longpress', 'inline operation'),
+             touchdrag_icon: icon('#iD-walkthrough-touchdrag', 'inline operation'),
+             pinch_icon: icon('#iD-walkthrough-pinch-apart', 'inline operation'),
+             // insert keys; may be localized and platform-dependent
+             shift: uiCmd.display('⇧'),
+             alt: uiCmd.display('⌥'),
+             "return": uiCmd.display('↵'),
+             esc: _t.html('shortcuts.key.esc'),
+             space: _t.html('shortcuts.key.space'),
+             add_note_key: _t.html('modes.add_note.key'),
+             help_key: _t.html('help.key'),
+             shortcuts_key: _t.html('shortcuts.toggle.key'),
+             // reference localized UI labels directly so that they'll always match
+             save: _t.html('save.title'),
+             undo: _t.html('undo.title'),
+             redo: _t.html('redo.title'),
+             upload: _t.html('commit.save'),
+             point: _t.html('modes.add_point.title'),
+             line: _t.html('modes.add_line.title'),
+             area: _t.html('modes.add_area.title'),
+             note: _t.html('modes.add_note.label'),
+             circularize: _t.html('operations.circularize.title'),
+             "continue": _t.html('operations.continue.title'),
+             copy: _t.html('operations.copy.title'),
+             "delete": _t.html('operations.delete.title'),
+             disconnect: _t.html('operations.disconnect.title'),
+             downgrade: _t.html('operations.downgrade.title'),
+             extract: _t.html('operations.extract.title'),
+             merge: _t.html('operations.merge.title'),
+             move: _t.html('operations.move.title'),
+             orthogonalize: _t.html('operations.orthogonalize.title'),
+             paste: _t.html('operations.paste.title'),
+             reflect_long: _t.html('operations.reflect.title.long'),
+             reflect_short: _t.html('operations.reflect.title.short'),
+             reverse: _t.html('operations.reverse.title'),
+             rotate: _t.html('operations.rotate.title'),
+             split: _t.html('operations.split.title'),
+             straighten: _t.html('operations.straighten.title'),
+             map_data: _t.html('map_data.title'),
+             osm_notes: _t.html('map_data.layers.notes.title'),
+             fields: _t.html('inspector.fields'),
+             tags: _t.html('inspector.tags'),
+             relations: _t.html('inspector.relations'),
+             new_relation: _t.html('inspector.new_relation'),
+             turn_restrictions: _t.html('_tagging.presets.fields.restrictions.label'),
+             background_settings: _t.html('background.description'),
+             imagery_offset: _t.html('background.fix_misalignment'),
+             start_the_walkthrough: _t.html('splash.walkthrough'),
+             help: _t.html('help.title'),
+             ok: _t.html('intro.ok')
+           };
          }
 
-         if (deep) {
-           var child = node.firstChild;
+         var reps;
 
-           while (child) {
-             node2.appendChild(_cloneNode(doc, child, deep));
-             child = child.nextSibling;
-           }
+         if (replacements) {
+           reps = Object.assign(replacements, helpStringReplacements);
+         } else {
+           reps = helpStringReplacements;
          }
 
-         return node2;
+         return _t.html(id, reps) // use keyboard key styling for shortcuts
+         .replace(/\`(.*?)\`/g, '<kbd>$1</kbd>');
        }
 
-       function __set__(object, key, value) {
-         object[key] = value;
-       } //do dynamic
+       function slugify(text) {
+         return text.toString().toLowerCase().replace(/\s+/g, '-') // Replace spaces with -
+         .replace(/[^\w\-]+/g, '') // Remove all non-word chars
+         .replace(/\-\-+/g, '-') // Replace multiple - with single -
+         .replace(/^-+/, '') // Trim - from start of text
+         .replace(/-+$/, ''); // Trim - from end of text
+       } // console warning for missing walkthrough names
 
 
-       try {
-         if (Object.defineProperty) {
-           var getTextContent = function getTextContent(node) {
-             switch (node.nodeType) {
-               case ELEMENT_NODE:
-               case DOCUMENT_FRAGMENT_NODE:
-                 var buf = [];
-                 node = node.firstChild;
+       var missingStrings = {};
+
+       function checkKey(key, text) {
+         if (_t(key, {
+           "default": undefined
+         }) === undefined) {
+           if (missingStrings.hasOwnProperty(key)) return; // warn once
+
+           missingStrings[key] = text;
+           var missing = key + ': ' + text;
+           if (typeof console !== 'undefined') console.log(missing); // eslint-disable-line
+         }
+       }
 
-                 while (node) {
-                   if (node.nodeType !== 7 && node.nodeType !== 8) {
-                     buf.push(getTextContent(node));
-                   }
+       function localize(obj) {
+         var key; // Assign name if entity has one..
 
-                   node = node.nextSibling;
-                 }
+         var name = obj.tags && obj.tags.name;
 
-                 return buf.join('');
+         if (name) {
+           key = 'intro.graph.name.' + slugify(name);
+           obj.tags.name = _t(key, {
+             "default": name
+           });
+           checkKey(key, name);
+         } // Assign street name if entity has one..
 
-               default:
-                 return node.nodeValue;
-             }
-           };
 
-           Object.defineProperty(LiveNodeList.prototype, 'length', {
-             get: function get() {
-               _updateLiveList(this);
+         var street = obj.tags && obj.tags['addr:street'];
 
-               return this.$$length;
-             }
+         if (street) {
+           key = 'intro.graph.name.' + slugify(street);
+           obj.tags['addr:street'] = _t(key, {
+             "default": street
            });
-           Object.defineProperty(Node.prototype, 'textContent', {
-             get: function get() {
-               return getTextContent(this);
-             },
-             set: function set(data) {
-               switch (this.nodeType) {
-                 case ELEMENT_NODE:
-                 case DOCUMENT_FRAGMENT_NODE:
-                   while (this.firstChild) {
-                     this.removeChild(this.firstChild);
-                   }
-
-                   if (data || String(data)) {
-                     this.appendChild(this.ownerDocument.createTextNode(data));
-                   }
+           checkKey(key, street); // Add address details common across walkthrough..
 
-                   break;
+           var addrTags = ['block_number', 'city', 'county', 'district', 'hamlet', 'neighbourhood', 'postcode', 'province', 'quarter', 'state', 'subdistrict', 'suburb'];
+           addrTags.forEach(function (k) {
+             var key = 'intro.graph.' + k;
+             var tag = 'addr:' + k;
+             var val = obj.tags && obj.tags[tag];
+             var str = _t(key, {
+               "default": val
+             });
 
-                 default:
-                   //TODO:
-                   this.data = data;
-                   this.value = data;
-                   this.nodeValue = data;
+             if (str) {
+               if (str.match(/^<.*>$/) !== null) {
+                 delete obj.tags[tag];
+               } else {
+                 obj.tags[tag] = str;
                }
              }
            });
-
-           __set__ = function __set__(object, key, value) {
-             //console.log(value)
-             object['$$' + key] = value;
-           };
          }
-       } catch (e) {//ie8
-       } //if(typeof require == 'function'){
-
 
-       var DOMImplementation_1 = DOMImplementation;
-       var XMLSerializer_1 = XMLSerializer$1; //}
+         return obj;
+       } // Used to detect squareness.. some duplicataion of code from actionOrthogonalize.
 
-       var dom = {
-         DOMImplementation: DOMImplementation_1,
-         XMLSerializer: XMLSerializer_1
-       };
+       function isMostlySquare(points) {
+         // note: uses 15 here instead of the 12 from actionOrthogonalize because
+         // actionOrthogonalize can actually straighten some larger angles as it iterates
+         var threshold = 15; // degrees within right or straight
 
-       var domParser = createCommonjsModule(function (module, exports) {
-         function DOMParser(options) {
-           this.options = options || {
-             locator: {}
-           };
-         }
+         var lowerBound = Math.cos((90 - threshold) * Math.PI / 180); // near right
 
-         DOMParser.prototype.parseFromString = function (source, mimeType) {
-           var options = this.options;
-           var sax = new XMLReader();
-           var domBuilder = options.domBuilder || new DOMHandler(); //contentHandler and LexicalHandler
+         var upperBound = Math.cos(threshold * Math.PI / 180); // near straight
 
-           var errorHandler = options.errorHandler;
-           var locator = options.locator;
-           var defaultNSMap = options.xmlns || {};
-           var entityMap = {
-             'lt': '<',
-             'gt': '>',
-             'amp': '&',
-             'quot': '"',
-             'apos': "'"
-           };
+         for (var i = 0; i < points.length; i++) {
+           var a = points[(i - 1 + points.length) % points.length];
+           var origin = points[i];
+           var b = points[(i + 1) % points.length];
+           var dotp = geoVecNormalizedDot(a, b, origin);
+           var mag = Math.abs(dotp);
 
-           if (locator) {
-             domBuilder.setDocumentLocator(locator);
+           if (mag > lowerBound && mag < upperBound) {
+             return false;
            }
+         }
 
-           sax.errorHandler = buildErrorHandler(errorHandler, domBuilder, locator);
-           sax.domBuilder = options.domBuilder || domBuilder;
-
-           if (/\/x?html?$/.test(mimeType)) {
-             entityMap.nbsp = '\xa0';
-             entityMap.copy = '\xa9';
-             defaultNSMap[''] = 'http://www.w3.org/1999/xhtml';
-           }
+         return true;
+       }
+       function selectMenuItem(context, operation) {
+         return context.container().select('.edit-menu .edit-menu-item-' + operation);
+       }
+       function transitionTime(point1, point2) {
+         var distance = geoSphericalDistance(point1, point2);
 
-           defaultNSMap.xml = defaultNSMap.xml || 'http://www.w3.org/XML/1998/namespace';
+         if (distance === 0) {
+           return 0;
+         } else if (distance < 80) {
+           return 500;
+         } else {
+           return 1000;
+         }
+       }
 
-           if (source) {
-             sax.parse(source, defaultNSMap, entityMap);
-           } else {
-             sax.errorHandler.error("invalid doc source");
-           }
+       // hide class, which sets display=none, and a d3 transition for opacity.
+       // this will cause blinking when called repeatedly, so check that the
+       // value actually changes between calls.
 
-           return domBuilder.doc;
+       function uiToggle(show, callback) {
+         return function (selection) {
+           selection.style('opacity', show ? 0 : 1).classed('hide', false).transition().style('opacity', show ? 1 : 0).on('end', function () {
+             select(this).classed('hide', !show).style('opacity', null);
+             if (callback) callback.apply(this);
+           });
          };
+       }
 
-         function buildErrorHandler(errorImpl, domBuilder, locator) {
-           if (!errorImpl) {
-             if (domBuilder instanceof DOMHandler) {
-               return domBuilder;
-             }
-
-             errorImpl = domBuilder;
-           }
-
-           var errorHandler = {};
-           var isCallback = errorImpl instanceof Function;
-           locator = locator || {};
-
-           function build(key) {
-             var fn = errorImpl[key];
+       function uiCurtain(containerNode) {
+         var surface = select(null),
+             tooltip = select(null),
+             darkness = select(null);
 
-             if (!fn && isCallback) {
-               fn = errorImpl.length == 2 ? function (msg) {
-                 errorImpl(key, msg);
-               } : errorImpl;
-             }
+         function curtain(selection) {
+           surface = selection.append('svg').attr('class', 'curtain').style('top', 0).style('left', 0);
+           darkness = surface.append('path').attr('x', 0).attr('y', 0).attr('class', 'curtain-darkness');
+           select(window).on('resize.curtain', resize);
+           tooltip = selection.append('div').attr('class', 'tooltip');
+           tooltip.append('div').attr('class', 'popover-arrow');
+           tooltip.append('div').attr('class', 'popover-inner');
+           resize();
 
-             errorHandler[key] = fn && function (msg) {
-               fn('[xmldom ' + key + ']\t' + msg + _locator(locator));
-             } || function () {};
+           function resize() {
+             surface.attr('width', containerNode.clientWidth).attr('height', containerNode.clientHeight);
+             curtain.cut(darkness.datum());
            }
-
-           build('warning');
-           build('error');
-           build('fatalError');
-           return errorHandler;
-         } //console.log('#\n\n\n\n\n\n\n####')
-
+         }
          /**
-          * +ContentHandler+ErrorHandler
-          * +LexicalHandler+EntityResolver2
-          * -DeclHandler-DTDHandler 
-          * 
-          * DefaultHandler:EntityResolver, DTDHandler, ContentHandler, ErrorHandler
-          * DefaultHandler2:DefaultHandler,LexicalHandler, DeclHandler, EntityResolver2
-          * @link http://www.saxproject.org/apidoc/org/xml/sax/helpers/DefaultHandler.html
+          * Reveal cuts the curtain to highlight the given box,
+          * and shows a tooltip with instructions next to the box.
+          *
+          * @param  {String|ClientRect} [box]   box used to cut the curtain
+          * @param  {String}    [text]          text for a tooltip
+          * @param  {Object}    [options]
+          * @param  {string}    [options.tooltipClass]    optional class to add to the tooltip
+          * @param  {integer}   [options.duration]        transition time in milliseconds
+          * @param  {string}    [options.buttonText]      if set, create a button with this text label
+          * @param  {function}  [options.buttonCallback]  if set, the callback for the button
+          * @param  {function}  [options.padding]         extra margin in px to put around bbox
+          * @param  {String|ClientRect} [options.tooltipBox]  box for tooltip position, if different from box for the curtain
           */
 
 
-         function DOMHandler() {
-           this.cdata = false;
-         }
+         curtain.reveal = function (box, html, options) {
+           options = options || {};
 
-         function position(locator, node) {
-           node.lineNumber = locator.lineNumber;
-           node.columnNumber = locator.columnNumber;
-         }
-         /**
-          * @see org.xml.sax.ContentHandler#startDocument
-          * @link http://www.saxproject.org/apidoc/org/xml/sax/ContentHandler.html
-          */
+           if (typeof box === 'string') {
+             box = select(box).node();
+           }
 
+           if (box && box.getBoundingClientRect) {
+             box = copyBox(box.getBoundingClientRect());
+             var containerRect = containerNode.getBoundingClientRect();
+             box.top -= containerRect.top;
+             box.left -= containerRect.left;
+           }
 
-         DOMHandler.prototype = {
-           startDocument: function startDocument() {
-             this.doc = new DOMImplementation().createDocument(null, null, null);
+           if (box && options.padding) {
+             box.top -= options.padding;
+             box.left -= options.padding;
+             box.bottom += options.padding;
+             box.right += options.padding;
+             box.height += options.padding * 2;
+             box.width += options.padding * 2;
+           }
 
-             if (this.locator) {
-               this.doc.documentURI = this.locator.systemId;
+           var tooltipBox;
+
+           if (options.tooltipBox) {
+             tooltipBox = options.tooltipBox;
+
+             if (typeof tooltipBox === 'string') {
+               tooltipBox = select(tooltipBox).node();
              }
-           },
-           startElement: function startElement(namespaceURI, localName, qName, attrs) {
-             var doc = this.doc;
-             var el = doc.createElementNS(namespaceURI, qName || localName);
-             var len = attrs.length;
-             appendElement(this, el);
-             this.currentElement = el;
-             this.locator && position(this.locator, el);
 
-             for (var i = 0; i < len; i++) {
-               var namespaceURI = attrs.getURI(i);
-               var value = attrs.getValue(i);
-               var qName = attrs.getQName(i);
-               var attr = doc.createAttributeNS(namespaceURI, qName);
-               this.locator && position(attrs.getLocator(i), attr);
-               attr.value = attr.nodeValue = value;
-               el.setAttributeNode(attr);
+             if (tooltipBox && tooltipBox.getBoundingClientRect) {
+               tooltipBox = copyBox(tooltipBox.getBoundingClientRect());
              }
-           },
-           endElement: function endElement(namespaceURI, localName, qName) {
-             var current = this.currentElement;
-             var tagName = current.tagName;
-             this.currentElement = current.parentNode;
-           },
-           startPrefixMapping: function startPrefixMapping(prefix, uri) {},
-           endPrefixMapping: function endPrefixMapping(prefix) {},
-           processingInstruction: function processingInstruction(target, data) {
-             var ins = this.doc.createProcessingInstruction(target, data);
-             this.locator && position(this.locator, ins);
-             appendElement(this, ins);
-           },
-           ignorableWhitespace: function ignorableWhitespace(ch, start, length) {},
-           characters: function characters(chars, start, length) {
-             chars = _toString.apply(this, arguments); //console.log(chars)
+           } else {
+             tooltipBox = box;
+           }
 
-             if (chars) {
-               if (this.cdata) {
-                 var charNode = this.doc.createCDATASection(chars);
+           if (tooltipBox && html) {
+             if (html.indexOf('**') !== -1) {
+               if (html.indexOf('<span') === 0) {
+                 html = html.replace(/^(<span.*?>)(.+?)(\*\*)/, '$1<span>$2</span>$3');
                } else {
-                 var charNode = this.doc.createTextNode(chars);
-               }
-
-               if (this.currentElement) {
-                 this.currentElement.appendChild(charNode);
-               } else if (/^\s*$/.test(chars)) {
-                 this.doc.appendChild(charNode); //process xml
-               }
+                 html = html.replace(/^(.+?)(\*\*)/, '<span>$1</span>$2');
+               } // pseudo markdown bold text for the instruction section..
 
-               this.locator && position(this.locator, charNode);
-             }
-           },
-           skippedEntity: function skippedEntity(name) {},
-           endDocument: function endDocument() {
-             this.doc.normalize();
-           },
-           setDocumentLocator: function setDocumentLocator(locator) {
-             if (this.locator = locator) {
-               // && !('lineNumber' in locator)){
-               locator.lineNumber = 0;
-             }
-           },
-           //LexicalHandler
-           comment: function comment(chars, start, length) {
-             chars = _toString.apply(this, arguments);
-             var comm = this.doc.createComment(chars);
-             this.locator && position(this.locator, comm);
-             appendElement(this, comm);
-           },
-           startCDATA: function startCDATA() {
-             //used in characters() methods
-             this.cdata = true;
-           },
-           endCDATA: function endCDATA() {
-             this.cdata = false;
-           },
-           startDTD: function startDTD(name, publicId, systemId) {
-             var impl = this.doc.implementation;
 
-             if (impl && impl.createDocumentType) {
-               var dt = impl.createDocumentType(name, publicId, systemId);
-               this.locator && position(this.locator, dt);
-               appendElement(this, dt);
+               html = html.replace(/\*\*(.*?)\*\*/g, '<span class="instruction">$1</span>');
              }
-           },
 
-           /**
-            * @see org.xml.sax.ErrorHandler
-            * @link http://www.saxproject.org/apidoc/org/xml/sax/ErrorHandler.html
-            */
-           warning: function warning(error) {
-             console.warn('[xmldom warning]\t' + error, _locator(this.locator));
-           },
-           error: function error(_error) {
-             console.error('[xmldom error]\t' + _error, _locator(this.locator));
-           },
-           fatalError: function fatalError(error) {
-             console.error('[xmldom fatalError]\t' + error, _locator(this.locator));
-             throw error;
-           }
-         };
+             html = html.replace(/\*(.*?)\*/g, '<em>$1</em>'); // emphasis
 
-         function _locator(l) {
-           if (l) {
-             return '\n@' + (l.systemId || '') + '#[line:' + l.lineNumber + ',col:' + l.columnNumber + ']';
-           }
-         }
+             html = html.replace(/\{br\}/g, '<br/><br/>'); // linebreak
 
-         function _toString(chars, start, length) {
-           if (typeof chars == 'string') {
-             return chars.substr(start, length);
-           } else {
-             //java sax connect width xmldom on rhino(what about: "? && !(chars instanceof String)")
-             if (chars.length >= start + length || start) {
-               return new java.lang.String(chars, start, length) + '';
+             if (options.buttonText && options.buttonCallback) {
+               html += '<div class="button-section">' + '<button href="#" class="button action">' + options.buttonText + '</button></div>';
              }
 
-             return chars;
-           }
-         }
-         /*
-          * @link http://www.saxproject.org/apidoc/org/xml/sax/ext/LexicalHandler.html
-          * used method of org.xml.sax.ext.LexicalHandler:
-          *  #comment(chars, start, length)
-          *  #startCDATA()
-          *  #endCDATA()
-          *  #startDTD(name, publicId, systemId)
-          *
-          *
-          * IGNORED method of org.xml.sax.ext.LexicalHandler:
-          *  #endDTD()
-          *  #startEntity(name)
-          *  #endEntity(name)
-          *
-          *
-          * @link http://www.saxproject.org/apidoc/org/xml/sax/ext/DeclHandler.html
-          * IGNORED method of org.xml.sax.ext.DeclHandler
-          *    #attributeDecl(eName, aName, type, mode, value)
-          *  #elementDecl(name, model)
-          *  #externalEntityDecl(name, publicId, systemId)
-          *  #internalEntityDecl(name, value)
-          * @link http://www.saxproject.org/apidoc/org/xml/sax/ext/EntityResolver2.html
-          * IGNORED method of org.xml.sax.EntityResolver2
-          *  #resolveEntity(String name,String publicId,String baseURI,String systemId)
-          *  #resolveEntity(publicId, systemId)
-          *  #getExternalSubset(name, baseURI)
-          * @link http://www.saxproject.org/apidoc/org/xml/sax/DTDHandler.html
-          * IGNORED method of org.xml.sax.DTDHandler
-          *  #notationDecl(name, publicId, systemId) {};
-          *  #unparsedEntityDecl(name, publicId, systemId, notationName) {};
-          */
+             var classes = 'curtain-tooltip popover tooltip arrowed in ' + (options.tooltipClass || '');
+             tooltip.classed(classes, true).selectAll('.popover-inner').html(html);
 
+             if (options.buttonText && options.buttonCallback) {
+               var button = tooltip.selectAll('.button-section .button.action');
+               button.on('click', function (d3_event) {
+                 d3_event.preventDefault();
+                 options.buttonCallback();
+               });
+             }
 
-         "endDTD,startEntity,endEntity,attributeDecl,elementDecl,externalEntityDecl,internalEntityDecl,resolveEntity,getExternalSubset,notationDecl,unparsedEntityDecl".replace(/\w+/g, function (key) {
-           DOMHandler.prototype[key] = function () {
-             return null;
-           };
-         });
-         /* Private static helpers treated below as private instance methods, so don't need to add these to the public API; we might use a Relator to also get rid of non-standard public properties */
+             var tip = copyBox(tooltip.node().getBoundingClientRect()),
+                 w = containerNode.clientWidth,
+                 h = containerNode.clientHeight,
+                 tooltipWidth = 200,
+                 tooltipArrow = 5,
+                 side,
+                 pos; // hack: this will have bottom placement,
+             // so need to reserve extra space for the tooltip illustration.
 
-         function appendElement(hander, node) {
-           if (!hander.currentElement) {
-             hander.doc.appendChild(node);
-           } else {
-             hander.currentElement.appendChild(node);
-           }
-         } //appendChild and setAttributeNS are preformance key
-         //if(typeof require == 'function'){
+             if (options.tooltipClass === 'intro-mouse') {
+               tip.height += 80;
+             } // trim box dimensions to just the portion that fits in the container..
 
 
-         var XMLReader = sax.XMLReader;
-         var DOMImplementation = exports.DOMImplementation = dom.DOMImplementation;
-         exports.XMLSerializer = dom.XMLSerializer;
-         exports.DOMParser = DOMParser; //}
-       });
+             if (tooltipBox.top + tooltipBox.height > h) {
+               tooltipBox.height -= tooltipBox.top + tooltipBox.height - h;
+             }
 
-       var togeojson = createCommonjsModule(function (module, exports) {
-         var toGeoJSON = function () {
+             if (tooltipBox.left + tooltipBox.width > w) {
+               tooltipBox.width -= tooltipBox.left + tooltipBox.width - w;
+             } // determine tooltip placement..
 
-           var removeSpace = /\s*/g,
-               trimSpace = /^\s*|\s*$/g,
-               splitSpace = /\s+/; // generate a short, numeric hash of a string
 
-           function okhash(x) {
-             if (!x || !x.length) return 0;
+             if (tooltipBox.top + tooltipBox.height < 100) {
+               // tooltip below box..
+               side = 'bottom';
+               pos = [tooltipBox.left + tooltipBox.width / 2 - tip.width / 2, tooltipBox.top + tooltipBox.height];
+             } else if (tooltipBox.top > h - 140) {
+               // tooltip above box..
+               side = 'top';
+               pos = [tooltipBox.left + tooltipBox.width / 2 - tip.width / 2, tooltipBox.top - tip.height];
+             } else {
+               // tooltip to the side of the tooltipBox..
+               var tipY = tooltipBox.top + tooltipBox.height / 2 - tip.height / 2;
 
-             for (var i = 0, h = 0; i < x.length; i++) {
-               h = (h << 5) - h + x.charCodeAt(i) | 0;
+               if (_mainLocalizer.textDirection() === 'rtl') {
+                 if (tooltipBox.left - tooltipWidth - tooltipArrow < 70) {
+                   side = 'right';
+                   pos = [tooltipBox.left + tooltipBox.width + tooltipArrow, tipY];
+                 } else {
+                   side = 'left';
+                   pos = [tooltipBox.left - tooltipWidth - tooltipArrow, tipY];
+                 }
+               } else {
+                 if (tooltipBox.left + tooltipBox.width + tooltipArrow + tooltipWidth > w - 70) {
+                   side = 'left';
+                   pos = [tooltipBox.left - tooltipWidth - tooltipArrow, tipY];
+                 } else {
+                   side = 'right';
+                   pos = [tooltipBox.left + tooltipBox.width + tooltipArrow, tipY];
+                 }
+               }
              }
 
-             return h;
-           } // all Y children of X
+             if (options.duration !== 0 || !tooltip.classed(side)) {
+               tooltip.call(uiToggle(true));
+             }
+
+             tooltip.style('top', pos[1] + 'px').style('left', pos[0] + 'px').attr('class', classes + ' ' + side); // shift popover-inner if it is very close to the top or bottom edge
+             // (doesn't affect the placement of the popover-arrow)
 
+             var shiftY = 0;
 
-           function get(x, y) {
-             return x.getElementsByTagName(y);
-           }
+             if (side === 'left' || side === 'right') {
+               if (pos[1] < 60) {
+                 shiftY = 60 - pos[1];
+               } else if (pos[1] + tip.height > h - 100) {
+                 shiftY = h - pos[1] - tip.height - 100;
+               }
+             }
 
-           function attr(x, y) {
-             return x.getAttribute(y);
+             tooltip.selectAll('.popover-inner').style('top', shiftY + 'px');
+           } else {
+             tooltip.classed('in', false).call(uiToggle(false));
            }
 
-           function attrf(x, y) {
-             return parseFloat(attr(x, y));
-           } // one Y child of X, if any, otherwise null
-
+           curtain.cut(box, options.duration);
+           return tooltip;
+         };
 
-           function get1(x, y) {
-             var n = get(x, y);
-             return n.length ? n[0] : null;
-           } // https://developer.mozilla.org/en-US/docs/Web/API/Node.normalize
+         curtain.cut = function (datum, duration) {
+           darkness.datum(datum).interrupt();
+           var selection;
 
+           if (duration === 0) {
+             selection = darkness;
+           } else {
+             selection = darkness.transition().duration(duration || 600).ease(linear$1);
+           }
 
-           function norm(el) {
-             if (el.normalize) {
-               el.normalize();
-             }
+           selection.attr('d', function (d) {
+             var containerWidth = containerNode.clientWidth;
+             var containerHeight = containerNode.clientHeight;
+             var string = 'M 0,0 L 0,' + containerHeight + ' L ' + containerWidth + ',' + containerHeight + 'L' + containerWidth + ',0 Z';
+             if (!d) return string;
+             return string + 'M' + d.left + ',' + d.top + 'L' + d.left + ',' + (d.top + d.height) + 'L' + (d.left + d.width) + ',' + (d.top + d.height) + 'L' + (d.left + d.width) + ',' + d.top + 'Z';
+           });
+         };
 
-             return el;
-           } // cast array x into numbers
+         curtain.remove = function () {
+           surface.remove();
+           tooltip.remove();
+           select(window).on('resize.curtain', null);
+         }; // ClientRects are immutable, so copy them to an object,
+         // in case we need to trim the height/width.
 
 
-           function numarray(x) {
-             for (var j = 0, o = []; j < x.length; j++) {
-               o[j] = parseFloat(x[j]);
-             }
+         function copyBox(src) {
+           return {
+             top: src.top,
+             right: src.right,
+             bottom: src.bottom,
+             left: src.left,
+             width: src.width,
+             height: src.height
+           };
+         }
 
-             return o;
-           } // get the content of a text node, if any
+         return curtain;
+       }
 
+       function uiIntroWelcome(context, reveal) {
+         var dispatch = dispatch$8('done');
+         var chapter = {
+           title: 'intro.welcome.title'
+         };
 
-           function nodeVal(x) {
-             if (x) {
-               norm(x);
-             }
+         function welcome() {
+           context.map().centerZoom([-85.63591, 41.94285], 19);
+           reveal('.intro-nav-wrap .chapter-welcome', helpHtml('intro.welcome.welcome'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: practice
+           });
+         }
 
-             return x && x.textContent || '';
-           } // get the contents of multiple text nodes, if present
+         function practice() {
+           reveal('.intro-nav-wrap .chapter-welcome', helpHtml('intro.welcome.practice'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: words
+           });
+         }
 
+         function words() {
+           reveal('.intro-nav-wrap .chapter-welcome', helpHtml('intro.welcome.words'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: chapters
+           });
+         }
 
-           function getMulti(x, ys) {
-             var o = {},
-                 n,
-                 k;
+         function chapters() {
+           dispatch.call('done');
+           reveal('.intro-nav-wrap .chapter-navigation', helpHtml('intro.welcome.chapters', {
+             next: _t('intro.navigation.title')
+           }));
+         }
 
-             for (k = 0; k < ys.length; k++) {
-               n = get1(x, ys[k]);
-               if (n) o[ys[k]] = nodeVal(n);
-             }
+         chapter.enter = function () {
+           welcome();
+         };
 
-             return o;
-           } // add properties of Y to X, overwriting if present in both
+         chapter.exit = function () {
+           context.container().select('.curtain-tooltip.intro-mouse').selectAll('.counter').remove();
+         };
 
+         chapter.restart = function () {
+           chapter.exit();
+           chapter.enter();
+         };
 
-           function extend(x, y) {
-             for (var k in y) {
-               x[k] = y[k];
-             }
-           } // get one coordinate from a coordinate array, if any
+         return utilRebind(chapter, dispatch, 'on');
+       }
 
+       function uiIntroNavigation(context, reveal) {
+         var dispatch = dispatch$8('done');
+         var timeouts = [];
+         var hallId = 'n2061';
+         var townHall = [-85.63591, 41.94285];
+         var springStreetId = 'w397';
+         var springStreetEndId = 'n1834';
+         var springStreet = [-85.63582, 41.94255];
+         var onewayField = _mainPresetIndex.field('oneway');
+         var maxspeedField = _mainPresetIndex.field('maxspeed');
+         var chapter = {
+           title: 'intro.navigation.title'
+         };
 
-           function coord1(v) {
-             return numarray(v.replace(removeSpace, '').split(','));
-           } // get all coordinates from a coordinate array as [[],[]]
+         function timeout(f, t) {
+           timeouts.push(window.setTimeout(f, t));
+         }
 
+         function eventCancel(d3_event) {
+           d3_event.stopPropagation();
+           d3_event.preventDefault();
+         }
 
-           function coord(v) {
-             var coords = v.replace(trimSpace, '').split(splitSpace),
-                 o = [];
+         function isTownHallSelected() {
+           var ids = context.selectedIDs();
+           return ids.length === 1 && ids[0] === hallId;
+         }
 
-             for (var i = 0; i < coords.length; i++) {
-               o.push(coord1(coords[i]));
-             }
+         function dragMap() {
+           context.enter(modeBrowse(context));
+           context.history().reset('initial');
+           var msec = transitionTime(townHall, context.map().center());
 
-             return o;
+           if (msec) {
+             reveal(null, null, {
+               duration: 0
+             });
            }
 
-           function coordPair(x) {
-             var ll = [attrf(x, 'lon'), attrf(x, 'lat')],
-                 ele = get1(x, 'ele'),
-                 // handle namespaced attribute in browser
-             heartRate = get1(x, 'gpxtpx:hr') || get1(x, 'hr'),
-                 time = get1(x, 'time'),
-                 e;
-
-             if (ele) {
-               e = parseFloat(nodeVal(ele));
+           context.map().centerZoomEase(townHall, 19, msec);
+           timeout(function () {
+             var centerStart = context.map().center();
+             var textId = context.lastPointerType() === 'mouse' ? 'drag' : 'drag_touch';
+             var dragString = helpHtml('intro.navigation.map_info') + '{br}' + helpHtml('intro.navigation.' + textId);
+             reveal('.surface', dragString);
+             context.map().on('drawn.intro', function () {
+               reveal('.surface', dragString, {
+                 duration: 0
+               });
+             });
+             context.map().on('move.intro', function () {
+               var centerNow = context.map().center();
 
-               if (!isNaN(e)) {
-                 ll.push(e);
+               if (centerStart[0] !== centerNow[0] || centerStart[1] !== centerNow[1]) {
+                 context.map().on('move.intro', null);
+                 timeout(function () {
+                   continueTo(zoomMap);
+                 }, 3000);
                }
-             }
-
-             return {
-               coordinates: ll,
-               time: time ? nodeVal(time) : null,
-               heartRate: heartRate ? parseFloat(nodeVal(heartRate)) : null
-             };
-           } // create a new feature collection parent object
-
+             });
+           }, msec + 100);
 
-           function fc() {
-             return {
-               type: 'FeatureCollection',
-               features: []
-             };
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             nextStep();
            }
+         }
 
-           var serializer;
+         function zoomMap() {
+           var zoomStart = context.map().zoom();
+           var textId = context.lastPointerType() === 'mouse' ? 'zoom' : 'zoom_touch';
+           var zoomString = helpHtml('intro.navigation.' + textId);
+           reveal('.surface', zoomString);
+           context.map().on('drawn.intro', function () {
+             reveal('.surface', zoomString, {
+               duration: 0
+             });
+           });
+           context.map().on('move.intro', function () {
+             if (context.map().zoom() !== zoomStart) {
+               context.map().on('move.intro', null);
+               timeout(function () {
+                 continueTo(features);
+               }, 3000);
+             }
+           });
 
-           if (typeof XMLSerializer !== 'undefined') {
-             /* istanbul ignore next */
-             serializer = new XMLSerializer(); // only require xmldom in a node environment
-           } else if ( (typeof process === "undefined" ? "undefined" : _typeof(process)) === 'object' && !process.browser) {
-             serializer = new domParser.XMLSerializer();
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             nextStep();
            }
+         }
 
-           function xml2str(str) {
-             // IE9 will create a new XMLSerializer but it'll crash immediately.
-             // This line is ignored because we don't run coverage tests in IE9
+         function features() {
+           var onClick = function onClick() {
+             continueTo(pointsLinesAreas);
+           };
 
-             /* istanbul ignore next */
-             if (str.xml !== undefined) return str.xml;
-             return serializer.serializeToString(str);
+           reveal('.surface', helpHtml('intro.navigation.features'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: onClick
+           });
+           context.map().on('drawn.intro', function () {
+             reveal('.surface', helpHtml('intro.navigation.features'), {
+               duration: 0,
+               buttonText: _t.html('intro.ok'),
+               buttonCallback: onClick
+             });
+           });
+
+           function continueTo(nextStep) {
+             context.map().on('drawn.intro', null);
+             nextStep();
            }
+         }
 
-           var t = {
-             kml: function kml(doc) {
-               var gj = fc(),
-                   // styleindex keeps track of hashed styles in order to match features
-               styleIndex = {},
-                   styleByHash = {},
-                   // stylemapindex keeps track of style maps to expose in properties
-               styleMapIndex = {},
-                   // atomic geospatial types supported by KML - MultiGeometry is
-               // handled separately
-               geotypes = ['Polygon', 'LineString', 'Point', 'Track', 'gx:Track'],
-                   // all root placemarks in the file
-               placemarks = get(doc, 'Placemark'),
-                   styles = get(doc, 'Style'),
-                   styleMaps = get(doc, 'StyleMap');
+         function pointsLinesAreas() {
+           var onClick = function onClick() {
+             continueTo(nodesWays);
+           };
 
-               for (var k = 0; k < styles.length; k++) {
-                 var hash = okhash(xml2str(styles[k])).toString(16);
-                 styleIndex['#' + attr(styles[k], 'id')] = hash;
-                 styleByHash[hash] = styles[k];
-               }
+           reveal('.surface', helpHtml('intro.navigation.points_lines_areas'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: onClick
+           });
+           context.map().on('drawn.intro', function () {
+             reveal('.surface', helpHtml('intro.navigation.points_lines_areas'), {
+               duration: 0,
+               buttonText: _t.html('intro.ok'),
+               buttonCallback: onClick
+             });
+           });
 
-               for (var l = 0; l < styleMaps.length; l++) {
-                 styleIndex['#' + attr(styleMaps[l], 'id')] = okhash(xml2str(styleMaps[l])).toString(16);
-                 var pairs = get(styleMaps[l], 'Pair');
-                 var pairsMap = {};
+           function continueTo(nextStep) {
+             context.map().on('drawn.intro', null);
+             nextStep();
+           }
+         }
 
-                 for (var m = 0; m < pairs.length; m++) {
-                   pairsMap[nodeVal(get1(pairs[m], 'key'))] = nodeVal(get1(pairs[m], 'styleUrl'));
-                 }
+         function nodesWays() {
+           var onClick = function onClick() {
+             continueTo(clickTownHall);
+           };
 
-                 styleMapIndex['#' + attr(styleMaps[l], 'id')] = pairsMap;
-               }
+           reveal('.surface', helpHtml('intro.navigation.nodes_ways'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: onClick
+           });
+           context.map().on('drawn.intro', function () {
+             reveal('.surface', helpHtml('intro.navigation.nodes_ways'), {
+               duration: 0,
+               buttonText: _t.html('intro.ok'),
+               buttonCallback: onClick
+             });
+           });
 
-               for (var j = 0; j < placemarks.length; j++) {
-                 gj.features = gj.features.concat(getPlacemark(placemarks[j]));
-               }
+           function continueTo(nextStep) {
+             context.map().on('drawn.intro', null);
+             nextStep();
+           }
+         }
 
-               function kmlColor(v) {
-                 var color, opacity;
-                 v = v || '';
+         function clickTownHall() {
+           context.enter(modeBrowse(context));
+           context.history().reset('initial');
+           var entity = context.hasEntity(hallId);
+           if (!entity) return;
+           reveal(null, null, {
+             duration: 0
+           });
+           context.map().centerZoomEase(entity.loc, 19, 500);
+           timeout(function () {
+             var entity = context.hasEntity(hallId);
+             if (!entity) return;
+             var box = pointBox(entity.loc, context);
+             var textId = context.lastPointerType() === 'mouse' ? 'click_townhall' : 'tap_townhall';
+             reveal(box, helpHtml('intro.navigation.' + textId));
+             context.map().on('move.intro drawn.intro', function () {
+               var entity = context.hasEntity(hallId);
+               if (!entity) return;
+               var box = pointBox(entity.loc, context);
+               reveal(box, helpHtml('intro.navigation.' + textId), {
+                 duration: 0
+               });
+             });
+             context.on('enter.intro', function () {
+               if (isTownHallSelected()) continueTo(selectedTownHall);
+             });
+           }, 550); // after centerZoomEase
 
-                 if (v.substr(0, 1) === '#') {
-                   v = v.substr(1);
-                 }
+           context.history().on('change.intro', function () {
+             if (!context.hasEntity(hallId)) {
+               continueTo(clickTownHall);
+             }
+           });
 
-                 if (v.length === 6 || v.length === 3) {
-                   color = v;
-                 }
+           function continueTo(nextStep) {
+             context.on('enter.intro', null);
+             context.map().on('move.intro drawn.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
+           }
+         }
 
-                 if (v.length === 8) {
-                   opacity = parseInt(v.substr(0, 2), 16) / 255;
-                   color = '#' + v.substr(6, 2) + v.substr(4, 2) + v.substr(2, 2);
-                 }
+         function selectedTownHall() {
+           if (!isTownHallSelected()) return clickTownHall();
+           var entity = context.hasEntity(hallId);
+           if (!entity) return clickTownHall();
+           var box = pointBox(entity.loc, context);
 
-                 return [color, isNaN(opacity) ? undefined : opacity];
-               }
+           var onClick = function onClick() {
+             continueTo(editorTownHall);
+           };
 
-               function gxCoord(v) {
-                 return numarray(v.split(' '));
-               }
+           reveal(box, helpHtml('intro.navigation.selected_townhall'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: onClick
+           });
+           context.map().on('move.intro drawn.intro', function () {
+             var entity = context.hasEntity(hallId);
+             if (!entity) return;
+             var box = pointBox(entity.loc, context);
+             reveal(box, helpHtml('intro.navigation.selected_townhall'), {
+               duration: 0,
+               buttonText: _t.html('intro.ok'),
+               buttonCallback: onClick
+             });
+           });
+           context.history().on('change.intro', function () {
+             if (!context.hasEntity(hallId)) {
+               continueTo(clickTownHall);
+             }
+           });
 
-               function gxCoords(root) {
-                 var elems = get(root, 'coord'),
-                     coords = [],
-                     times = [];
-                 if (elems.length === 0) elems = get(root, 'gx:coord');
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
+           }
+         }
 
-                 for (var i = 0; i < elems.length; i++) {
-                   coords.push(gxCoord(nodeVal(elems[i])));
-                 }
+         function editorTownHall() {
+           if (!isTownHallSelected()) return clickTownHall(); // disallow scrolling
 
-                 var timeElems = get(root, 'when');
+           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
 
-                 for (var j = 0; j < timeElems.length; j++) {
-                   times.push(nodeVal(timeElems[j]));
-                 }
+           var onClick = function onClick() {
+             continueTo(presetTownHall);
+           };
 
-                 return {
-                   coords: coords,
-                   times: times
-                 };
-               }
+           reveal('.entity-editor-pane', helpHtml('intro.navigation.editor_townhall'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: onClick
+           });
+           context.on('exit.intro', function () {
+             continueTo(clickTownHall);
+           });
+           context.history().on('change.intro', function () {
+             if (!context.hasEntity(hallId)) {
+               continueTo(clickTownHall);
+             }
+           });
 
-               function getGeometry(root) {
-                 var geomNode,
-                     geomNodes,
-                     i,
-                     j,
-                     k,
-                     geoms = [],
-                     coordTimes = [];
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             context.history().on('change.intro', null);
+             context.container().select('.inspector-wrap').on('wheel.intro', null);
+             nextStep();
+           }
+         }
 
-                 if (get1(root, 'MultiGeometry')) {
-                   return getGeometry(get1(root, 'MultiGeometry'));
-                 }
+         function presetTownHall() {
+           if (!isTownHallSelected()) return clickTownHall(); // reset pane, in case user happened to change it..
 
-                 if (get1(root, 'MultiTrack')) {
-                   return getGeometry(get1(root, 'MultiTrack'));
-                 }
+           context.container().select('.inspector-wrap .panewrap').style('right', '0%'); // disallow scrolling
 
-                 if (get1(root, 'gx:MultiTrack')) {
-                   return getGeometry(get1(root, 'gx:MultiTrack'));
-                 }
+           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel); // preset match, in case the user happened to change it.
 
-                 for (i = 0; i < geotypes.length; i++) {
-                   geomNodes = get(root, geotypes[i]);
+           var entity = context.entity(context.selectedIDs()[0]);
+           var preset = _mainPresetIndex.match(entity, context.graph());
 
-                   if (geomNodes) {
-                     for (j = 0; j < geomNodes.length; j++) {
-                       geomNode = geomNodes[j];
+           var onClick = function onClick() {
+             continueTo(fieldsTownHall);
+           };
+
+           reveal('.entity-editor-pane .section-feature-type', helpHtml('intro.navigation.preset_townhall', {
+             preset: preset.name()
+           }), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: onClick
+           });
+           context.on('exit.intro', function () {
+             continueTo(clickTownHall);
+           });
+           context.history().on('change.intro', function () {
+             if (!context.hasEntity(hallId)) {
+               continueTo(clickTownHall);
+             }
+           });
 
-                       if (geotypes[i] === 'Point') {
-                         geoms.push({
-                           type: 'Point',
-                           coordinates: coord1(nodeVal(get1(geomNode, 'coordinates')))
-                         });
-                       } else if (geotypes[i] === 'LineString') {
-                         geoms.push({
-                           type: 'LineString',
-                           coordinates: coord(nodeVal(get1(geomNode, 'coordinates')))
-                         });
-                       } else if (geotypes[i] === 'Polygon') {
-                         var rings = get(geomNode, 'LinearRing'),
-                             coords = [];
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             context.history().on('change.intro', null);
+             context.container().select('.inspector-wrap').on('wheel.intro', null);
+             nextStep();
+           }
+         }
 
-                         for (k = 0; k < rings.length; k++) {
-                           coords.push(coord(nodeVal(get1(rings[k], 'coordinates'))));
-                         }
+         function fieldsTownHall() {
+           if (!isTownHallSelected()) return clickTownHall(); // reset pane, in case user happened to change it..
 
-                         geoms.push({
-                           type: 'Polygon',
-                           coordinates: coords
-                         });
-                       } else if (geotypes[i] === 'Track' || geotypes[i] === 'gx:Track') {
-                         var track = gxCoords(geomNode);
-                         geoms.push({
-                           type: 'LineString',
-                           coordinates: track.coords
-                         });
-                         if (track.times.length) coordTimes.push(track.times);
-                       }
-                     }
-                   }
-                 }
+           context.container().select('.inspector-wrap .panewrap').style('right', '0%'); // disallow scrolling
 
-                 return {
-                   geoms: geoms,
-                   coordTimes: coordTimes
-                 };
-               }
+           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
 
-               function getPlacemark(root) {
-                 var geomsAndTimes = getGeometry(root),
-                     i,
-                     properties = {},
-                     name = nodeVal(get1(root, 'name')),
-                     address = nodeVal(get1(root, 'address')),
-                     styleUrl = nodeVal(get1(root, 'styleUrl')),
-                     description = nodeVal(get1(root, 'description')),
-                     timeSpan = get1(root, 'TimeSpan'),
-                     timeStamp = get1(root, 'TimeStamp'),
-                     extendedData = get1(root, 'ExtendedData'),
-                     lineStyle = get1(root, 'LineStyle'),
-                     polyStyle = get1(root, 'PolyStyle'),
-                     visibility = get1(root, 'visibility');
-                 if (!geomsAndTimes.geoms.length) return [];
-                 if (name) properties.name = name;
-                 if (address) properties.address = address;
+           var onClick = function onClick() {
+             continueTo(closeTownHall);
+           };
 
-                 if (styleUrl) {
-                   if (styleUrl[0] !== '#') {
-                     styleUrl = '#' + styleUrl;
-                   }
+           reveal('.entity-editor-pane .section-preset-fields', helpHtml('intro.navigation.fields_townhall'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: onClick
+           });
+           context.on('exit.intro', function () {
+             continueTo(clickTownHall);
+           });
+           context.history().on('change.intro', function () {
+             if (!context.hasEntity(hallId)) {
+               continueTo(clickTownHall);
+             }
+           });
 
-                   properties.styleUrl = styleUrl;
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             context.history().on('change.intro', null);
+             context.container().select('.inspector-wrap').on('wheel.intro', null);
+             nextStep();
+           }
+         }
 
-                   if (styleIndex[styleUrl]) {
-                     properties.styleHash = styleIndex[styleUrl];
-                   }
+         function closeTownHall() {
+           if (!isTownHallSelected()) return clickTownHall();
+           var selector = '.entity-editor-pane button.close svg use';
+           var href = select(selector).attr('href') || '#iD-icon-close';
+           reveal('.entity-editor-pane', helpHtml('intro.navigation.close_townhall', {
+             button: icon(href, 'inline')
+           }));
+           context.on('exit.intro', function () {
+             continueTo(searchStreet);
+           });
+           context.history().on('change.intro', function () {
+             // update the close icon in the tooltip if the user edits something.
+             var selector = '.entity-editor-pane button.close svg use';
+             var href = select(selector).attr('href') || '#iD-icon-close';
+             reveal('.entity-editor-pane', helpHtml('intro.navigation.close_townhall', {
+               button: icon(href, 'inline')
+             }), {
+               duration: 0
+             });
+           });
 
-                   if (styleMapIndex[styleUrl]) {
-                     properties.styleMapHash = styleMapIndex[styleUrl];
-                     properties.styleHash = styleIndex[styleMapIndex[styleUrl].normal];
-                   } // Try to populate the lineStyle or polyStyle since we got the style hash
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
+           }
+         }
 
+         function searchStreet() {
+           context.enter(modeBrowse(context));
+           context.history().reset('initial'); // ensure spring street exists
 
-                   var style = styleByHash[properties.styleHash];
+           var msec = transitionTime(springStreet, context.map().center());
 
-                   if (style) {
-                     if (!lineStyle) lineStyle = get1(style, 'LineStyle');
-                     if (!polyStyle) polyStyle = get1(style, 'PolyStyle');
-                   }
-                 }
+           if (msec) {
+             reveal(null, null, {
+               duration: 0
+             });
+           }
 
-                 if (description) properties.description = description;
+           context.map().centerZoomEase(springStreet, 19, msec); // ..and user can see it
 
-                 if (timeSpan) {
-                   var begin = nodeVal(get1(timeSpan, 'begin'));
-                   var end = nodeVal(get1(timeSpan, 'end'));
-                   properties.timespan = {
-                     begin: begin,
-                     end: end
-                   };
-                 }
+           timeout(function () {
+             reveal('.search-header input', helpHtml('intro.navigation.search_street', {
+               name: _t('intro.graph.name.spring-street')
+             }));
+             context.container().select('.search-header input').on('keyup.intro', checkSearchResult);
+           }, msec + 100);
+         }
 
-                 if (timeStamp) {
-                   properties.timestamp = nodeVal(get1(timeStamp, 'when'));
-                 }
+         function checkSearchResult() {
+           var first = context.container().select('.feature-list-item:nth-child(0n+2)'); // skip "No Results" item
 
-                 if (lineStyle) {
-                   var linestyles = kmlColor(nodeVal(get1(lineStyle, 'color'))),
-                       color = linestyles[0],
-                       opacity = linestyles[1],
-                       width = parseFloat(nodeVal(get1(lineStyle, 'width')));
-                   if (color) properties.stroke = color;
-                   if (!isNaN(opacity)) properties['stroke-opacity'] = opacity;
-                   if (!isNaN(width)) properties['stroke-width'] = width;
-                 }
+           var firstName = first.select('.entity-name');
+           var name = _t('intro.graph.name.spring-street');
 
-                 if (polyStyle) {
-                   var polystyles = kmlColor(nodeVal(get1(polyStyle, 'color'))),
-                       pcolor = polystyles[0],
-                       popacity = polystyles[1],
-                       fill = nodeVal(get1(polyStyle, 'fill')),
-                       outline = nodeVal(get1(polyStyle, 'outline'));
-                   if (pcolor) properties.fill = pcolor;
-                   if (!isNaN(popacity)) properties['fill-opacity'] = popacity;
-                   if (fill) properties['fill-opacity'] = fill === '1' ? properties['fill-opacity'] || 1 : 0;
-                   if (outline) properties['stroke-opacity'] = outline === '1' ? properties['stroke-opacity'] || 1 : 0;
-                 }
+           if (!firstName.empty() && firstName.html() === name) {
+             reveal(first.node(), helpHtml('intro.navigation.choose_street', {
+               name: name
+             }), {
+               duration: 300
+             });
+             context.on('exit.intro', function () {
+               continueTo(selectedStreet);
+             });
+             context.container().select('.search-header input').on('keydown.intro', eventCancel, true).on('keyup.intro', null);
+           }
 
-                 if (extendedData) {
-                   var datas = get(extendedData, 'Data'),
-                       simpleDatas = get(extendedData, 'SimpleData');
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             context.container().select('.search-header input').on('keydown.intro', null).on('keyup.intro', null);
+             nextStep();
+           }
+         }
 
-                   for (i = 0; i < datas.length; i++) {
-                     properties[datas[i].getAttribute('name')] = nodeVal(get1(datas[i], 'value'));
-                   }
+         function selectedStreet() {
+           if (!context.hasEntity(springStreetEndId) || !context.hasEntity(springStreetId)) {
+             return searchStreet();
+           }
 
-                   for (i = 0; i < simpleDatas.length; i++) {
-                     properties[simpleDatas[i].getAttribute('name')] = nodeVal(simpleDatas[i]);
-                   }
-                 }
+           var onClick = function onClick() {
+             continueTo(editorStreet);
+           };
 
-                 if (visibility) {
-                   properties.visibility = nodeVal(visibility);
-                 }
+           var entity = context.entity(springStreetEndId);
+           var box = pointBox(entity.loc, context);
+           box.height = 500;
+           reveal(box, helpHtml('intro.navigation.selected_street', {
+             name: _t('intro.graph.name.spring-street')
+           }), {
+             duration: 600,
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: onClick
+           });
+           timeout(function () {
+             context.map().on('move.intro drawn.intro', function () {
+               var entity = context.hasEntity(springStreetEndId);
+               if (!entity) return;
+               var box = pointBox(entity.loc, context);
+               box.height = 500;
+               reveal(box, helpHtml('intro.navigation.selected_street', {
+                 name: _t('intro.graph.name.spring-street')
+               }), {
+                 duration: 0,
+                 buttonText: _t.html('intro.ok'),
+                 buttonCallback: onClick
+               });
+             });
+           }, 600); // after reveal.
 
-                 if (geomsAndTimes.coordTimes.length) {
-                   properties.coordTimes = geomsAndTimes.coordTimes.length === 1 ? geomsAndTimes.coordTimes[0] : geomsAndTimes.coordTimes;
-                 }
+           context.on('enter.intro', function (mode) {
+             if (!context.hasEntity(springStreetId)) {
+               return continueTo(searchStreet);
+             }
 
-                 var feature = {
-                   type: 'Feature',
-                   geometry: geomsAndTimes.geoms.length === 1 ? geomsAndTimes.geoms[0] : {
-                     type: 'GeometryCollection',
-                     geometries: geomsAndTimes.geoms
-                   },
-                   properties: properties
-                 };
-                 if (attr(root, 'id')) feature.id = attr(root, 'id');
-                 return [feature];
-               }
+             var ids = context.selectedIDs();
 
-               return gj;
-             },
-             gpx: function gpx(doc) {
-               var i,
-                   tracks = get(doc, 'trk'),
-                   routes = get(doc, 'rte'),
-                   waypoints = get(doc, 'wpt'),
-                   // a feature collection
-               gj = fc(),
-                   feature;
+             if (mode.id !== 'select' || !ids.length || ids[0] !== springStreetId) {
+               // keep Spring Street selected..
+               context.enter(modeSelect(context, [springStreetId]));
+             }
+           });
+           context.history().on('change.intro', function () {
+             if (!context.hasEntity(springStreetEndId) || !context.hasEntity(springStreetId)) {
+               timeout(function () {
+                 continueTo(searchStreet);
+               }, 300); // after any transition (e.g. if user deleted intersection)
+             }
+           });
 
-               for (i = 0; i < tracks.length; i++) {
-                 feature = getTrack(tracks[i]);
-                 if (feature) gj.features.push(feature);
-               }
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('enter.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
+           }
+         }
 
-               for (i = 0; i < routes.length; i++) {
-                 feature = getRoute(routes[i]);
-                 if (feature) gj.features.push(feature);
-               }
+         function editorStreet() {
+           var selector = '.entity-editor-pane button.close svg use';
+           var href = select(selector).attr('href') || '#iD-icon-close';
+           reveal('.entity-editor-pane', helpHtml('intro.navigation.street_different_fields') + '{br}' + helpHtml('intro.navigation.editor_street', {
+             button: icon(href, 'inline'),
+             field1: onewayField.label(),
+             field2: maxspeedField.label()
+           }));
+           context.on('exit.intro', function () {
+             continueTo(play);
+           });
+           context.history().on('change.intro', function () {
+             // update the close icon in the tooltip if the user edits something.
+             var selector = '.entity-editor-pane button.close svg use';
+             var href = select(selector).attr('href') || '#iD-icon-close';
+             reveal('.entity-editor-pane', helpHtml('intro.navigation.street_different_fields') + '{br}' + helpHtml('intro.navigation.editor_street', {
+               button: icon(href, 'inline'),
+               field1: onewayField.label(),
+               field2: maxspeedField.label()
+             }), {
+               duration: 0
+             });
+           });
 
-               for (i = 0; i < waypoints.length; i++) {
-                 gj.features.push(getPoint(waypoints[i]));
-               }
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
+           }
+         }
 
-               function getPoints(node, pointname) {
-                 var pts = get(node, pointname),
-                     line = [],
-                     times = [],
-                     heartRates = [],
-                     l = pts.length;
-                 if (l < 2) return {}; // Invalid line in GeoJSON
+         function play() {
+           dispatch.call('done');
+           reveal('.ideditor', helpHtml('intro.navigation.play', {
+             next: _t('intro.points.title')
+           }), {
+             tooltipBox: '.intro-nav-wrap .chapter-point',
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: function buttonCallback() {
+               reveal('.ideditor');
+             }
+           });
+         }
 
-                 for (var i = 0; i < l; i++) {
-                   var c = coordPair(pts[i]);
-                   line.push(c.coordinates);
-                   if (c.time) times.push(c.time);
-                   if (c.heartRate) heartRates.push(c.heartRate);
-                 }
+         chapter.enter = function () {
+           dragMap();
+         };
 
-                 return {
-                   line: line,
-                   times: times,
-                   heartRates: heartRates
-                 };
-               }
+         chapter.exit = function () {
+           timeouts.forEach(window.clearTimeout);
+           context.on('enter.intro exit.intro', null);
+           context.map().on('move.intro drawn.intro', null);
+           context.history().on('change.intro', null);
+           context.container().select('.inspector-wrap').on('wheel.intro', null);
+           context.container().select('.search-header input').on('keydown.intro keyup.intro', null);
+         };
 
-               function getTrack(node) {
-                 var segments = get(node, 'trkseg'),
-                     track = [],
-                     times = [],
-                     heartRates = [],
-                     line;
+         chapter.restart = function () {
+           chapter.exit();
+           chapter.enter();
+         };
 
-                 for (var i = 0; i < segments.length; i++) {
-                   line = getPoints(segments[i], 'trkpt');
+         return utilRebind(chapter, dispatch, 'on');
+       }
 
-                   if (line) {
-                     if (line.line) track.push(line.line);
-                     if (line.times && line.times.length) times.push(line.times);
-                     if (line.heartRates && line.heartRates.length) heartRates.push(line.heartRates);
-                   }
-                 }
+       function uiIntroPoint(context, reveal) {
+         var dispatch = dispatch$8('done');
+         var timeouts = [];
+         var intersection = [-85.63279, 41.94394];
+         var building = [-85.632422, 41.944045];
+         var cafePreset = _mainPresetIndex.item('amenity/cafe');
+         var _pointID = null;
+         var chapter = {
+           title: 'intro.points.title'
+         };
 
-                 if (track.length === 0) return;
-                 var properties = getProperties(node);
-                 extend(properties, getLineStyle(get1(node, 'extensions')));
-                 if (times.length) properties.coordTimes = track.length === 1 ? times[0] : times;
-                 if (heartRates.length) properties.heartRates = track.length === 1 ? heartRates[0] : heartRates;
-                 return {
-                   type: 'Feature',
-                   properties: properties,
-                   geometry: {
-                     type: track.length === 1 ? 'LineString' : 'MultiLineString',
-                     coordinates: track.length === 1 ? track[0] : track
-                   }
-                 };
-               }
+         function timeout(f, t) {
+           timeouts.push(window.setTimeout(f, t));
+         }
 
-               function getRoute(node) {
-                 var line = getPoints(node, 'rtept');
-                 if (!line.line) return;
-                 var prop = getProperties(node);
-                 extend(prop, getLineStyle(get1(node, 'extensions')));
-                 var routeObj = {
-                   type: 'Feature',
-                   properties: prop,
-                   geometry: {
-                     type: 'LineString',
-                     coordinates: line.line
-                   }
-                 };
-                 return routeObj;
-               }
+         function eventCancel(d3_event) {
+           d3_event.stopPropagation();
+           d3_event.preventDefault();
+         }
 
-               function getPoint(node) {
-                 var prop = getProperties(node);
-                 extend(prop, getMulti(node, ['sym']));
-                 return {
-                   type: 'Feature',
-                   properties: prop,
-                   geometry: {
-                     type: 'Point',
-                     coordinates: coordPair(node).coordinates
-                   }
-                 };
-               }
+         function addPoint() {
+           context.enter(modeBrowse(context));
+           context.history().reset('initial');
+           var msec = transitionTime(intersection, context.map().center());
 
-               function getLineStyle(extensions) {
-                 var style = {};
+           if (msec) {
+             reveal(null, null, {
+               duration: 0
+             });
+           }
 
-                 if (extensions) {
-                   var lineStyle = get1(extensions, 'line');
+           context.map().centerZoomEase(intersection, 19, msec);
+           timeout(function () {
+             var tooltip = reveal('button.add-point', helpHtml('intro.points.points_info') + '{br}' + helpHtml('intro.points.add_point'));
+             _pointID = null;
+             tooltip.selectAll('.popover-inner').insert('svg', 'span').attr('class', 'tooltip-illustration').append('use').attr('xlink:href', '#iD-graphic-points');
+             context.on('enter.intro', function (mode) {
+               if (mode.id !== 'add-point') return;
+               continueTo(placePoint);
+             });
+           }, msec + 100);
 
-                   if (lineStyle) {
-                     var color = nodeVal(get1(lineStyle, 'color')),
-                         opacity = parseFloat(nodeVal(get1(lineStyle, 'opacity'))),
-                         width = parseFloat(nodeVal(get1(lineStyle, 'width')));
-                     if (color) style.stroke = color;
-                     if (!isNaN(opacity)) style['stroke-opacity'] = opacity; // GPX width is in mm, convert to px with 96 px per inch
+           function continueTo(nextStep) {
+             context.on('enter.intro', null);
+             nextStep();
+           }
+         }
 
-                     if (!isNaN(width)) style['stroke-width'] = width * 96 / 25.4;
-                   }
-                 }
+         function placePoint() {
+           if (context.mode().id !== 'add-point') {
+             return chapter.restart();
+           }
 
-                 return style;
-               }
+           var pointBox = pad(building, 150, context);
+           var textId = context.lastPointerType() === 'mouse' ? 'place_point' : 'place_point_touch';
+           reveal(pointBox, helpHtml('intro.points.' + textId));
+           context.map().on('move.intro drawn.intro', function () {
+             pointBox = pad(building, 150, context);
+             reveal(pointBox, helpHtml('intro.points.' + textId), {
+               duration: 0
+             });
+           });
+           context.on('enter.intro', function (mode) {
+             if (mode.id !== 'select') return chapter.restart();
+             _pointID = context.mode().selectedIDs()[0];
+             continueTo(searchPreset);
+           });
 
-               function getProperties(node) {
-                 var prop = getMulti(node, ['name', 'cmt', 'desc', 'type', 'time', 'keywords']),
-                     links = get(node, 'link');
-                 if (links.length) prop.links = [];
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
+           }
+         }
 
-                 for (var i = 0, link; i < links.length; i++) {
-                   link = {
-                     href: attr(links[i], 'href')
-                   };
-                   extend(link, getMulti(links[i], ['text', 'type']));
-                   prop.links.push(link);
-                 }
+         function searchPreset() {
+           if (context.mode().id !== 'select' || !_pointID || !context.hasEntity(_pointID)) {
+             return addPoint();
+           } // disallow scrolling
 
-                 return prop;
-               }
 
-               return gj;
+           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
+           context.container().select('.preset-search-input').on('keydown.intro', null).on('keyup.intro', checkPresetSearch);
+           reveal('.preset-search-input', helpHtml('intro.points.search_cafe', {
+             preset: cafePreset.name()
+           }));
+           context.on('enter.intro', function (mode) {
+             if (!_pointID || !context.hasEntity(_pointID)) {
+               return continueTo(addPoint);
              }
-           };
-           return t;
-         }();
-
-         module.exports = toGeoJSON;
-       });
 
-       var _initialized = false;
-       var _enabled = false;
+             var ids = context.selectedIDs();
 
-       var _geojson;
+             if (mode.id !== 'select' || !ids.length || ids[0] !== _pointID) {
+               // keep the user's point selected..
+               context.enter(modeSelect(context, [_pointID])); // disallow scrolling
 
-       function svgData(projection, context, dispatch) {
-         var throttledRedraw = throttle(function () {
-           dispatch.call('change');
-         }, 1000);
+               context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
+               context.container().select('.preset-search-input').on('keydown.intro', null).on('keyup.intro', checkPresetSearch);
+               reveal('.preset-search-input', helpHtml('intro.points.search_cafe', {
+                 preset: cafePreset.name()
+               }));
+               context.history().on('change.intro', null);
+             }
+           });
 
-         var _showLabels = true;
-         var detected = utilDetect();
-         var layer = select(null);
+           function checkPresetSearch() {
+             var first = context.container().select('.preset-list-item:first-child');
 
-         var _vtService;
+             if (first.classed('preset-amenity-cafe')) {
+               context.container().select('.preset-search-input').on('keydown.intro', eventCancel, true).on('keyup.intro', null);
+               reveal(first.select('.preset-list-button').node(), helpHtml('intro.points.choose_cafe', {
+                 preset: cafePreset.name()
+               }), {
+                 duration: 300
+               });
+               context.history().on('change.intro', function () {
+                 continueTo(aboutFeatureEditor);
+               });
+             }
+           }
 
-         var _fileList;
+           function continueTo(nextStep) {
+             context.on('enter.intro', null);
+             context.history().on('change.intro', null);
+             context.container().select('.inspector-wrap').on('wheel.intro', null);
+             context.container().select('.preset-search-input').on('keydown.intro keyup.intro', null);
+             nextStep();
+           }
+         }
 
-         var _template;
+         function aboutFeatureEditor() {
+           if (context.mode().id !== 'select' || !_pointID || !context.hasEntity(_pointID)) {
+             return addPoint();
+           }
 
-         var _src;
+           timeout(function () {
+             reveal('.entity-editor-pane', helpHtml('intro.points.feature_editor'), {
+               tooltipClass: 'intro-points-describe',
+               buttonText: _t.html('intro.ok'),
+               buttonCallback: function buttonCallback() {
+                 continueTo(addName);
+               }
+             });
+           }, 400);
+           context.on('exit.intro', function () {
+             // if user leaves select mode here, just continue with the tutorial.
+             continueTo(reselectPoint);
+           });
 
-         function init() {
-           if (_initialized) return; // run once
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             nextStep();
+           }
+         }
 
-           _geojson = {};
-           _enabled = true;
+         function addName() {
+           if (context.mode().id !== 'select' || !_pointID || !context.hasEntity(_pointID)) {
+             return addPoint();
+           } // reset pane, in case user happened to change it..
 
-           function over(d3_event) {
-             d3_event.stopPropagation();
-             d3_event.preventDefault();
-             d3_event.dataTransfer.dropEffect = 'copy';
-           }
 
-           context.container().attr('dropzone', 'copy').on('drop.svgData', function (d3_event) {
-             d3_event.stopPropagation();
-             d3_event.preventDefault();
-             if (!detected.filedrop) return;
-             drawData.fileList(d3_event.dataTransfer.files);
-           }).on('dragenter.svgData', over).on('dragexit.svgData', over).on('dragover.svgData', over);
-           _initialized = true;
-         }
+           context.container().select('.inspector-wrap .panewrap').style('right', '0%');
+           var addNameString = helpHtml('intro.points.fields_info') + '{br}' + helpHtml('intro.points.add_name');
+           timeout(function () {
+             // It's possible for the user to add a name in a previous step..
+             // If so, don't tell them to add the name in this step.
+             // Give them an OK button instead.
+             var entity = context.entity(_pointID);
 
-         function getService() {
-           if (services.vectorTile && !_vtService) {
-             _vtService = services.vectorTile;
+             if (entity.tags.name) {
+               var tooltip = reveal('.entity-editor-pane', addNameString, {
+                 tooltipClass: 'intro-points-describe',
+                 buttonText: _t.html('intro.ok'),
+                 buttonCallback: function buttonCallback() {
+                   continueTo(addCloseEditor);
+                 }
+               });
+               tooltip.select('.instruction').style('display', 'none');
+             } else {
+               reveal('.entity-editor-pane', addNameString, {
+                 tooltipClass: 'intro-points-describe'
+               });
+             }
+           }, 400);
+           context.history().on('change.intro', function () {
+             continueTo(addCloseEditor);
+           });
+           context.on('exit.intro', function () {
+             // if user leaves select mode here, just continue with the tutorial.
+             continueTo(reselectPoint);
+           });
 
-             _vtService.event.on('loadedData', throttledRedraw);
-           } else if (!services.vectorTile && _vtService) {
-             _vtService = null;
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
            }
-
-           return _vtService;
          }
 
-         function showLayer() {
-           layerOn();
-           layer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end', function () {
-             dispatch.call('change');
+         function addCloseEditor() {
+           // reset pane, in case user happened to change it..
+           context.container().select('.inspector-wrap .panewrap').style('right', '0%');
+           var selector = '.entity-editor-pane button.close svg use';
+           var href = select(selector).attr('href') || '#iD-icon-close';
+           context.on('exit.intro', function () {
+             continueTo(reselectPoint);
            });
-         }
+           reveal('.entity-editor-pane', helpHtml('intro.points.add_close', {
+             button: icon(href, 'inline')
+           }));
 
-         function hideLayer() {
-           throttledRedraw.cancel();
-           layer.transition().duration(250).style('opacity', 0).on('end', layerOff);
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             nextStep();
+           }
          }
 
-         function layerOn() {
-           layer.style('display', 'block');
-         }
+         function reselectPoint() {
+           if (!_pointID) return chapter.restart();
+           var entity = context.hasEntity(_pointID);
+           if (!entity) return chapter.restart(); // make sure it's still a cafe, in case user somehow changed it..
 
-         function layerOff() {
-           layer.selectAll('.viewfield-group').remove();
-           layer.style('display', 'none');
-         } // ensure that all geojson features in a collection have IDs
+           var oldPreset = _mainPresetIndex.match(entity, context.graph());
+           context.replace(actionChangePreset(_pointID, oldPreset, cafePreset));
+           context.enter(modeBrowse(context));
+           var msec = transitionTime(entity.loc, context.map().center());
 
+           if (msec) {
+             reveal(null, null, {
+               duration: 0
+             });
+           }
 
-         function ensureIDs(gj) {
-           if (!gj) return null;
+           context.map().centerEase(entity.loc, msec);
+           timeout(function () {
+             var box = pointBox(entity.loc, context);
+             reveal(box, helpHtml('intro.points.reselect'), {
+               duration: 600
+             });
+             timeout(function () {
+               context.map().on('move.intro drawn.intro', function () {
+                 var entity = context.hasEntity(_pointID);
+                 if (!entity) return chapter.restart();
+                 var box = pointBox(entity.loc, context);
+                 reveal(box, helpHtml('intro.points.reselect'), {
+                   duration: 0
+                 });
+               });
+             }, 600); // after reveal..
 
-           if (gj.type === 'FeatureCollection') {
-             for (var i = 0; i < gj.features.length; i++) {
-               ensureFeatureID(gj.features[i]);
-             }
-           } else {
-             ensureFeatureID(gj);
+             context.on('enter.intro', function (mode) {
+               if (mode.id !== 'select') return;
+               continueTo(updatePoint);
+             });
+           }, msec + 100);
+
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
            }
+         }
 
-           return gj;
-         } // ensure that each single Feature object has a unique ID
+         function updatePoint() {
+           if (context.mode().id !== 'select' || !_pointID || !context.hasEntity(_pointID)) {
+             return continueTo(reselectPoint);
+           } // reset pane, in case user happened to untag the point..
 
 
-         function ensureFeatureID(feature) {
-           if (!feature) return;
-           feature.__featurehash__ = utilHashcode(fastJsonStableStringify(feature));
-           return feature;
-         } // Prefer an array of Features instead of a FeatureCollection
+           context.container().select('.inspector-wrap .panewrap').style('right', '0%');
+           context.on('exit.intro', function () {
+             continueTo(reselectPoint);
+           });
+           context.history().on('change.intro', function () {
+             continueTo(updateCloseEditor);
+           });
+           timeout(function () {
+             reveal('.entity-editor-pane', helpHtml('intro.points.update'), {
+               tooltipClass: 'intro-points-describe'
+             });
+           }, 400);
 
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
+           }
+         }
 
-         function getFeatures(gj) {
-           if (!gj) return [];
+         function updateCloseEditor() {
+           if (context.mode().id !== 'select' || !_pointID || !context.hasEntity(_pointID)) {
+             return continueTo(reselectPoint);
+           } // reset pane, in case user happened to change it..
 
-           if (gj.type === 'FeatureCollection') {
-             return gj.features;
-           } else {
-             return [gj];
+
+           context.container().select('.inspector-wrap .panewrap').style('right', '0%');
+           context.on('exit.intro', function () {
+             continueTo(rightClickPoint);
+           });
+           timeout(function () {
+             reveal('.entity-editor-pane', helpHtml('intro.points.update_close', {
+               button: icon('#iD-icon-close', 'inline')
+             }));
+           }, 500);
+
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             nextStep();
            }
          }
 
-         function featureKey(d) {
-           return d.__featurehash__;
-         }
+         function rightClickPoint() {
+           if (!_pointID) return chapter.restart();
+           var entity = context.hasEntity(_pointID);
+           if (!entity) return chapter.restart();
+           context.enter(modeBrowse(context));
+           var box = pointBox(entity.loc, context);
+           var textId = context.lastPointerType() === 'mouse' ? 'rightclick' : 'edit_menu_touch';
+           reveal(box, helpHtml('intro.points.' + textId), {
+             duration: 600
+           });
+           timeout(function () {
+             context.map().on('move.intro', function () {
+               var entity = context.hasEntity(_pointID);
+               if (!entity) return chapter.restart();
+               var box = pointBox(entity.loc, context);
+               reveal(box, helpHtml('intro.points.' + textId), {
+                 duration: 0
+               });
+             });
+           }, 600); // after reveal
 
-         function isPolygon(d) {
-           return d.geometry.type === 'Polygon' || d.geometry.type === 'MultiPolygon';
-         }
+           context.on('enter.intro', function (mode) {
+             if (mode.id !== 'select') return;
+             var ids = context.selectedIDs();
+             if (ids.length !== 1 || ids[0] !== _pointID) return;
+             timeout(function () {
+               var node = selectMenuItem(context, 'delete').node();
+               if (!node) return;
+               continueTo(enterDelete);
+             }, 50); // after menu visible
+           });
 
-         function clipPathID(d) {
-           return 'ideditor-data-' + d.__featurehash__ + '-clippath';
+           function continueTo(nextStep) {
+             context.on('enter.intro', null);
+             context.map().on('move.intro', null);
+             nextStep();
+           }
          }
 
-         function featureClasses(d) {
-           return ['data' + d.__featurehash__, d.geometry.type, isPolygon(d) ? 'area' : '', d.__layerID__ || ''].filter(Boolean).join(' ');
-         }
+         function enterDelete() {
+           if (!_pointID) return chapter.restart();
+           var entity = context.hasEntity(_pointID);
+           if (!entity) return chapter.restart();
+           var node = selectMenuItem(context, 'delete').node();
 
-         function drawData(selection) {
-           var vtService = getService();
-           var getPath = svgPath(projection).geojson;
-           var getAreaPath = svgPath(projection, null, true).geojson;
-           var hasData = drawData.hasData();
-           layer = selection.selectAll('.layer-mapdata').data(_enabled && hasData ? [0] : []);
-           layer.exit().remove();
-           layer = layer.enter().append('g').attr('class', 'layer-mapdata').merge(layer);
-           var surface = context.surface();
-           if (!surface || surface.empty()) return; // not ready to draw yet, starting up
-           // Gather data
+           if (!node) {
+             return continueTo(rightClickPoint);
+           }
 
-           var geoData, polygonData;
+           reveal('.edit-menu', helpHtml('intro.points.delete'), {
+             padding: 50
+           });
+           timeout(function () {
+             context.map().on('move.intro', function () {
+               reveal('.edit-menu', helpHtml('intro.points.delete'), {
+                 duration: 0,
+                 padding: 50
+               });
+             });
+           }, 300); // after menu visible
 
-           if (_template && vtService) {
-             // fetch data from vector tile service
-             var sourceID = _template;
-             vtService.loadTiles(sourceID, _template, projection);
-             geoData = vtService.data(sourceID, projection);
-           } else {
-             geoData = getFeatures(_geojson);
+           context.on('exit.intro', function () {
+             if (!_pointID) return chapter.restart();
+             var entity = context.hasEntity(_pointID);
+             if (entity) return continueTo(rightClickPoint); // point still exists
+           });
+           context.history().on('change.intro', function (changed) {
+             if (changed.deleted().length) {
+               continueTo(undo);
+             }
+           });
+
+           function continueTo(nextStep) {
+             context.map().on('move.intro', null);
+             context.history().on('change.intro', null);
+             context.on('exit.intro', null);
+             nextStep();
            }
+         }
 
-           geoData = geoData.filter(getPath);
-           polygonData = geoData.filter(isPolygon); // Draw clip paths for polygons
+         function undo() {
+           context.history().on('change.intro', function () {
+             continueTo(play);
+           });
+           reveal('.top-toolbar button.undo-button', helpHtml('intro.points.undo'));
 
-           var clipPaths = surface.selectAll('defs').selectAll('.clipPath-data').data(polygonData, featureKey);
-           clipPaths.exit().remove();
-           var clipPathsEnter = clipPaths.enter().append('clipPath').attr('class', 'clipPath-data').attr('id', clipPathID);
-           clipPathsEnter.append('path');
-           clipPaths.merge(clipPathsEnter).selectAll('path').attr('d', getAreaPath); // Draw fill, shadow, stroke layers
+           function continueTo(nextStep) {
+             context.history().on('change.intro', null);
+             nextStep();
+           }
+         }
 
-           var datagroups = layer.selectAll('g.datagroup').data(['fill', 'shadow', 'stroke']);
-           datagroups = datagroups.enter().append('g').attr('class', function (d) {
-             return 'datagroup datagroup-' + d;
-           }).merge(datagroups); // Draw paths
+         function play() {
+           dispatch.call('done');
+           reveal('.ideditor', helpHtml('intro.points.play', {
+             next: _t('intro.areas.title')
+           }), {
+             tooltipBox: '.intro-nav-wrap .chapter-area',
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: function buttonCallback() {
+               reveal('.ideditor');
+             }
+           });
+         }
 
-           var pathData = {
-             fill: polygonData,
-             shadow: geoData,
-             stroke: geoData
-           };
-           var paths = datagroups.selectAll('path').data(function (layer) {
-             return pathData[layer];
-           }, featureKey); // exit
+         chapter.enter = function () {
+           addPoint();
+         };
 
-           paths.exit().remove(); // enter/update
+         chapter.exit = function () {
+           timeouts.forEach(window.clearTimeout);
+           context.on('enter.intro exit.intro', null);
+           context.map().on('move.intro drawn.intro', null);
+           context.history().on('change.intro', null);
+           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
+           context.container().select('.preset-search-input').on('keydown.intro keyup.intro', null);
+         };
 
-           paths = paths.enter().append('path').attr('class', function (d) {
-             var datagroup = this.parentNode.__data__;
-             return 'pathdata ' + datagroup + ' ' + featureClasses(d);
-           }).attr('clip-path', function (d) {
-             var datagroup = this.parentNode.__data__;
-             return datagroup === 'fill' ? 'url(#' + clipPathID(d) + ')' : null;
-           }).merge(paths).attr('d', function (d) {
-             var datagroup = this.parentNode.__data__;
-             return datagroup === 'fill' ? getAreaPath(d) : getPath(d);
-           }); // Draw labels
+         chapter.restart = function () {
+           chapter.exit();
+           chapter.enter();
+         };
 
-           layer.call(drawLabels, 'label-halo', geoData).call(drawLabels, 'label', geoData);
+         return utilRebind(chapter, dispatch, 'on');
+       }
 
-           function drawLabels(selection, textClass, data) {
-             var labelPath = d3_geoPath(projection);
-             var labelData = data.filter(function (d) {
-               return _showLabels && d.properties && (d.properties.desc || d.properties.name);
-             });
-             var labels = selection.selectAll('text.' + textClass).data(labelData, featureKey); // exit
+       function uiIntroArea(context, reveal) {
+         var dispatch = dispatch$8('done');
+         var playground = [-85.63552, 41.94159];
+         var playgroundPreset = _mainPresetIndex.item('leisure/playground');
+         var nameField = _mainPresetIndex.field('name');
+         var descriptionField = _mainPresetIndex.field('description');
+         var timeouts = [];
 
-             labels.exit().remove(); // enter/update
+         var _areaID;
 
-             labels = labels.enter().append('text').attr('class', function (d) {
-               return textClass + ' ' + featureClasses(d);
-             }).merge(labels).text(function (d) {
-               return d.properties.desc || d.properties.name;
-             }).attr('x', function (d) {
-               var centroid = labelPath.centroid(d);
-               return centroid[0] + 11;
-             }).attr('y', function (d) {
-               var centroid = labelPath.centroid(d);
-               return centroid[1];
-             });
-           }
+         var chapter = {
+           title: 'intro.areas.title'
+         };
+
+         function timeout(f, t) {
+           timeouts.push(window.setTimeout(f, t));
          }
 
-         function getExtension(fileName) {
-           if (!fileName) return;
-           var re = /\.(gpx|kml|(geo)?json)$/i;
-           var match = fileName.toLowerCase().match(re);
-           return match && match.length && match[0];
+         function eventCancel(d3_event) {
+           d3_event.stopPropagation();
+           d3_event.preventDefault();
          }
 
-         function xmlToDom(textdata) {
-           return new DOMParser().parseFromString(textdata, 'text/xml');
+         function revealPlayground(center, text, options) {
+           var padding = 180 * Math.pow(2, context.map().zoom() - 19.5);
+           var box = pad(center, padding, context);
+           reveal(box, text, options);
          }
 
-         drawData.setFile = function (extension, data) {
-           _template = null;
-           _fileList = null;
-           _geojson = null;
-           _src = null;
-           var gj;
-
-           switch (extension) {
-             case '.gpx':
-               gj = togeojson.gpx(xmlToDom(data));
-               break;
-
-             case '.kml':
-               gj = togeojson.kml(xmlToDom(data));
-               break;
+         function addArea() {
+           context.enter(modeBrowse(context));
+           context.history().reset('initial');
+           _areaID = null;
+           var msec = transitionTime(playground, context.map().center());
 
-             case '.geojson':
-             case '.json':
-               gj = JSON.parse(data);
-               break;
+           if (msec) {
+             reveal(null, null, {
+               duration: 0
+             });
            }
 
-           gj = gj || {};
+           context.map().centerZoomEase(playground, 19, msec);
+           timeout(function () {
+             var tooltip = reveal('button.add-area', helpHtml('intro.areas.add_playground'));
+             tooltip.selectAll('.popover-inner').insert('svg', 'span').attr('class', 'tooltip-illustration').append('use').attr('xlink:href', '#iD-graphic-areas');
+             context.on('enter.intro', function (mode) {
+               if (mode.id !== 'add-area') return;
+               continueTo(startPlayground);
+             });
+           }, msec + 100);
 
-           if (Object.keys(gj).length) {
-             _geojson = ensureIDs(gj);
-             _src = extension + ' data file';
-             this.fitZoom();
+           function continueTo(nextStep) {
+             context.on('enter.intro', null);
+             nextStep();
            }
+         }
 
-           dispatch.call('change');
-           return this;
-         };
-
-         drawData.showLabels = function (val) {
-           if (!arguments.length) return _showLabels;
-           _showLabels = val;
-           return this;
-         };
+         function startPlayground() {
+           if (context.mode().id !== 'add-area') {
+             return chapter.restart();
+           }
 
-         drawData.enabled = function (val) {
-           if (!arguments.length) return _enabled;
-           _enabled = val;
+           _areaID = null;
+           context.map().zoomEase(19.5, 500);
+           timeout(function () {
+             var textId = context.lastPointerType() === 'mouse' ? 'starting_node_click' : 'starting_node_tap';
+             var startDrawString = helpHtml('intro.areas.start_playground') + helpHtml('intro.areas.' + textId);
+             revealPlayground(playground, startDrawString, {
+               duration: 250
+             });
+             timeout(function () {
+               context.map().on('move.intro drawn.intro', function () {
+                 revealPlayground(playground, startDrawString, {
+                   duration: 0
+                 });
+               });
+               context.on('enter.intro', function (mode) {
+                 if (mode.id !== 'draw-area') return chapter.restart();
+                 continueTo(continuePlayground);
+               });
+             }, 250); // after reveal
+           }, 550); // after easing
 
-           if (_enabled) {
-             showLayer();
-           } else {
-             hideLayer();
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
            }
+         }
 
-           dispatch.call('change');
-           return this;
-         };
+         function continuePlayground() {
+           if (context.mode().id !== 'draw-area') {
+             return chapter.restart();
+           }
 
-         drawData.hasData = function () {
-           var gj = _geojson || {};
-           return !!(_template || Object.keys(gj).length);
-         };
+           _areaID = null;
+           revealPlayground(playground, helpHtml('intro.areas.continue_playground'), {
+             duration: 250
+           });
+           timeout(function () {
+             context.map().on('move.intro drawn.intro', function () {
+               revealPlayground(playground, helpHtml('intro.areas.continue_playground'), {
+                 duration: 0
+               });
+             });
+           }, 250); // after reveal
 
-         drawData.template = function (val, src) {
-           if (!arguments.length) return _template; // test source against OSM imagery blocklists..
+           context.on('enter.intro', function (mode) {
+             if (mode.id === 'draw-area') {
+               var entity = context.hasEntity(context.selectedIDs()[0]);
 
-           var osm = context.connection();
+               if (entity && entity.nodes.length >= 6) {
+                 return continueTo(finishPlayground);
+               } else {
+                 return;
+               }
+             } else if (mode.id === 'select') {
+               _areaID = context.selectedIDs()[0];
+               return continueTo(searchPresets);
+             } else {
+               return chapter.restart();
+             }
+           });
 
-           if (osm) {
-             var blocklists = osm.imageryBlocklists();
-             var fail = false;
-             var tested = 0;
-             var regex;
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
+           }
+         }
 
-             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.
+         function finishPlayground() {
+           if (context.mode().id !== 'draw-area') {
+             return chapter.restart();
+           }
 
+           _areaID = null;
+           var finishString = helpHtml('intro.areas.finish_area_' + (context.lastPointerType() === 'mouse' ? 'click' : 'tap')) + helpHtml('intro.areas.finish_playground');
+           revealPlayground(playground, finishString, {
+             duration: 250
+           });
+           timeout(function () {
+             context.map().on('move.intro drawn.intro', function () {
+               revealPlayground(playground, finishString, {
+                 duration: 0
+               });
+             });
+           }, 250); // after reveal
 
-             if (!tested) {
-               regex = /.*\.google(apis)?\..*\/(vt|kh)[\?\/].*([xyz]=.*){3}.*/;
-               fail = regex.test(val);
+           context.on('enter.intro', function (mode) {
+             if (mode.id === 'draw-area') {
+               return;
+             } else if (mode.id === 'select') {
+               _areaID = context.selectedIDs()[0];
+               return continueTo(searchPresets);
+             } else {
+               return chapter.restart();
              }
+           });
+
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
            }
+         }
 
-           _template = val;
-           _fileList = null;
-           _geojson = null; // strip off the querystring/hash from the template,
-           // it often includes the access token
+         function searchPresets() {
+           if (!_areaID || !context.hasEntity(_areaID)) {
+             return addArea();
+           }
 
-           _src = src || 'vectortile:' + val.split(/[?#]/)[0];
-           dispatch.call('change');
-           return this;
-         };
+           var ids = context.selectedIDs();
 
-         drawData.geojson = function (gj, src) {
-           if (!arguments.length) return _geojson;
-           _template = null;
-           _fileList = null;
-           _geojson = null;
-           _src = null;
-           gj = gj || {};
+           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _areaID) {
+             context.enter(modeSelect(context, [_areaID]));
+           } // disallow scrolling
 
-           if (Object.keys(gj).length) {
-             _geojson = ensureIDs(gj);
-             _src = src || 'unknown.geojson';
-           }
 
-           dispatch.call('change');
-           return this;
-         };
+           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
+           timeout(function () {
+             // reset pane, in case user somehow happened to change it..
+             context.container().select('.inspector-wrap .panewrap').style('right', '-100%');
+             context.container().select('.preset-search-input').on('keydown.intro', null).on('keyup.intro', checkPresetSearch);
+             reveal('.preset-search-input', helpHtml('intro.areas.search_playground', {
+               preset: playgroundPreset.name()
+             }));
+           }, 400); // after preset list pane visible..
 
-         drawData.fileList = function (fileList) {
-           if (!arguments.length) return _fileList;
-           _template = null;
-           _fileList = fileList;
-           _geojson = null;
-           _src = null;
-           if (!fileList || !fileList.length) return this;
-           var f = fileList[0];
-           var extension = getExtension(f.name);
-           var reader = new FileReader();
+           context.on('enter.intro', function (mode) {
+             if (!_areaID || !context.hasEntity(_areaID)) {
+               return continueTo(addArea);
+             }
 
-           reader.onload = function () {
-             return function (e) {
-               drawData.setFile(extension, e.target.result);
-             };
-           }();
+             var ids = context.selectedIDs();
 
-           reader.readAsText(f);
-           return this;
-         };
+             if (mode.id !== 'select' || !ids.length || ids[0] !== _areaID) {
+               // keep the user's area selected..
+               context.enter(modeSelect(context, [_areaID])); // reset pane, in case user somehow happened to change it..
 
-         drawData.url = function (url, defaultExtension) {
-           _template = null;
-           _fileList = null;
-           _geojson = null;
-           _src = null; // strip off any querystring/hash from the url before checking extension
+               context.container().select('.inspector-wrap .panewrap').style('right', '-100%'); // disallow scrolling
 
-           var testUrl = url.split(/[?#]/)[0];
-           var extension = getExtension(testUrl) || defaultExtension;
+               context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
+               context.container().select('.preset-search-input').on('keydown.intro', null).on('keyup.intro', checkPresetSearch);
+               reveal('.preset-search-input', helpHtml('intro.areas.search_playground', {
+                 preset: playgroundPreset.name()
+               }));
+               context.history().on('change.intro', null);
+             }
+           });
 
-           if (extension) {
-             _template = null;
-             d3_text(url).then(function (data) {
-               drawData.setFile(extension, data);
-             })["catch"](function () {
-               /* ignore */
-             });
-           } else {
-             drawData.template(url);
+           function checkPresetSearch() {
+             var first = context.container().select('.preset-list-item:first-child');
+
+             if (first.classed('preset-leisure-playground')) {
+               reveal(first.select('.preset-list-button').node(), helpHtml('intro.areas.choose_playground', {
+                 preset: playgroundPreset.name()
+               }), {
+                 duration: 300
+               });
+               context.container().select('.preset-search-input').on('keydown.intro', eventCancel, true).on('keyup.intro', null);
+               context.history().on('change.intro', function () {
+                 continueTo(clickAddField);
+               });
+             }
            }
 
-           return this;
-         };
+           function continueTo(nextStep) {
+             context.container().select('.inspector-wrap').on('wheel.intro', null);
+             context.on('enter.intro', null);
+             context.history().on('change.intro', null);
+             context.container().select('.preset-search-input').on('keydown.intro keyup.intro', null);
+             nextStep();
+           }
+         }
 
-         drawData.getSrc = function () {
-           return _src || '';
-         };
+         function clickAddField() {
+           if (!_areaID || !context.hasEntity(_areaID)) {
+             return addArea();
+           }
 
-         drawData.fitZoom = function () {
-           var features = getFeatures(_geojson);
-           if (!features.length) return;
-           var map = context.map();
-           var viewport = map.trimmedExtent().polygon();
-           var coords = features.reduce(function (coords, feature) {
-             var geom = feature.geometry;
-             if (!geom) return coords;
-             var c = geom.coordinates;
-             /* eslint-disable no-fallthrough */
+           var ids = context.selectedIDs();
 
-             switch (geom.type) {
-               case 'Point':
-                 c = [c];
+           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _areaID) {
+             return searchPresets();
+           }
 
-               case 'MultiPoint':
-               case 'LineString':
-                 break;
+           if (!context.container().select('.form-field-description').empty()) {
+             return continueTo(describePlayground);
+           } // disallow scrolling
 
-               case 'MultiPolygon':
-                 c = utilArrayFlatten(c);
 
-               case 'Polygon':
-               case 'MultiLineString':
-                 c = utilArrayFlatten(c);
-                 break;
-             }
-             /* eslint-enable no-fallthrough */
+           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
+           timeout(function () {
+             // reset pane, in case user somehow happened to change it..
+             context.container().select('.inspector-wrap .panewrap').style('right', '0%'); // It's possible for the user to add a description in a previous step..
+             // If they did this already, just continue to next step.
 
+             var entity = context.entity(_areaID);
 
-             return utilArrayUnion(coords, c);
-           }, []);
+             if (entity.tags.description) {
+               return continueTo(play);
+             } // scroll "Add field" into view
 
-           if (!geoPolygonIntersectsPolygon(viewport, coords, true)) {
-             var extent = geoExtent(d3_geoBounds({
-               type: 'LineString',
-               coordinates: coords
-             }));
-             map.centerZoom(extent.center(), map.trimmedExtentZoom(extent));
-           }
 
-           return this;
-         };
+             var box = context.container().select('.more-fields').node().getBoundingClientRect();
 
-         init();
-         return drawData;
-       }
+             if (box.top > 300) {
+               var pane = context.container().select('.entity-editor-pane .inspector-body');
+               var start = pane.node().scrollTop;
+               var end = start + (box.top - 300);
+               pane.transition().duration(250).tween('scroll.inspector', function () {
+                 var node = this;
+                 var i = d3_interpolateNumber(start, end);
+                 return function (t) {
+                   node.scrollTop = i(t);
+                 };
+               });
+             }
 
-       function svgDebug(projection, context) {
-         function drawDebug(selection) {
-           var showTile = context.getDebug('tile');
-           var showCollision = context.getDebug('collision');
-           var showImagery = context.getDebug('imagery');
-           var showTouchTargets = context.getDebug('target');
-           var showDownloaded = context.getDebug('downloaded');
-           var debugData = [];
+             timeout(function () {
+               reveal('.more-fields .combobox-input', helpHtml('intro.areas.add_field', {
+                 name: nameField.label(),
+                 description: descriptionField.label()
+               }), {
+                 duration: 300
+               });
+               context.container().select('.more-fields .combobox-input').on('click.intro', function () {
+                 // Watch for the combobox to appear...
+                 var watcher;
+                 watcher = window.setInterval(function () {
+                   if (!context.container().select('div.combobox').empty()) {
+                     window.clearInterval(watcher);
+                     continueTo(chooseDescriptionField);
+                   }
+                 }, 300);
+               });
+             }, 300); // after "Add Field" visible
+           }, 400); // after editor pane visible
 
-           if (showTile) {
-             debugData.push({
-               "class": 'red',
-               label: 'tile'
-             });
-           }
+           context.on('exit.intro', function () {
+             return continueTo(searchPresets);
+           });
 
-           if (showCollision) {
-             debugData.push({
-               "class": 'yellow',
-               label: 'collision'
-             });
+           function continueTo(nextStep) {
+             context.container().select('.inspector-wrap').on('wheel.intro', null);
+             context.container().select('.more-fields .combobox-input').on('click.intro', null);
+             context.on('exit.intro', null);
+             nextStep();
            }
+         }
 
-           if (showImagery) {
-             debugData.push({
-               "class": 'orange',
-               label: 'imagery'
-             });
+         function chooseDescriptionField() {
+           if (!_areaID || !context.hasEntity(_areaID)) {
+             return addArea();
            }
 
-           if (showTouchTargets) {
-             debugData.push({
-               "class": 'pink',
-               label: 'touchTargets'
-             });
-           }
+           var ids = context.selectedIDs();
 
-           if (showDownloaded) {
-             debugData.push({
-               "class": 'purple',
-               label: 'downloaded'
-             });
+           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _areaID) {
+             return searchPresets();
            }
 
-           var legend = context.container().select('.main-content').selectAll('.debug-legend').data(debugData.length ? [0] : []);
-           legend.exit().remove();
-           legend = legend.enter().append('div').attr('class', 'fillD debug-legend').merge(legend);
-           var legendItems = legend.selectAll('.debug-legend-item').data(debugData, function (d) {
-             return d.label;
-           });
-           legendItems.exit().remove();
-           legendItems.enter().append('span').attr('class', function (d) {
-             return "debug-legend-item ".concat(d["class"]);
-           }).text(function (d) {
-             return d.label;
-           });
-           var layer = selection.selectAll('.layer-debug').data(showImagery || showDownloaded ? [0] : []);
-           layer.exit().remove();
-           layer = layer.enter().append('g').attr('class', 'layer-debug').merge(layer); // imagery
-
-           var extent = context.map().extent();
-           _mainFileFetcher.get('imagery').then(function (d) {
-             var hits = showImagery && d.query.bbox(extent.rectangle(), true) || [];
-             var features = hits.map(function (d) {
-               return d.features[d.id];
-             });
-             var imagery = layer.selectAll('path.debug-imagery').data(features);
-             imagery.exit().remove();
-             imagery.enter().append('path').attr('class', 'debug-imagery debug orange');
-           })["catch"](function () {
-             /* ignore */
-           }); // downloaded
+           if (!context.container().select('.form-field-description').empty()) {
+             return continueTo(describePlayground);
+           } // Make sure combobox is ready..
 
-           var osm = context.connection();
-           var dataDownloaded = [];
 
-           if (osm && showDownloaded) {
-             var rtree = osm.caches('get').tile.rtree;
-             dataDownloaded = rtree.all().map(function (bbox) {
-               return {
-                 type: 'Feature',
-                 properties: {
-                   id: bbox.id
-                 },
-                 geometry: {
-                   type: 'Polygon',
-                   coordinates: [[[bbox.minX, bbox.minY], [bbox.minX, bbox.maxY], [bbox.maxX, bbox.maxY], [bbox.maxX, bbox.minY], [bbox.minX, bbox.minY]]]
-                 }
-               };
-             });
-           }
+           if (context.container().select('div.combobox').empty()) {
+             return continueTo(clickAddField);
+           } // Watch for the combobox to go away..
 
-           var downloaded = layer.selectAll('path.debug-downloaded').data(showDownloaded ? dataDownloaded : []);
-           downloaded.exit().remove();
-           downloaded.enter().append('path').attr('class', 'debug-downloaded debug purple'); // update
 
-           layer.selectAll('path').attr('d', svgPath(projection).geojson);
-         } // This looks strange because `enabled` methods on other layers are
-         // chainable getter/setters, and this one is just a getter.
+           var watcher;
+           watcher = window.setInterval(function () {
+             if (context.container().select('div.combobox').empty()) {
+               window.clearInterval(watcher);
+               timeout(function () {
+                 if (context.container().select('.form-field-description').empty()) {
+                   continueTo(retryChooseDescription);
+                 } else {
+                   continueTo(describePlayground);
+                 }
+               }, 300); // after description field added.
+             }
+           }, 300);
+           reveal('div.combobox', helpHtml('intro.areas.choose_field', {
+             field: descriptionField.label()
+           }), {
+             duration: 300
+           });
+           context.on('exit.intro', function () {
+             return continueTo(searchPresets);
+           });
 
+           function continueTo(nextStep) {
+             if (watcher) window.clearInterval(watcher);
+             context.on('exit.intro', null);
+             nextStep();
+           }
+         }
 
-         drawDebug.enabled = function () {
-           if (!arguments.length) {
-             return context.getDebug('tile') || context.getDebug('collision') || context.getDebug('imagery') || context.getDebug('target') || context.getDebug('downloaded');
-           } else {
-             return this;
+         function describePlayground() {
+           if (!_areaID || !context.hasEntity(_areaID)) {
+             return addArea();
            }
-         };
 
-         return drawDebug;
-       }
+           var ids = context.selectedIDs();
 
-       /*
-           A standalone SVG element that contains only a `defs` sub-element. To be
-           used once globally, since defs IDs must be unique within a document.
-       */
+           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _areaID) {
+             return searchPresets();
+           } // reset pane, in case user happened to change it..
 
-       function svgDefs(context) {
-         var _defsSelection = select(null);
 
-         var _spritesheetIds = ['iD-sprite', 'maki-sprite', 'temaki-sprite', 'fa-sprite', 'community-sprite'];
+           context.container().select('.inspector-wrap .panewrap').style('right', '0%');
 
-         function drawDefs(selection) {
-           _defsSelection = selection.append('defs'); // add markers
+           if (context.container().select('.form-field-description').empty()) {
+             return continueTo(retryChooseDescription);
+           }
 
-           _defsSelection.append('marker').attr('id', 'ideditor-oneway-marker').attr('viewBox', '0 0 10 5').attr('refX', 2.5).attr('refY', 2.5).attr('markerWidth', 2).attr('markerHeight', 2).attr('markerUnits', 'strokeWidth').attr('orient', 'auto').append('path').attr('class', 'oneway-marker-path').attr('d', 'M 5,3 L 0,3 L 0,2 L 5,2 L 5,0 L 10,2.5 L 5,5 z').attr('stroke', 'none').attr('fill', '#000').attr('opacity', '0.75'); // SVG markers have to be given a colour where they're defined
-           // (they can't inherit it from the line they're attached to),
-           // so we need to manually define markers for each color of tag
-           // (also, it's slightly nicer if we can control the
-           // positioning for different tags)
+           context.on('exit.intro', function () {
+             continueTo(play);
+           });
+           reveal('.entity-editor-pane', helpHtml('intro.areas.describe_playground', {
+             button: icon('#iD-icon-close', 'inline')
+           }), {
+             duration: 300
+           });
 
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             nextStep();
+           }
+         }
 
-           function addSidedMarker(name, color, offset) {
-             _defsSelection.append('marker').attr('id', 'ideditor-sided-marker-' + name).attr('viewBox', '0 0 2 2').attr('refX', 1).attr('refY', -offset).attr('markerWidth', 1.5).attr('markerHeight', 1.5).attr('markerUnits', 'strokeWidth').attr('orient', 'auto').append('path').attr('class', 'sided-marker-path sided-marker-' + name + '-path').attr('d', 'M 0,0 L 1,1 L 2,0 z').attr('stroke', 'none').attr('fill', color);
+         function retryChooseDescription() {
+           if (!_areaID || !context.hasEntity(_areaID)) {
+             return addArea();
            }
 
-           addSidedMarker('natural', 'rgb(170, 170, 170)', 0); // for a coastline, the arrows are (somewhat unintuitively) on
-           // the water side, so let's color them blue (with a gap) to
-           // give a stronger indication
+           var ids = context.selectedIDs();
 
-           addSidedMarker('coastline', '#77dede', 1);
-           addSidedMarker('waterway', '#77dede', 1); // barriers have a dashed line, and separating the triangle
-           // from the line visually suits that
+           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _areaID) {
+             return searchPresets();
+           } // reset pane, in case user happened to change it..
 
-           addSidedMarker('barrier', '#ddd', 1);
-           addSidedMarker('man_made', '#fff', 0);
 
-           _defsSelection.append('marker').attr('id', 'ideditor-viewfield-marker').attr('viewBox', '0 0 16 16').attr('refX', 8).attr('refY', 16).attr('markerWidth', 4).attr('markerHeight', 4).attr('markerUnits', 'strokeWidth').attr('orient', 'auto').append('path').attr('class', 'viewfield-marker-path').attr('d', 'M 6,14 C 8,13.4 8,13.4 10,14 L 16,3 C 12,0 4,0 0,3 z').attr('fill', '#333').attr('fill-opacity', '0.75').attr('stroke', '#fff').attr('stroke-width', '0.5px').attr('stroke-opacity', '0.75');
+           context.container().select('.inspector-wrap .panewrap').style('right', '0%');
+           reveal('.entity-editor-pane', helpHtml('intro.areas.retry_add_field', {
+             field: descriptionField.label()
+           }), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: function buttonCallback() {
+               continueTo(clickAddField);
+             }
+           });
+           context.on('exit.intro', function () {
+             return continueTo(searchPresets);
+           });
 
-           _defsSelection.append('marker').attr('id', 'ideditor-viewfield-marker-wireframe').attr('viewBox', '0 0 16 16').attr('refX', 8).attr('refY', 16).attr('markerWidth', 4).attr('markerHeight', 4).attr('markerUnits', 'strokeWidth').attr('orient', 'auto').append('path').attr('class', 'viewfield-marker-path').attr('d', 'M 6,14 C 8,13.4 8,13.4 10,14 L 16,3 C 12,0 4,0 0,3 z').attr('fill', 'none').attr('stroke', '#fff').attr('stroke-width', '0.5px').attr('stroke-opacity', '0.75'); // add patterns
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             nextStep();
+           }
+         }
 
+         function play() {
+           dispatch.call('done');
+           reveal('.ideditor', helpHtml('intro.areas.play', {
+             next: _t('intro.lines.title')
+           }), {
+             tooltipBox: '.intro-nav-wrap .chapter-line',
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: function buttonCallback() {
+               reveal('.ideditor');
+             }
+           });
+         }
 
-           var patterns = _defsSelection.selectAll('pattern').data([// pattern name, pattern image name
-           ['beach', 'dots'], ['construction', 'construction'], ['cemetery', 'cemetery'], ['cemetery_christian', 'cemetery_christian'], ['cemetery_buddhist', 'cemetery_buddhist'], ['cemetery_muslim', 'cemetery_muslim'], ['cemetery_jewish', 'cemetery_jewish'], ['farmland', 'farmland'], ['farmyard', 'farmyard'], ['forest', 'forest'], ['forest_broadleaved', 'forest_broadleaved'], ['forest_needleleaved', 'forest_needleleaved'], ['forest_leafless', 'forest_leafless'], ['golf_green', 'grass'], ['grass', 'grass'], ['landfill', 'landfill'], ['meadow', 'grass'], ['orchard', 'orchard'], ['pond', 'pond'], ['quarry', 'quarry'], ['scrub', 'bushes'], ['vineyard', 'vineyard'], ['water_standing', 'lines'], ['waves', 'waves'], ['wetland', 'wetland'], ['wetland_marsh', 'wetland_marsh'], ['wetland_swamp', 'wetland_swamp'], ['wetland_bog', 'wetland_bog'], ['wetland_reedbed', 'wetland_reedbed']]).enter().append('pattern').attr('id', function (d) {
-             return 'ideditor-pattern-' + d[0];
-           }).attr('width', 32).attr('height', 32).attr('patternUnits', 'userSpaceOnUse');
+         chapter.enter = function () {
+           addArea();
+         };
 
-           patterns.append('rect').attr('x', 0).attr('y', 0).attr('width', 32).attr('height', 32).attr('class', function (d) {
-             return 'pattern-color-' + d[0];
-           });
-           patterns.append('image').attr('x', 0).attr('y', 0).attr('width', 32).attr('height', 32).attr('xlink:href', function (d) {
-             return context.imagePath('pattern/' + d[1] + '.png');
-           }); // add clip paths
+         chapter.exit = function () {
+           timeouts.forEach(window.clearTimeout);
+           context.on('enter.intro exit.intro', null);
+           context.map().on('move.intro drawn.intro', null);
+           context.history().on('change.intro', null);
+           context.container().select('.inspector-wrap').on('wheel.intro', null);
+           context.container().select('.preset-search-input').on('keydown.intro keyup.intro', null);
+           context.container().select('.more-fields .combobox-input').on('click.intro', null);
+         };
 
-           _defsSelection.selectAll('clipPath').data([12, 18, 20, 32, 45]).enter().append('clipPath').attr('id', function (d) {
-             return 'ideditor-clip-square-' + d;
-           }).append('rect').attr('x', 0).attr('y', 0).attr('width', function (d) {
-             return d;
-           }).attr('height', function (d) {
-             return d;
-           }); // add symbol spritesheets
+         chapter.restart = function () {
+           chapter.exit();
+           chapter.enter();
+         };
 
+         return utilRebind(chapter, dispatch, 'on');
+       }
 
-           addSprites(_spritesheetIds, true);
+       function uiIntroLine(context, reveal) {
+         var dispatch = dispatch$8('done');
+         var timeouts = [];
+         var _tulipRoadID = null;
+         var flowerRoadID = 'w646';
+         var tulipRoadStart = [-85.6297754121684, 41.95805253325314];
+         var tulipRoadMidpoint = [-85.62975395449628, 41.95787501510204];
+         var tulipRoadIntersection = [-85.62974496187628, 41.95742515554585];
+         var roadCategory = _mainPresetIndex.item('category-road_minor');
+         var residentialPreset = _mainPresetIndex.item('highway/residential');
+         var woodRoadID = 'w525';
+         var woodRoadEndID = 'n2862';
+         var woodRoadAddNode = [-85.62390110349587, 41.95397111462291];
+         var woodRoadDragEndpoint = [-85.623867390213, 41.95466987786487];
+         var woodRoadDragMidpoint = [-85.62386254803509, 41.95430395953872];
+         var washingtonStreetID = 'w522';
+         var twelfthAvenueID = 'w1';
+         var eleventhAvenueEndID = 'n3550';
+         var twelfthAvenueEndID = 'n5';
+         var _washingtonSegmentID = null;
+         var eleventhAvenueEnd = context.entity(eleventhAvenueEndID).loc;
+         var twelfthAvenueEnd = context.entity(twelfthAvenueEndID).loc;
+         var deleteLinesLoc = [-85.6219395542764, 41.95228033922477];
+         var twelfthAvenue = [-85.62219310052491, 41.952505413152956];
+         var chapter = {
+           title: 'intro.lines.title'
+         };
+
+         function timeout(f, t) {
+           timeouts.push(window.setTimeout(f, t));
          }
 
-         function addSprites(ids, overrideColors) {
-           _spritesheetIds = utilArrayUniq(_spritesheetIds.concat(ids));
+         function eventCancel(d3_event) {
+           d3_event.stopPropagation();
+           d3_event.preventDefault();
+         }
 
-           var spritesheets = _defsSelection.selectAll('.spritesheet').data(_spritesheetIds);
+         function addLine() {
+           context.enter(modeBrowse(context));
+           context.history().reset('initial');
+           var msec = transitionTime(tulipRoadStart, context.map().center());
 
-           spritesheets.enter().append('g').attr('class', function (d) {
-             return 'spritesheet spritesheet-' + d;
-           }).each(function (d) {
-             var url = context.imagePath(d + '.svg');
-             var node = select(this).node();
-             svg(url).then(function (svg) {
-               node.appendChild(select(svg.documentElement).attr('id', 'ideditor-' + d).node());
+           if (msec) {
+             reveal(null, null, {
+               duration: 0
+             });
+           }
 
-               if (overrideColors && d !== 'iD-sprite') {
-                 // allow icon colors to be overridden..
-                 select(node).selectAll('path').attr('fill', 'currentColor');
-               }
-             })["catch"](function () {
-               /* ignore */
+           context.map().centerZoomEase(tulipRoadStart, 18.5, msec);
+           timeout(function () {
+             var tooltip = reveal('button.add-line', helpHtml('intro.lines.add_line'));
+             tooltip.selectAll('.popover-inner').insert('svg', 'span').attr('class', 'tooltip-illustration').append('use').attr('xlink:href', '#iD-graphic-lines');
+             context.on('enter.intro', function (mode) {
+               if (mode.id !== 'add-line') return;
+               continueTo(startLine);
+             });
+           }, msec + 100);
+
+           function continueTo(nextStep) {
+             context.on('enter.intro', null);
+             nextStep();
+           }
+         }
+
+         function startLine() {
+           if (context.mode().id !== 'add-line') return chapter.restart();
+           _tulipRoadID = null;
+           var padding = 70 * Math.pow(2, context.map().zoom() - 18);
+           var box = pad(tulipRoadStart, padding, context);
+           box.height = box.height + 100;
+           var textId = context.lastPointerType() === 'mouse' ? 'start_line' : 'start_line_tap';
+           var startLineString = helpHtml('intro.lines.missing_road') + '{br}' + helpHtml('intro.lines.line_draw_info') + helpHtml('intro.lines.' + textId);
+           reveal(box, startLineString);
+           context.map().on('move.intro drawn.intro', function () {
+             padding = 70 * Math.pow(2, context.map().zoom() - 18);
+             box = pad(tulipRoadStart, padding, context);
+             box.height = box.height + 100;
+             reveal(box, startLineString, {
+               duration: 0
              });
            });
-           spritesheets.exit().remove();
+           context.on('enter.intro', function (mode) {
+             if (mode.id !== 'draw-line') return chapter.restart();
+             continueTo(drawLine);
+           });
+
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
+           }
          }
 
-         drawDefs.addSprites = addSprites;
-         return drawDefs;
-       }
+         function drawLine() {
+           if (context.mode().id !== 'draw-line') return chapter.restart();
+           _tulipRoadID = context.mode().selectedIDs()[0];
+           context.map().centerEase(tulipRoadMidpoint, 500);
+           timeout(function () {
+             var padding = 200 * Math.pow(2, context.map().zoom() - 18.5);
+             var box = pad(tulipRoadMidpoint, padding, context);
+             box.height = box.height * 2;
+             reveal(box, helpHtml('intro.lines.intersect', {
+               name: _t('intro.graph.name.flower-street')
+             }));
+             context.map().on('move.intro drawn.intro', function () {
+               padding = 200 * Math.pow(2, context.map().zoom() - 18.5);
+               box = pad(tulipRoadMidpoint, padding, context);
+               box.height = box.height * 2;
+               reveal(box, helpHtml('intro.lines.intersect', {
+                 name: _t('intro.graph.name.flower-street')
+               }), {
+                 duration: 0
+               });
+             });
+           }, 550); // after easing..
 
-       var _layerEnabled = false;
+           context.history().on('change.intro', function () {
+             if (isLineConnected()) {
+               continueTo(continueLine);
+             }
+           });
+           context.on('enter.intro', function (mode) {
+             if (mode.id === 'draw-line') {
+               return;
+             } else if (mode.id === 'select') {
+               continueTo(retryIntersect);
+               return;
+             } else {
+               return chapter.restart();
+             }
+           });
 
-       var _qaService;
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.history().on('change.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
+           }
+         }
 
-       function svgKeepRight(projection, context, dispatch) {
-         var throttledRedraw = throttle(function () {
-           return dispatch.call('change');
-         }, 1000);
+         function isLineConnected() {
+           var entity = _tulipRoadID && context.hasEntity(_tulipRoadID);
 
-         var minZoom = 12;
-         var touchLayer = select(null);
-         var drawLayer = select(null);
-         var layerVisible = false;
+           if (!entity) return false;
+           var drawNodes = context.graph().childNodes(entity);
+           return drawNodes.some(function (node) {
+             return context.graph().parentWays(node).some(function (parent) {
+               return parent.id === flowerRoadID;
+             });
+           });
+         }
 
-         function markerPath(selection, klass) {
-           selection.attr('class', klass).attr('transform', 'translate(-4, -24)').attr('d', 'M11.6,6.2H7.1l1.4-5.1C8.6,0.6,8.1,0,7.5,0H2.2C1.7,0,1.3,0.3,1.3,0.8L0,10.2c-0.1,0.6,0.4,1.1,0.9,1.1h4.6l-1.8,7.6C3.6,19.4,4.1,20,4.7,20c0.3,0,0.6-0.2,0.8-0.5l6.9-11.9C12.7,7,12.3,6.2,11.6,6.2z');
-         } // Loosely-coupled keepRight service for fetching issues.
+         function retryIntersect() {
+           select(window).on('pointerdown.intro mousedown.intro', eventCancel, true);
+           var box = pad(tulipRoadIntersection, 80, context);
+           reveal(box, helpHtml('intro.lines.retry_intersect', {
+             name: _t('intro.graph.name.flower-street')
+           }));
+           timeout(chapter.restart, 3000);
+         }
 
+         function continueLine() {
+           if (context.mode().id !== 'draw-line') return chapter.restart();
 
-         function getService() {
-           if (services.keepRight && !_qaService) {
-             _qaService = services.keepRight;
+           var entity = _tulipRoadID && context.hasEntity(_tulipRoadID);
 
-             _qaService.on('loaded', throttledRedraw);
-           } else if (!services.keepRight && _qaService) {
-             _qaService = null;
+           if (!entity) return chapter.restart();
+           context.map().centerEase(tulipRoadIntersection, 500);
+           var continueLineText = helpHtml('intro.lines.continue_line') + '{br}' + helpHtml('intro.lines.finish_line_' + (context.lastPointerType() === 'mouse' ? 'click' : 'tap')) + helpHtml('intro.lines.finish_road');
+           reveal('.surface', continueLineText);
+           context.on('enter.intro', function (mode) {
+             if (mode.id === 'draw-line') {
+               return;
+             } else if (mode.id === 'select') {
+               return continueTo(chooseCategoryRoad);
+             } else {
+               return chapter.restart();
+             }
+           });
+
+           function continueTo(nextStep) {
+             context.on('enter.intro', null);
+             nextStep();
            }
+         }
 
-           return _qaService;
-         } // Show the markers
+         function chooseCategoryRoad() {
+           if (context.mode().id !== 'select') return chapter.restart();
+           context.on('exit.intro', function () {
+             return chapter.restart();
+           });
+           var button = context.container().select('.preset-category-road_minor .preset-list-button');
+           if (button.empty()) return chapter.restart(); // disallow scrolling
 
+           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
+           timeout(function () {
+             // reset pane, in case user somehow happened to change it..
+             context.container().select('.inspector-wrap .panewrap').style('right', '-100%');
+             reveal(button.node(), helpHtml('intro.lines.choose_category_road', {
+               category: roadCategory.name()
+             }));
+             button.on('click.intro', function () {
+               continueTo(choosePresetResidential);
+             });
+           }, 400); // after editor pane visible
 
-         function editOn() {
-           if (!layerVisible) {
-             layerVisible = true;
-             drawLayer.style('display', 'block');
+           function continueTo(nextStep) {
+             context.container().select('.inspector-wrap').on('wheel.intro', null);
+             context.container().select('.preset-list-button').on('click.intro', null);
+             context.on('exit.intro', null);
+             nextStep();
            }
-         } // Immediately remove the markers and their touch targets
+         }
 
+         function choosePresetResidential() {
+           if (context.mode().id !== 'select') return chapter.restart();
+           context.on('exit.intro', function () {
+             return chapter.restart();
+           });
+           var subgrid = context.container().select('.preset-category-road_minor .subgrid');
+           if (subgrid.empty()) return chapter.restart();
+           subgrid.selectAll(':not(.preset-highway-residential) .preset-list-button').on('click.intro', function () {
+             continueTo(retryPresetResidential);
+           });
+           subgrid.selectAll('.preset-highway-residential .preset-list-button').on('click.intro', function () {
+             continueTo(nameRoad);
+           });
+           timeout(function () {
+             reveal(subgrid.node(), helpHtml('intro.lines.choose_preset_residential', {
+               preset: residentialPreset.name()
+             }), {
+               tooltipBox: '.preset-highway-residential .preset-list-button',
+               duration: 300
+             });
+           }, 300);
 
-         function editOff() {
-           if (layerVisible) {
-             layerVisible = false;
-             drawLayer.style('display', 'none');
-             drawLayer.selectAll('.qaItem.keepRight').remove();
-             touchLayer.selectAll('.qaItem.keepRight').remove();
+           function continueTo(nextStep) {
+             context.container().select('.preset-list-button').on('click.intro', null);
+             context.on('exit.intro', null);
+             nextStep();
            }
-         } // Enable the layer.  This shows the markers and transitions them to visible.
+         } // selected wrong road type
 
 
-         function layerOn() {
-           editOn();
-           drawLayer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end interrupt', function () {
-             return dispatch.call('change');
-           });
-         } // Disable the layer.  This transitions the layer invisible and then hides the markers.
+         function retryPresetResidential() {
+           if (context.mode().id !== 'select') return chapter.restart();
+           context.on('exit.intro', function () {
+             return chapter.restart();
+           }); // disallow scrolling
 
+           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
+           timeout(function () {
+             var button = context.container().select('.entity-editor-pane .preset-list-button');
+             reveal(button.node(), helpHtml('intro.lines.retry_preset_residential', {
+               preset: residentialPreset.name()
+             }));
+             button.on('click.intro', function () {
+               continueTo(chooseCategoryRoad);
+             });
+           }, 500);
 
-         function layerOff() {
-           throttledRedraw.cancel();
-           drawLayer.interrupt();
-           touchLayer.selectAll('.qaItem.keepRight').remove();
-           drawLayer.transition().duration(250).style('opacity', 0).on('end interrupt', function () {
-             editOff();
-             dispatch.call('change');
+           function continueTo(nextStep) {
+             context.container().select('.inspector-wrap').on('wheel.intro', null);
+             context.container().select('.preset-list-button').on('click.intro', null);
+             context.on('exit.intro', null);
+             nextStep();
+           }
+         }
+
+         function nameRoad() {
+           context.on('exit.intro', function () {
+             continueTo(didNameRoad);
            });
-         } // Update the issue markers
+           timeout(function () {
+             reveal('.entity-editor-pane', helpHtml('intro.lines.name_road', {
+               button: icon('#iD-icon-close', 'inline')
+             }), {
+               tooltipClass: 'intro-lines-name_road'
+             });
+           }, 500);
 
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             nextStep();
+           }
+         }
 
-         function updateMarkers() {
-           if (!layerVisible || !_layerEnabled) return;
-           var service = getService();
-           var selectedID = context.selectedErrorID();
-           var data = service ? service.getItems(projection) : [];
-           var getTransform = svgPointTransform(projection); // Draw markers..
+         function didNameRoad() {
+           context.history().checkpoint('doneAddLine');
+           timeout(function () {
+             reveal('.surface', helpHtml('intro.lines.did_name_road'), {
+               buttonText: _t.html('intro.ok'),
+               buttonCallback: function buttonCallback() {
+                 continueTo(updateLine);
+               }
+             });
+           }, 500);
 
-           var markers = drawLayer.selectAll('.qaItem.keepRight').data(data, function (d) {
-             return d.id;
-           }); // exit
+           function continueTo(nextStep) {
+             nextStep();
+           }
+         }
 
-           markers.exit().remove(); // enter
+         function updateLine() {
+           context.history().reset('doneAddLine');
 
-           var markersEnter = markers.enter().append('g').attr('class', function (d) {
-             return "qaItem ".concat(d.service, " itemId-").concat(d.id, " itemType-").concat(d.parentIssueType);
-           });
-           markersEnter.append('ellipse').attr('cx', 0.5).attr('cy', 1).attr('rx', 6.5).attr('ry', 3).attr('class', 'stroke');
-           markersEnter.append('path').call(markerPath, 'shadow');
-           markersEnter.append('use').attr('class', 'qaItem-fill').attr('width', '20px').attr('height', '20px').attr('x', '-8px').attr('y', '-22px').attr('xlink:href', '#iD-icon-bolt'); // update
+           if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+             return chapter.restart();
+           }
 
-           markers.merge(markersEnter).sort(sortY).classed('selected', function (d) {
-             return d.id === selectedID;
-           }).attr('transform', getTransform); // Draw targets..
+           var msec = transitionTime(woodRoadDragMidpoint, context.map().center());
 
-           if (touchLayer.empty()) return;
-           var fillClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
-           var targets = touchLayer.selectAll('.qaItem.keepRight').data(data, function (d) {
-             return d.id;
-           }); // exit
+           if (msec) {
+             reveal(null, null, {
+               duration: 0
+             });
+           }
+
+           context.map().centerZoomEase(woodRoadDragMidpoint, 19, msec);
+           timeout(function () {
+             var padding = 250 * Math.pow(2, context.map().zoom() - 19);
+             var box = pad(woodRoadDragMidpoint, padding, context);
 
-           targets.exit().remove(); // enter/update
+             var advance = function advance() {
+               continueTo(addNode);
+             };
 
-           targets.enter().append('rect').attr('width', '20px').attr('height', '20px').attr('x', '-8px').attr('y', '-22px').merge(targets).sort(sortY).attr('class', function (d) {
-             return "qaItem ".concat(d.service, " target ").concat(fillClass, " itemId-").concat(d.id);
-           }).attr('transform', getTransform);
+             reveal(box, helpHtml('intro.lines.update_line'), {
+               buttonText: _t.html('intro.ok'),
+               buttonCallback: advance
+             });
+             context.map().on('move.intro drawn.intro', function () {
+               var padding = 250 * Math.pow(2, context.map().zoom() - 19);
+               var box = pad(woodRoadDragMidpoint, padding, context);
+               reveal(box, helpHtml('intro.lines.update_line'), {
+                 duration: 0,
+                 buttonText: _t.html('intro.ok'),
+                 buttonCallback: advance
+               });
+             });
+           }, msec + 100);
 
-           function sortY(a, b) {
-             return a.id === selectedID ? 1 : b.id === selectedID ? -1 : a.severity === 'error' && b.severity !== 'error' ? 1 : b.severity === 'error' && a.severity !== 'error' ? -1 : b.loc[1] - a.loc[1];
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             nextStep();
            }
-         } // Draw the keepRight layer and schedule loading issues and updating markers.
-
+         }
 
-         function drawKeepRight(selection) {
-           var service = getService();
-           var surface = context.surface();
+         function addNode() {
+           context.history().reset('doneAddLine');
 
-           if (surface && !surface.empty()) {
-             touchLayer = surface.selectAll('.data-layer.touch .layer-touch.markers');
+           if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+             return chapter.restart();
            }
 
-           drawLayer = selection.selectAll('.layer-keepRight').data(service ? [0] : []);
-           drawLayer.exit().remove();
-           drawLayer = drawLayer.enter().append('g').attr('class', 'layer-keepRight').style('display', _layerEnabled ? 'block' : 'none').merge(drawLayer);
+           var padding = 40 * Math.pow(2, context.map().zoom() - 19);
+           var box = pad(woodRoadAddNode, padding, context);
+           var addNodeString = helpHtml('intro.lines.add_node' + (context.lastPointerType() === 'mouse' ? '' : '_touch'));
+           reveal(box, addNodeString);
+           context.map().on('move.intro drawn.intro', function () {
+             var padding = 40 * Math.pow(2, context.map().zoom() - 19);
+             var box = pad(woodRoadAddNode, padding, context);
+             reveal(box, addNodeString, {
+               duration: 0
+             });
+           });
+           context.history().on('change.intro', function (changed) {
+             if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+               return continueTo(updateLine);
+             }
 
-           if (_layerEnabled) {
-             if (service && ~~context.map().zoom() >= minZoom) {
-               editOn();
-               service.loadIssues(projection);
-               updateMarkers();
-             } else {
-               editOff();
+             if (changed.created().length === 1) {
+               timeout(function () {
+                 continueTo(startDragEndpoint);
+               }, 500);
+             }
+           });
+           context.on('enter.intro', function (mode) {
+             if (mode.id !== 'select') {
+               continueTo(updateLine);
              }
+           });
+
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.history().on('change.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
            }
-         } // Toggles the layer on and off
+         }
 
+         function startDragEndpoint() {
+           if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+             return continueTo(updateLine);
+           }
 
-         drawKeepRight.enabled = function (val) {
-           if (!arguments.length) return _layerEnabled;
-           _layerEnabled = val;
+           var padding = 100 * Math.pow(2, context.map().zoom() - 19);
+           var box = pad(woodRoadDragEndpoint, padding, context);
+           var startDragString = helpHtml('intro.lines.start_drag_endpoint' + (context.lastPointerType() === 'mouse' ? '' : '_touch')) + helpHtml('intro.lines.drag_to_intersection');
+           reveal(box, startDragString);
+           context.map().on('move.intro drawn.intro', function () {
+             if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+               return continueTo(updateLine);
+             }
 
-           if (_layerEnabled) {
-             layerOn();
-           } else {
-             layerOff();
+             var padding = 100 * Math.pow(2, context.map().zoom() - 19);
+             var box = pad(woodRoadDragEndpoint, padding, context);
+             reveal(box, startDragString, {
+               duration: 0
+             });
+             var entity = context.entity(woodRoadEndID);
 
-             if (context.selectedErrorID()) {
-               context.enter(modeBrowse(context));
+             if (geoSphericalDistance(entity.loc, woodRoadDragEndpoint) <= 4) {
+               continueTo(finishDragEndpoint);
              }
-           }
-
-           dispatch.call('change');
-           return this;
-         };
+           });
 
-         drawKeepRight.supported = function () {
-           return !!getService();
-         };
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             nextStep();
+           }
+         }
 
-         return drawKeepRight;
-       }
+         function finishDragEndpoint() {
+           if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+             return continueTo(updateLine);
+           }
 
-       function svgGeolocate(projection) {
-         var layer = select(null);
+           var padding = 100 * Math.pow(2, context.map().zoom() - 19);
+           var box = pad(woodRoadDragEndpoint, padding, context);
+           var finishDragString = helpHtml('intro.lines.spot_looks_good') + helpHtml('intro.lines.finish_drag_endpoint' + (context.lastPointerType() === 'mouse' ? '' : '_touch'));
+           reveal(box, finishDragString);
+           context.map().on('move.intro drawn.intro', function () {
+             if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+               return continueTo(updateLine);
+             }
 
-         var _position;
+             var padding = 100 * Math.pow(2, context.map().zoom() - 19);
+             var box = pad(woodRoadDragEndpoint, padding, context);
+             reveal(box, finishDragString, {
+               duration: 0
+             });
+             var entity = context.entity(woodRoadEndID);
 
-         function init() {
-           if (svgGeolocate.initialized) return; // run once
+             if (geoSphericalDistance(entity.loc, woodRoadDragEndpoint) > 4) {
+               continueTo(startDragEndpoint);
+             }
+           });
+           context.on('enter.intro', function () {
+             continueTo(startDragMidpoint);
+           });
 
-           svgGeolocate.enabled = false;
-           svgGeolocate.initialized = true;
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
+           }
          }
 
-         function showLayer() {
-           layer.style('display', 'block');
-         }
+         function startDragMidpoint() {
+           if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+             return continueTo(updateLine);
+           }
 
-         function hideLayer() {
-           layer.transition().duration(250).style('opacity', 0);
-         }
+           if (context.selectedIDs().indexOf(woodRoadID) === -1) {
+             context.enter(modeSelect(context, [woodRoadID]));
+           }
 
-         function layerOn() {
-           layer.style('opacity', 0).transition().duration(250).style('opacity', 1);
-         }
+           var padding = 80 * Math.pow(2, context.map().zoom() - 19);
+           var box = pad(woodRoadDragMidpoint, padding, context);
+           reveal(box, helpHtml('intro.lines.start_drag_midpoint'));
+           context.map().on('move.intro drawn.intro', function () {
+             if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+               return continueTo(updateLine);
+             }
 
-         function layerOff() {
-           layer.style('display', 'none');
-         }
+             var padding = 80 * Math.pow(2, context.map().zoom() - 19);
+             var box = pad(woodRoadDragMidpoint, padding, context);
+             reveal(box, helpHtml('intro.lines.start_drag_midpoint'), {
+               duration: 0
+             });
+           });
+           context.history().on('change.intro', function (changed) {
+             if (changed.created().length === 1) {
+               continueTo(continueDragMidpoint);
+             }
+           });
+           context.on('enter.intro', function (mode) {
+             if (mode.id !== 'select') {
+               // keep Wood Road selected so midpoint triangles are drawn..
+               context.enter(modeSelect(context, [woodRoadID]));
+             }
+           });
 
-         function transform(d) {
-           return svgPointTransform(projection)(d);
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.history().on('change.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
+           }
          }
 
-         function accuracy(accuracy, loc) {
-           // converts accuracy to pixels...
-           var degreesRadius = geoMetersToLat(accuracy),
-               tangentLoc = [loc[0], loc[1] + degreesRadius],
-               projectedTangent = projection(tangentLoc),
-               projectedLoc = projection([loc[0], loc[1]]); // southern most point will have higher pixel value...
+         function continueDragMidpoint() {
+           if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+             return continueTo(updateLine);
+           }
 
-           return Math.round(projectedLoc[1] - projectedTangent[1]).toString();
-         }
+           var padding = 100 * Math.pow(2, context.map().zoom() - 19);
+           var box = pad(woodRoadDragEndpoint, padding, context);
+           box.height += 400;
 
-         function update() {
-           var geolocation = {
-             loc: [_position.coords.longitude, _position.coords.latitude]
+           var advance = function advance() {
+             context.history().checkpoint('doneUpdateLine');
+             continueTo(deleteLines);
            };
-           var groups = layer.selectAll('.geolocations').selectAll('.geolocation').data([geolocation]);
-           groups.exit().remove();
-           var pointsEnter = groups.enter().append('g').attr('class', 'geolocation');
-           pointsEnter.append('circle').attr('class', 'geolocate-radius').attr('dx', '0').attr('dy', '0').attr('fill', 'rgb(15,128,225)').attr('fill-opacity', '0.3').attr('r', '0');
-           pointsEnter.append('circle').attr('dx', '0').attr('dy', '0').attr('fill', 'rgb(15,128,225)').attr('stroke', 'white').attr('stroke-width', '1.5').attr('r', '6');
-           groups.merge(pointsEnter).attr('transform', transform);
-           layer.select('.geolocate-radius').attr('r', accuracy(_position.coords.accuracy, geolocation.loc));
-         }
 
-         function drawLocation(selection) {
-           var enabled = svgGeolocate.enabled;
-           layer = selection.selectAll('.layer-geolocate').data([0]);
-           layer.exit().remove();
-           var layerEnter = layer.enter().append('g').attr('class', 'layer-geolocate').style('display', enabled ? 'block' : 'none');
-           layerEnter.append('g').attr('class', 'geolocations');
-           layer = layerEnter.merge(layer);
+           reveal(box, helpHtml('intro.lines.continue_drag_midpoint'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: advance
+           });
+           context.map().on('move.intro drawn.intro', function () {
+             if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+               return continueTo(updateLine);
+             }
 
-           if (enabled) {
-             update();
-           } else {
-             layerOff();
+             var padding = 100 * Math.pow(2, context.map().zoom() - 19);
+             var box = pad(woodRoadDragEndpoint, padding, context);
+             box.height += 400;
+             reveal(box, helpHtml('intro.lines.continue_drag_midpoint'), {
+               duration: 0,
+               buttonText: _t.html('intro.ok'),
+               buttonCallback: advance
+             });
+           });
+
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             nextStep();
            }
          }
 
-         drawLocation.enabled = function (position, enabled) {
-           if (!arguments.length) return svgGeolocate.enabled;
-           _position = position;
-           svgGeolocate.enabled = enabled;
+         function deleteLines() {
+           context.history().reset('doneUpdateLine');
+           context.enter(modeBrowse(context));
 
-           if (svgGeolocate.enabled) {
-             showLayer();
-             layerOn();
-           } else {
-             hideLayer();
+           if (!context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+             return chapter.restart();
            }
 
-           return this;
-         };
-
-         init();
-         return drawLocation;
-       }
-
-       function svgLabels(projection, context) {
-         var path = d3_geoPath(projection);
-         var detected = utilDetect();
-         var baselineHack = detected.ie || detected.browser.toLowerCase() === 'edge' || detected.browser.toLowerCase() === 'firefox' && detected.version >= 70;
-
-         var _rdrawn = new RBush();
+           var msec = transitionTime(deleteLinesLoc, context.map().center());
 
-         var _rskipped = new RBush();
+           if (msec) {
+             reveal(null, null, {
+               duration: 0
+             });
+           }
 
-         var _textWidthCache = {};
-         var _entitybboxes = {}; // Listed from highest to lowest priority
+           context.map().centerZoomEase(deleteLinesLoc, 18, msec);
+           timeout(function () {
+             var padding = 200 * Math.pow(2, context.map().zoom() - 18);
+             var box = pad(deleteLinesLoc, padding, context);
+             box.top -= 200;
+             box.height += 400;
 
-         var labelStack = [['line', 'aeroway', '*', 12], ['line', 'highway', 'motorway', 12], ['line', 'highway', 'trunk', 12], ['line', 'highway', 'primary', 12], ['line', 'highway', 'secondary', 12], ['line', 'highway', 'tertiary', 12], ['line', 'highway', '*', 12], ['line', 'railway', '*', 12], ['line', 'waterway', '*', 12], ['area', 'aeroway', '*', 12], ['area', 'amenity', '*', 12], ['area', 'building', '*', 12], ['area', 'historic', '*', 12], ['area', 'leisure', '*', 12], ['area', 'man_made', '*', 12], ['area', 'natural', '*', 12], ['area', 'shop', '*', 12], ['area', 'tourism', '*', 12], ['area', 'camp_site', '*', 12], ['point', 'aeroway', '*', 10], ['point', 'amenity', '*', 10], ['point', 'building', '*', 10], ['point', 'historic', '*', 10], ['point', 'leisure', '*', 10], ['point', 'man_made', '*', 10], ['point', 'natural', '*', 10], ['point', 'shop', '*', 10], ['point', 'tourism', '*', 10], ['point', 'camp_site', '*', 10], ['line', 'name', '*', 12], ['area', 'name', '*', 12], ['point', 'name', '*', 10]];
+             var advance = function advance() {
+               continueTo(rightClickIntersection);
+             };
 
-         function shouldSkipIcon(preset) {
-           var noIcons = ['building', 'landuse', 'natural'];
-           return noIcons.some(function (s) {
-             return preset.id.indexOf(s) >= 0;
-           });
-         }
+             reveal(box, helpHtml('intro.lines.delete_lines', {
+               street: _t('intro.graph.name.12th-avenue')
+             }), {
+               buttonText: _t.html('intro.ok'),
+               buttonCallback: advance
+             });
+             context.map().on('move.intro drawn.intro', function () {
+               var padding = 200 * Math.pow(2, context.map().zoom() - 18);
+               var box = pad(deleteLinesLoc, padding, context);
+               box.top -= 200;
+               box.height += 400;
+               reveal(box, helpHtml('intro.lines.delete_lines', {
+                 street: _t('intro.graph.name.12th-avenue')
+               }), {
+                 duration: 0,
+                 buttonText: _t.html('intro.ok'),
+                 buttonCallback: advance
+               });
+             });
+             context.history().on('change.intro', function () {
+               timeout(function () {
+                 continueTo(deleteLines);
+               }, 500); // after any transition (e.g. if user deleted intersection)
+             });
+           }, msec + 100);
 
-         function get(array, prop) {
-           return function (d, i) {
-             return array[i][prop];
-           };
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
+           }
          }
 
-         function textWidth(text, size, elem) {
-           var c = _textWidthCache[size];
-           if (!c) c = _textWidthCache[size] = {};
-
-           if (c[text]) {
-             return c[text];
-           } else if (elem) {
-             c[text] = elem.getComputedTextLength();
-             return c[text];
-           } else {
-             var str = encodeURIComponent(text).match(/%[CDEFcdef]/g);
+         function rightClickIntersection() {
+           context.history().reset('doneUpdateLine');
+           context.enter(modeBrowse(context));
+           context.map().centerZoomEase(eleventhAvenueEnd, 18, 500);
+           var rightClickString = helpHtml('intro.lines.split_street', {
+             street1: _t('intro.graph.name.11th-avenue'),
+             street2: _t('intro.graph.name.washington-street')
+           }) + helpHtml('intro.lines.' + (context.lastPointerType() === 'mouse' ? 'rightclick_intersection' : 'edit_menu_intersection_touch'));
+           timeout(function () {
+             var padding = 60 * Math.pow(2, context.map().zoom() - 18);
+             var box = pad(eleventhAvenueEnd, padding, context);
+             reveal(box, rightClickString);
+             context.map().on('move.intro drawn.intro', function () {
+               var padding = 60 * Math.pow(2, context.map().zoom() - 18);
+               var box = pad(eleventhAvenueEnd, padding, context);
+               reveal(box, rightClickString, {
+                 duration: 0
+               });
+             });
+             context.on('enter.intro', function (mode) {
+               if (mode.id !== 'select') return;
+               var ids = context.selectedIDs();
+               if (ids.length !== 1 || ids[0] !== eleventhAvenueEndID) return;
+               timeout(function () {
+                 var node = selectMenuItem(context, 'split').node();
+                 if (!node) return;
+                 continueTo(splitIntersection);
+               }, 50); // after menu visible
+             });
+             context.history().on('change.intro', function () {
+               timeout(function () {
+                 continueTo(deleteLines);
+               }, 300); // after any transition (e.g. if user deleted intersection)
+             });
+           }, 600);
 
-             if (str === null) {
-               return size / 3 * 2 * text.length;
-             } else {
-               return size / 3 * (2 * text.length + str.length);
-             }
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('enter.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
            }
          }
 
-         function drawLinePaths(selection, entities, filter, classes, labels) {
-           var paths = selection.selectAll('path').filter(filter).data(entities, osmEntity.key); // exit
+         function splitIntersection() {
+           if (!context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+             return continueTo(deleteLines);
+           }
 
-           paths.exit().remove(); // enter/update
+           var node = selectMenuItem(context, 'split').node();
 
-           paths.enter().append('path').style('stroke-width', get(labels, 'font-size')).attr('id', function (d) {
-             return 'ideditor-labelpath-' + d.id;
-           }).attr('class', classes).merge(paths).attr('d', get(labels, 'lineString'));
-         }
+           if (!node) {
+             return continueTo(rightClickIntersection);
+           }
 
-         function drawLineLabels(selection, entities, filter, classes, labels) {
-           var texts = selection.selectAll('text.' + classes).filter(filter).data(entities, osmEntity.key); // exit
+           var wasChanged = false;
+           _washingtonSegmentID = null;
+           reveal('.edit-menu', helpHtml('intro.lines.split_intersection', {
+             street: _t('intro.graph.name.washington-street')
+           }), {
+             padding: 50
+           });
+           context.map().on('move.intro drawn.intro', function () {
+             var node = selectMenuItem(context, 'split').node();
 
-           texts.exit().remove(); // enter
+             if (!wasChanged && !node) {
+               return continueTo(rightClickIntersection);
+             }
 
-           texts.enter().append('text').attr('class', function (d, i) {
-             return classes + ' ' + labels[i].classes + ' ' + d.id;
-           }).attr('dy', baselineHack ? '0.35em' : null).append('textPath').attr('class', 'textpath'); // update
+             reveal('.edit-menu', helpHtml('intro.lines.split_intersection', {
+               street: _t('intro.graph.name.washington-street')
+             }), {
+               duration: 0,
+               padding: 50
+             });
+           });
+           context.history().on('change.intro', function (changed) {
+             wasChanged = true;
+             timeout(function () {
+               if (context.history().undoAnnotation() === _t('operations.split.annotation.line', {
+                 n: 1
+               })) {
+                 _washingtonSegmentID = changed.created()[0].id;
+                 continueTo(didSplit);
+               } else {
+                 _washingtonSegmentID = null;
+                 continueTo(retrySplit);
+               }
+             }, 300); // after any transition (e.g. if user deleted intersection)
+           });
 
-           selection.selectAll('text.' + classes).selectAll('.textpath').filter(filter).data(entities, osmEntity.key).attr('startOffset', '50%').attr('xlink:href', function (d) {
-             return '#ideditor-labelpath-' + d.id;
-           }).text(utilDisplayNameForPath);
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
+           }
          }
 
-         function drawPointLabels(selection, entities, filter, classes, labels) {
-           var texts = selection.selectAll('text.' + classes).filter(filter).data(entities, osmEntity.key); // exit
+         function retrySplit() {
+           context.enter(modeBrowse(context));
+           context.map().centerZoomEase(eleventhAvenueEnd, 18, 500);
 
-           texts.exit().remove(); // enter/update
+           var advance = function advance() {
+             continueTo(rightClickIntersection);
+           };
 
-           texts.enter().append('text').attr('class', function (d, i) {
-             return classes + ' ' + labels[i].classes + ' ' + d.id;
-           }).merge(texts).attr('x', get(labels, 'x')).attr('y', get(labels, 'y')).style('text-anchor', get(labels, 'textAnchor')).text(utilDisplayName).each(function (d, i) {
-             textWidth(utilDisplayName(d), labels[i].height, this);
+           var padding = 60 * Math.pow(2, context.map().zoom() - 18);
+           var box = pad(eleventhAvenueEnd, padding, context);
+           reveal(box, helpHtml('intro.lines.retry_split'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: advance
+           });
+           context.map().on('move.intro drawn.intro', function () {
+             var padding = 60 * Math.pow(2, context.map().zoom() - 18);
+             var box = pad(eleventhAvenueEnd, padding, context);
+             reveal(box, helpHtml('intro.lines.retry_split'), {
+               duration: 0,
+               buttonText: _t.html('intro.ok'),
+               buttonCallback: advance
+             });
            });
-         }
-
-         function drawAreaLabels(selection, entities, filter, classes, labels) {
-           entities = entities.filter(hasText);
-           labels = labels.filter(hasText);
-           drawPointLabels(selection, entities, filter, classes, labels);
 
-           function hasText(d, i) {
-             return labels[i].hasOwnProperty('x') && labels[i].hasOwnProperty('y');
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             nextStep();
            }
          }
 
-         function drawAreaIcons(selection, entities, filter, classes, labels) {
-           var icons = selection.selectAll('use.' + classes).filter(filter).data(entities, osmEntity.key); // exit
+         function didSplit() {
+           if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+             return continueTo(rightClickIntersection);
+           }
 
-           icons.exit().remove(); // enter/update
+           var ids = context.selectedIDs();
+           var string = 'intro.lines.did_split_' + (ids.length > 1 ? 'multi' : 'single');
+           var street = _t('intro.graph.name.washington-street');
+           var padding = 200 * Math.pow(2, context.map().zoom() - 18);
+           var box = pad(twelfthAvenue, padding, context);
+           box.width = box.width / 2;
+           reveal(box, helpHtml(string, {
+             street1: street,
+             street2: street
+           }), {
+             duration: 500
+           });
+           timeout(function () {
+             context.map().centerZoomEase(twelfthAvenue, 18, 500);
+             context.map().on('move.intro drawn.intro', function () {
+               var padding = 200 * Math.pow(2, context.map().zoom() - 18);
+               var box = pad(twelfthAvenue, padding, context);
+               box.width = box.width / 2;
+               reveal(box, helpHtml(string, {
+                 street1: street,
+                 street2: street
+               }), {
+                 duration: 0
+               });
+             });
+           }, 600); // after initial reveal and curtain cut
 
-           icons.enter().append('use').attr('class', 'icon ' + classes).attr('width', '17px').attr('height', '17px').merge(icons).attr('transform', get(labels, 'transform')).attr('xlink:href', function (d) {
-             var preset = _mainPresetIndex.match(d, context.graph());
-             var picon = preset && preset.icon;
+           context.on('enter.intro', function () {
+             var ids = context.selectedIDs();
 
-             if (!picon) {
-               return '';
-             } else {
-               var isMaki = /^maki-/.test(picon);
-               return '#' + picon + (isMaki ? '-15' : '');
+             if (ids.length === 1 && ids[0] === _washingtonSegmentID) {
+               continueTo(multiSelect);
+             }
+           });
+           context.history().on('change.intro', function () {
+             if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+               return continueTo(rightClickIntersection);
              }
            });
-         }
-
-         function drawCollisionBoxes(selection, rtree, which) {
-           var classes = 'debug ' + which + ' ' + (which === 'debug-skipped' ? 'orange' : 'yellow');
-           var gj = [];
 
-           if (context.getDebug('collision')) {
-             gj = rtree.all().map(function (d) {
-               return {
-                 type: 'Polygon',
-                 coordinates: [[[d.minX, d.minY], [d.maxX, d.minY], [d.maxX, d.maxY], [d.minX, d.maxY], [d.minX, d.minY]]]
-               };
-             });
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('enter.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
            }
-
-           var boxes = selection.selectAll('.' + which).data(gj); // exit
-
-           boxes.exit().remove(); // enter/update
-
-           boxes.enter().append('path').attr('class', classes).merge(boxes).attr('d', d3_geoPath());
          }
 
-         function drawLabels(selection, graph, entities, filter, dimensions, fullRedraw) {
-           var wireframe = context.surface().classed('fill-wireframe');
-           var zoom = geoScaleToZoom(projection.scale());
-           var labelable = [];
-           var renderNodeAs = {};
-           var i, j, k, entity, geometry;
-
-           for (i = 0; i < labelStack.length; i++) {
-             labelable.push([]);
+         function multiSelect() {
+           if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+             return continueTo(rightClickIntersection);
            }
 
-           if (fullRedraw) {
-             _rdrawn.clear();
-
-             _rskipped.clear();
+           var ids = context.selectedIDs();
+           var hasWashington = ids.indexOf(_washingtonSegmentID) !== -1;
+           var hasTwelfth = ids.indexOf(twelfthAvenueID) !== -1;
 
-             _entitybboxes = {};
-           } else {
-             for (i = 0; i < entities.length; i++) {
-               entity = entities[i];
-               var toRemove = [].concat(_entitybboxes[entity.id] || []).concat(_entitybboxes[entity.id + 'I'] || []);
+           if (hasWashington && hasTwelfth) {
+             return continueTo(multiRightClick);
+           } else if (!hasWashington && !hasTwelfth) {
+             return continueTo(didSplit);
+           }
 
-               for (j = 0; j < toRemove.length; j++) {
-                 _rdrawn.remove(toRemove[j]);
+           context.map().centerZoomEase(twelfthAvenue, 18, 500);
+           timeout(function () {
+             var selected, other, padding, box;
 
-                 _rskipped.remove(toRemove[j]);
-               }
+             if (hasWashington) {
+               selected = _t('intro.graph.name.washington-street');
+               other = _t('intro.graph.name.12th-avenue');
+               padding = 60 * Math.pow(2, context.map().zoom() - 18);
+               box = pad(twelfthAvenueEnd, padding, context);
+               box.width *= 3;
+             } else {
+               selected = _t('intro.graph.name.12th-avenue');
+               other = _t('intro.graph.name.washington-street');
+               padding = 200 * Math.pow(2, context.map().zoom() - 18);
+               box = pad(twelfthAvenue, padding, context);
+               box.width /= 2;
              }
-           } // Loop through all the entities to do some preprocessing
-
-
-           for (i = 0; i < entities.length; i++) {
-             entity = entities[i];
-             geometry = entity.geometry(graph); // Insert collision boxes around interesting points/vertices
-
-             if (geometry === 'point' || geometry === 'vertex' && isInterestingVertex(entity)) {
-               var hasDirections = entity.directions(graph, projection).length;
-               var markerPadding;
 
-               if (!wireframe && geometry === 'point' && !(zoom >= 18 && hasDirections)) {
-                 renderNodeAs[entity.id] = 'point';
-                 markerPadding = 20; // extra y for marker height
+             reveal(box, helpHtml('intro.lines.multi_select', {
+               selected: selected,
+               other1: other
+             }) + ' ' + helpHtml('intro.lines.add_to_selection_' + (context.lastPointerType() === 'mouse' ? 'click' : 'touch'), {
+               selected: selected,
+               other2: other
+             }));
+             context.map().on('move.intro drawn.intro', function () {
+               if (hasWashington) {
+                 selected = _t('intro.graph.name.washington-street');
+                 other = _t('intro.graph.name.12th-avenue');
+                 padding = 60 * Math.pow(2, context.map().zoom() - 18);
+                 box = pad(twelfthAvenueEnd, padding, context);
+                 box.width *= 3;
                } else {
-                 renderNodeAs[entity.id] = 'vertex';
-                 markerPadding = 0;
+                 selected = _t('intro.graph.name.12th-avenue');
+                 other = _t('intro.graph.name.washington-street');
+                 padding = 200 * Math.pow(2, context.map().zoom() - 18);
+                 box = pad(twelfthAvenue, padding, context);
+                 box.width /= 2;
                }
 
-               var coord = projection(entity.loc);
-               var nodePadding = 10;
-               var bbox = {
-                 minX: coord[0] - nodePadding,
-                 minY: coord[1] - nodePadding - markerPadding,
-                 maxX: coord[0] + nodePadding,
-                 maxY: coord[1] + nodePadding
-               };
-               doInsert(bbox, entity.id + 'P');
-             } // From here on, treat vertices like points
-
-
-             if (geometry === 'vertex') {
-               geometry = 'point';
-             } // Determine which entities are label-able
-
-
-             var preset = geometry === 'area' && _mainPresetIndex.match(entity, graph);
-             var icon = preset && !shouldSkipIcon(preset) && preset.icon;
-             if (!icon && !utilDisplayName(entity)) continue;
-
-             for (k = 0; k < labelStack.length; k++) {
-               var matchGeom = labelStack[k][0];
-               var matchKey = labelStack[k][1];
-               var matchVal = labelStack[k][2];
-               var hasVal = entity.tags[matchKey];
-
-               if (geometry === matchGeom && hasVal && (matchVal === '*' || matchVal === hasVal)) {
-                 labelable[k].push(entity);
-                 break;
+               reveal(box, helpHtml('intro.lines.multi_select', {
+                 selected: selected,
+                 other1: other
+               }) + ' ' + helpHtml('intro.lines.add_to_selection_' + (context.lastPointerType() === 'mouse' ? 'click' : 'touch'), {
+                 selected: selected,
+                 other2: other
+               }), {
+                 duration: 0
+               });
+             });
+             context.on('enter.intro', function () {
+               continueTo(multiSelect);
+             });
+             context.history().on('change.intro', function () {
+               if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+                 return continueTo(rightClickIntersection);
                }
-             }
-           }
-
-           var positions = {
-             point: [],
-             line: [],
-             area: []
-           };
-           var labelled = {
-             point: [],
-             line: [],
-             area: []
-           }; // Try and find a valid label for labellable entities
-
-           for (k = 0; k < labelable.length; k++) {
-             var fontSize = labelStack[k][3];
-
-             for (i = 0; i < labelable[k].length; i++) {
-               entity = labelable[k][i];
-               geometry = entity.geometry(graph);
-               var getName = geometry === 'line' ? utilDisplayNameForPath : utilDisplayName;
-               var name = getName(entity);
-               var width = name && textWidth(name, fontSize);
-               var p = null;
+             });
+           }, 600);
 
-               if (geometry === 'point' || geometry === 'vertex') {
-                 // no point or vertex labels in wireframe mode
-                 // no vertex labels at low zooms (vertices have no icons)
-                 if (wireframe) continue;
-                 var renderAs = renderNodeAs[entity.id];
-                 if (renderAs === 'vertex' && zoom < 17) continue;
-                 p = getPointLabel(entity, width, fontSize, renderAs);
-               } else if (geometry === 'line') {
-                 p = getLineLabel(entity, width, fontSize);
-               } else if (geometry === 'area') {
-                 p = getAreaLabel(entity, width, fontSize);
-               }
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('enter.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
+           }
+         }
 
-               if (p) {
-                 if (geometry === 'vertex') {
-                   geometry = 'point';
-                 } // treat vertex like point
+         function multiRightClick() {
+           if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+             return continueTo(rightClickIntersection);
+           }
 
+           var padding = 200 * Math.pow(2, context.map().zoom() - 18);
+           var box = pad(twelfthAvenue, padding, context);
+           var rightClickString = helpHtml('intro.lines.multi_select_success') + helpHtml('intro.lines.multi_' + (context.lastPointerType() === 'mouse' ? 'rightclick' : 'edit_menu_touch'));
+           reveal(box, rightClickString);
+           context.map().on('move.intro drawn.intro', function () {
+             var padding = 200 * Math.pow(2, context.map().zoom() - 18);
+             var box = pad(twelfthAvenue, padding, context);
+             reveal(box, rightClickString, {
+               duration: 0
+             });
+           });
+           context.ui().editMenu().on('toggled.intro', function (open) {
+             if (!open) return;
+             timeout(function () {
+               var ids = context.selectedIDs();
 
-                 p.classes = geometry + ' tag-' + labelStack[k][1];
-                 positions[geometry].push(p);
-                 labelled[geometry].push(entity);
+               if (ids.length === 2 && ids.indexOf(twelfthAvenueID) !== -1 && ids.indexOf(_washingtonSegmentID) !== -1) {
+                 var node = selectMenuItem(context, 'delete').node();
+                 if (!node) return;
+                 continueTo(multiDelete);
+               } else if (ids.length === 1 && ids.indexOf(_washingtonSegmentID) !== -1) {
+                 return continueTo(multiSelect);
+               } else {
+                 return continueTo(didSplit);
                }
+             }, 300); // after edit menu visible
+           });
+           context.history().on('change.intro', function () {
+             if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+               return continueTo(rightClickIntersection);
              }
-           }
+           });
 
-           function isInterestingVertex(entity) {
-             var selectedIDs = context.selectedIDs();
-             return entity.hasInterestingTags() || entity.isEndpoint(graph) || entity.isConnected(graph) || selectedIDs.indexOf(entity.id) !== -1 || graph.parentWays(entity).some(function (parent) {
-               return selectedIDs.indexOf(parent.id) !== -1;
-             });
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.ui().editMenu().on('toggled.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
            }
+         }
 
-           function getPointLabel(entity, width, height, geometry) {
-             var y = geometry === 'point' ? -12 : 0;
-             var pointOffsets = {
-               ltr: [15, y, 'start'],
-               rtl: [-15, y, 'end']
-             };
-             var textDirection = _mainLocalizer.textDirection();
-             var coord = projection(entity.loc);
-             var textPadding = 2;
-             var offset = pointOffsets[textDirection];
-             var p = {
-               height: height,
-               width: width,
-               x: coord[0] + offset[0],
-               y: coord[1] + offset[1],
-               textAnchor: offset[2]
-             }; // insert a collision box for the text label..
-
-             var bbox;
+         function multiDelete() {
+           if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+             return continueTo(rightClickIntersection);
+           }
 
-             if (textDirection === 'rtl') {
-               bbox = {
-                 minX: p.x - width - textPadding,
-                 minY: p.y - height / 2 - textPadding,
-                 maxX: p.x + textPadding,
-                 maxY: p.y + height / 2 + textPadding
-               };
+           var node = selectMenuItem(context, 'delete').node();
+           if (!node) return continueTo(multiRightClick);
+           reveal('.edit-menu', helpHtml('intro.lines.multi_delete'), {
+             padding: 50
+           });
+           context.map().on('move.intro drawn.intro', function () {
+             reveal('.edit-menu', helpHtml('intro.lines.multi_delete'), {
+               duration: 0,
+               padding: 50
+             });
+           });
+           context.on('exit.intro', function () {
+             if (context.hasEntity(_washingtonSegmentID) || context.hasEntity(twelfthAvenueID)) {
+               return continueTo(multiSelect); // left select mode but roads still exist
+             }
+           });
+           context.history().on('change.intro', function () {
+             if (context.hasEntity(_washingtonSegmentID) || context.hasEntity(twelfthAvenueID)) {
+               continueTo(retryDelete); // changed something but roads still exist
              } else {
-               bbox = {
-                 minX: p.x - textPadding,
-                 minY: p.y - height / 2 - textPadding,
-                 maxX: p.x + width + textPadding,
-                 maxY: p.y + height / 2 + textPadding
-               };
+               continueTo(play);
              }
+           });
 
-             if (tryInsert([bbox], entity.id, true)) {
-               return p;
-             }
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('exit.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
            }
+         }
 
-           function getLineLabel(entity, width, height) {
-             var viewport = geoExtent(context.projection.clipExtent()).polygon();
-             var points = graph.childNodes(entity).map(function (node) {
-               return projection(node.loc);
-             });
-             var length = geoPathLength(points);
-             if (length < width + 20) return; // % along the line to attempt to place the label
-
-             var lineOffsets = [50, 45, 55, 40, 60, 35, 65, 30, 70, 25, 75, 20, 80, 15, 95, 10, 90, 5, 95];
-             var padding = 3;
-
-             for (var i = 0; i < lineOffsets.length; i++) {
-               var offset = lineOffsets[i];
-               var middle = offset / 100 * length;
-               var start = middle - width / 2;
-               if (start < 0 || start + width > length) continue; // generate subpath and ignore paths that are invalid or don't cross viewport.
-
-               var sub = subpath(points, start, start + width);
-
-               if (!sub || !geoPolygonIntersectsPolygon(viewport, sub, true)) {
-                 continue;
-               }
-
-               var isReverse = reverse(sub);
-
-               if (isReverse) {
-                 sub = sub.reverse();
-               }
+         function retryDelete() {
+           context.enter(modeBrowse(context));
+           var padding = 200 * Math.pow(2, context.map().zoom() - 18);
+           var box = pad(twelfthAvenue, padding, context);
+           reveal(box, helpHtml('intro.lines.retry_delete'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: function buttonCallback() {
+               continueTo(multiSelect);
+             }
+           });
 
-               var bboxes = [];
-               var boxsize = (height + 2) / 2;
+           function continueTo(nextStep) {
+             nextStep();
+           }
+         }
 
-               for (var j = 0; j < sub.length - 1; j++) {
-                 var a = sub[j];
-                 var b = sub[j + 1]; // split up the text into small collision boxes
+         function play() {
+           dispatch.call('done');
+           reveal('.ideditor', helpHtml('intro.lines.play', {
+             next: _t('intro.buildings.title')
+           }), {
+             tooltipBox: '.intro-nav-wrap .chapter-building',
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: function buttonCallback() {
+               reveal('.ideditor');
+             }
+           });
+         }
 
-                 var num = Math.max(1, Math.floor(geoVecLength(a, b) / boxsize / 2));
+         chapter.enter = function () {
+           addLine();
+         };
 
-                 for (var box = 0; box < num; box++) {
-                   var p = geoVecInterp(a, b, box / num);
-                   var x0 = p[0] - boxsize - padding;
-                   var y0 = p[1] - boxsize - padding;
-                   var x1 = p[0] + boxsize + padding;
-                   var y1 = p[1] + boxsize + padding;
-                   bboxes.push({
-                     minX: Math.min(x0, x1),
-                     minY: Math.min(y0, y1),
-                     maxX: Math.max(x0, x1),
-                     maxY: Math.max(y0, y1)
-                   });
-                 }
-               }
+         chapter.exit = function () {
+           timeouts.forEach(window.clearTimeout);
+           select(window).on('pointerdown.intro mousedown.intro', null, true);
+           context.on('enter.intro exit.intro', null);
+           context.map().on('move.intro drawn.intro', null);
+           context.history().on('change.intro', null);
+           context.container().select('.inspector-wrap').on('wheel.intro', null);
+           context.container().select('.preset-list-button').on('click.intro', null);
+         };
 
-               if (tryInsert(bboxes, entity.id, false)) {
-                 // accept this one
-                 return {
-                   'font-size': height + 2,
-                   lineString: lineString(sub),
-                   startOffset: offset + '%'
-                 };
-               }
-             }
+         chapter.restart = function () {
+           chapter.exit();
+           chapter.enter();
+         };
 
-             function reverse(p) {
-               var angle = Math.atan2(p[1][1] - p[0][1], p[1][0] - p[0][0]);
-               return !(p[0][0] < p[p.length - 1][0] && angle < Math.PI / 2 && angle > -Math.PI / 2);
-             }
+         return utilRebind(chapter, dispatch, 'on');
+       }
 
-             function lineString(points) {
-               return 'M' + points.join('L');
-             }
+       function uiIntroBuilding(context, reveal) {
+         var dispatch = dispatch$8('done');
+         var house = [-85.62815, 41.95638];
+         var tank = [-85.62732, 41.95347];
+         var buildingCatetory = _mainPresetIndex.item('category-building');
+         var housePreset = _mainPresetIndex.item('building/house');
+         var tankPreset = _mainPresetIndex.item('man_made/storage_tank');
+         var timeouts = [];
+         var _houseID = null;
+         var _tankID = null;
+         var chapter = {
+           title: 'intro.buildings.title'
+         };
 
-             function subpath(points, from, to) {
-               var sofar = 0;
-               var start, end, i0, i1;
+         function timeout(f, t) {
+           timeouts.push(window.setTimeout(f, t));
+         }
 
-               for (var i = 0; i < points.length - 1; i++) {
-                 var a = points[i];
-                 var b = points[i + 1];
-                 var current = geoVecLength(a, b);
-                 var portion;
+         function eventCancel(d3_event) {
+           d3_event.stopPropagation();
+           d3_event.preventDefault();
+         }
 
-                 if (!start && sofar + current >= from) {
-                   portion = (from - sofar) / current;
-                   start = [a[0] + portion * (b[0] - a[0]), a[1] + portion * (b[1] - a[1])];
-                   i0 = i + 1;
-                 }
+         function revealHouse(center, text, options) {
+           var padding = 160 * Math.pow(2, context.map().zoom() - 20);
+           var box = pad(center, padding, context);
+           reveal(box, text, options);
+         }
 
-                 if (!end && sofar + current >= to) {
-                   portion = (to - sofar) / current;
-                   end = [a[0] + portion * (b[0] - a[0]), a[1] + portion * (b[1] - a[1])];
-                   i1 = i + 1;
-                 }
+         function revealTank(center, text, options) {
+           var padding = 190 * Math.pow(2, context.map().zoom() - 19.5);
+           var box = pad(center, padding, context);
+           reveal(box, text, options);
+         }
 
-                 sofar += current;
-               }
+         function addHouse() {
+           context.enter(modeBrowse(context));
+           context.history().reset('initial');
+           _houseID = null;
+           var msec = transitionTime(house, context.map().center());
 
-               var result = points.slice(i0, i1);
-               result.unshift(start);
-               result.push(end);
-               return result;
-             }
+           if (msec) {
+             reveal(null, null, {
+               duration: 0
+             });
            }
 
-           function getAreaLabel(entity, width, height) {
-             var centroid = path.centroid(entity.asGeoJSON(graph, true));
-             var extent = entity.extent(graph);
-             var areaWidth = projection(extent[1])[0] - projection(extent[0])[0];
-             if (isNaN(centroid[0]) || areaWidth < 20) return;
-             var preset = _mainPresetIndex.match(entity, context.graph());
-             var picon = preset && preset.icon;
-             var iconSize = 17;
-             var padding = 2;
-             var p = {};
+           context.map().centerZoomEase(house, 19, msec);
+           timeout(function () {
+             var tooltip = reveal('button.add-area', helpHtml('intro.buildings.add_building'));
+             tooltip.selectAll('.popover-inner').insert('svg', 'span').attr('class', 'tooltip-illustration').append('use').attr('xlink:href', '#iD-graphic-buildings');
+             context.on('enter.intro', function (mode) {
+               if (mode.id !== 'add-area') return;
+               continueTo(startHouse);
+             });
+           }, msec + 100);
 
-             if (picon) {
-               // icon and label..
-               if (addIcon()) {
-                 addLabel(iconSize + padding);
-                 return p;
-               }
-             } else {
-               // label only..
-               if (addLabel(0)) {
-                 return p;
-               }
-             }
+           function continueTo(nextStep) {
+             context.on('enter.intro', null);
+             nextStep();
+           }
+         }
 
-             function addIcon() {
-               var iconX = centroid[0] - iconSize / 2;
-               var iconY = centroid[1] - iconSize / 2;
-               var bbox = {
-                 minX: iconX,
-                 minY: iconY,
-                 maxX: iconX + iconSize,
-                 maxY: iconY + iconSize
-               };
+         function startHouse() {
+           if (context.mode().id !== 'add-area') {
+             return continueTo(addHouse);
+           }
 
-               if (tryInsert([bbox], entity.id + 'I', true)) {
-                 p.transform = 'translate(' + iconX + ',' + iconY + ')';
-                 return true;
-               }
+           _houseID = null;
+           context.map().zoomEase(20, 500);
+           timeout(function () {
+             var startString = helpHtml('intro.buildings.start_building') + helpHtml('intro.buildings.building_corner_' + (context.lastPointerType() === 'mouse' ? 'click' : 'tap'));
+             revealHouse(house, startString);
+             context.map().on('move.intro drawn.intro', function () {
+               revealHouse(house, startString, {
+                 duration: 0
+               });
+             });
+             context.on('enter.intro', function (mode) {
+               if (mode.id !== 'draw-area') return chapter.restart();
+               continueTo(continueHouse);
+             });
+           }, 550); // after easing
 
-               return false;
-             }
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
+           }
+         }
 
-             function addLabel(yOffset) {
-               if (width && areaWidth >= width + 20) {
-                 var labelX = centroid[0];
-                 var labelY = centroid[1] + yOffset;
-                 var bbox = {
-                   minX: labelX - width / 2 - padding,
-                   minY: labelY - height / 2 - padding,
-                   maxX: labelX + width / 2 + padding,
-                   maxY: labelY + height / 2 + padding
-                 };
+         function continueHouse() {
+           if (context.mode().id !== 'draw-area') {
+             return continueTo(addHouse);
+           }
 
-                 if (tryInsert([bbox], entity.id, true)) {
-                   p.x = labelX;
-                   p.y = labelY;
-                   p.textAnchor = 'middle';
-                   p.height = height;
-                   return true;
-                 }
-               }
+           _houseID = null;
+           var continueString = helpHtml('intro.buildings.continue_building') + '{br}' + helpHtml('intro.areas.finish_area_' + (context.lastPointerType() === 'mouse' ? 'click' : 'tap')) + helpHtml('intro.buildings.finish_building');
+           revealHouse(house, continueString);
+           context.map().on('move.intro drawn.intro', function () {
+             revealHouse(house, continueString, {
+               duration: 0
+             });
+           });
+           context.on('enter.intro', function (mode) {
+             if (mode.id === 'draw-area') {
+               return;
+             } else if (mode.id === 'select') {
+               var graph = context.graph();
+               var way = context.entity(context.selectedIDs()[0]);
+               var nodes = graph.childNodes(way);
+               var points = utilArrayUniq(nodes).map(function (n) {
+                 return context.projection(n.loc);
+               });
 
-               return false;
+               if (isMostlySquare(points)) {
+                 _houseID = way.id;
+                 return continueTo(chooseCategoryBuilding);
+               } else {
+                 return continueTo(retryHouse);
+               }
+             } else {
+               return chapter.restart();
              }
-           } // force insert a singular bounding box
-           // singular box only, no array, id better be unique
+           });
 
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
+           }
+         }
 
-           function doInsert(bbox, id) {
-             bbox.id = id;
-             var oldbox = _entitybboxes[id];
+         function retryHouse() {
+           var onClick = function onClick() {
+             continueTo(addHouse);
+           };
 
-             if (oldbox) {
-               _rdrawn.remove(oldbox);
-             }
+           revealHouse(house, helpHtml('intro.buildings.retry_building'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: onClick
+           });
+           context.map().on('move.intro drawn.intro', function () {
+             revealHouse(house, helpHtml('intro.buildings.retry_building'), {
+               duration: 0,
+               buttonText: _t.html('intro.ok'),
+               buttonCallback: onClick
+             });
+           });
 
-             _entitybboxes[id] = bbox;
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             nextStep();
+           }
+         }
 
-             _rdrawn.insert(bbox);
+         function chooseCategoryBuilding() {
+           if (!_houseID || !context.hasEntity(_houseID)) {
+             return addHouse();
            }
 
-           function tryInsert(bboxes, id, saveSkipped) {
-             var skipped = false;
+           var ids = context.selectedIDs();
 
-             for (var i = 0; i < bboxes.length; i++) {
-               var bbox = bboxes[i];
-               bbox.id = id; // Check that label is visible
+           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _houseID) {
+             context.enter(modeSelect(context, [_houseID]));
+           } // disallow scrolling
 
-               if (bbox.minX < 0 || bbox.minY < 0 || bbox.maxX > dimensions[0] || bbox.maxY > dimensions[1]) {
-                 skipped = true;
-                 break;
-               }
 
-               if (_rdrawn.collides(bbox)) {
-                 skipped = true;
-                 break;
-               }
+           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
+           timeout(function () {
+             // reset pane, in case user somehow happened to change it..
+             context.container().select('.inspector-wrap .panewrap').style('right', '-100%');
+             var button = context.container().select('.preset-category-building .preset-list-button');
+             reveal(button.node(), helpHtml('intro.buildings.choose_category_building', {
+               category: buildingCatetory.name()
+             }));
+             button.on('click.intro', function () {
+               button.on('click.intro', null);
+               continueTo(choosePresetHouse);
+             });
+           }, 400); // after preset list pane visible..
+
+           context.on('enter.intro', function (mode) {
+             if (!_houseID || !context.hasEntity(_houseID)) {
+               return continueTo(addHouse);
              }
 
-             _entitybboxes[id] = bboxes;
+             var ids = context.selectedIDs();
 
-             if (skipped) {
-               if (saveSkipped) {
-                 _rskipped.load(bboxes);
-               }
-             } else {
-               _rdrawn.load(bboxes);
+             if (mode.id !== 'select' || !ids.length || ids[0] !== _houseID) {
+               return continueTo(chooseCategoryBuilding);
              }
+           });
 
-             return !skipped;
+           function continueTo(nextStep) {
+             context.container().select('.inspector-wrap').on('wheel.intro', null);
+             context.container().select('.preset-list-button').on('click.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
            }
+         }
 
-           var layer = selection.selectAll('.layer-osm.labels');
-           layer.selectAll('.labels-group').data(['halo', 'label', 'debug']).enter().append('g').attr('class', function (d) {
-             return 'labels-group ' + d;
-           });
-           var halo = layer.selectAll('.labels-group.halo');
-           var label = layer.selectAll('.labels-group.label');
-           var debug = layer.selectAll('.labels-group.debug'); // points
+         function choosePresetHouse() {
+           if (!_houseID || !context.hasEntity(_houseID)) {
+             return addHouse();
+           }
 
-           drawPointLabels(label, labelled.point, filter, 'pointlabel', positions.point);
-           drawPointLabels(halo, labelled.point, filter, 'pointlabel-halo', positions.point); // lines
+           var ids = context.selectedIDs();
 
-           drawLinePaths(layer, labelled.line, filter, '', positions.line);
-           drawLineLabels(label, labelled.line, filter, 'linelabel', positions.line);
-           drawLineLabels(halo, labelled.line, filter, 'linelabel-halo', positions.line); // areas
+           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _houseID) {
+             context.enter(modeSelect(context, [_houseID]));
+           } // disallow scrolling
 
-           drawAreaLabels(label, labelled.area, filter, 'arealabel', positions.area);
-           drawAreaLabels(halo, labelled.area, filter, 'arealabel-halo', positions.area);
-           drawAreaIcons(label, labelled.area, filter, 'areaicon', positions.area);
-           drawAreaIcons(halo, labelled.area, filter, 'areaicon-halo', positions.area); // debug
 
-           drawCollisionBoxes(debug, _rskipped, 'debug-skipped');
-           drawCollisionBoxes(debug, _rdrawn, 'debug-drawn');
-           layer.call(filterLabels);
-         }
+           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
+           timeout(function () {
+             // reset pane, in case user somehow happened to change it..
+             context.container().select('.inspector-wrap .panewrap').style('right', '-100%');
+             var button = context.container().select('.preset-building-house .preset-list-button');
+             reveal(button.node(), helpHtml('intro.buildings.choose_preset_house', {
+               preset: housePreset.name()
+             }), {
+               duration: 300
+             });
+             button.on('click.intro', function () {
+               button.on('click.intro', null);
+               continueTo(closeEditorHouse);
+             });
+           }, 400); // after preset list pane visible..
 
-         function filterLabels(selection) {
-           var drawLayer = selection.selectAll('.layer-osm.labels');
-           var layers = drawLayer.selectAll('.labels-group.halo, .labels-group.label');
-           layers.selectAll('.nolabel').classed('nolabel', false);
-           var mouse = context.map().mouse();
-           var graph = context.graph();
-           var selectedIDs = context.selectedIDs();
-           var ids = [];
-           var pad, bbox; // hide labels near the mouse
+           context.on('enter.intro', function (mode) {
+             if (!_houseID || !context.hasEntity(_houseID)) {
+               return continueTo(addHouse);
+             }
 
-           if (mouse) {
-             pad = 20;
-             bbox = {
-               minX: mouse[0] - pad,
-               minY: mouse[1] - pad,
-               maxX: mouse[0] + pad,
-               maxY: mouse[1] + pad
-             };
+             var ids = context.selectedIDs();
 
-             var nearMouse = _rdrawn.search(bbox).map(function (entity) {
-               return entity.id;
-             });
+             if (mode.id !== 'select' || !ids.length || ids[0] !== _houseID) {
+               return continueTo(chooseCategoryBuilding);
+             }
+           });
 
-             ids.push.apply(ids, nearMouse);
-           } // hide labels on selected nodes (they look weird when dragging / haloed)
+           function continueTo(nextStep) {
+             context.container().select('.inspector-wrap').on('wheel.intro', null);
+             context.container().select('.preset-list-button').on('click.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
+           }
+         }
 
+         function closeEditorHouse() {
+           if (!_houseID || !context.hasEntity(_houseID)) {
+             return addHouse();
+           }
 
-           for (var i = 0; i < selectedIDs.length; i++) {
-             var entity = graph.hasEntity(selectedIDs[i]);
+           var ids = context.selectedIDs();
 
-             if (entity && entity.type === 'node') {
-               ids.push(selectedIDs[i]);
-             }
+           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _houseID) {
+             context.enter(modeSelect(context, [_houseID]));
            }
 
-           layers.selectAll(utilEntitySelector(ids)).classed('nolabel', true); // draw the mouse bbox if debugging is on..
-
-           var debug = selection.selectAll('.labels-group.debug');
-           var gj = [];
+           context.history().checkpoint('hasHouse');
+           context.on('exit.intro', function () {
+             continueTo(rightClickHouse);
+           });
+           timeout(function () {
+             reveal('.entity-editor-pane', helpHtml('intro.buildings.close', {
+               button: icon('#iD-icon-close', 'inline')
+             }));
+           }, 500);
 
-           if (context.getDebug('collision')) {
-             gj = bbox ? [{
-               type: 'Polygon',
-               coordinates: [[[bbox.minX, bbox.minY], [bbox.maxX, bbox.minY], [bbox.maxX, bbox.maxY], [bbox.minX, bbox.maxY], [bbox.minX, bbox.minY]]]
-             }] : [];
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             nextStep();
            }
+         }
 
-           var box = debug.selectAll('.debug-mouse').data(gj); // exit
+         function rightClickHouse() {
+           if (!_houseID) return chapter.restart();
+           context.enter(modeBrowse(context));
+           context.history().reset('hasHouse');
+           var zoom = context.map().zoom();
 
-           box.exit().remove(); // enter/update
+           if (zoom < 20) {
+             zoom = 20;
+           }
 
-           box.enter().append('path').attr('class', 'debug debug-mouse yellow').merge(box).attr('d', d3_geoPath());
+           context.map().centerZoomEase(house, zoom, 500);
+           context.on('enter.intro', function (mode) {
+             if (mode.id !== 'select') return;
+             var ids = context.selectedIDs();
+             if (ids.length !== 1 || ids[0] !== _houseID) return;
+             timeout(function () {
+               var node = selectMenuItem(context, 'orthogonalize').node();
+               if (!node) return;
+               continueTo(clickSquare);
+             }, 50); // after menu visible
+           });
+           context.map().on('move.intro drawn.intro', function () {
+             var rightclickString = helpHtml('intro.buildings.' + (context.lastPointerType() === 'mouse' ? 'rightclick_building' : 'edit_menu_building_touch'));
+             revealHouse(house, rightclickString, {
+               duration: 0
+             });
+           });
+           context.history().on('change.intro', function () {
+             continueTo(rightClickHouse);
+           });
+
+           function continueTo(nextStep) {
+             context.on('enter.intro', null);
+             context.map().on('move.intro drawn.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
+           }
          }
 
-         var throttleFilterLabels = throttle(filterLabels, 100);
+         function clickSquare() {
+           if (!_houseID) return chapter.restart();
+           var entity = context.hasEntity(_houseID);
+           if (!entity) return continueTo(rightClickHouse);
+           var node = selectMenuItem(context, 'orthogonalize').node();
 
-         drawLabels.observe = function (selection) {
-           var listener = function listener() {
-             throttleFilterLabels(selection);
-           };
+           if (!node) {
+             return continueTo(rightClickHouse);
+           }
 
-           selection.on('mousemove.hidelabels', listener);
-           context.on('enter.hidelabels', listener);
-         };
+           var wasChanged = false;
+           reveal('.edit-menu', helpHtml('intro.buildings.square_building'), {
+             padding: 50
+           });
+           context.on('enter.intro', function (mode) {
+             if (mode.id === 'browse') {
+               continueTo(rightClickHouse);
+             } else if (mode.id === 'move' || mode.id === 'rotate') {
+               continueTo(retryClickSquare);
+             }
+           });
+           context.map().on('move.intro', function () {
+             var node = selectMenuItem(context, 'orthogonalize').node();
 
-         drawLabels.off = function (selection) {
-           throttleFilterLabels.cancel();
-           selection.on('mousemove.hidelabels', null);
-           context.on('enter.hidelabels', null);
-         };
+             if (!wasChanged && !node) {
+               return continueTo(rightClickHouse);
+             }
 
-         return drawLabels;
-       }
+             reveal('.edit-menu', helpHtml('intro.buildings.square_building'), {
+               duration: 0,
+               padding: 50
+             });
+           });
+           context.history().on('change.intro', function () {
+             wasChanged = true;
+             context.history().on('change.intro', null); // Something changed.  Wait for transition to complete and check undo annotation.
 
-       var _layerEnabled$1 = false;
+             timeout(function () {
+               if (context.history().undoAnnotation() === _t('operations.orthogonalize.annotation.feature', {
+                 n: 1
+               })) {
+                 continueTo(doneSquare);
+               } else {
+                 continueTo(retryClickSquare);
+               }
+             }, 500); // after transitioned actions
+           });
 
-       var _qaService$1;
+           function continueTo(nextStep) {
+             context.on('enter.intro', null);
+             context.map().on('move.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
+           }
+         }
 
-       function svgImproveOSM(projection, context, dispatch) {
-         var throttledRedraw = throttle(function () {
-           return dispatch.call('change');
-         }, 1000);
+         function retryClickSquare() {
+           context.enter(modeBrowse(context));
+           revealHouse(house, helpHtml('intro.buildings.retry_square'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: function buttonCallback() {
+               continueTo(rightClickHouse);
+             }
+           });
 
-         var minZoom = 12;
-         var touchLayer = select(null);
-         var drawLayer = select(null);
-         var layerVisible = false;
+           function continueTo(nextStep) {
+             nextStep();
+           }
+         }
 
-         function markerPath(selection, klass) {
-           selection.attr('class', klass).attr('transform', 'translate(-10, -28)').attr('points', '16,3 4,3 1,6 1,17 4,20 7,20 10,27 13,20 16,20 19,17.033 19,6');
-         } // Loosely-coupled improveOSM service for fetching issues
+         function doneSquare() {
+           context.history().checkpoint('doneSquare');
+           revealHouse(house, helpHtml('intro.buildings.done_square'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: function buttonCallback() {
+               continueTo(addTank);
+             }
+           });
 
+           function continueTo(nextStep) {
+             nextStep();
+           }
+         }
 
-         function getService() {
-           if (services.improveOSM && !_qaService$1) {
-             _qaService$1 = services.improveOSM;
+         function addTank() {
+           context.enter(modeBrowse(context));
+           context.history().reset('doneSquare');
+           _tankID = null;
+           var msec = transitionTime(tank, context.map().center());
 
-             _qaService$1.on('loaded', throttledRedraw);
-           } else if (!services.improveOSM && _qaService$1) {
-             _qaService$1 = null;
+           if (msec) {
+             reveal(null, null, {
+               duration: 0
+             });
            }
 
-           return _qaService$1;
-         } // Show the markers
+           context.map().centerZoomEase(tank, 19.5, msec);
+           timeout(function () {
+             reveal('button.add-area', helpHtml('intro.buildings.add_tank'));
+             context.on('enter.intro', function (mode) {
+               if (mode.id !== 'add-area') return;
+               continueTo(startTank);
+             });
+           }, msec + 100);
 
+           function continueTo(nextStep) {
+             context.on('enter.intro', null);
+             nextStep();
+           }
+         }
 
-         function editOn() {
-           if (!layerVisible) {
-             layerVisible = true;
-             drawLayer.style('display', 'block');
+         function startTank() {
+           if (context.mode().id !== 'add-area') {
+             return continueTo(addTank);
            }
-         } // Immediately remove the markers and their touch targets
 
+           _tankID = null;
+           timeout(function () {
+             var startString = helpHtml('intro.buildings.start_tank') + helpHtml('intro.buildings.tank_edge_' + (context.lastPointerType() === 'mouse' ? 'click' : 'tap'));
+             revealTank(tank, startString);
+             context.map().on('move.intro drawn.intro', function () {
+               revealTank(tank, startString, {
+                 duration: 0
+               });
+             });
+             context.on('enter.intro', function (mode) {
+               if (mode.id !== 'draw-area') return chapter.restart();
+               continueTo(continueTank);
+             });
+           }, 550); // after easing
 
-         function editOff() {
-           if (layerVisible) {
-             layerVisible = false;
-             drawLayer.style('display', 'none');
-             drawLayer.selectAll('.qaItem.improveOSM').remove();
-             touchLayer.selectAll('.qaItem.improveOSM').remove();
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
            }
-         } // Enable the layer.  This shows the markers and transitions them to visible.
+         }
 
+         function continueTank() {
+           if (context.mode().id !== 'draw-area') {
+             return continueTo(addTank);
+           }
 
-         function layerOn() {
-           editOn();
-           drawLayer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end interrupt', function () {
-             return dispatch.call('change');
+           _tankID = null;
+           var continueString = helpHtml('intro.buildings.continue_tank') + '{br}' + helpHtml('intro.areas.finish_area_' + (context.lastPointerType() === 'mouse' ? 'click' : 'tap')) + helpHtml('intro.buildings.finish_tank');
+           revealTank(tank, continueString);
+           context.map().on('move.intro drawn.intro', function () {
+             revealTank(tank, continueString, {
+               duration: 0
+             });
            });
-         } // Disable the layer.  This transitions the layer invisible and then hides the markers.
-
-
-         function layerOff() {
-           throttledRedraw.cancel();
-           drawLayer.interrupt();
-           touchLayer.selectAll('.qaItem.improveOSM').remove();
-           drawLayer.transition().duration(250).style('opacity', 0).on('end interrupt', function () {
-             editOff();
-             dispatch.call('change');
+           context.on('enter.intro', function (mode) {
+             if (mode.id === 'draw-area') {
+               return;
+             } else if (mode.id === 'select') {
+               _tankID = context.selectedIDs()[0];
+               return continueTo(searchPresetTank);
+             } else {
+               return continueTo(addTank);
+             }
            });
-         } // Update the issue markers
 
+           function continueTo(nextStep) {
+             context.map().on('move.intro drawn.intro', null);
+             context.on('enter.intro', null);
+             nextStep();
+           }
+         }
 
-         function updateMarkers() {
-           if (!layerVisible || !_layerEnabled$1) return;
-           var service = getService();
-           var selectedID = context.selectedErrorID();
-           var data = service ? service.getItems(projection) : [];
-           var getTransform = svgPointTransform(projection); // Draw markers..
-
-           var markers = drawLayer.selectAll('.qaItem.improveOSM').data(data, function (d) {
-             return d.id;
-           }); // exit
+         function searchPresetTank() {
+           if (!_tankID || !context.hasEntity(_tankID)) {
+             return addTank();
+           }
 
-           markers.exit().remove(); // enter
+           var ids = context.selectedIDs();
 
-           var markersEnter = markers.enter().append('g').attr('class', function (d) {
-             return "qaItem ".concat(d.service, " itemId-").concat(d.id, " itemType-").concat(d.itemType);
-           });
-           markersEnter.append('polygon').call(markerPath, 'shadow');
-           markersEnter.append('ellipse').attr('cx', 0).attr('cy', 0).attr('rx', 4.5).attr('ry', 2).attr('class', 'stroke');
-           markersEnter.append('polygon').attr('fill', 'currentColor').call(markerPath, 'qaItem-fill');
-           markersEnter.append('use').attr('transform', 'translate(-6.5, -23)').attr('class', 'icon-annotation').attr('width', '13px').attr('height', '13px').attr('xlink:href', function (d) {
-             var picon = d.icon;
+           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _tankID) {
+             context.enter(modeSelect(context, [_tankID]));
+           } // disallow scrolling
 
-             if (!picon) {
-               return '';
-             } else {
-               var isMaki = /^maki-/.test(picon);
-               return "#".concat(picon).concat(isMaki ? '-11' : '');
-             }
-           }); // update
 
-           markers.merge(markersEnter).sort(sortY).classed('selected', function (d) {
-             return d.id === selectedID;
-           }).attr('transform', getTransform); // Draw targets..
+           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
+           timeout(function () {
+             // reset pane, in case user somehow happened to change it..
+             context.container().select('.inspector-wrap .panewrap').style('right', '-100%');
+             context.container().select('.preset-search-input').on('keydown.intro', null).on('keyup.intro', checkPresetSearch);
+             reveal('.preset-search-input', helpHtml('intro.buildings.search_tank', {
+               preset: tankPreset.name()
+             }));
+           }, 400); // after preset list pane visible..
 
-           if (touchLayer.empty()) return;
-           var fillClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
-           var targets = touchLayer.selectAll('.qaItem.improveOSM').data(data, function (d) {
-             return d.id;
-           }); // exit
+           context.on('enter.intro', function (mode) {
+             if (!_tankID || !context.hasEntity(_tankID)) {
+               return continueTo(addTank);
+             }
 
-           targets.exit().remove(); // enter/update
+             var ids = context.selectedIDs();
 
-           targets.enter().append('rect').attr('width', '20px').attr('height', '30px').attr('x', '-10px').attr('y', '-28px').merge(targets).sort(sortY).attr('class', function (d) {
-             return "qaItem ".concat(d.service, " target ").concat(fillClass, " itemId-").concat(d.id);
-           }).attr('transform', getTransform);
+             if (mode.id !== 'select' || !ids.length || ids[0] !== _tankID) {
+               // keep the user's area selected..
+               context.enter(modeSelect(context, [_tankID])); // reset pane, in case user somehow happened to change it..
 
-           function sortY(a, b) {
-             return a.id === selectedID ? 1 : b.id === selectedID ? -1 : b.loc[1] - a.loc[1];
-           }
-         } // Draw the ImproveOSM layer and schedule loading issues and updating markers.
+               context.container().select('.inspector-wrap .panewrap').style('right', '-100%'); // disallow scrolling
 
+               context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
+               context.container().select('.preset-search-input').on('keydown.intro', null).on('keyup.intro', checkPresetSearch);
+               reveal('.preset-search-input', helpHtml('intro.buildings.search_tank', {
+                 preset: tankPreset.name()
+               }));
+               context.history().on('change.intro', null);
+             }
+           });
 
-         function drawImproveOSM(selection) {
-           var service = getService();
-           var surface = context.surface();
+           function checkPresetSearch() {
+             var first = context.container().select('.preset-list-item:first-child');
 
-           if (surface && !surface.empty()) {
-             touchLayer = surface.selectAll('.data-layer.touch .layer-touch.markers');
+             if (first.classed('preset-man_made-storage_tank')) {
+               reveal(first.select('.preset-list-button').node(), helpHtml('intro.buildings.choose_tank', {
+                 preset: tankPreset.name()
+               }), {
+                 duration: 300
+               });
+               context.container().select('.preset-search-input').on('keydown.intro', eventCancel, true).on('keyup.intro', null);
+               context.history().on('change.intro', function () {
+                 continueTo(closeEditorTank);
+               });
+             }
            }
 
-           drawLayer = selection.selectAll('.layer-improveOSM').data(service ? [0] : []);
-           drawLayer.exit().remove();
-           drawLayer = drawLayer.enter().append('g').attr('class', 'layer-improveOSM').style('display', _layerEnabled$1 ? 'block' : 'none').merge(drawLayer);
+           function continueTo(nextStep) {
+             context.container().select('.inspector-wrap').on('wheel.intro', null);
+             context.on('enter.intro', null);
+             context.history().on('change.intro', null);
+             context.container().select('.preset-search-input').on('keydown.intro keyup.intro', null);
+             nextStep();
+           }
+         }
 
-           if (_layerEnabled$1) {
-             if (service && ~~context.map().zoom() >= minZoom) {
-               editOn();
-               service.loadIssues(projection);
-               updateMarkers();
-             } else {
-               editOff();
-             }
+         function closeEditorTank() {
+           if (!_tankID || !context.hasEntity(_tankID)) {
+             return addTank();
            }
-         } // Toggles the layer on and off
 
+           var ids = context.selectedIDs();
 
-         drawImproveOSM.enabled = function (val) {
-           if (!arguments.length) return _layerEnabled$1;
-           _layerEnabled$1 = val;
+           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _tankID) {
+             context.enter(modeSelect(context, [_tankID]));
+           }
 
-           if (_layerEnabled$1) {
-             layerOn();
-           } else {
-             layerOff();
+           context.history().checkpoint('hasTank');
+           context.on('exit.intro', function () {
+             continueTo(rightClickTank);
+           });
+           timeout(function () {
+             reveal('.entity-editor-pane', helpHtml('intro.buildings.close', {
+               button: icon('#iD-icon-close', 'inline')
+             }));
+           }, 500);
 
-             if (context.selectedErrorID()) {
-               context.enter(modeBrowse(context));
-             }
+           function continueTo(nextStep) {
+             context.on('exit.intro', null);
+             nextStep();
            }
+         }
 
-           dispatch.call('change');
-           return this;
-         };
+         function rightClickTank() {
+           if (!_tankID) return continueTo(addTank);
+           context.enter(modeBrowse(context));
+           context.history().reset('hasTank');
+           context.map().centerEase(tank, 500);
+           timeout(function () {
+             context.on('enter.intro', function (mode) {
+               if (mode.id !== 'select') return;
+               var ids = context.selectedIDs();
+               if (ids.length !== 1 || ids[0] !== _tankID) return;
+               timeout(function () {
+                 var node = selectMenuItem(context, 'circularize').node();
+                 if (!node) return;
+                 continueTo(clickCircle);
+               }, 50); // after menu visible
+             });
+             var rightclickString = helpHtml('intro.buildings.' + (context.lastPointerType() === 'mouse' ? 'rightclick_tank' : 'edit_menu_tank_touch'));
+             revealTank(tank, rightclickString);
+             context.map().on('move.intro drawn.intro', function () {
+               revealTank(tank, rightclickString, {
+                 duration: 0
+               });
+             });
+             context.history().on('change.intro', function () {
+               continueTo(rightClickTank);
+             });
+           }, 600);
 
-         drawImproveOSM.supported = function () {
-           return !!getService();
-         };
+           function continueTo(nextStep) {
+             context.on('enter.intro', null);
+             context.map().on('move.intro drawn.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
+           }
+         }
 
-         return drawImproveOSM;
-       }
+         function clickCircle() {
+           if (!_tankID) return chapter.restart();
+           var entity = context.hasEntity(_tankID);
+           if (!entity) return continueTo(rightClickTank);
+           var node = selectMenuItem(context, 'circularize').node();
 
-       var _layerEnabled$2 = false;
+           if (!node) {
+             return continueTo(rightClickTank);
+           }
 
-       var _qaService$2;
+           var wasChanged = false;
+           reveal('.edit-menu', helpHtml('intro.buildings.circle_tank'), {
+             padding: 50
+           });
+           context.on('enter.intro', function (mode) {
+             if (mode.id === 'browse') {
+               continueTo(rightClickTank);
+             } else if (mode.id === 'move' || mode.id === 'rotate') {
+               continueTo(retryClickCircle);
+             }
+           });
+           context.map().on('move.intro', function () {
+             var node = selectMenuItem(context, 'circularize').node();
 
-       function svgOsmose(projection, context, dispatch) {
-         var throttledRedraw = throttle(function () {
-           return dispatch.call('change');
-         }, 1000);
+             if (!wasChanged && !node) {
+               return continueTo(rightClickTank);
+             }
 
-         var minZoom = 12;
-         var touchLayer = select(null);
-         var drawLayer = select(null);
-         var layerVisible = false;
+             reveal('.edit-menu', helpHtml('intro.buildings.circle_tank'), {
+               duration: 0,
+               padding: 50
+             });
+           });
+           context.history().on('change.intro', function () {
+             wasChanged = true;
+             context.history().on('change.intro', null); // Something changed.  Wait for transition to complete and check undo annotation.
 
-         function markerPath(selection, klass) {
-           selection.attr('class', klass).attr('transform', 'translate(-10, -28)').attr('points', '16,3 4,3 1,6 1,17 4,20 7,20 10,27 13,20 16,20 19,17.033 19,6');
-         } // Loosely-coupled osmose service for fetching issues
+             timeout(function () {
+               if (context.history().undoAnnotation() === _t('operations.circularize.annotation.feature', {
+                 n: 1
+               })) {
+                 continueTo(play);
+               } else {
+                 continueTo(retryClickCircle);
+               }
+             }, 500); // after transitioned actions
+           });
 
+           function continueTo(nextStep) {
+             context.on('enter.intro', null);
+             context.map().on('move.intro', null);
+             context.history().on('change.intro', null);
+             nextStep();
+           }
+         }
 
-         function getService() {
-           if (services.osmose && !_qaService$2) {
-             _qaService$2 = services.osmose;
+         function retryClickCircle() {
+           context.enter(modeBrowse(context));
+           revealTank(tank, helpHtml('intro.buildings.retry_circle'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: function buttonCallback() {
+               continueTo(rightClickTank);
+             }
+           });
 
-             _qaService$2.on('loaded', throttledRedraw);
-           } else if (!services.osmose && _qaService$2) {
-             _qaService$2 = null;
+           function continueTo(nextStep) {
+             nextStep();
            }
+         }
 
-           return _qaService$2;
-         } // Show the markers
+         function play() {
+           dispatch.call('done');
+           reveal('.ideditor', helpHtml('intro.buildings.play', {
+             next: _t('intro.startediting.title')
+           }), {
+             tooltipBox: '.intro-nav-wrap .chapter-startEditing',
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: function buttonCallback() {
+               reveal('.ideditor');
+             }
+           });
+         }
 
+         chapter.enter = function () {
+           addHouse();
+         };
 
-         function editOn() {
-           if (!layerVisible) {
-             layerVisible = true;
-             drawLayer.style('display', 'block');
-           }
-         } // Immediately remove the markers and their touch targets
+         chapter.exit = function () {
+           timeouts.forEach(window.clearTimeout);
+           context.on('enter.intro exit.intro', null);
+           context.map().on('move.intro drawn.intro', null);
+           context.history().on('change.intro', null);
+           context.container().select('.inspector-wrap').on('wheel.intro', null);
+           context.container().select('.preset-search-input').on('keydown.intro keyup.intro', null);
+           context.container().select('.more-fields .combobox-input').on('click.intro', null);
+         };
 
+         chapter.restart = function () {
+           chapter.exit();
+           chapter.enter();
+         };
 
-         function editOff() {
-           if (layerVisible) {
-             layerVisible = false;
-             drawLayer.style('display', 'none');
-             drawLayer.selectAll('.qaItem.osmose').remove();
-             touchLayer.selectAll('.qaItem.osmose').remove();
-           }
-         } // Enable the layer.  This shows the markers and transitions them to visible.
+         return utilRebind(chapter, dispatch, 'on');
+       }
 
+       function uiIntroStartEditing(context, reveal) {
+         var dispatch = dispatch$8('done', 'startEditing');
+         var modalSelection = select(null);
+         var chapter = {
+           title: 'intro.startediting.title'
+         };
 
-         function layerOn() {
-           editOn();
-           drawLayer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end interrupt', function () {
-             return dispatch.call('change');
+         function showHelp() {
+           reveal('.map-control.help-control', helpHtml('intro.startediting.help'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: function buttonCallback() {
+               shortcuts();
+             }
            });
-         } // Disable the layer.  This transitions the layer invisible and then hides the markers.
-
+         }
 
-         function layerOff() {
-           throttledRedraw.cancel();
-           drawLayer.interrupt();
-           touchLayer.selectAll('.qaItem.osmose').remove();
-           drawLayer.transition().duration(250).style('opacity', 0).on('end interrupt', function () {
-             editOff();
-             dispatch.call('change');
+         function shortcuts() {
+           reveal('.map-control.help-control', helpHtml('intro.startediting.shortcuts'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: function buttonCallback() {
+               showSave();
+             }
            });
-         } // Update the issue markers
-
+         }
 
-         function updateMarkers() {
-           if (!layerVisible || !_layerEnabled$2) return;
-           var service = getService();
-           var selectedID = context.selectedErrorID();
-           var data = service ? service.getItems(projection) : [];
-           var getTransform = svgPointTransform(projection); // Draw markers..
+         function showSave() {
+           context.container().selectAll('.shaded').remove(); // in case user opened keyboard shortcuts
 
-           var markers = drawLayer.selectAll('.qaItem.osmose').data(data, function (d) {
-             return d.id;
-           }); // exit
+           reveal('.top-toolbar button.save', helpHtml('intro.startediting.save'), {
+             buttonText: _t.html('intro.ok'),
+             buttonCallback: function buttonCallback() {
+               showStart();
+             }
+           });
+         }
 
-           markers.exit().remove(); // enter
+         function showStart() {
+           context.container().selectAll('.shaded').remove(); // in case user opened keyboard shortcuts
 
-           var markersEnter = markers.enter().append('g').attr('class', function (d) {
-             return "qaItem ".concat(d.service, " itemId-").concat(d.id, " itemType-").concat(d.itemType);
+           modalSelection = uiModal(context.container());
+           modalSelection.select('.modal').attr('class', 'modal-splash modal');
+           modalSelection.selectAll('.close').remove();
+           var startbutton = modalSelection.select('.content').attr('class', 'fillL').append('button').attr('class', 'modal-section huge-modal-button').on('click', function () {
+             modalSelection.remove();
            });
-           markersEnter.append('polygon').call(markerPath, 'shadow');
-           markersEnter.append('ellipse').attr('cx', 0).attr('cy', 0).attr('rx', 4.5).attr('ry', 2).attr('class', 'stroke');
-           markersEnter.append('polygon').attr('fill', function (d) {
-             return service.getColor(d.item);
-           }).call(markerPath, 'qaItem-fill');
-           markersEnter.append('use').attr('transform', 'translate(-6.5, -23)').attr('class', 'icon-annotation').attr('width', '13px').attr('height', '13px').attr('xlink:href', function (d) {
-             var picon = d.icon;
+           startbutton.append('svg').attr('class', 'illustration').append('use').attr('xlink:href', '#iD-logo-walkthrough');
+           startbutton.append('h2').html(_t.html('intro.startediting.start'));
+           dispatch.call('startEditing');
+         }
 
-             if (!picon) {
-               return '';
-             } else {
-               var isMaki = /^maki-/.test(picon);
-               return "#".concat(picon).concat(isMaki ? '-11' : '');
-             }
-           }); // update
+         chapter.enter = function () {
+           showHelp();
+         };
 
-           markers.merge(markersEnter).sort(sortY).classed('selected', function (d) {
-             return d.id === selectedID;
-           }).attr('transform', getTransform); // Draw targets..
+         chapter.exit = function () {
+           modalSelection.remove();
+           context.container().selectAll('.shaded').remove(); // in case user opened keyboard shortcuts
+         };
 
-           if (touchLayer.empty()) return;
-           var fillClass = context.getDebug('target') ? 'pink' : 'nocolor';
-           var targets = touchLayer.selectAll('.qaItem.osmose').data(data, function (d) {
-             return d.id;
-           }); // exit
+         return utilRebind(chapter, dispatch, 'on');
+       }
 
-           targets.exit().remove(); // enter/update
+       var chapterUi = {
+         welcome: uiIntroWelcome,
+         navigation: uiIntroNavigation,
+         point: uiIntroPoint,
+         area: uiIntroArea,
+         line: uiIntroLine,
+         building: uiIntroBuilding,
+         startEditing: uiIntroStartEditing
+       };
+       var chapterFlow = ['welcome', 'navigation', 'point', 'area', 'line', 'building', 'startEditing'];
+       function uiIntro(context) {
+         var INTRO_IMAGERY = 'EsriWorldImageryClarity';
+         var _introGraph = {};
 
-           targets.enter().append('rect').attr('width', '20px').attr('height', '30px').attr('x', '-10px').attr('y', '-28px').merge(targets).sort(sortY).attr('class', function (d) {
-             return "qaItem ".concat(d.service, " target ").concat(fillClass, " itemId-").concat(d.id);
-           }).attr('transform', getTransform);
+         var _currChapter;
+
+         function intro(selection) {
+           _mainFileFetcher.get('intro_graph').then(function (dataIntroGraph) {
+             // create entities for intro graph and localize names
+             for (var id in dataIntroGraph) {
+               if (!_introGraph[id]) {
+                 _introGraph[id] = osmEntity(localize(dataIntroGraph[id]));
+               }
+             }
 
-           function sortY(a, b) {
-             return a.id === selectedID ? 1 : b.id === selectedID ? -1 : b.loc[1] - a.loc[1];
-           }
-         } // Draw the Osmose layer and schedule loading issues and updating markers.
+             selection.call(startIntro);
+           })["catch"](function () {
+             /* ignore */
+           });
+         }
 
+         function startIntro(selection) {
+           context.enter(modeBrowse(context)); // Save current map state
 
-         function drawOsmose(selection) {
-           var service = getService();
-           var surface = context.surface();
+           var osm = context.connection();
+           var history = context.history().toJSON();
+           var hash = window.location.hash;
+           var center = context.map().center();
+           var zoom = context.map().zoom();
+           var background = context.background().baseLayerSource();
+           var overlays = context.background().overlayLayerSources();
+           var opacity = context.container().selectAll('.main-map .layer-background').style('opacity');
+           var caches = osm && osm.caches();
+           var baseEntities = context.history().graph().base().entities; // Show sidebar and disable the sidebar resizing button
+           // (this needs to be before `context.inIntro(true)`)
 
-           if (surface && !surface.empty()) {
-             touchLayer = surface.selectAll('.data-layer.touch .layer-touch.markers');
-           }
+           context.ui().sidebar.expand();
+           context.container().selectAll('button.sidebar-toggle').classed('disabled', true); // Block saving
 
-           drawLayer = selection.selectAll('.layer-osmose').data(service ? [0] : []);
-           drawLayer.exit().remove();
-           drawLayer = drawLayer.enter().append('g').attr('class', 'layer-osmose').style('display', _layerEnabled$2 ? 'block' : 'none').merge(drawLayer);
+           context.inIntro(true); // Load semi-real data used in intro
 
-           if (_layerEnabled$2) {
-             if (service && ~~context.map().zoom() >= minZoom) {
-               editOn();
-               service.loadIssues(projection);
-               updateMarkers();
-             } else {
-               editOff();
-             }
+           if (osm) {
+             osm.toggle(false).reset();
            }
-         } // Toggles the layer on and off
 
+           context.history().reset();
+           context.history().merge(Object.values(coreGraph().load(_introGraph).entities));
+           context.history().checkpoint('initial'); // Setup imagery
 
-         drawOsmose.enabled = function (val) {
-           if (!arguments.length) return _layerEnabled$2;
-           _layerEnabled$2 = val;
+           var imagery = context.background().findSource(INTRO_IMAGERY);
 
-           if (_layerEnabled$2) {
-             // Strings supplied by Osmose fetched before showing layer for first time
-             // NOTE: Currently no way to change locale in iD at runtime, would need to re-call this method if that's ever implemented
-             // Also, If layer is toggled quickly multiple requests are sent
-             getService().loadStrings().then(layerOn)["catch"](function (err) {
-               console.log(err); // eslint-disable-line no-console
-             });
+           if (imagery) {
+             context.background().baseLayerSource(imagery);
            } else {
-             layerOff();
-
-             if (context.selectedErrorID()) {
-               context.enter(modeBrowse(context));
-             }
+             context.background().bing();
            }
 
-           dispatch.call('change');
-           return this;
-         };
-
-         drawOsmose.supported = function () {
-           return !!getService();
-         };
+           overlays.forEach(function (d) {
+             return context.background().toggleOverlayLayer(d);
+           }); // Setup data layers (only OSM)
 
-         return drawOsmose;
-       }
+           var layers = context.layers();
+           layers.all().forEach(function (item) {
+             // if the layer has the function `enabled`
+             if (typeof item.layer.enabled === 'function') {
+               item.layer.enabled(item.id === 'osm');
+             }
+           });
+           context.container().selectAll('.main-map .layer-background').style('opacity', 1);
+           var curtain = uiCurtain(context.container().node());
+           selection.call(curtain); // Store that the user started the walkthrough..
 
-       function svgStreetside(projection, context, dispatch) {
-         var throttledRedraw = throttle(function () {
-           dispatch.call('change');
-         }, 1000);
+           corePreferences('walkthrough_started', 'yes'); // Restore previous walkthrough progress..
 
-         var minZoom = 14;
-         var minMarkerZoom = 16;
-         var minViewfieldZoom = 18;
-         var layer = select(null);
-         var _viewerYaw = 0;
-         var _selectedSequence = null;
+           var storedProgress = corePreferences('walkthrough_progress') || '';
+           var progress = storedProgress.split(';').filter(Boolean);
+           var chapters = chapterFlow.map(function (chapter, i) {
+             var s = chapterUi[chapter](context, curtain.reveal).on('done', function () {
+               buttons.filter(function (d) {
+                 return d.title === s.title;
+               }).classed('finished', true);
 
-         var _streetside;
-         /**
-          * init().
-          */
+               if (i < chapterFlow.length - 1) {
+                 var next = chapterFlow[i + 1];
+                 context.container().select("button.chapter-".concat(next)).classed('next', true);
+               } // Store walkthrough progress..
 
 
-         function init() {
-           if (svgStreetside.initialized) return; // run once
+               progress.push(chapter);
+               corePreferences('walkthrough_progress', utilArrayUniq(progress).join(';'));
+             });
+             return s;
+           });
+           chapters[chapters.length - 1].on('startEditing', function () {
+             // Store walkthrough progress..
+             progress.push('startEditing');
+             corePreferences('walkthrough_progress', utilArrayUniq(progress).join(';')); // Store if walkthrough is completed..
 
-           svgStreetside.enabled = false;
-           svgStreetside.initialized = true;
-         }
-         /**
-          * getService().
-          */
+             var incomplete = utilArrayDifference(chapterFlow, progress);
 
+             if (!incomplete.length) {
+               corePreferences('walkthrough_completed', 'yes');
+             }
 
-         function getService() {
-           if (services.streetside && !_streetside) {
-             _streetside = services.streetside;
+             curtain.remove();
+             navwrap.remove();
+             context.container().selectAll('.main-map .layer-background').style('opacity', opacity);
+             context.container().selectAll('button.sidebar-toggle').classed('disabled', false);
 
-             _streetside.event.on('viewerChanged.svgStreetside', viewerChanged).on('loadedImages.svgStreetside', throttledRedraw);
-           } else if (!services.streetside && _streetside) {
-             _streetside = null;
-           }
+             if (osm) {
+               osm.toggle(true).reset().caches(caches);
+             }
 
-           return _streetside;
-         }
-         /**
-          * showLayer().
-          */
+             context.history().reset().merge(Object.values(baseEntities));
+             context.background().baseLayerSource(background);
+             overlays.forEach(function (d) {
+               return context.background().toggleOverlayLayer(d);
+             });
 
+             if (history) {
+               context.history().fromJSON(history, false);
+             }
 
-         function showLayer() {
-           var service = getService();
-           if (!service) return;
-           editOn();
-           layer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end', function () {
-             dispatch.call('change');
+             context.map().centerZoom(center, zoom);
+             window.location.replace(hash);
+             context.inIntro(false);
            });
-         }
-         /**
-          * hideLayer().
-          */
+           var navwrap = selection.append('div').attr('class', 'intro-nav-wrap fillD');
+           navwrap.append('svg').attr('class', 'intro-nav-wrap-logo').append('use').attr('xlink:href', '#iD-logo-walkthrough');
+           var buttonwrap = navwrap.append('div').attr('class', 'joined').selectAll('button.chapter');
+           var buttons = buttonwrap.data(chapters).enter().append('button').attr('class', function (d, i) {
+             return "chapter chapter-".concat(chapterFlow[i]);
+           }).on('click', enterChapter);
+           buttons.append('span').html(function (d) {
+             return _t.html(d.title);
+           });
+           buttons.append('span').attr('class', 'status').call(svgIcon(_mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-backward' : '#iD-icon-forward', 'inline'));
+           enterChapter(null, chapters[0]);
 
+           function enterChapter(d3_event, newChapter) {
+             if (_currChapter) {
+               _currChapter.exit();
+             }
 
-         function hideLayer() {
-           throttledRedraw.cancel();
-           layer.transition().duration(250).style('opacity', 0).on('end', editOff);
-         }
-         /**
-          * editOn().
-          */
+             context.enter(modeBrowse(context));
+             _currChapter = newChapter;
 
+             _currChapter.enter();
 
-         function editOn() {
-           layer.style('display', 'block');
+             buttons.classed('next', false).classed('active', function (d) {
+               return d.title === _currChapter.title;
+             });
+           }
          }
-         /**
-          * editOff().
-          */
 
+         return intro;
+       }
 
-         function editOff() {
-           layer.selectAll('.viewfield-group').remove();
-           layer.style('display', 'none');
-         }
-         /**
-          * click() Handles 'bubble' point click event.
-          */
+       function uiIssuesInfo(context) {
+         var warningsItem = {
+           id: 'warnings',
+           count: 0,
+           iconID: 'iD-icon-alert',
+           descriptionID: 'issues.warnings_and_errors'
+         };
+         var resolvedItem = {
+           id: 'resolved',
+           count: 0,
+           iconID: 'iD-icon-apply',
+           descriptionID: 'issues.user_resolved_issues'
+         };
 
+         function update(selection) {
+           var shownItems = [];
+           var liveIssues = context.validator().getIssues({
+             what: corePreferences('validate-what') || 'edited',
+             where: corePreferences('validate-where') || 'all'
+           });
 
-         function click(d3_event, d) {
-           var service = getService();
-           if (!service) return; // try to preserve the viewer rotation when staying on the same sequence
+           if (liveIssues.length) {
+             warningsItem.count = liveIssues.length;
+             shownItems.push(warningsItem);
+           }
 
-           if (d.sequenceKey !== _selectedSequence) {
-             _viewerYaw = 0; // reset
+           if (corePreferences('validate-what') === 'all') {
+             var resolvedIssues = context.validator().getResolvedIssues();
+
+             if (resolvedIssues.length) {
+               resolvedItem.count = resolvedIssues.length;
+               shownItems.push(resolvedItem);
+             }
            }
 
-           _selectedSequence = d.sequenceKey;
-           service.ensureViewerLoaded(context).then(function () {
-             service.selectImage(context, d.key).yaw(_viewerYaw).showViewer(context);
+           var chips = selection.selectAll('.chip').data(shownItems, function (d) {
+             return d.id;
            });
-           context.map().centerEase(d.loc);
-         }
-         /**
-          * mouseover().
-          */
-
+           chips.exit().remove();
+           var enter = chips.enter().append('a').attr('class', function (d) {
+             return 'chip ' + d.id + '-count';
+           }).attr('href', '#').each(function (d) {
+             var chipSelection = select(this);
+             var tooltipBehavior = uiTooltip().placement('top').title(_t.html(d.descriptionID));
+             chipSelection.call(tooltipBehavior).on('click', function (d3_event) {
+               d3_event.preventDefault();
+               tooltipBehavior.hide(select(this)); // open the Issues pane
 
-         function mouseover(d3_event, d) {
-           var service = getService();
-           if (service) service.setStyles(context, d);
+               context.ui().togglePanes(context.container().select('.map-panes .issues-pane'));
+             });
+             chipSelection.call(svgIcon('#' + d.iconID));
+           });
+           enter.append('span').attr('class', 'count');
+           enter.merge(chips).selectAll('span.count').html(function (d) {
+             return d.count.toString();
+           });
          }
-         /**
-          * mouseout().
-          */
 
+         return function (selection) {
+           update(selection);
+           context.validator().on('validated.infobox', function () {
+             update(selection);
+           });
+         };
+       }
 
-         function mouseout() {
-           var service = getService();
-           if (service) service.setStyles(context, null);
-         }
-         /**
-          * transform().
-          */
+       function uiMapInMap(context) {
+         function mapInMap(selection) {
+           var backgroundLayer = rendererTileLayer(context);
+           var overlayLayers = {};
+           var projection = geoRawMercator();
+           var dataLayer = svgData(projection, context).showLabels(false);
+           var debugLayer = svgDebug(projection, context);
+           var zoom = d3_zoom().scaleExtent([geoZoomToScale(0.5), geoZoomToScale(24)]).on('start', zoomStarted).on('zoom', zoomed).on('end', zoomEnded);
+           var wrap = select(null);
+           var tiles = select(null);
+           var viewport = select(null);
+           var _isTransformed = false;
+           var _isHidden = true;
+           var _skipEvents = false;
+           var _gesture = null;
+           var _zDiff = 6; // by default, minimap renders at (main zoom - 6)
 
+           var _dMini; // dimensions of minimap
 
-         function transform(d) {
-           var t = svgPointTransform(projection)(d);
-           var rot = d.ca + _viewerYaw;
 
-           if (rot) {
-             t += ' rotate(' + Math.floor(rot) + ',0,0)';
-           }
+           var _cMini; // center pixel of minimap
 
-           return t;
-         }
 
-         function viewerChanged() {
-           var service = getService();
-           if (!service) return;
-           var viewer = service.viewer();
-           if (!viewer) return; // update viewfield rotation
+           var _tStart; // transform at start of gesture
 
-           _viewerYaw = viewer.getYaw(); // avoid updating if the map is currently transformed
-           // e.g. during drags or easing.
 
-           if (context.map().isTransformed()) return;
-           layer.selectAll('.viewfield-group.currentView').attr('transform', transform);
-         }
+           var _tCurr; // transform at most recent event
 
-         function filterBubbles(bubbles) {
-           var fromDate = context.photos().fromDate();
-           var toDate = context.photos().toDate();
-           var usernames = context.photos().usernames();
 
-           if (fromDate) {
-             var fromTimestamp = new Date(fromDate).getTime();
-             bubbles = bubbles.filter(function (bubble) {
-               return new Date(bubble.captured_at).getTime() >= fromTimestamp;
-             });
-           }
+           var _timeoutID;
 
-           if (toDate) {
-             var toTimestamp = new Date(toDate).getTime();
-             bubbles = bubbles.filter(function (bubble) {
-               return new Date(bubble.captured_at).getTime() <= toTimestamp;
-             });
+           function zoomStarted() {
+             if (_skipEvents) return;
+             _tStart = _tCurr = projection.transform();
+             _gesture = null;
            }
 
-           if (usernames) {
-             bubbles = bubbles.filter(function (bubble) {
-               return usernames.indexOf(bubble.captured_by) !== -1;
-             });
-           }
+           function zoomed(d3_event) {
+             if (_skipEvents) return;
+             var x = d3_event.transform.x;
+             var y = d3_event.transform.y;
+             var k = d3_event.transform.k;
+             var isZooming = k !== _tStart.k;
+             var isPanning = x !== _tStart.x || y !== _tStart.y;
 
-           return bubbles;
-         }
+             if (!isZooming && !isPanning) {
+               return; // no change
+             } // lock in either zooming or panning, don't allow both in minimap.
 
-         function filterSequences(sequences) {
-           var fromDate = context.photos().fromDate();
-           var toDate = context.photos().toDate();
-           var usernames = context.photos().usernames();
 
-           if (fromDate) {
-             var fromTimestamp = new Date(fromDate).getTime();
-             sequences = sequences.filter(function (sequences) {
-               return new Date(sequences.properties.captured_at).getTime() >= fromTimestamp;
-             });
-           }
+             if (!_gesture) {
+               _gesture = isZooming ? 'zoom' : 'pan';
+             }
 
-           if (toDate) {
-             var toTimestamp = new Date(toDate).getTime();
-             sequences = sequences.filter(function (sequences) {
-               return new Date(sequences.properties.captured_at).getTime() <= toTimestamp;
-             });
-           }
+             var tMini = projection.transform();
+             var tX, tY, scale;
 
-           if (usernames) {
-             sequences = sequences.filter(function (sequences) {
-               return usernames.indexOf(sequences.properties.captured_by) !== -1;
-             });
+             if (_gesture === 'zoom') {
+               scale = k / tMini.k;
+               tX = (_cMini[0] / scale - _cMini[0]) * scale;
+               tY = (_cMini[1] / scale - _cMini[1]) * scale;
+             } else {
+               k = tMini.k;
+               scale = 1;
+               tX = x - tMini.x;
+               tY = y - tMini.y;
+             }
+
+             utilSetTransform(tiles, tX, tY, scale);
+             utilSetTransform(viewport, 0, 0, scale);
+             _isTransformed = true;
+             _tCurr = identity$2.translate(x, y).scale(k);
+             var zMain = geoScaleToZoom(context.projection.scale());
+             var zMini = geoScaleToZoom(k);
+             _zDiff = zMain - zMini;
+             queueRedraw();
            }
 
-           return sequences;
-         }
-         /**
-          * update().
-          */
+           function zoomEnded() {
+             if (_skipEvents) return;
+             if (_gesture !== 'pan') return;
+             updateProjection();
+             _gesture = null;
+             context.map().center(projection.invert(_cMini)); // recenter main map..
+           }
 
+           function updateProjection() {
+             var loc = context.map().center();
+             var tMain = context.projection.transform();
+             var zMain = geoScaleToZoom(tMain.k);
+             var zMini = Math.max(zMain - _zDiff, 0.5);
+             var kMini = geoZoomToScale(zMini);
+             projection.translate([tMain.x, tMain.y]).scale(kMini);
+             var point = projection(loc);
+             var mouse = _gesture === 'pan' ? geoVecSubtract([_tCurr.x, _tCurr.y], [_tStart.x, _tStart.y]) : [0, 0];
+             var xMini = _cMini[0] - point[0] + tMain.x + mouse[0];
+             var yMini = _cMini[1] - point[1] + tMain.y + mouse[1];
+             projection.translate([xMini, yMini]).clipExtent([[0, 0], _dMini]);
+             _tCurr = projection.transform();
 
-         function update() {
-           var viewer = context.container().select('.photoviewer');
-           var selected = viewer.empty() ? undefined : viewer.datum();
-           var z = ~~context.map().zoom();
-           var showMarkers = z >= minMarkerZoom;
-           var showViewfields = z >= minViewfieldZoom;
-           var service = getService();
-           var sequences = [];
-           var bubbles = [];
+             if (_isTransformed) {
+               utilSetTransform(tiles, 0, 0);
+               utilSetTransform(viewport, 0, 0);
+               _isTransformed = false;
+             }
 
-           if (context.photos().showsPanoramic()) {
-             sequences = service ? service.sequences(projection) : [];
-             bubbles = service && showMarkers ? service.bubbles(projection) : [];
-             sequences = filterSequences(sequences);
-             bubbles = filterBubbles(bubbles);
+             zoom.scaleExtent([geoZoomToScale(0.5), geoZoomToScale(zMain - 3)]);
+             _skipEvents = true;
+             wrap.call(zoom.transform, _tCurr);
+             _skipEvents = false;
            }
 
-           var traces = layer.selectAll('.sequences').selectAll('.sequence').data(sequences, function (d) {
-             return d.properties.key;
-           }); // exit
+           function redraw() {
+             clearTimeout(_timeoutID);
+             if (_isHidden) return;
+             updateProjection();
+             var zMini = geoScaleToZoom(projection.scale()); // setup tile container
 
-           traces.exit().remove(); // enter/update
+             tiles = wrap.selectAll('.map-in-map-tiles').data([0]);
+             tiles = tiles.enter().append('div').attr('class', 'map-in-map-tiles').merge(tiles); // redraw background
 
-           traces = traces.enter().append('path').attr('class', 'sequence').merge(traces).attr('d', svgPath(projection).geojson);
-           var groups = layer.selectAll('.markers').selectAll('.viewfield-group').data(bubbles, function (d) {
-             // force reenter once bubbles are attached to a sequence
-             return d.key + (d.sequenceKey ? 'v1' : 'v0');
-           }); // exit
+             backgroundLayer.source(context.background().baseLayerSource()).projection(projection).dimensions(_dMini);
+             var background = tiles.selectAll('.map-in-map-background').data([0]);
+             background.enter().append('div').attr('class', 'map-in-map-background').merge(background).call(backgroundLayer); // redraw overlay
 
-           groups.exit().remove(); // enter
+             var overlaySources = context.background().overlayLayerSources();
+             var activeOverlayLayers = [];
 
-           var groupsEnter = groups.enter().append('g').attr('class', 'viewfield-group').on('mouseenter', mouseover).on('mouseleave', mouseout).on('click', click);
-           groupsEnter.append('g').attr('class', 'viewfield-scale'); // update
+             for (var i = 0; i < overlaySources.length; i++) {
+               if (overlaySources[i].validZoom(zMini)) {
+                 if (!overlayLayers[i]) overlayLayers[i] = rendererTileLayer(context);
+                 activeOverlayLayers.push(overlayLayers[i].source(overlaySources[i]).projection(projection).dimensions(_dMini));
+               }
+             }
+
+             var overlay = tiles.selectAll('.map-in-map-overlay').data([0]);
+             overlay = overlay.enter().append('div').attr('class', 'map-in-map-overlay').merge(overlay);
+             var overlays = overlay.selectAll('div').data(activeOverlayLayers, function (d) {
+               return d.source().name();
+             });
+             overlays.exit().remove();
+             overlays = overlays.enter().append('div').merge(overlays).each(function (layer) {
+               select(this).call(layer);
+             });
+             var dataLayers = tiles.selectAll('.map-in-map-data').data([0]);
+             dataLayers.exit().remove();
+             dataLayers = dataLayers.enter().append('svg').attr('class', 'map-in-map-data').merge(dataLayers).call(dataLayer).call(debugLayer); // redraw viewport bounding box
 
-           var markers = groups.merge(groupsEnter).sort(function (a, b) {
-             return a === selected ? 1 : b === selected ? -1 : b.loc[1] - a.loc[1];
-           }).attr('transform', transform).select('.viewfield-scale');
-           markers.selectAll('circle').data([0]).enter().append('circle').attr('dx', '0').attr('dy', '0').attr('r', '6');
-           var viewfields = markers.selectAll('.viewfield').data(showViewfields ? [0] : []);
-           viewfields.exit().remove(); // viewfields may or may not be drawn...
-           // but if they are, draw below the circles
+             if (_gesture !== 'pan') {
+               var getPath = d3_geoPath(projection);
+               var bbox = {
+                 type: 'Polygon',
+                 coordinates: [context.map().extent().polygon()]
+               };
+               viewport = wrap.selectAll('.map-in-map-viewport').data([0]);
+               viewport = viewport.enter().append('svg').attr('class', 'map-in-map-viewport').merge(viewport);
+               var path = viewport.selectAll('.map-in-map-bbox').data([bbox]);
+               path.enter().append('path').attr('class', 'map-in-map-bbox').merge(path).attr('d', getPath).classed('thick', function (d) {
+                 return getPath.area(d) < 30;
+               });
+             }
+           }
 
-           viewfields.enter().insert('path', 'circle').attr('class', 'viewfield').attr('transform', 'scale(1.5,1.5),translate(-8, -13)').attr('d', viewfieldPath);
+           function queueRedraw() {
+             clearTimeout(_timeoutID);
+             _timeoutID = setTimeout(function () {
+               redraw();
+             }, 750);
+           }
 
-           function viewfieldPath() {
-             var d = this.parentNode.__data__;
+           function toggle(d3_event) {
+             if (d3_event) d3_event.preventDefault();
+             _isHidden = !_isHidden;
+             context.container().select('.minimap-toggle-item').classed('active', !_isHidden).select('input').property('checked', !_isHidden);
 
-             if (d.pano) {
-               return 'M 8,13 m -10,0 a 10,10 0 1,0 20,0 a 10,10 0 1,0 -20,0';
+             if (_isHidden) {
+               wrap.style('display', 'block').style('opacity', '1').transition().duration(200).style('opacity', '0').on('end', function () {
+                 selection.selectAll('.map-in-map').style('display', 'none');
+               });
              } else {
-               return 'M 6,9 C 8,8.4 8,8.4 10,9 L 16,-2 C 12,-5 4,-5 0,-2 z';
+               wrap.style('display', 'block').style('opacity', '0').transition().duration(200).style('opacity', '1').on('end', function () {
+                 redraw();
+               });
              }
            }
-         }
-         /**
-          * drawImages()
-          * drawImages is the method that is returned (and that runs) every time 'svgStreetside()' is called.
-          * 'svgStreetside()' is called from index.js
-          */
 
+           uiMapInMap.toggle = toggle;
+           wrap = selection.selectAll('.map-in-map').data([0]);
+           wrap = wrap.enter().append('div').attr('class', 'map-in-map').style('display', _isHidden ? 'none' : 'block').call(zoom).on('dblclick.zoom', null).merge(wrap); // reflow warning: Hardcode dimensions - currently can't resize it anyway..
 
-         function drawImages(selection) {
-           var enabled = svgStreetside.enabled;
-           var service = getService();
-           layer = selection.selectAll('.layer-streetside-images').data(service ? [0] : []);
-           layer.exit().remove();
-           var layerEnter = layer.enter().append('g').attr('class', 'layer-streetside-images').style('display', enabled ? 'block' : 'none');
-           layerEnter.append('g').attr('class', 'sequences');
-           layerEnter.append('g').attr('class', 'markers');
-           layer = layerEnter.merge(layer);
+           _dMini = [200, 150]; //utilGetDimensions(wrap);
 
-           if (enabled) {
-             if (service && ~~context.map().zoom() >= minZoom) {
-               editOn();
-               update();
-               service.loadBubbles(projection);
-             } else {
-               editOff();
+           _cMini = geoVecScale(_dMini, 0.5);
+           context.map().on('drawn.map-in-map', function (drawn) {
+             if (drawn.full === true) {
+               redraw();
              }
-           }
+           });
+           redraw();
+           context.keybinding().on(_t('background.minimap.key'), toggle);
          }
-         /**
-          * drawImages.enabled().
-          */
 
+         return mapInMap;
+       }
 
-         drawImages.enabled = function (_) {
-           if (!arguments.length) return svgStreetside.enabled;
-           svgStreetside.enabled = _;
+       function uiNotice(context) {
+         return function (selection) {
+           var div = selection.append('div').attr('class', 'notice');
+           var button = div.append('button').attr('class', 'zoom-to notice fillD').on('click', function () {
+             context.map().zoomEase(context.minEditableZoom());
+           }).on('wheel', function (d3_event) {
+             // let wheel events pass through #4482
+             var e2 = new WheelEvent(d3_event.type, d3_event);
+             context.surface().node().dispatchEvent(e2);
+           });
+           button.call(svgIcon('#iD-icon-plus', 'pre-text')).append('span').attr('class', 'label').html(_t.html('zoom_in_edit'));
 
-           if (svgStreetside.enabled) {
-             showLayer();
-             context.photos().on('change.streetside', update);
-           } else {
-             hideLayer();
-             context.photos().on('change.streetside', null);
+           function disableTooHigh() {
+             var canEdit = context.map().zoom() >= context.minEditableZoom();
+             div.style('display', canEdit ? 'none' : 'block');
            }
 
-           dispatch.call('change');
-           return this;
+           context.map().on('move.notice', debounce(disableTooHigh, 500));
+           disableTooHigh();
          };
-         /**
-          * drawImages.supported().
-          */
+       }
 
+       function uiPhotoviewer(context) {
+         var dispatch = dispatch$8('resize');
 
-         drawImages.supported = function () {
-           return !!getService();
-         };
+         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
 
-         init();
-         return drawImages;
-       }
+         function photoviewer(selection) {
+           selection.append('button').attr('class', 'thumb-hide').on('click', function () {
+             if (services.streetside) {
+               services.streetside.hideViewer(context);
+             }
 
-       function svgMapillaryImages(projection, context, dispatch) {
-         var throttledRedraw = throttle(function () {
-           dispatch.call('change');
-         }, 1000);
+             if (services.mapillary) {
+               services.mapillary.hideViewer(context);
+             }
 
-         var minZoom = 12;
-         var minMarkerZoom = 16;
-         var minViewfieldZoom = 18;
-         var layer = select(null);
+             if (services.openstreetcam) {
+               services.openstreetcam.hideViewer(context);
+             }
+           }).append('div').call(svgIcon('#iD-icon-close'));
 
-         var _mapillary;
+           function preventDefault(d3_event) {
+             d3_event.preventDefault();
+           }
 
-         var viewerCompassAngle;
+           selection.append('button').attr('class', 'resize-handle-xy').on('touchstart touchdown touchend', preventDefault).on(_pointerPrefix + 'down', buildResizeListener(selection, 'resize', dispatch, {
+             resizeOnX: true,
+             resizeOnY: true
+           }));
+           selection.append('button').attr('class', 'resize-handle-x').on('touchstart touchdown touchend', preventDefault).on(_pointerPrefix + 'down', buildResizeListener(selection, 'resize', dispatch, {
+             resizeOnX: true
+           }));
+           selection.append('button').attr('class', 'resize-handle-y').on('touchstart touchdown touchend', preventDefault).on(_pointerPrefix + 'down', buildResizeListener(selection, 'resize', dispatch, {
+             resizeOnY: true
+           }));
 
-         function init() {
-           if (svgMapillaryImages.initialized) return; // run once
+           function buildResizeListener(target, eventName, dispatch, options) {
+             var resizeOnX = !!options.resizeOnX;
+             var resizeOnY = !!options.resizeOnY;
+             var minHeight = options.minHeight || 240;
+             var minWidth = options.minWidth || 320;
+             var pointerId;
+             var startX;
+             var startY;
+             var startWidth;
+             var startHeight;
 
-           svgMapillaryImages.enabled = false;
-           svgMapillaryImages.initialized = true;
-         }
+             function startResize(d3_event) {
+               if (pointerId !== (d3_event.pointerId || 'mouse')) return;
+               d3_event.preventDefault();
+               d3_event.stopPropagation();
+               var mapSize = context.map().dimensions();
 
-         function getService() {
-           if (services.mapillary && !_mapillary) {
-             _mapillary = services.mapillary;
+               if (resizeOnX) {
+                 var maxWidth = mapSize[0];
+                 var newWidth = clamp(startWidth + d3_event.clientX - startX, minWidth, maxWidth);
+                 target.style('width', newWidth + 'px');
+               }
 
-             _mapillary.event.on('loadedImages', throttledRedraw);
-           } else if (!services.mapillary && _mapillary) {
-             _mapillary = null;
-           }
+               if (resizeOnY) {
+                 var maxHeight = mapSize[1] - 90; // preserve space at top/bottom of map
 
-           return _mapillary;
-         }
+                 var newHeight = clamp(startHeight + startY - d3_event.clientY, minHeight, maxHeight);
+                 target.style('height', newHeight + 'px');
+               }
 
-         function showLayer() {
-           var service = getService();
-           if (!service) return;
-           editOn();
-           layer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end', function () {
-             dispatch.call('change');
-           });
-         }
+               dispatch.call(eventName, target, utilGetDimensions(target, true));
+             }
 
-         function hideLayer() {
-           throttledRedraw.cancel();
-           layer.transition().duration(250).style('opacity', 0).on('end', editOff);
-         }
+             function clamp(num, min, max) {
+               return Math.max(min, Math.min(num, max));
+             }
 
-         function editOn() {
-           layer.style('display', 'block');
-         }
+             function stopResize(d3_event) {
+               if (pointerId !== (d3_event.pointerId || 'mouse')) return;
+               d3_event.preventDefault();
+               d3_event.stopPropagation(); // remove all the listeners we added
 
-         function editOff() {
-           layer.selectAll('.viewfield-group').remove();
-           layer.style('display', 'none');
-         }
+               select(window).on('.' + eventName, null);
+             }
 
-         function click(d3_event, d) {
-           var service = getService();
-           if (!service) return;
-           service.ensureViewerLoaded(context).then(function () {
-             service.selectImage(context, d.key).showViewer(context);
-           });
-           context.map().centerEase(d.loc);
-         }
+             return function initResize(d3_event) {
+               d3_event.preventDefault();
+               d3_event.stopPropagation();
+               pointerId = d3_event.pointerId || 'mouse';
+               startX = d3_event.clientX;
+               startY = d3_event.clientY;
+               var targetRect = target.node().getBoundingClientRect();
+               startWidth = targetRect.width;
+               startHeight = targetRect.height;
+               select(window).on(_pointerPrefix + 'move.' + eventName, startResize, false).on(_pointerPrefix + 'up.' + eventName, stopResize, false);
 
-         function mouseover(d) {
-           var service = getService();
-           if (service) service.setStyles(context, d);
+               if (_pointerPrefix === 'pointer') {
+                 select(window).on('pointercancel.' + eventName, stopResize, false);
+               }
+             };
+           }
          }
 
-         function mouseout() {
-           var service = getService();
-           if (service) service.setStyles(context, null);
-         }
+         photoviewer.onMapResize = function () {
+           var photoviewer = context.container().select('.photoviewer');
+           var content = context.container().select('.main-content');
+           var mapDimensions = utilGetDimensions(content, true); // shrink photo viewer if it is too big
+           // (-90 preserves space at top and bottom of map used by menus)
 
-         function transform(d) {
-           var t = svgPointTransform(projection)(d);
+           var photoDimensions = utilGetDimensions(photoviewer, true);
 
-           if (d.pano && viewerCompassAngle !== null && isFinite(viewerCompassAngle)) {
-             t += ' rotate(' + Math.floor(viewerCompassAngle) + ',0,0)';
-           } else if (d.ca) {
-             t += ' rotate(' + Math.floor(d.ca) + ',0,0)';
+           if (photoDimensions[0] > mapDimensions[0] || photoDimensions[1] > mapDimensions[1] - 90) {
+             var setPhotoDimensions = [Math.min(photoDimensions[0], mapDimensions[0]), Math.min(photoDimensions[1], mapDimensions[1] - 90)];
+             photoviewer.style('width', setPhotoDimensions[0] + 'px').style('height', setPhotoDimensions[1] + 'px');
+             dispatch.call('resize', photoviewer, setPhotoDimensions);
            }
+         };
 
-           return t;
-         }
+         return utilRebind(photoviewer, dispatch, 'on');
+       }
 
-         function filterImages(images) {
-           var showsPano = context.photos().showsPanoramic();
-           var showsFlat = context.photos().showsFlat();
-           var fromDate = context.photos().fromDate();
-           var toDate = context.photos().toDate();
-           var usernames = context.photos().usernames();
+       function uiRestore(context) {
+         return function (selection) {
+           if (!context.history().hasRestorableChanges()) return;
+           var modalSelection = uiModal(selection, true);
+           modalSelection.select('.modal').attr('class', 'modal fillL');
+           var introModal = modalSelection.select('.content');
+           introModal.append('div').attr('class', 'modal-section').append('h3').html(_t.html('restore.heading'));
+           introModal.append('div').attr('class', 'modal-section').append('p').html(_t.html('restore.description'));
+           var buttonWrap = introModal.append('div').attr('class', 'modal-actions');
+           var restore = buttonWrap.append('button').attr('class', 'restore').on('click', function () {
+             context.history().restore();
+             modalSelection.remove();
+           });
+           restore.append('svg').attr('class', 'logo logo-restore').append('use').attr('xlink:href', '#iD-logo-restore');
+           restore.append('div').html(_t.html('restore.restore'));
+           var reset = buttonWrap.append('button').attr('class', 'reset').on('click', function () {
+             context.history().clearSaved();
+             modalSelection.remove();
+           });
+           reset.append('svg').attr('class', 'logo logo-reset').append('use').attr('xlink:href', '#iD-logo-reset');
+           reset.append('div').html(_t.html('restore.reset'));
+           restore.node().focus();
+         };
+       }
 
-           if (!showsPano || !showsFlat) {
-             images = images.filter(function (image) {
-               if (image.pano) return showsPano;
-               return showsFlat;
-             });
-           }
+       function uiScale(context) {
+         var projection = context.projection,
+             isImperial = !_mainLocalizer.usesMetric(),
+             maxLength = 180,
+             tickHeight = 8;
 
-           if (fromDate) {
-             var fromTimestamp = new Date(fromDate).getTime();
-             images = images.filter(function (image) {
-               return new Date(image.captured_at).getTime() >= fromTimestamp;
-             });
-           }
+         function scaleDefs(loc1, loc2) {
+           var lat = (loc2[1] + loc1[1]) / 2,
+               conversion = isImperial ? 3.28084 : 1,
+               dist = geoLonToMeters(loc2[0] - loc1[0], lat) * conversion,
+               scale = {
+             dist: 0,
+             px: 0,
+             text: ''
+           },
+               buckets,
+               i,
+               val,
+               dLon;
 
-           if (toDate) {
-             var toTimestamp = new Date(toDate).getTime();
-             images = images.filter(function (image) {
-               return new Date(image.captured_at).getTime() <= toTimestamp;
-             });
-           }
+           if (isImperial) {
+             buckets = [5280000, 528000, 52800, 5280, 500, 50, 5, 1];
+           } else {
+             buckets = [5000000, 500000, 50000, 5000, 500, 50, 5, 1];
+           } // determine a user-friendly endpoint for the scale
 
-           if (usernames) {
-             images = images.filter(function (image) {
-               return usernames.indexOf(image.captured_by) !== -1;
-             });
+
+           for (i = 0; i < buckets.length; i++) {
+             val = buckets[i];
+
+             if (dist >= val) {
+               scale.dist = Math.floor(dist / val) * val;
+               break;
+             } else {
+               scale.dist = +dist.toFixed(2);
+             }
            }
 
-           return images;
+           dLon = geoMetersToLon(scale.dist / conversion, lat);
+           scale.px = Math.round(projection([loc1[0] + dLon, loc1[1]])[0]);
+           scale.text = displayLength(scale.dist / conversion, isImperial);
+           return scale;
          }
 
-         function filterSequences(sequences, service) {
-           var showsPano = context.photos().showsPanoramic();
-           var showsFlat = context.photos().showsFlat();
-           var fromDate = context.photos().fromDate();
-           var toDate = context.photos().toDate();
-           var usernames = context.photos().usernames();
+         function update(selection) {
+           // choose loc1, loc2 along bottom of viewport (near where the scale will be drawn)
+           var dims = context.map().dimensions(),
+               loc1 = projection.invert([0, dims[1]]),
+               loc2 = projection.invert([maxLength, dims[1]]),
+               scale = scaleDefs(loc1, loc2);
+           selection.select('.scale-path').attr('d', 'M0.5,0.5v' + tickHeight + 'h' + scale.px + 'v-' + tickHeight);
+           selection.select('.scale-text').style(_mainLocalizer.textDirection() === 'ltr' ? 'left' : 'right', scale.px + 16 + 'px').html(scale.text);
+         }
 
-           if (!showsPano || !showsFlat) {
-             sequences = sequences.filter(function (sequence) {
-               if (sequence.properties.hasOwnProperty('pano')) {
-                 if (sequence.properties.pano) return showsPano;
-                 return showsFlat;
-               } else {
-                 // if the sequence doesn't specify pano or not, search its images
-                 var cProps = sequence.properties.coordinateProperties;
+         return function (selection) {
+           function switchUnits() {
+             isImperial = !isImperial;
+             selection.call(update);
+           }
 
-                 if (cProps && cProps.image_keys && cProps.image_keys.length > 0) {
-                   for (var index in cProps.image_keys) {
-                     var imageKey = cProps.image_keys[index];
-                     var image = service.cachedImage(imageKey);
+           var scalegroup = selection.append('svg').attr('class', 'scale').on('click', switchUnits).append('g').attr('transform', 'translate(10,11)');
+           scalegroup.append('path').attr('class', 'scale-path');
+           selection.append('div').attr('class', 'scale-text');
+           selection.call(update);
+           context.map().on('move.scale', function () {
+             update(selection);
+           });
+         };
+       }
 
-                     if (image && image.hasOwnProperty('pano')) {
-                       if (image.pano) return showsPano;
-                       return showsFlat;
-                     }
-                   }
-                 }
-               }
+       function uiShortcuts(context) {
+         var detected = utilDetect();
+         var _activeTab = 0;
 
-               return false;
-             });
-           }
+         var _modalSelection;
 
-           if (fromDate) {
-             var fromTimestamp = new Date(fromDate).getTime();
-             sequences = sequences.filter(function (sequence) {
-               return new Date(sequence.properties.captured_at).getTime() >= fromTimestamp;
-             });
-           }
+         var _selection = select(null);
 
-           if (toDate) {
-             var toTimestamp = new Date(toDate).getTime();
-             sequences = sequences.filter(function (sequence) {
-               return new Date(sequence.properties.captured_at).getTime() <= toTimestamp;
-             });
-           }
+         var _dataShortcuts;
 
-           if (usernames) {
-             sequences = sequences.filter(function (sequence) {
-               return usernames.indexOf(sequence.properties.username) !== -1;
-             });
-           }
+         function shortcutsModal(_modalSelection) {
+           _modalSelection.select('.modal').classed('modal-shortcuts', true);
 
-           return sequences;
+           var content = _modalSelection.select('.content');
+
+           content.append('div').attr('class', 'modal-section').append('h3').html(_t.html('shortcuts.title'));
+           _mainFileFetcher.get('shortcuts').then(function (data) {
+             _dataShortcuts = data;
+             content.call(render);
+           })["catch"](function () {
+             /* ignore */
+           });
          }
 
-         function update() {
-           var z = ~~context.map().zoom();
-           var showMarkers = z >= minMarkerZoom;
-           var showViewfields = z >= minViewfieldZoom;
-           var service = getService();
-           var sequences = service ? service.sequences(projection) : [];
-           var images = service && showMarkers ? service.images(projection) : [];
-           images = filterImages(images);
-           sequences = filterSequences(sequences, service);
-           service.filterViewer(context);
-           var traces = layer.selectAll('.sequences').selectAll('.sequence').data(sequences, function (d) {
-             return d.properties.key;
-           }); // exit
+         function render(selection) {
+           if (!_dataShortcuts) return;
+           var wrapper = selection.selectAll('.wrapper').data([0]);
+           var wrapperEnter = wrapper.enter().append('div').attr('class', 'wrapper modal-section');
+           var tabsBar = wrapperEnter.append('div').attr('class', 'tabs-bar');
+           var shortcutsList = wrapperEnter.append('div').attr('class', 'shortcuts-list');
+           wrapper = wrapper.merge(wrapperEnter);
+           var tabs = tabsBar.selectAll('.tab').data(_dataShortcuts);
+           var tabsEnter = tabs.enter().append('a').attr('class', 'tab').attr('href', '#').on('click', function (d3_event, d) {
+             d3_event.preventDefault();
 
-           traces.exit().remove(); // enter/update
+             var i = _dataShortcuts.indexOf(d);
 
-           traces = traces.enter().append('path').attr('class', 'sequence').merge(traces).attr('d', svgPath(projection).geojson);
-           var groups = layer.selectAll('.markers').selectAll('.viewfield-group').data(images, function (d) {
-             return d.key;
-           }); // exit
+             _activeTab = i;
+             render(selection);
+           });
+           tabsEnter.append('span').html(function (d) {
+             return _t.html(d.text);
+           }); // Update
 
-           groups.exit().remove(); // enter
+           wrapper.selectAll('.tab').classed('active', function (d, i) {
+             return i === _activeTab;
+           });
+           var shortcuts = shortcutsList.selectAll('.shortcut-tab').data(_dataShortcuts);
+           var shortcutsEnter = shortcuts.enter().append('div').attr('class', function (d) {
+             return 'shortcut-tab shortcut-tab-' + d.tab;
+           });
+           var columnsEnter = shortcutsEnter.selectAll('.shortcut-column').data(function (d) {
+             return d.columns;
+           }).enter().append('table').attr('class', 'shortcut-column');
+           var rowsEnter = columnsEnter.selectAll('.shortcut-row').data(function (d) {
+             return d.rows;
+           }).enter().append('tr').attr('class', 'shortcut-row');
+           var sectionRows = rowsEnter.filter(function (d) {
+             return !d.shortcuts;
+           });
+           sectionRows.append('td');
+           sectionRows.append('td').attr('class', 'shortcut-section').append('h3').html(function (d) {
+             return _t.html(d.text);
+           });
+           var shortcutRows = rowsEnter.filter(function (d) {
+             return d.shortcuts;
+           });
+           var shortcutKeys = shortcutRows.append('td').attr('class', 'shortcut-keys');
+           var modifierKeys = shortcutKeys.filter(function (d) {
+             return d.modifiers;
+           });
+           modifierKeys.selectAll('kbd.modifier').data(function (d) {
+             if (detected.os === 'win' && d.text === 'shortcuts.editing.commands.redo') {
+               return ['⌘'];
+             } else if (detected.os !== 'mac' && d.text === 'shortcuts.browsing.display_options.fullscreen') {
+               return [];
+             } else {
+               return d.modifiers;
+             }
+           }).enter().each(function () {
+             var selection = select(this);
+             selection.append('kbd').attr('class', 'modifier').html(function (d) {
+               return uiCmd.display(d);
+             });
+             selection.append('span').html('+');
+           });
+           shortcutKeys.selectAll('kbd.shortcut').data(function (d) {
+             var arr = d.shortcuts;
 
-           var groupsEnter = groups.enter().append('g').attr('class', 'viewfield-group').on('mouseenter', mouseover).on('mouseleave', mouseout).on('click', click);
-           groupsEnter.append('g').attr('class', 'viewfield-scale'); // update
+             if (detected.os === 'win' && d.text === 'shortcuts.editing.commands.redo') {
+               arr = ['Y'];
+             } else if (detected.os !== 'mac' && d.text === 'shortcuts.browsing.display_options.fullscreen') {
+               arr = ['F11'];
+             } // replace translations
 
-           var markers = groups.merge(groupsEnter).sort(function (a, b) {
-             return b.loc[1] - a.loc[1]; // sort Y
-           }).attr('transform', transform).select('.viewfield-scale');
-           markers.selectAll('circle').data([0]).enter().append('circle').attr('dx', '0').attr('dy', '0').attr('r', '6');
-           var viewfields = markers.selectAll('.viewfield').data(showViewfields ? [0] : []);
-           viewfields.exit().remove();
-           viewfields.enter() // viewfields may or may not be drawn...
-           .insert('path', 'circle') // but if they are, draw below the circles
-           .attr('class', 'viewfield').classed('pano', function () {
-             return this.parentNode.__data__.pano;
-           }).attr('transform', 'scale(1.5,1.5),translate(-8, -13)').attr('d', viewfieldPath);
 
-           function viewfieldPath() {
-             var d = this.parentNode.__data__;
+             arr = arr.map(function (s) {
+               return uiCmd.display(s.indexOf('.') !== -1 ? _t(s) : s);
+             });
+             return utilArrayUniq(arr).map(function (s) {
+               return {
+                 shortcut: s,
+                 separator: d.separator,
+                 suffix: d.suffix
+               };
+             });
+           }).enter().each(function (d, i, nodes) {
+             var selection = select(this);
+             var click = d.shortcut.toLowerCase().match(/(.*).click/);
 
-             if (d.pano) {
-               return 'M 8,13 m -10,0 a 10,10 0 1,0 20,0 a 10,10 0 1,0 -20,0';
+             if (click && click[1]) {
+               // replace "left_click", "right_click" with mouse icon
+               selection.call(svgIcon('#iD-walkthrough-mouse-' + click[1], 'operation'));
+             } else if (d.shortcut.toLowerCase() === 'long-press') {
+               selection.call(svgIcon('#iD-walkthrough-longpress', 'longpress operation'));
+             } else if (d.shortcut.toLowerCase() === 'tap') {
+               selection.call(svgIcon('#iD-walkthrough-tap', 'tap operation'));
              } else {
-               return 'M 6,9 C 8,8.4 8,8.4 10,9 L 16,-2 C 12,-5 4,-5 0,-2 z';
+               selection.append('kbd').attr('class', 'shortcut').html(function (d) {
+                 return d.shortcut;
+               });
              }
-           }
-         }
 
-         function drawImages(selection) {
-           var enabled = svgMapillaryImages.enabled;
-           var service = getService();
-           layer = selection.selectAll('.layer-mapillary').data(service ? [0] : []);
-           layer.exit().remove();
-           var layerEnter = layer.enter().append('g').attr('class', 'layer-mapillary').style('display', enabled ? 'block' : 'none');
-           layerEnter.append('g').attr('class', 'sequences');
-           layerEnter.append('g').attr('class', 'markers');
-           layer = layerEnter.merge(layer);
-
-           if (enabled) {
-             if (service && ~~context.map().zoom() >= minZoom) {
-               editOn();
-               update();
-               service.loadImages(projection);
-             } else {
-               editOff();
+             if (i < nodes.length - 1) {
+               selection.append('span').html(d.separator || "\xA0" + _t.html('shortcuts.or') + "\xA0");
+             } else if (i === nodes.length - 1 && d.suffix) {
+               selection.append('span').html(d.suffix);
              }
-           }
+           });
+           shortcutKeys.filter(function (d) {
+             return d.gesture;
+           }).each(function () {
+             var selection = select(this);
+             selection.append('span').html('+');
+             selection.append('span').attr('class', 'gesture').html(function (d) {
+               return _t.html(d.gesture);
+             });
+           });
+           shortcutRows.append('td').attr('class', 'shortcut-desc').html(function (d) {
+             return d.text ? _t.html(d.text) : "\xA0";
+           }); // Update
+
+           wrapper.selectAll('.shortcut-tab').style('display', function (d, i) {
+             return i === _activeTab ? 'flex' : 'none';
+           });
          }
 
-         drawImages.enabled = function (_) {
-           if (!arguments.length) return svgMapillaryImages.enabled;
-           svgMapillaryImages.enabled = _;
+         return function (selection, show) {
+           _selection = selection;
 
-           if (svgMapillaryImages.enabled) {
-             showLayer();
-             context.photos().on('change.mapillary_images', update);
+           if (show) {
+             _modalSelection = uiModal(selection);
+
+             _modalSelection.call(shortcutsModal);
            } else {
-             hideLayer();
-             context.photos().on('change.mapillary_images', null);
-           }
+             context.keybinding().on([_t('shortcuts.toggle.key'), '?'], function () {
+               if (context.container().selectAll('.modal-shortcuts').size()) {
+                 // already showing
+                 if (_modalSelection) {
+                   _modalSelection.close();
 
-           dispatch.call('change');
-           return this;
+                   _modalSelection = null;
+                 }
+               } else {
+                 _modalSelection = uiModal(_selection);
+
+                 _modalSelection.call(shortcutsModal);
+               }
+             });
+           }
          };
+       }
 
-         drawImages.supported = function () {
-           return !!getService();
+       function uiDataHeader() {
+         var _datum;
+
+         function dataHeader(selection) {
+           var header = selection.selectAll('.data-header').data(_datum ? [_datum] : [], function (d) {
+             return d.__featurehash__;
+           });
+           header.exit().remove();
+           var headerEnter = header.enter().append('div').attr('class', 'data-header');
+           var iconEnter = headerEnter.append('div').attr('class', 'data-header-icon');
+           iconEnter.append('div').attr('class', 'preset-icon-28').call(svgIcon('#iD-icon-data', 'note-fill'));
+           headerEnter.append('div').attr('class', 'data-header-label').html(_t.html('map_data.layers.custom.title'));
+         }
+
+         dataHeader.datum = function (val) {
+           if (!arguments.length) return _datum;
+           _datum = val;
+           return this;
          };
 
-         init();
-         return drawImages;
+         return dataHeader;
        }
 
-       function svgMapillaryPosition(projection, context) {
-         var throttledRedraw = throttle(function () {
-           update();
-         }, 1000);
+       // It is keyed on the `value` of the entry. Data should be an array of objects like:
+       //   [{
+       //       value:   'string value',  // required
+       //       display: 'label html'     // optional
+       //       title:   'hover text'     // optional
+       //       terms:   ['search terms'] // optional
+       //   }, ...]
 
-         var minZoom = 12;
-         var minViewfieldZoom = 18;
-         var layer = select(null);
+       var _comboHideTimerID;
 
-         var _mapillary;
+       function uiCombobox(context, klass) {
+         var dispatch = dispatch$8('accept', 'cancel');
+         var container = context.container();
+         var _suggestions = [];
+         var _data = [];
+         var _fetched = {};
+         var _selected = null;
+         var _canAutocomplete = true;
+         var _caseSensitive = false;
+         var _cancelFetch = false;
+         var _minItems = 2;
+         var _tDown = 0;
 
-         var viewerCompassAngle;
+         var _mouseEnterHandler, _mouseLeaveHandler;
 
-         function init() {
-           if (svgMapillaryPosition.initialized) return; // run once
+         var _fetcher = function _fetcher(val, cb) {
+           cb(_data.filter(function (d) {
+             var terms = d.terms || [];
+             terms.push(d.value);
+             return terms.some(function (term) {
+               return term.toString().toLowerCase().indexOf(val.toLowerCase()) !== -1;
+             });
+           }));
+         };
 
-           svgMapillaryPosition.initialized = true;
-         }
+         var combobox = function combobox(input, attachTo) {
+           if (!input || input.empty()) return;
+           input.classed('combobox-input', true).on('focus.combo-input', focus).on('blur.combo-input', blur).on('keydown.combo-input', keydown).on('keyup.combo-input', keyup).on('input.combo-input', change).on('mousedown.combo-input', mousedown).each(function () {
+             var parent = this.parentNode;
+             var sibling = this.nextSibling;
+             select(parent).selectAll('.combobox-caret').filter(function (d) {
+               return d === input.node();
+             }).data([input.node()]).enter().insert('div', function () {
+               return sibling;
+             }).attr('class', 'combobox-caret').on('mousedown.combo-caret', function (d3_event) {
+               d3_event.preventDefault(); // don't steal focus from input
 
-         function getService() {
-           if (services.mapillary && !_mapillary) {
-             _mapillary = services.mapillary;
+               input.node().focus(); // focus the input as if it was clicked
 
-             _mapillary.event.on('nodeChanged', throttledRedraw);
+               mousedown(d3_event);
+             }).on('mouseup.combo-caret', function (d3_event) {
+               d3_event.preventDefault(); // don't steal focus from input
 
-             _mapillary.event.on('bearingChanged', function (e) {
-               viewerCompassAngle = e;
-               if (context.map().isTransformed()) return;
-               layer.selectAll('.viewfield-group.currentView').filter(function (d) {
-                 return d.pano;
-               }).attr('transform', transform);
+               mouseup(d3_event);
              });
-           } else if (!services.mapillary && _mapillary) {
-             _mapillary = null;
-           }
+           });
 
-           return _mapillary;
-         }
+           function mousedown(d3_event) {
+             if (d3_event.button !== 0) return; // left click only
 
-         function editOn() {
-           layer.style('display', 'block');
-         }
+             _tDown = +new Date(); // clear selection
 
-         function editOff() {
-           layer.selectAll('.viewfield-group').remove();
-           layer.style('display', 'none');
-         }
+             var start = input.property('selectionStart');
+             var end = input.property('selectionEnd');
 
-         function transform(d) {
-           var t = svgPointTransform(projection)(d);
+             if (start !== end) {
+               var val = utilGetSetValue(input);
+               input.node().setSelectionRange(val.length, val.length);
+               return;
+             }
 
-           if (d.pano && viewerCompassAngle !== null && isFinite(viewerCompassAngle)) {
-             t += ' rotate(' + Math.floor(viewerCompassAngle) + ',0,0)';
-           } else if (d.ca) {
-             t += ' rotate(' + Math.floor(d.ca) + ',0,0)';
+             input.on('mouseup.combo-input', mouseup);
            }
 
-           return t;
-         }
-
-         function update() {
-           var z = ~~context.map().zoom();
-           var showViewfields = z >= minViewfieldZoom;
-           var service = getService();
-           var node = service && service.getActiveImage();
-           var groups = layer.selectAll('.markers').selectAll('.viewfield-group').data(node ? [node] : [], function (d) {
-             return d.key;
-           }); // exit
+           function mouseup(d3_event) {
+             input.on('mouseup.combo-input', null);
+             if (d3_event.button !== 0) return; // left click only
 
-           groups.exit().remove(); // enter
+             if (input.node() !== document.activeElement) return; // exit if this input is not focused
 
-           var groupsEnter = groups.enter().append('g').attr('class', 'viewfield-group currentView highlighted');
-           groupsEnter.append('g').attr('class', 'viewfield-scale'); // update
+             var start = input.property('selectionStart');
+             var end = input.property('selectionEnd');
+             if (start !== end) return; // exit if user is selecting
+             // not showing or showing for a different field - try to show it.
 
-           var markers = groups.merge(groupsEnter).attr('transform', transform).select('.viewfield-scale');
-           markers.selectAll('circle').data([0]).enter().append('circle').attr('dx', '0').attr('dy', '0').attr('r', '6');
-           var viewfields = markers.selectAll('.viewfield').data(showViewfields ? [0] : []);
-           viewfields.exit().remove();
-           viewfields.enter().insert('path', 'circle').attr('class', 'viewfield').classed('pano', function () {
-             return this.parentNode.__data__.pano;
-           }).attr('transform', 'scale(1.5,1.5),translate(-8, -13)').attr('d', viewfieldPath);
+             var combo = container.selectAll('.combobox');
 
-           function viewfieldPath() {
-             var d = this.parentNode.__data__;
+             if (combo.empty() || combo.datum() !== input.node()) {
+               var tOrig = _tDown;
+               window.setTimeout(function () {
+                 if (tOrig !== _tDown) return; // exit if user double clicked
 
-             if (d.pano) {
-               return 'M 8,13 m -10,0 a 10,10 0 1,0 20,0 a 10,10 0 1,0 -20,0';
+                 fetchComboData('', function () {
+                   show();
+                   render();
+                 });
+               }, 250);
              } else {
-               return 'M 6,9 C 8,8.4 8,8.4 10,9 L 16,-2 C 12,-5 4,-5 0,-2 z';
+               hide();
              }
            }
-         }
-
-         function drawImages(selection) {
-           var service = getService();
-           layer = selection.selectAll('.layer-mapillary-position').data(service ? [0] : []);
-           layer.exit().remove();
-           var layerEnter = layer.enter().append('g').attr('class', 'layer-mapillary-position');
-           layerEnter.append('g').attr('class', 'markers');
-           layer = layerEnter.merge(layer);
 
-           if (service && ~~context.map().zoom() >= minZoom) {
-             editOn();
-             update();
-           } else {
-             editOff();
+           function focus() {
+             fetchComboData(''); // prefetch values (may warm taginfo cache)
            }
-         }
-
-         drawImages.enabled = function () {
-           update();
-           return this;
-         };
-
-         drawImages.supported = function () {
-           return !!getService();
-         };
-
-         init();
-         return drawImages;
-       }
-
-       function svgMapillarySigns(projection, context, dispatch) {
-         var throttledRedraw = throttle(function () {
-           dispatch.call('change');
-         }, 1000);
-
-         var minZoom = 12;
-         var layer = select(null);
-
-         var _mapillary;
-
-         function init() {
-           if (svgMapillarySigns.initialized) return; // run once
 
-           svgMapillarySigns.enabled = false;
-           svgMapillarySigns.initialized = true;
-         }
-
-         function getService() {
-           if (services.mapillary && !_mapillary) {
-             _mapillary = services.mapillary;
-
-             _mapillary.event.on('loadedSigns', throttledRedraw);
-           } else if (!services.mapillary && _mapillary) {
-             _mapillary = null;
+           function blur() {
+             _comboHideTimerID = window.setTimeout(hide, 75);
            }
 
-           return _mapillary;
-         }
-
-         function showLayer() {
-           var service = getService();
-           if (!service) return;
-           service.loadSignResources(context);
-           editOn();
-         }
-
-         function hideLayer() {
-           throttledRedraw.cancel();
-           editOff();
-         }
-
-         function editOn() {
-           layer.style('display', 'block');
-         }
-
-         function editOff() {
-           layer.selectAll('.icon-sign').remove();
-           layer.style('display', 'none');
-         }
-
-         function click(d3_event, d) {
-           var service = getService();
-           if (!service) return;
-           context.map().centerEase(d.loc);
-           var selectedImageKey = service.getSelectedImageKey();
-           var imageKey;
-           var highlightedDetection; // Pick one of the images the sign was detected in,
-           // preference given to an image already selected.
-
-           d.detections.forEach(function (detection) {
-             if (!imageKey || selectedImageKey === detection.image_key) {
-               imageKey = detection.image_key;
-               highlightedDetection = detection;
-             }
-           });
+           function show() {
+             hide(); // remove any existing
 
-           if (imageKey === selectedImageKey) {
-             service.highlightDetection(highlightedDetection).selectImage(context, imageKey);
-           } else {
-             service.ensureViewerLoaded(context).then(function () {
-               service.highlightDetection(highlightedDetection).selectImage(context, imageKey).showViewer(context);
+             container.insert('div', ':first-child').datum(input.node()).attr('class', 'combobox' + (klass ? ' combobox-' + klass : '')).style('position', 'absolute').style('display', 'block').style('left', '0px').on('mousedown.combo-container', function (d3_event) {
+               // prevent moving focus out of the input field
+               d3_event.preventDefault();
              });
+             container.on('scroll.combo-scroll', render, true);
            }
-         }
 
-         function filterData(detectedFeatures) {
-           var service = getService();
-           var fromDate = context.photos().fromDate();
-           var toDate = context.photos().toDate();
-           var usernames = context.photos().usernames();
+           function hide() {
+             if (_comboHideTimerID) {
+               window.clearTimeout(_comboHideTimerID);
+               _comboHideTimerID = undefined;
+             }
 
-           if (fromDate) {
-             var fromTimestamp = new Date(fromDate).getTime();
-             detectedFeatures = detectedFeatures.filter(function (feature) {
-               return new Date(feature.last_seen_at).getTime() >= fromTimestamp;
-             });
+             container.selectAll('.combobox').remove();
+             container.on('scroll.combo-scroll', null);
            }
 
-           if (toDate) {
-             var toTimestamp = new Date(toDate).getTime();
-             detectedFeatures = detectedFeatures.filter(function (feature) {
-               return new Date(feature.first_seen_at).getTime() <= toTimestamp;
-             });
-           }
+           function keydown(d3_event) {
+             var shown = !container.selectAll('.combobox').empty();
+             var tagName = input.node() ? input.node().tagName.toLowerCase() : '';
 
-           if (usernames && service) {
-             detectedFeatures = detectedFeatures.filter(function (feature) {
-               return feature.detections.some(function (detection) {
-                 var imageKey = detection.image_key;
-                 var image = service.cachedImage(imageKey);
-                 return image && usernames.indexOf(image.captured_by) !== -1;
-               });
-             });
-           }
+             switch (d3_event.keyCode) {
+               case 8: // ⌫ Backspace
 
-           return detectedFeatures;
-         }
+               case 46:
+                 // ⌦ Delete
+                 d3_event.stopPropagation();
+                 _selected = null;
+                 render();
+                 input.on('input.combo-input', function () {
+                   var start = input.property('selectionStart');
+                   input.node().setSelectionRange(start, start);
+                   input.on('input.combo-input', change);
+                 });
+                 break;
 
-         function update() {
-           var service = getService();
-           var data = service ? service.signs(projection) : [];
-           data = filterData(data);
-           var selectedImageKey = service.getSelectedImageKey();
-           var transform = svgPointTransform(projection);
-           var signs = layer.selectAll('.icon-sign').data(data, function (d) {
-             return d.key;
-           }); // exit
+               case 9:
+                 // ⇥ Tab
+                 accept();
+                 break;
 
-           signs.exit().remove(); // enter
+               case 13:
+                 // ↩ Return
+                 d3_event.preventDefault();
+                 d3_event.stopPropagation();
+                 break;
 
-           var enter = signs.enter().append('g').attr('class', 'icon-sign icon-detected').on('click', click);
-           enter.append('use').attr('width', '24px').attr('height', '24px').attr('x', '-12px').attr('y', '-12px').attr('xlink:href', function (d) {
-             return '#' + d.value;
-           });
-           enter.append('rect').attr('width', '24px').attr('height', '24px').attr('x', '-12px').attr('y', '-12px'); // update
+               case 38:
+                 // ↑ Up arrow
+                 if (tagName === 'textarea' && !shown) return;
+                 d3_event.preventDefault();
 
-           signs.merge(enter).attr('transform', transform).classed('currentView', function (d) {
-             return d.detections.some(function (detection) {
-               return detection.image_key === selectedImageKey;
-             });
-           }).sort(function (a, b) {
-             var aSelected = a.detections.some(function (detection) {
-               return detection.image_key === selectedImageKey;
-             });
-             var bSelected = b.detections.some(function (detection) {
-               return detection.image_key === selectedImageKey;
-             });
+                 if (tagName === 'input' && !shown) {
+                   show();
+                 }
 
-             if (aSelected === bSelected) {
-               return b.loc[1] - a.loc[1]; // sort Y
-             } else if (aSelected) {
-               return 1;
-             }
+                 nav(-1);
+                 break;
 
-             return -1;
-           });
-         }
+               case 40:
+                 // ↓ Down arrow
+                 if (tagName === 'textarea' && !shown) return;
+                 d3_event.preventDefault();
 
-         function drawSigns(selection) {
-           var enabled = svgMapillarySigns.enabled;
-           var service = getService();
-           layer = selection.selectAll('.layer-mapillary-signs').data(service ? [0] : []);
-           layer.exit().remove();
-           layer = layer.enter().append('g').attr('class', 'layer-mapillary-signs layer-mapillary-detections').style('display', enabled ? 'block' : 'none').merge(layer);
+                 if (tagName === 'input' && !shown) {
+                   show();
+                 }
 
-           if (enabled) {
-             if (service && ~~context.map().zoom() >= minZoom) {
-               editOn();
-               update();
-               service.loadSigns(projection);
-               service.showSignDetections(true);
-             } else {
-               editOff();
+                 nav(+1);
+                 break;
              }
-           } else if (service) {
-             service.showSignDetections(false);
            }
-         }
 
-         drawSigns.enabled = function (_) {
-           if (!arguments.length) return svgMapillarySigns.enabled;
-           svgMapillarySigns.enabled = _;
+           function keyup(d3_event) {
+             switch (d3_event.keyCode) {
+               case 27:
+                 // ⎋ Escape
+                 cancel();
+                 break;
 
-           if (svgMapillarySigns.enabled) {
-             showLayer();
-             context.photos().on('change.mapillary_signs', update);
-           } else {
-             hideLayer();
-             context.photos().on('change.mapillary_signs', null);
-           }
+               case 13:
+                 // ↩ Return
+                 accept();
+                 break;
+             }
+           } // Called whenever the input value is changed (e.g. on typing)
+
+
+           function change() {
+             fetchComboData(value(), function () {
+               _selected = null;
+               var val = input.property('value');
 
-           dispatch.call('change');
-           return this;
-         };
+               if (_suggestions.length) {
+                 if (input.property('selectionEnd') === val.length) {
+                   _selected = tryAutocomplete();
+                 }
 
-         drawSigns.supported = function () {
-           return !!getService();
-         };
+                 if (!_selected) {
+                   _selected = val;
+                 }
+               }
 
-         init();
-         return drawSigns;
-       }
+               if (val.length) {
+                 var combo = container.selectAll('.combobox');
 
-       function svgMapillaryMapFeatures(projection, context, dispatch) {
-         var throttledRedraw = throttle(function () {
-           dispatch.call('change');
-         }, 1000);
+                 if (combo.empty()) {
+                   show();
+                 }
+               } else {
+                 hide();
+               }
 
-         var minZoom = 12;
-         var layer = select(null);
+               render();
+             });
+           } // Called when the user presses up/down arrows to navigate the list
 
-         var _mapillary;
 
-         function init() {
-           if (svgMapillaryMapFeatures.initialized) return; // run once
+           function nav(dir) {
+             if (_suggestions.length) {
+               // try to determine previously selected index..
+               var index = -1;
 
-           svgMapillaryMapFeatures.enabled = false;
-           svgMapillaryMapFeatures.initialized = true;
-         }
+               for (var i = 0; i < _suggestions.length; i++) {
+                 if (_selected && _suggestions[i].value === _selected) {
+                   index = i;
+                   break;
+                 }
+               } // pick new _selected
 
-         function getService() {
-           if (services.mapillary && !_mapillary) {
-             _mapillary = services.mapillary;
 
-             _mapillary.event.on('loadedMapFeatures', throttledRedraw);
-           } else if (!services.mapillary && _mapillary) {
-             _mapillary = null;
+               index = Math.max(Math.min(index + dir, _suggestions.length - 1), 0);
+               _selected = _suggestions[index].value;
+               input.property('value', _selected);
+             }
+
+             render();
+             ensureVisible();
            }
 
-           return _mapillary;
-         }
+           function ensureVisible() {
+             var combo = container.selectAll('.combobox');
+             if (combo.empty()) return;
+             var containerRect = container.node().getBoundingClientRect();
+             var comboRect = combo.node().getBoundingClientRect();
 
-         function showLayer() {
-           var service = getService();
-           if (!service) return;
-           service.loadObjectResources(context);
-           editOn();
-         }
+             if (comboRect.bottom > containerRect.bottom) {
+               var node = attachTo ? attachTo.node() : input.node();
+               node.scrollIntoView({
+                 behavior: 'instant',
+                 block: 'center'
+               });
+               render();
+             } // https://stackoverflow.com/questions/11039885/scrollintoview-causing-the-whole-page-to-move
 
-         function hideLayer() {
-           throttledRedraw.cancel();
-           editOff();
-         }
 
-         function editOn() {
-           layer.style('display', 'block');
-         }
+             var selected = combo.selectAll('.combobox-option.selected').node();
 
-         function editOff() {
-           layer.selectAll('.icon-map-feature').remove();
-           layer.style('display', 'none');
-         }
+             if (selected) {
+               selected.scrollIntoView({
+                 behavior: 'smooth',
+                 block: 'nearest'
+               });
+             }
+           }
 
-         function click(d3_event, d) {
-           var service = getService();
-           if (!service) return;
-           context.map().centerEase(d.loc);
-           var selectedImageKey = service.getSelectedImageKey();
-           var imageKey;
-           var highlightedDetection; // Pick one of the images the map feature was detected in,
-           // preference given to an image already selected.
+           function value() {
+             var value = input.property('value');
+             var start = input.property('selectionStart');
+             var end = input.property('selectionEnd');
 
-           d.detections.forEach(function (detection) {
-             if (!imageKey || selectedImageKey === detection.image_key) {
-               imageKey = detection.image_key;
-               highlightedDetection = detection;
+             if (start && end) {
+               value = value.substring(0, start);
              }
-           });
 
-           if (imageKey === selectedImageKey) {
-             service.highlightDetection(highlightedDetection).selectImage(context, imageKey);
-           } else {
-             service.ensureViewerLoaded(context).then(function () {
-               service.highlightDetection(highlightedDetection).selectImage(context, imageKey).showViewer(context);
-             });
+             return value;
            }
-         }
 
-         function filterData(detectedFeatures) {
-           var service = getService();
-           var fromDate = context.photos().fromDate();
-           var toDate = context.photos().toDate();
-           var usernames = context.photos().usernames();
+           function fetchComboData(v, cb) {
+             _cancelFetch = false;
 
-           if (fromDate) {
-             var fromTimestamp = new Date(fromDate).getTime();
-             detectedFeatures = detectedFeatures.filter(function (feature) {
-               return new Date(feature.last_seen_at).getTime() >= fromTimestamp;
-             });
-           }
+             _fetcher.call(input, v, function (results) {
+               // already chose a value, don't overwrite or autocomplete it
+               if (_cancelFetch) return;
+               _suggestions = results;
+               results.forEach(function (d) {
+                 _fetched[d.value] = d;
+               });
 
-           if (toDate) {
-             var toTimestamp = new Date(toDate).getTime();
-             detectedFeatures = detectedFeatures.filter(function (feature) {
-               return new Date(feature.first_seen_at).getTime() <= toTimestamp;
+               if (cb) {
+                 cb();
+               }
              });
            }
 
-           if (usernames && service) {
-             detectedFeatures = detectedFeatures.filter(function (feature) {
-               return feature.detections.some(function (detection) {
-                 var imageKey = detection.image_key;
-                 var image = service.cachedImage(imageKey);
-                 return image && usernames.indexOf(image.captured_by) !== -1;
-               });
-             });
-           }
+           function tryAutocomplete() {
+             if (!_canAutocomplete) return;
+             var val = _caseSensitive ? value() : value().toLowerCase();
+             if (!val) return; // Don't autocomplete if user is typing a number - #4935
 
-           return detectedFeatures;
-         }
+             if (!isNaN(parseFloat(val)) && isFinite(val)) return;
+             var bestIndex = -1;
 
-         function update() {
-           var service = getService();
-           var data = service ? service.mapFeatures(projection) : [];
-           data = filterData(data);
-           var selectedImageKey = service && service.getSelectedImageKey();
-           var transform = svgPointTransform(projection);
-           var mapFeatures = layer.selectAll('.icon-map-feature').data(data, function (d) {
-             return d.key;
-           }); // exit
+             for (var i = 0; i < _suggestions.length; i++) {
+               var suggestion = _suggestions[i].value;
+               var compare = _caseSensitive ? suggestion : suggestion.toLowerCase(); // if search string matches suggestion exactly, pick it..
 
-           mapFeatures.exit().remove(); // enter
+               if (compare === val) {
+                 bestIndex = i;
+                 break; // otherwise lock in the first result that starts with the search string..
+               } else if (bestIndex === -1 && compare.indexOf(val) === 0) {
+                 bestIndex = i;
+               }
+             }
 
-           var enter = mapFeatures.enter().append('g').attr('class', 'icon-map-feature icon-detected').on('click', click);
-           enter.append('title').text(function (d) {
-             var id = d.value.replace(/--/g, '.').replace(/-/g, '_');
-             return _t('mapillary_map_features.' + id);
-           });
-           enter.append('use').attr('width', '24px').attr('height', '24px').attr('x', '-12px').attr('y', '-12px').attr('xlink:href', function (d) {
-             if (d.value === 'object--billboard') {
-               // no billboard icon right now, so use the advertisement icon
-               return '#object--sign--advertisement';
+             if (bestIndex !== -1) {
+               var bestVal = _suggestions[bestIndex].value;
+               input.property('value', bestVal);
+               input.node().setSelectionRange(val.length, bestVal.length);
+               return bestVal;
              }
+           }
 
-             return '#' + d.value;
-           });
-           enter.append('rect').attr('width', '24px').attr('height', '24px').attr('x', '-12px').attr('y', '-12px'); // update
+           function render() {
+             if (_suggestions.length < _minItems || document.activeElement !== input.node()) {
+               hide();
+               return;
+             }
 
-           mapFeatures.merge(enter).attr('transform', transform).classed('currentView', function (d) {
-             return d.detections.some(function (detection) {
-               return detection.image_key === selectedImageKey;
-             });
-           }).sort(function (a, b) {
-             var aSelected = a.detections.some(function (detection) {
-               return detection.image_key === selectedImageKey;
-             });
-             var bSelected = b.detections.some(function (detection) {
-               return detection.image_key === selectedImageKey;
+             var shown = !container.selectAll('.combobox').empty();
+             if (!shown) return;
+             var combo = container.selectAll('.combobox');
+             var options = combo.selectAll('.combobox-option').data(_suggestions, function (d) {
+               return d.value;
              });
+             options.exit().remove(); // enter/update
 
-             if (aSelected === bSelected) {
-               return b.loc[1] - a.loc[1]; // sort Y
-             } else if (aSelected) {
-               return 1;
-             }
-
-             return -1;
-           });
-         }
+             options.enter().append('a').attr('class', function (d) {
+               return 'combobox-option ' + (d.klass || '');
+             }).attr('title', function (d) {
+               return d.title;
+             }).html(function (d) {
+               return d.display || d.value;
+             }).on('mouseenter', _mouseEnterHandler).on('mouseleave', _mouseLeaveHandler).merge(options).classed('selected', function (d) {
+               return d.value === _selected;
+             }).on('click.combo-option', accept).order();
+             var node = attachTo ? attachTo.node() : input.node();
+             var containerRect = container.node().getBoundingClientRect();
+             var rect = node.getBoundingClientRect();
+             combo.style('left', rect.left + 5 - containerRect.left + 'px').style('width', rect.width - 10 + 'px').style('top', rect.height + rect.top - containerRect.top + 'px');
+           } // Dispatches an 'accept' event
+           // Then hides the combobox.
 
-         function drawMapFeatures(selection) {
-           var enabled = svgMapillaryMapFeatures.enabled;
-           var service = getService();
-           layer = selection.selectAll('.layer-mapillary-map-features').data(service ? [0] : []);
-           layer.exit().remove();
-           layer = layer.enter().append('g').attr('class', 'layer-mapillary-map-features layer-mapillary-detections').style('display', enabled ? 'block' : 'none').merge(layer);
 
-           if (enabled) {
-             if (service && ~~context.map().zoom() >= minZoom) {
-               editOn();
-               update();
-               service.loadMapFeatures(projection);
-               service.showFeatureDetections(true);
-             } else {
-               editOff();
-             }
-           } else if (service) {
-             service.showFeatureDetections(false);
-           }
-         }
+           function accept(d3_event, d) {
+             _cancelFetch = true;
+             var thiz = input.node();
 
-         drawMapFeatures.enabled = function (_) {
-           if (!arguments.length) return svgMapillaryMapFeatures.enabled;
-           svgMapillaryMapFeatures.enabled = _;
+             if (d) {
+               // user clicked on a suggestion
+               utilGetSetValue(input, d.value); // replace field contents
 
-           if (svgMapillaryMapFeatures.enabled) {
-             showLayer();
-             context.photos().on('change.mapillary_map_features', update);
-           } else {
-             hideLayer();
-             context.photos().on('change.mapillary_map_features', null);
-           }
+               utilTriggerEvent(input, 'change');
+             } // clear (and keep) selection
 
-           dispatch.call('change');
-           return this;
-         };
 
-         drawMapFeatures.supported = function () {
-           return !!getService();
-         };
+             var val = utilGetSetValue(input);
+             thiz.setSelectionRange(val.length, val.length);
+             d = _fetched[val];
+             dispatch.call('accept', thiz, d, val);
+             hide();
+           } // Dispatches an 'cancel' event
+           // Then hides the combobox.
 
-         init();
-         return drawMapFeatures;
-       }
 
-       function svgOpenstreetcamImages(projection, context, dispatch) {
-         var throttledRedraw = throttle(function () {
-           dispatch.call('change');
-         }, 1000);
+           function cancel() {
+             _cancelFetch = true;
+             var thiz = input.node(); // clear (and remove) selection, and replace field contents
 
-         var minZoom = 12;
-         var minMarkerZoom = 16;
-         var minViewfieldZoom = 18;
-         var layer = select(null);
+             var val = utilGetSetValue(input);
+             var start = input.property('selectionStart');
+             var end = input.property('selectionEnd');
+             val = val.slice(0, start) + val.slice(end);
+             utilGetSetValue(input, val);
+             thiz.setSelectionRange(val.length, val.length);
+             dispatch.call('cancel', thiz);
+             hide();
+           }
+         };
 
-         var _openstreetcam;
+         combobox.canAutocomplete = function (val) {
+           if (!arguments.length) return _canAutocomplete;
+           _canAutocomplete = val;
+           return combobox;
+         };
 
-         function init() {
-           if (svgOpenstreetcamImages.initialized) return; // run once
+         combobox.caseSensitive = function (val) {
+           if (!arguments.length) return _caseSensitive;
+           _caseSensitive = val;
+           return combobox;
+         };
 
-           svgOpenstreetcamImages.enabled = false;
-           svgOpenstreetcamImages.initialized = true;
-         }
+         combobox.data = function (val) {
+           if (!arguments.length) return _data;
+           _data = val;
+           return combobox;
+         };
 
-         function getService() {
-           if (services.openstreetcam && !_openstreetcam) {
-             _openstreetcam = services.openstreetcam;
+         combobox.fetcher = function (val) {
+           if (!arguments.length) return _fetcher;
+           _fetcher = val;
+           return combobox;
+         };
 
-             _openstreetcam.event.on('loadedImages', throttledRedraw);
-           } else if (!services.openstreetcam && _openstreetcam) {
-             _openstreetcam = null;
-           }
+         combobox.minItems = function (val) {
+           if (!arguments.length) return _minItems;
+           _minItems = val;
+           return combobox;
+         };
 
-           return _openstreetcam;
-         }
+         combobox.itemsMouseEnter = function (val) {
+           if (!arguments.length) return _mouseEnterHandler;
+           _mouseEnterHandler = val;
+           return combobox;
+         };
 
-         function showLayer() {
-           var service = getService();
-           if (!service) return;
-           editOn();
-           layer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end', function () {
-             dispatch.call('change');
-           });
-         }
+         combobox.itemsMouseLeave = function (val) {
+           if (!arguments.length) return _mouseLeaveHandler;
+           _mouseLeaveHandler = val;
+           return combobox;
+         };
 
-         function hideLayer() {
-           throttledRedraw.cancel();
-           layer.transition().duration(250).style('opacity', 0).on('end', editOff);
-         }
+         return utilRebind(combobox, dispatch, 'on');
+       }
 
-         function editOn() {
-           layer.style('display', 'block');
-         }
+       uiCombobox.off = function (input, context) {
+         input.on('focus.combo-input', null).on('blur.combo-input', null).on('keydown.combo-input', null).on('keyup.combo-input', null).on('input.combo-input', null).on('mousedown.combo-input', null).on('mouseup.combo-input', null);
+         context.container().on('scroll.combo-scroll', null);
+       };
 
-         function editOff() {
-           layer.selectAll('.viewfield-group').remove();
-           layer.style('display', 'none');
-         }
+       function uiDisclosure(context, key, expandedDefault) {
+         var dispatch = dispatch$8('toggled');
 
-         function click(d3_event, d) {
-           var service = getService();
-           if (!service) return;
-           service.ensureViewerLoaded(context).then(function () {
-             service.selectImage(context, d.key).showViewer(context);
-           });
-           context.map().centerEase(d.loc);
-         }
+         var _expanded;
 
-         function mouseover(d3_event, d) {
-           var service = getService();
-           if (service) service.setStyles(context, d);
-         }
+         var _label = utilFunctor('');
 
-         function mouseout() {
-           var service = getService();
-           if (service) service.setStyles(context, null);
-         }
+         var _updatePreference = true;
 
-         function transform(d) {
-           var t = svgPointTransform(projection)(d);
+         var _content = function _content() {};
 
-           if (d.ca) {
-             t += ' rotate(' + Math.floor(d.ca) + ',0,0)';
+         var disclosure = function disclosure(selection) {
+           if (_expanded === undefined || _expanded === null) {
+             // loading _expanded here allows it to be reset by calling `disclosure.expanded(null)`
+             var preference = corePreferences('disclosure.' + key + '.expanded');
+             _expanded = preference === null ? !!expandedDefault : preference === 'true';
            }
 
-           return t;
-         }
+           var hideToggle = selection.selectAll('.hide-toggle-' + key).data([0]); // enter
 
-         function filterImages(images) {
-           var fromDate = context.photos().fromDate();
-           var toDate = context.photos().toDate();
-           var usernames = context.photos().usernames();
+           var hideToggleEnter = hideToggle.enter().append('a').attr('href', '#').attr('class', 'hide-toggle hide-toggle-' + key).call(svgIcon('', 'pre-text', 'hide-toggle-icon'));
+           hideToggleEnter.append('span').attr('class', 'hide-toggle-text'); // update
 
-           if (fromDate) {
-             var fromTimestamp = new Date(fromDate).getTime();
-             images = images.filter(function (item) {
-               return new Date(item.captured_at).getTime() >= fromTimestamp;
-             });
-           }
+           hideToggle = hideToggleEnter.merge(hideToggle);
+           hideToggle.on('click', toggle).classed('expanded', _expanded);
+           hideToggle.selectAll('.hide-toggle-text').html(_label());
+           hideToggle.selectAll('.hide-toggle-icon').attr('xlink:href', _expanded ? '#iD-icon-down' : _mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-backward' : '#iD-icon-forward');
+           var wrap = selection.selectAll('.disclosure-wrap').data([0]); // enter/update
 
-           if (toDate) {
-             var toTimestamp = new Date(toDate).getTime();
-             images = images.filter(function (item) {
-               return new Date(item.captured_at).getTime() <= toTimestamp;
-             });
-           }
+           wrap = wrap.enter().append('div').attr('class', 'disclosure-wrap disclosure-wrap-' + key).merge(wrap).classed('hide', !_expanded);
 
-           if (usernames) {
-             images = images.filter(function (item) {
-               return usernames.indexOf(item.captured_by) !== -1;
-             });
+           if (_expanded) {
+             wrap.call(_content);
            }
 
-           return images;
-         }
+           function toggle(d3_event) {
+             d3_event.preventDefault();
+             _expanded = !_expanded;
 
-         function filterSequences(sequences) {
-           var fromDate = context.photos().fromDate();
-           var toDate = context.photos().toDate();
-           var usernames = context.photos().usernames();
+             if (_updatePreference) {
+               corePreferences('disclosure.' + key + '.expanded', _expanded);
+             }
 
-           if (fromDate) {
-             var fromTimestamp = new Date(fromDate).getTime();
-             sequences = sequences.filter(function (image) {
-               return new Date(image.properties.captured_at).getTime() >= fromTimestamp;
-             });
-           }
+             hideToggle.classed('expanded', _expanded);
+             hideToggle.selectAll('.hide-toggle-icon').attr('xlink:href', _expanded ? '#iD-icon-down' : _mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-backward' : '#iD-icon-forward');
+             wrap.call(uiToggle(_expanded));
 
-           if (toDate) {
-             var toTimestamp = new Date(toDate).getTime();
-             sequences = sequences.filter(function (image) {
-               return new Date(image.properties.captured_at).getTime() <= toTimestamp;
-             });
-           }
+             if (_expanded) {
+               wrap.call(_content);
+             }
 
-           if (usernames) {
-             sequences = sequences.filter(function (image) {
-               return usernames.indexOf(image.properties.captured_by) !== -1;
-             });
+             dispatch.call('toggled', this, _expanded);
            }
+         };
 
-           return sequences;
-         }
+         disclosure.label = function (val) {
+           if (!arguments.length) return _label;
+           _label = utilFunctor(val);
+           return disclosure;
+         };
 
-         function update() {
-           var viewer = context.container().select('.photoviewer');
-           var selected = viewer.empty() ? undefined : viewer.datum();
-           var z = ~~context.map().zoom();
-           var showMarkers = z >= minMarkerZoom;
-           var showViewfields = z >= minViewfieldZoom;
-           var service = getService();
-           var sequences = [];
-           var images = [];
+         disclosure.expanded = function (val) {
+           if (!arguments.length) return _expanded;
+           _expanded = val;
+           return disclosure;
+         };
 
-           if (context.photos().showsFlat()) {
-             sequences = service ? service.sequences(projection) : [];
-             images = service && showMarkers ? service.images(projection) : [];
-             sequences = filterSequences(sequences);
-             images = filterImages(images);
-           }
+         disclosure.updatePreference = function (val) {
+           if (!arguments.length) return _updatePreference;
+           _updatePreference = val;
+           return disclosure;
+         };
 
-           var traces = layer.selectAll('.sequences').selectAll('.sequence').data(sequences, function (d) {
-             return d.properties.key;
-           }); // exit
+         disclosure.content = function (val) {
+           if (!arguments.length) return _content;
+           _content = val;
+           return disclosure;
+         };
+
+         return utilRebind(disclosure, dispatch, 'on');
+       }
 
-           traces.exit().remove(); // enter/update
+       // Can be labeled and collapsible.
 
-           traces = traces.enter().append('path').attr('class', 'sequence').merge(traces).attr('d', svgPath(projection).geojson);
-           var groups = layer.selectAll('.markers').selectAll('.viewfield-group').data(images, function (d) {
-             return d.key;
-           }); // exit
+       function uiSection(id, context) {
+         var _classes = utilFunctor('');
 
-           groups.exit().remove(); // enter
+         var _shouldDisplay;
 
-           var groupsEnter = groups.enter().append('g').attr('class', 'viewfield-group').on('mouseenter', mouseover).on('mouseleave', mouseout).on('click', click);
-           groupsEnter.append('g').attr('class', 'viewfield-scale'); // update
+         var _content;
 
-           var markers = groups.merge(groupsEnter).sort(function (a, b) {
-             return a === selected ? 1 : b === selected ? -1 : b.loc[1] - a.loc[1]; // sort Y
-           }).attr('transform', transform).select('.viewfield-scale');
-           markers.selectAll('circle').data([0]).enter().append('circle').attr('dx', '0').attr('dy', '0').attr('r', '6');
-           var viewfields = markers.selectAll('.viewfield').data(showViewfields ? [0] : []);
-           viewfields.exit().remove();
-           viewfields.enter() // viewfields may or may not be drawn...
-           .insert('path', 'circle') // but if they are, draw below the circles
-           .attr('class', 'viewfield').attr('transform', 'scale(1.5,1.5),translate(-8, -13)').attr('d', 'M 6,9 C 8,8.4 8,8.4 10,9 L 16,-2 C 12,-5 4,-5 0,-2 z');
-         }
+         var _disclosure;
 
-         function drawImages(selection) {
-           var enabled = svgOpenstreetcamImages.enabled,
-               service = getService();
-           layer = selection.selectAll('.layer-openstreetcam').data(service ? [0] : []);
-           layer.exit().remove();
-           var layerEnter = layer.enter().append('g').attr('class', 'layer-openstreetcam').style('display', enabled ? 'block' : 'none');
-           layerEnter.append('g').attr('class', 'sequences');
-           layerEnter.append('g').attr('class', 'markers');
-           layer = layerEnter.merge(layer);
+         var _label;
 
-           if (enabled) {
-             if (service && ~~context.map().zoom() >= minZoom) {
-               editOn();
-               update();
-               service.loadImages(projection);
-             } else {
-               editOff();
-             }
-           }
-         }
+         var _expandedByDefault = utilFunctor(true);
 
-         drawImages.enabled = function (_) {
-           if (!arguments.length) return svgOpenstreetcamImages.enabled;
-           svgOpenstreetcamImages.enabled = _;
+         var _disclosureContent;
 
-           if (svgOpenstreetcamImages.enabled) {
-             showLayer();
-             context.photos().on('change.openstreetcam_images', update);
-           } else {
-             hideLayer();
-             context.photos().on('change.openstreetcam_images', null);
-           }
+         var _disclosureExpanded;
 
-           dispatch.call('change');
-           return this;
+         var _containerSelection = select(null);
+
+         var section = {
+           id: id
          };
 
-         drawImages.supported = function () {
-           return !!getService();
+         section.classes = function (val) {
+           if (!arguments.length) return _classes;
+           _classes = utilFunctor(val);
+           return section;
          };
 
-         init();
-         return drawImages;
-       }
+         section.label = function (val) {
+           if (!arguments.length) return _label;
+           _label = utilFunctor(val);
+           return section;
+         };
 
-       function svgOsm(projection, context, dispatch) {
-         var enabled = true;
+         section.expandedByDefault = function (val) {
+           if (!arguments.length) return _expandedByDefault;
+           _expandedByDefault = utilFunctor(val);
+           return section;
+         };
 
-         function drawOsm(selection) {
-           selection.selectAll('.layer-osm').data(['covered', 'areas', 'lines', 'points', 'labels']).enter().append('g').attr('class', function (d) {
-             return 'layer-osm ' + d;
-           });
-           selection.selectAll('.layer-osm.points').selectAll('.points-group').data(['points', 'midpoints', 'vertices', 'turns']).enter().append('g').attr('class', function (d) {
-             return 'points-group ' + d;
-           });
-         }
+         section.shouldDisplay = function (val) {
+           if (!arguments.length) return _shouldDisplay;
+           _shouldDisplay = utilFunctor(val);
+           return section;
+         };
 
-         function showLayer() {
-           var layer = context.surface().selectAll('.data-layer.osm');
-           layer.interrupt();
-           layer.classed('disabled', false).style('opacity', 0).transition().duration(250).style('opacity', 1).on('end interrupt', function () {
-             dispatch.call('change');
-           });
-         }
+         section.content = function (val) {
+           if (!arguments.length) return _content;
+           _content = val;
+           return section;
+         };
 
-         function hideLayer() {
-           var layer = context.surface().selectAll('.data-layer.osm');
-           layer.interrupt();
-           layer.transition().duration(250).style('opacity', 0).on('end interrupt', function () {
-             layer.classed('disabled', true);
-             dispatch.call('change');
-           });
-         }
+         section.disclosureContent = function (val) {
+           if (!arguments.length) return _disclosureContent;
+           _disclosureContent = val;
+           return section;
+         };
 
-         drawOsm.enabled = function (val) {
-           if (!arguments.length) return enabled;
-           enabled = val;
+         section.disclosureExpanded = function (val) {
+           if (!arguments.length) return _disclosureExpanded;
+           _disclosureExpanded = val;
+           return section;
+         }; // may be called multiple times
 
-           if (enabled) {
-             showLayer();
-           } else {
-             hideLayer();
-           }
 
-           dispatch.call('change');
-           return this;
-         };
+         section.render = function (selection) {
+           _containerSelection = selection.selectAll('.section-' + id).data([0]);
 
-         return drawOsm;
-       }
+           var sectionEnter = _containerSelection.enter().append('div').attr('class', 'section section-' + id + ' ' + (_classes && _classes() || ''));
 
-       var _notesEnabled = false;
+           _containerSelection = sectionEnter.merge(_containerSelection);
 
-       var _osmService;
+           _containerSelection.call(renderContent);
+         };
 
-       function svgNotes(projection, context, dispatch$1) {
-         if (!dispatch$1) {
-           dispatch$1 = dispatch('change');
-         }
+         section.reRender = function () {
+           _containerSelection.call(renderContent);
+         };
 
-         var throttledRedraw = throttle(function () {
-           dispatch$1.call('change');
-         }, 1000);
+         section.selection = function () {
+           return _containerSelection;
+         };
 
-         var minZoom = 12;
-         var touchLayer = select(null);
-         var drawLayer = select(null);
-         var _notesVisible = false;
+         section.disclosure = function () {
+           return _disclosure;
+         }; // may be called multiple times
 
-         function markerPath(selection, klass) {
-           selection.attr('class', klass).attr('transform', 'translate(-8, -22)').attr('d', 'm17.5,0l-15,0c-1.37,0 -2.5,1.12 -2.5,2.5l0,11.25c0,1.37 1.12,2.5 2.5,2.5l3.75,0l0,3.28c0,0.38 0.43,0.6 0.75,0.37l4.87,-3.65l5.62,0c1.37,0 2.5,-1.12 2.5,-2.5l0,-11.25c0,-1.37 -1.12,-2.5 -2.5,-2.5z');
-         } // Loosely-coupled osm service for fetching notes.
 
+         function renderContent(selection) {
+           if (_shouldDisplay) {
+             var shouldDisplay = _shouldDisplay();
 
-         function getService() {
-           if (services.osm && !_osmService) {
-             _osmService = services.osm;
+             selection.classed('hide', !shouldDisplay);
 
-             _osmService.on('loadedNotes', throttledRedraw);
-           } else if (!services.osm && _osmService) {
-             _osmService = null;
+             if (!shouldDisplay) {
+               selection.html('');
+               return;
+             }
            }
 
-           return _osmService;
-         } // Show the notes
-
+           if (_disclosureContent) {
+             if (!_disclosure) {
+               _disclosure = uiDisclosure(context, id.replace(/-/g, '_'), _expandedByDefault()).label(_label || '')
+               /*.on('toggled', function(expanded) {
+                   if (expanded) { selection.node().parentNode.scrollTop += 200; }
+               })*/
+               .content(_disclosureContent);
+             }
 
-         function editOn() {
-           if (!_notesVisible) {
-             _notesVisible = true;
-             drawLayer.style('display', 'block');
-           }
-         } // Immediately remove the notes and their touch targets
+             if (_disclosureExpanded !== undefined) {
+               _disclosure.expanded(_disclosureExpanded);
 
+               _disclosureExpanded = undefined;
+             }
 
-         function editOff() {
-           if (_notesVisible) {
-             _notesVisible = false;
-             drawLayer.style('display', 'none');
-             drawLayer.selectAll('.note').remove();
-             touchLayer.selectAll('.note').remove();
+             selection.call(_disclosure);
+             return;
            }
-         } // Enable the layer.  This shows the notes and transitions them to visible.
 
+           if (_content) {
+             selection.call(_content);
+           }
+         }
 
-         function layerOn() {
-           editOn();
-           drawLayer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end interrupt', function () {
-             dispatch$1.call('change');
-           });
-         } // Disable the layer.  This transitions the layer invisible and then hides the notes.
+         return section;
+       }
 
+       // {
+       //   key: 'string',     // required
+       //   value: 'string'    // optional
+       // }
+       //   -or-
+       // {
+       //   qid: 'string'      // brand wikidata  (e.g. 'Q37158')
+       // }
+       //
 
-         function layerOff() {
-           throttledRedraw.cancel();
-           drawLayer.interrupt();
-           touchLayer.selectAll('.note').remove();
-           drawLayer.transition().duration(250).style('opacity', 0).on('end interrupt', function () {
-             editOff();
-             dispatch$1.call('change');
-           });
-         } // Update the note markers
+       function uiTagReference(what) {
+         var wikibase = what.qid ? services.wikidata : services.osmWikibase;
+         var tagReference = {};
 
+         var _button = select(null);
 
-         function updateMarkers() {
-           if (!_notesVisible || !_notesEnabled) return;
-           var service = getService();
-           var selectedID = context.selectedNoteID();
-           var data = service ? service.notes(projection) : [];
-           var getTransform = svgPointTransform(projection); // Draw markers..
+         var _body = select(null);
 
-           var notes = drawLayer.selectAll('.note').data(data, function (d) {
-             return d.status + d.id;
-           }); // exit
+         var _loaded;
 
-           notes.exit().remove(); // enter
+         var _showing;
 
-           var notesEnter = notes.enter().append('g').attr('class', function (d) {
-             return 'note note-' + d.id + ' ' + d.status;
-           }).classed('new', function (d) {
-             return d.id < 0;
-           });
-           notesEnter.append('ellipse').attr('cx', 0.5).attr('cy', 1).attr('rx', 6.5).attr('ry', 3).attr('class', 'stroke');
-           notesEnter.append('path').call(markerPath, 'shadow');
-           notesEnter.append('use').attr('class', 'note-fill').attr('width', '20px').attr('height', '20px').attr('x', '-8px').attr('y', '-22px').attr('xlink:href', '#iD-icon-note');
-           notesEnter.selectAll('.icon-annotation').data(function (d) {
-             return [d];
-           }).enter().append('use').attr('class', 'icon-annotation').attr('width', '10px').attr('height', '10px').attr('x', '-3px').attr('y', '-19px').attr('xlink:href', function (d) {
-             return '#iD-icon-' + (d.id < 0 ? 'plus' : d.status === 'open' ? 'close' : 'apply');
-           }); // update
+         function load() {
+           if (!wikibase) return;
 
-           notes.merge(notesEnter).sort(sortY).classed('selected', function (d) {
-             var mode = context.mode();
-             var isMoving = mode && mode.id === 'drag-note'; // no shadows when dragging
+           _button.classed('tag-reference-loading', true);
 
-             return !isMoving && d.id === selectedID;
-           }).attr('transform', getTransform); // Draw targets..
+           wikibase.getDocs(what, gotDocs);
+         }
 
-           if (touchLayer.empty()) return;
-           var fillClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
-           var targets = touchLayer.selectAll('.note').data(data, function (d) {
-             return d.id;
-           }); // exit
+         function gotDocs(err, docs) {
+           _body.html('');
 
-           targets.exit().remove(); // enter/update
+           if (!docs || !docs.title) {
+             _body.append('p').attr('class', 'tag-reference-description').html(_t.html('inspector.no_documentation_key'));
 
-           targets.enter().append('rect').attr('width', '20px').attr('height', '20px').attr('x', '-8px').attr('y', '-22px').merge(targets).sort(sortY).attr('class', function (d) {
-             var newClass = d.id < 0 ? 'new' : '';
-             return 'note target note-' + d.id + ' ' + fillClass + newClass;
-           }).attr('transform', getTransform);
+             done();
+             return;
+           }
 
-           function sortY(a, b) {
-             return a.id === selectedID ? 1 : b.id === selectedID ? -1 : b.loc[1] - a.loc[1];
+           if (docs.imageURL) {
+             _body.append('img').attr('class', 'tag-reference-wiki-image').attr('src', docs.imageURL).on('load', function () {
+               done();
+             }).on('error', function () {
+               select(this).remove();
+               done();
+             });
+           } else {
+             done();
            }
-         } // Draw the notes layer and schedule loading notes and updating markers.
 
+           _body.append('p').attr('class', 'tag-reference-description').html(docs.description ? _mainLocalizer.htmlForLocalizedText(docs.description, docs.descriptionLocaleCode) : _t.html('inspector.no_documentation_key')).append('a').attr('class', 'tag-reference-edit').attr('target', '_blank').attr('title', _t('inspector.edit_reference')).attr('href', docs.editURL).call(svgIcon('#iD-icon-edit', 'inline'));
 
-         function drawNotes(selection) {
-           var service = getService();
-           var surface = context.surface();
+           if (docs.wiki) {
+             _body.append('a').attr('class', 'tag-reference-link').attr('target', '_blank').attr('href', docs.wiki.url).call(svgIcon('#iD-icon-out-link', 'inline')).append('span').html(_t.html(docs.wiki.text));
+           } // Add link to info about "good changeset comments" - #2923
 
-           if (surface && !surface.empty()) {
-             touchLayer = surface.selectAll('.data-layer.touch .layer-touch.markers');
+
+           if (what.key === 'comment') {
+             _body.append('a').attr('class', 'tag-reference-comment-link').attr('target', '_blank').call(svgIcon('#iD-icon-out-link', 'inline')).attr('href', _t('commit.about_changeset_comments_link')).append('span').html(_t.html('commit.about_changeset_comments'));
            }
+         }
 
-           drawLayer = selection.selectAll('.layer-notes').data(service ? [0] : []);
-           drawLayer.exit().remove();
-           drawLayer = drawLayer.enter().append('g').attr('class', 'layer-notes').style('display', _notesEnabled ? 'block' : 'none').merge(drawLayer);
+         function done() {
+           _loaded = true;
 
-           if (_notesEnabled) {
-             if (service && ~~context.map().zoom() >= minZoom) {
-               editOn();
-               service.loadNotes(projection);
-               updateMarkers();
-             } else {
-               editOff();
-             }
-           }
-         } // Toggles the layer on and off
+           _button.classed('tag-reference-loading', false);
 
+           _body.classed('expanded', true).transition().duration(200).style('max-height', '200px').style('opacity', '1');
 
-         drawNotes.enabled = function (val) {
-           if (!arguments.length) return _notesEnabled;
-           _notesEnabled = val;
+           _showing = true;
 
-           if (_notesEnabled) {
-             layerOn();
-           } else {
-             layerOff();
+           _button.selectAll('svg.icon use').each(function () {
+             var iconUse = select(this);
 
-             if (context.selectedNoteID()) {
-               context.enter(modeBrowse(context));
+             if (iconUse.attr('href') === '#iD-icon-info') {
+               iconUse.attr('href', '#iD-icon-info-filled');
              }
-           }
+           });
+         }
 
-           dispatch$1.call('change');
-           return this;
-         };
+         function hide() {
+           _body.transition().duration(200).style('max-height', '0px').style('opacity', '0').on('end', function () {
+             _body.classed('expanded', false);
+           });
 
-         return drawNotes;
-       }
+           _showing = false;
 
-       function svgTouch() {
-         function drawTouch(selection) {
-           selection.selectAll('.layer-touch').data(['areas', 'lines', 'points', 'turns', 'markers']).enter().append('g').attr('class', function (d) {
-             return 'layer-touch ' + d;
+           _button.selectAll('svg.icon use').each(function () {
+             var iconUse = select(this);
+
+             if (iconUse.attr('href') === '#iD-icon-info-filled') {
+               iconUse.attr('href', '#iD-icon-info');
+             }
            });
          }
 
-         return drawTouch;
-       }
+         tagReference.button = function (selection, klass, iconName) {
+           _button = selection.selectAll('.tag-reference-button').data([0]);
+           _button = _button.enter().append('button').attr('class', 'tag-reference-button ' + (klass || '')).attr('title', _t('icons.information')).call(svgIcon('#iD-icon-' + (iconName || 'inspect'))).merge(_button);
 
-       function refresh(selection, node) {
-         var cr = node.getBoundingClientRect();
-         var prop = [cr.width, cr.height];
-         selection.property('__dimensions__', prop);
-         return prop;
-       }
+           _button.on('click', function (d3_event) {
+             d3_event.stopPropagation();
+             d3_event.preventDefault();
+             this.blur(); // avoid keeping focus on the button - #4641
 
-       function utilGetDimensions(selection, force) {
-         if (!selection || selection.empty()) {
-           return [0, 0];
-         }
+             if (_showing) {
+               hide();
+             } else if (_loaded) {
+               done();
+             } else {
+               load();
+             }
+           });
+         };
 
-         var node = selection.node(),
-             cached = selection.property('__dimensions__');
-         return !cached || force ? refresh(selection, node) : cached;
-       }
-       function utilSetDimensions(selection, dimensions) {
-         if (!selection || selection.empty()) {
-           return selection;
-         }
+         tagReference.body = function (selection) {
+           var itemID = what.qid || what.key + '-' + (what.value || '');
+           _body = selection.selectAll('.tag-reference-body').data([itemID], function (d) {
+             return d;
+           });
 
-         var node = selection.node();
+           _body.exit().remove();
 
-         if (dimensions === null) {
-           refresh(selection, node);
-           return selection;
-         }
+           _body = _body.enter().append('div').attr('class', 'tag-reference-body').style('max-height', '0').style('opacity', '0').merge(_body);
 
-         return selection.property('__dimensions__', [dimensions[0], dimensions[1]]).attr('width', dimensions[0]).attr('height', dimensions[1]);
+           if (_showing === false) {
+             hide();
+           }
+         };
+
+         tagReference.showing = function (val) {
+           if (!arguments.length) return _showing;
+           _showing = val;
+           return tagReference;
+         };
+
+         return tagReference;
        }
 
-       function svgLayers(projection, context) {
-         var dispatch$1 = dispatch('change');
-         var svg = select(null);
-         var _layers = [{
-           id: 'osm',
-           layer: svgOsm(projection, context, dispatch$1)
-         }, {
-           id: 'notes',
-           layer: svgNotes(projection, context, dispatch$1)
-         }, {
-           id: 'data',
-           layer: svgData(projection, context, dispatch$1)
-         }, {
-           id: 'keepRight',
-           layer: svgKeepRight(projection, context, dispatch$1)
-         }, {
-           id: 'improveOSM',
-           layer: svgImproveOSM(projection, context, dispatch$1)
-         }, {
-           id: 'osmose',
-           layer: svgOsmose(projection, context, dispatch$1)
-         }, {
-           id: 'streetside',
-           layer: svgStreetside(projection, context, dispatch$1)
-         }, {
-           id: 'mapillary',
-           layer: svgMapillaryImages(projection, context, dispatch$1)
-         }, {
-           id: 'mapillary-position',
-           layer: svgMapillaryPosition(projection, context)
-         }, {
-           id: 'mapillary-map-features',
-           layer: svgMapillaryMapFeatures(projection, context, dispatch$1)
-         }, {
-           id: 'mapillary-signs',
-           layer: svgMapillarySigns(projection, context, dispatch$1)
-         }, {
-           id: 'openstreetcam',
-           layer: svgOpenstreetcamImages(projection, context, dispatch$1)
-         }, {
-           id: 'debug',
-           layer: svgDebug(projection, context)
-         }, {
-           id: 'geolocate',
-           layer: svgGeolocate(projection)
+       function uiSectionRawTagEditor(id, context) {
+         var section = uiSection(id, context).classes('raw-tag-editor').label(function () {
+           var count = Object.keys(_tags).filter(function (d) {
+             return d;
+           }).length;
+           return _t('inspector.title_count', {
+             title: _t.html('inspector.tags'),
+             count: count
+           });
+         }).expandedByDefault(false).disclosureContent(renderDisclosureContent);
+         var taginfo = services.taginfo;
+         var dispatch = dispatch$8('change');
+         var availableViews = [{
+           id: 'list',
+           icon: '#fas-th-list'
          }, {
-           id: 'touch',
-           layer: svgTouch()
+           id: 'text',
+           icon: '#fas-i-cursor'
          }];
 
-         function drawLayers(selection) {
-           svg = selection.selectAll('.surface').data([0]);
-           svg = svg.enter().append('svg').attr('class', 'surface').merge(svg);
-           var defs = svg.selectAll('.surface-defs').data([0]);
-           defs.enter().append('defs').attr('class', 'surface-defs');
-           var groups = svg.selectAll('.data-layer').data(_layers);
-           groups.exit().remove();
-           groups.enter().append('g').attr('class', function (d) {
-             return 'data-layer ' + d.id;
-           }).merge(groups).each(function (d) {
-             select(this).call(d.layer);
-           });
-         }
+         var _tagView = corePreferences('raw-tag-editor-view') || 'list'; // 'list, 'text'
 
-         drawLayers.all = function () {
-           return _layers;
-         };
 
-         drawLayers.layer = function (id) {
-           var obj = _layers.find(function (o) {
-             return o.id === id;
-           });
+         var _readOnlyTags = []; // the keys in the order we want them to display
+
+         var _orderedKeys = [];
+         var _showBlank = false;
+         var _pendingChange = null;
+
+         var _state;
+
+         var _presets;
+
+         var _tags;
 
-           return obj && obj.layer;
-         };
+         var _entityIDs;
 
-         drawLayers.only = function (what) {
-           var arr = [].concat(what);
+         var _didInteract = false;
 
-           var all = _layers.map(function (layer) {
-             return layer.id;
-           });
+         function interacted() {
+           _didInteract = true;
+         }
 
-           return drawLayers.remove(utilArrayDifference(all, arr));
-         };
+         function renderDisclosureContent(wrap) {
+           // remove deleted keys
+           _orderedKeys = _orderedKeys.filter(function (key) {
+             return _tags[key] !== undefined;
+           }); // When switching to a different entity or changing the state (hover/select)
+           // reorder the keys alphabetically.
+           // We trigger this by emptying the `_orderedKeys` array, then it will be rebuilt here.
+           // Otherwise leave their order alone - #5857, #5927
 
-         drawLayers.remove = function (what) {
-           var arr = [].concat(what);
-           arr.forEach(function (id) {
-             _layers = _layers.filter(function (o) {
-               return o.id !== id;
-             });
-           });
-           dispatch$1.call('change');
-           return this;
-         };
+           var all = Object.keys(_tags).sort();
+           var missingKeys = utilArrayDifference(all, _orderedKeys);
 
-         drawLayers.add = function (what) {
-           var arr = [].concat(what);
-           arr.forEach(function (obj) {
-             if ('id' in obj && 'layer' in obj) {
-               _layers.push(obj);
-             }
-           });
-           dispatch$1.call('change');
-           return this;
-         };
+           for (var i in missingKeys) {
+             _orderedKeys.push(missingKeys[i]);
+           } // assemble row data
 
-         drawLayers.dimensions = function (val) {
-           if (!arguments.length) return utilGetDimensions(svg);
-           utilSetDimensions(svg, val);
-           return this;
-         };
 
-         return utilRebind(drawLayers, dispatch$1, 'on');
-       }
+           var rowData = _orderedKeys.map(function (key, i) {
+             return {
+               index: i,
+               key: key,
+               value: _tags[key]
+             };
+           }); // append blank row last, if necessary
 
-       function svgLines(projection, context) {
-         var detected = utilDetect();
-         var highway_stack = {
-           motorway: 0,
-           motorway_link: 1,
-           trunk: 2,
-           trunk_link: 3,
-           primary: 4,
-           primary_link: 5,
-           secondary: 6,
-           tertiary: 7,
-           unclassified: 8,
-           residential: 9,
-           service: 10,
-           footway: 11
-         };
 
-         function drawTargets(selection, graph, entities, filter) {
-           var targetClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
-           var nopeClass = context.getDebug('target') ? 'red ' : 'nocolor ';
-           var getPath = svgPath(projection).geojson;
-           var activeID = context.activeID();
-           var base = context.history().base(); // The targets and nopes will be MultiLineString sub-segments of the ways
+           if (!rowData.length || _showBlank) {
+             _showBlank = false;
+             rowData.push({
+               index: rowData.length,
+               key: '',
+               value: ''
+             });
+           } // View Options
 
-           var data = {
-             targets: [],
-             nopes: []
-           };
-           entities.forEach(function (way) {
-             var features = svgSegmentWay(way, graph, activeID);
-             data.targets.push.apply(data.targets, features.passive);
-             data.nopes.push.apply(data.nopes, features.active);
-           }); // Targets allow hover and vertex snapping
 
-           var targetData = data.targets.filter(getPath);
-           var targets = selection.selectAll('.line.target-allowed').filter(function (d) {
-             return filter(d.properties.entity);
-           }).data(targetData, function key(d) {
+           var options = wrap.selectAll('.raw-tag-options').data([0]);
+           options.exit().remove();
+           var optionsEnter = options.enter().insert('div', ':first-child').attr('class', 'raw-tag-options');
+           var optionEnter = optionsEnter.selectAll('.raw-tag-option').data(availableViews, function (d) {
              return d.id;
-           }); // exit
-
-           targets.exit().remove();
+           }).enter();
+           optionEnter.append('button').attr('class', function (d) {
+             return 'raw-tag-option raw-tag-option-' + d.id + (_tagView === d.id ? ' selected' : '');
+           }).attr('title', function (d) {
+             return _t('icons.' + d.id);
+           }).on('click', function (d3_event, d) {
+             _tagView = d.id;
+             corePreferences('raw-tag-editor-view', d.id);
+             wrap.selectAll('.raw-tag-option').classed('selected', function (datum) {
+               return datum === d;
+             });
+             wrap.selectAll('.tag-text').classed('hide', d.id !== 'text').each(setTextareaHeight);
+             wrap.selectAll('.tag-list, .add-row').classed('hide', d.id !== 'list');
+           }).each(function (d) {
+             select(this).call(svgIcon(d.icon));
+           }); // View as Text
 
-           var segmentWasEdited = function segmentWasEdited(d) {
-             var wayID = d.properties.entity.id; // if the whole line was edited, don't draw segment changes
+           var textData = rowsToText(rowData);
+           var textarea = wrap.selectAll('.tag-text').data([0]);
+           textarea = textarea.enter().append('textarea').attr('class', 'tag-text' + (_tagView !== 'text' ? ' hide' : '')).call(utilNoAuto).attr('placeholder', _t('inspector.key_value')).attr('spellcheck', 'false').merge(textarea);
+           textarea.call(utilGetSetValue, textData).each(setTextareaHeight).on('input', setTextareaHeight).on('focus', interacted).on('blur', textChanged).on('change', textChanged); // View as List
 
-             if (!base.entities[wayID] || !fastDeepEqual(graph.entities[wayID].nodes, base.entities[wayID].nodes)) {
-               return false;
-             }
+           var list = wrap.selectAll('.tag-list').data([0]);
+           list = list.enter().append('ul').attr('class', 'tag-list' + (_tagView !== 'list' ? ' hide' : '')).merge(list); // Container for the Add button
 
-             return d.properties.nodes.some(function (n) {
-               return !base.entities[n.id] || !fastDeepEqual(graph.entities[n.id].loc, base.entities[n.id].loc);
-             });
-           }; // enter/update
+           var addRowEnter = wrap.selectAll('.add-row').data([0]).enter().append('div').attr('class', 'add-row' + (_tagView !== 'list' ? ' hide' : ''));
+           addRowEnter.append('button').attr('class', 'add-tag').call(svgIcon('#iD-icon-plus', 'light')).on('click', addTag);
+           addRowEnter.append('div').attr('class', 'space-value'); // preserve space
 
+           addRowEnter.append('div').attr('class', 'space-buttons'); // preserve space
+           // Tag list items
 
-           targets.enter().append('path').merge(targets).attr('d', getPath).attr('class', function (d) {
-             return 'way line target target-allowed ' + targetClass + d.id;
-           }).classed('segment-edited', segmentWasEdited); // NOPE
+           var items = list.selectAll('.tag-row').data(rowData, function (d) {
+             return d.key;
+           });
+           items.exit().each(unbind).remove(); // Enter
 
-           var nopeData = data.nopes.filter(getPath);
-           var nopes = selection.selectAll('.line.target-nope').filter(function (d) {
-             return filter(d.properties.entity);
-           }).data(nopeData, function key(d) {
-             return d.id;
-           }); // exit
+           var itemsEnter = items.enter().append('li').attr('class', 'tag-row').classed('readonly', isReadOnly);
+           var innerWrap = itemsEnter.append('div').attr('class', 'inner-wrap');
+           innerWrap.append('div').attr('class', 'key-wrap').append('input').property('type', 'text').attr('class', 'key').call(utilNoAuto).on('focus', interacted).on('blur', keyChange).on('change', keyChange);
+           innerWrap.append('div').attr('class', 'value-wrap').append('input').property('type', 'text').attr('class', 'value').call(utilNoAuto).on('focus', interacted).on('blur', valueChange).on('change', valueChange).on('keydown.push-more', pushMore);
+           innerWrap.append('button').attr('class', 'form-field-button remove').attr('title', _t('icons.remove')).call(svgIcon('#iD-operation-delete')); // Update
 
-           nopes.exit().remove(); // enter/update
+           items = items.merge(itemsEnter).sort(function (a, b) {
+             return a.index - b.index;
+           });
+           items.each(function (d) {
+             var row = select(this);
+             var key = row.select('input.key'); // propagate bound data
 
-           nopes.enter().append('path').merge(nopes).attr('d', getPath).attr('class', function (d) {
-             return 'way line target target-nope ' + nopeClass + d.id;
-           }).classed('segment-edited', segmentWasEdited);
-         }
+             var value = row.select('input.value'); // propagate bound data
 
-         function drawLines(selection, graph, entities, filter) {
-           var base = context.history().base();
+             if (_entityIDs && taginfo && _state !== 'hover') {
+               bindTypeahead(key, value);
+             }
 
-           function waystack(a, b) {
-             var selected = context.selectedIDs();
-             var scoreA = selected.indexOf(a.id) !== -1 ? 20 : 0;
-             var scoreB = selected.indexOf(b.id) !== -1 ? 20 : 0;
+             var referenceOptions = {
+               key: d.key
+             };
 
-             if (a.tags.highway) {
-               scoreA -= highway_stack[a.tags.highway];
+             if (typeof d.value === 'string') {
+               referenceOptions.value = d.value;
              }
 
-             if (b.tags.highway) {
-               scoreB -= highway_stack[b.tags.highway];
+             var reference = uiTagReference(referenceOptions);
+
+             if (_state === 'hover') {
+               reference.showing(false);
              }
 
-             return scoreA - scoreB;
+             row.select('.inner-wrap') // propagate bound data
+             .call(reference.button);
+             row.call(reference.body);
+             row.select('button.remove'); // propagate bound data
+           });
+           items.selectAll('input.key').attr('title', function (d) {
+             return d.key;
+           }).call(utilGetSetValue, function (d) {
+             return d.key;
+           }).attr('readonly', function (d) {
+             return isReadOnly(d) || typeof d.value !== 'string' || null;
+           });
+           items.selectAll('input.value').attr('title', function (d) {
+             return Array.isArray(d.value) ? d.value.filter(Boolean).join('\n') : d.value;
+           }).classed('mixed', function (d) {
+             return Array.isArray(d.value);
+           }).attr('placeholder', function (d) {
+             return typeof d.value === 'string' ? null : _t('inspector.multiple_values');
+           }).call(utilGetSetValue, function (d) {
+             return typeof d.value === 'string' ? d.value : '';
+           }).attr('readonly', function (d) {
+             return isReadOnly(d) || null;
+           });
+           items.selectAll('button.remove').on(('PointerEvent' in window ? 'pointer' : 'mouse') + 'down', removeTag); // 'click' fires too late - #5878
+         }
+
+         function isReadOnly(d) {
+           for (var i = 0; i < _readOnlyTags.length; i++) {
+             if (d.key.match(_readOnlyTags[i]) !== null) {
+               return true;
+             }
            }
 
-           function drawLineGroup(selection, klass, isSelected) {
-             // Note: Don't add `.selected` class in draw modes
-             var mode = context.mode();
-             var isDrawing = mode && /^draw/.test(mode.id);
-             var selectedClass = !isDrawing && isSelected ? 'selected ' : '';
-             var lines = selection.selectAll('path').filter(filter).data(getPathData(isSelected), osmEntity.key);
-             lines.exit().remove(); // Optimization: Call expensive TagClasses only on enter selection. This
-             // works because osmEntity.key is defined to include the entity v attribute.
+           return false;
+         }
 
-             lines.enter().append('path').attr('class', function (d) {
-               var prefix = 'way line'; // if this line isn't styled by its own tags
+         function setTextareaHeight() {
+           if (_tagView !== 'text') return;
+           var selection = select(this);
+           var matches = selection.node().value.match(/\n/g);
+           var lineCount = 2 + Number(matches && matches.length);
+           var lineHeight = 20;
+           selection.style('height', lineCount * lineHeight + 'px');
+         }
 
-               if (!d.hasInterestingTags()) {
-                 var parentRelations = graph.parentRelations(d);
-                 var parentMultipolygons = parentRelations.filter(function (relation) {
-                   return relation.isMultipolygon();
-                 }); // and if it's a member of at least one multipolygon relation
+         function stringify(s) {
+           return JSON.stringify(s).slice(1, -1); // without leading/trailing "
+         }
 
-                 if (parentMultipolygons.length > 0 && // and only multipolygon relations
-                 parentRelations.length === parentMultipolygons.length) {
-                   // then fudge the classes to style this as an area edge
-                   prefix = 'relation area';
-                 }
-               }
+         function unstringify(s) {
+           var leading = '';
+           var trailing = '';
 
-               var oldMPClass = oldMultiPolygonOuters[d.id] ? 'old-multipolygon ' : '';
-               return prefix + ' ' + klass + ' ' + selectedClass + oldMPClass + d.id;
-             }).classed('added', function (d) {
-               return !base.entities[d.id];
-             }).classed('geometry-edited', function (d) {
-               return graph.entities[d.id] && base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].nodes, base.entities[d.id].nodes);
-             }).classed('retagged', function (d) {
-               return graph.entities[d.id] && base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].tags, base.entities[d.id].tags);
-             }).call(svgTagClasses()).merge(lines).sort(waystack).attr('d', getPath).call(svgTagClasses().tags(svgRelationMemberTags(graph)));
-             return selection;
+           if (s.length < 1 || s.charAt(0) !== '"') {
+             leading = '"';
            }
 
-           function getPathData(isSelected) {
-             return function () {
-               var layer = this.parentNode.__data__;
-               var data = pathdata[layer] || [];
-               return data.filter(function (d) {
-                 if (isSelected) return context.selectedIDs().indexOf(d.id) !== -1;else return context.selectedIDs().indexOf(d.id) === -1;
-               });
-             };
+           if (s.length < 2 || s.charAt(s.length - 1) !== '"' || s.charAt(s.length - 1) === '"' && s.charAt(s.length - 2) === '\\') {
+             trailing = '"';
            }
 
-           function addMarkers(layergroup, pathclass, groupclass, groupdata, marker) {
-             var markergroup = layergroup.selectAll('g.' + groupclass).data([pathclass]);
-             markergroup = markergroup.enter().append('g').attr('class', groupclass).merge(markergroup);
-             var markers = markergroup.selectAll('path').filter(filter).data(function data() {
-               return groupdata[this.parentNode.__data__] || [];
-             }, function key(d) {
-               return [d.id, d.index];
-             });
-             markers.exit().remove();
-             markers = markers.enter().append('path').attr('class', pathclass).merge(markers).attr('marker-mid', marker).attr('d', function (d) {
-               return d.d;
-             });
+           return JSON.parse(leading + s + trailing);
+         }
 
-             if (detected.ie) {
-               markers.each(function () {
-                 this.parentNode.insertBefore(this, this);
-               });
-             }
+         function rowsToText(rows) {
+           var str = rows.filter(function (row) {
+             return row.key && row.key.trim() !== '';
+           }).map(function (row) {
+             var rawVal = row.value;
+             if (typeof rawVal !== 'string') rawVal = '*';
+             var val = rawVal ? stringify(rawVal) : '';
+             return stringify(row.key) + '=' + val;
+           }).join('\n');
+
+           if (_state !== 'hover' && str.length) {
+             return str + '\n';
            }
 
-           var getPath = svgPath(projection, graph);
-           var ways = [];
-           var onewaydata = {};
-           var sideddata = {};
-           var oldMultiPolygonOuters = {};
+           return str;
+         }
 
-           for (var i = 0; i < entities.length; i++) {
-             var entity = entities[i];
-             var outer = osmOldMultipolygonOuterMember(entity, graph);
+         function textChanged() {
+           var newText = this.value.trim();
+           var newTags = {};
+           newText.split('\n').forEach(function (row) {
+             var m = row.match(/^\s*([^=]+)=(.*)$/);
 
-             if (outer) {
-               ways.push(entity.mergeTags(outer.tags));
-               oldMultiPolygonOuters[outer.id] = true;
-             } else if (entity.geometry(graph) === 'line') {
-               ways.push(entity);
+             if (m !== null) {
+               var k = context.cleanTagKey(unstringify(m[1].trim()));
+               var v = context.cleanTagValue(unstringify(m[2].trim()));
+               newTags[k] = v;
              }
-           }
-
-           ways = ways.filter(getPath);
-           var pathdata = utilArrayGroupBy(ways, function (way) {
-             return way.layer();
            });
-           Object.keys(pathdata).forEach(function (k) {
-             var v = pathdata[k];
-             var onewayArr = v.filter(function (d) {
-               return d.isOneWay();
-             });
-             var onewaySegments = svgMarkerSegments(projection, graph, 35, function shouldReverse(entity) {
-               return entity.tags.oneway === '-1';
-             }, function bothDirections(entity) {
-               return entity.tags.oneway === 'reversible' || entity.tags.oneway === 'alternating';
-             });
-             onewaydata[k] = utilArrayFlatten(onewayArr.map(onewaySegments));
-             var sidedArr = v.filter(function (d) {
-               return d.isSided();
-             });
-             var sidedSegments = svgMarkerSegments(projection, graph, 30, function shouldReverse() {
-               return false;
-             }, function bothDirections() {
-               return false;
-             });
-             sideddata[k] = utilArrayFlatten(sidedArr.map(sidedSegments));
-           });
-           var covered = selection.selectAll('.layer-osm.covered'); // under areas
+           var tagDiff = utilTagDiff(_tags, newTags);
+           if (!tagDiff.length) return;
+           _pendingChange = _pendingChange || {};
+           tagDiff.forEach(function (change) {
+             if (isReadOnly({
+               key: change.key
+             })) return; // skip unchanged multiselection placeholders
 
-           var uncovered = selection.selectAll('.layer-osm.lines'); // over areas
+             if (change.newVal === '*' && typeof change.oldVal !== 'string') return;
 
-           var touchLayer = selection.selectAll('.layer-touch.lines'); // Draw lines..
+             if (change.type === '-') {
+               _pendingChange[change.key] = undefined;
+             } else if (change.type === '+') {
+               _pendingChange[change.key] = change.newVal || '';
+             }
+           });
 
-           [covered, uncovered].forEach(function (selection) {
-             var range$1 = selection === covered ? range(-10, 0) : range(0, 11);
-             var layergroup = selection.selectAll('g.layergroup').data(range$1);
-             layergroup = layergroup.enter().append('g').attr('class', function (d) {
-               return 'layergroup layer' + String(d);
-             }).merge(layergroup);
-             layergroup.selectAll('g.linegroup').data(['shadow', 'casing', 'stroke', 'shadow-highlighted', 'casing-highlighted', 'stroke-highlighted']).enter().append('g').attr('class', function (d) {
-               return 'linegroup line-' + d;
-             });
-             layergroup.selectAll('g.line-shadow').call(drawLineGroup, 'shadow', false);
-             layergroup.selectAll('g.line-casing').call(drawLineGroup, 'casing', false);
-             layergroup.selectAll('g.line-stroke').call(drawLineGroup, 'stroke', false);
-             layergroup.selectAll('g.line-shadow-highlighted').call(drawLineGroup, 'shadow', true);
-             layergroup.selectAll('g.line-casing-highlighted').call(drawLineGroup, 'casing', true);
-             layergroup.selectAll('g.line-stroke-highlighted').call(drawLineGroup, 'stroke', true);
-             addMarkers(layergroup, 'oneway', 'onewaygroup', onewaydata, 'url(#ideditor-oneway-marker)');
-             addMarkers(layergroup, 'sided', 'sidedgroup', sideddata, function marker(d) {
-               var category = graph.entity(d.id).sidednessIdentifier();
-               return 'url(#ideditor-sided-marker-' + category + ')';
-             });
-           }); // Draw touch targets..
+           if (Object.keys(_pendingChange).length === 0) {
+             _pendingChange = null;
+             return;
+           }
 
-           touchLayer.call(drawTargets, graph, ways, filter);
+           scheduleChange();
          }
 
-         return drawLines;
-       }
-
-       function svgMidpoints(projection, context) {
-         var targetRadius = 8;
-
-         function drawTargets(selection, graph, entities, filter) {
-           var fillClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
-           var getTransform = svgPointTransform(projection).geojson;
-           var data = entities.map(function (midpoint) {
-             return {
-               type: 'Feature',
-               id: midpoint.id,
-               properties: {
-                 target: true,
-                 entity: midpoint
-               },
-               geometry: {
-                 type: 'Point',
-                 coordinates: midpoint.loc
-               }
-             };
-           });
-           var targets = selection.selectAll('.midpoint.target').filter(function (d) {
-             return filter(d.properties.entity);
-           }).data(data, function key(d) {
-             return d.id;
-           }); // exit
+         function pushMore(d3_event) {
+           // if pressing Tab on the last value field with content, add a blank row
+           if (d3_event.keyCode === 9 && !d3_event.shiftKey && section.selection().selectAll('.tag-list li:last-child input.value').node() === this && utilGetSetValue(select(this))) {
+             addTag();
+           }
+         }
 
-           targets.exit().remove(); // enter/update
+         function bindTypeahead(key, value) {
+           if (isReadOnly(key.datum())) return;
 
-           targets.enter().append('circle').attr('r', targetRadius).merge(targets).attr('class', function (d) {
-             return 'node midpoint target ' + fillClass + d.id;
-           }).attr('transform', getTransform);
-         }
+           if (Array.isArray(value.datum().value)) {
+             value.call(uiCombobox(context, 'tag-value').minItems(1).fetcher(function (value, callback) {
+               var keyString = utilGetSetValue(key);
+               if (!_tags[keyString]) return;
 
-         function drawMidpoints(selection, graph, entities, filter, extent) {
-           var drawLayer = selection.selectAll('.layer-osm.points .points-group.midpoints');
-           var touchLayer = selection.selectAll('.layer-touch.points');
-           var mode = context.mode();
+               var data = _tags[keyString].filter(Boolean).map(function (tagValue) {
+                 return {
+                   value: tagValue,
+                   title: tagValue
+                 };
+               });
 
-           if (mode && mode.id !== 'select' || !context.map().withinEditableZoom()) {
-             drawLayer.selectAll('.midpoint').remove();
-             touchLayer.selectAll('.midpoint.target').remove();
+               callback(data);
+             }));
              return;
            }
 
-           var poly = extent.polygon();
-           var midpoints = {};
+           var geometry = context.graph().geometry(_entityIDs[0]);
+           key.call(uiCombobox(context, 'tag-key').fetcher(function (value, callback) {
+             taginfo.keys({
+               debounce: true,
+               geometry: geometry,
+               query: value
+             }, function (err, data) {
+               if (!err) {
+                 var filtered = data.filter(function (d) {
+                   return _tags[d.value] === undefined;
+                 });
+                 callback(sort(value, filtered));
+               }
+             });
+           }));
+           value.call(uiCombobox(context, 'tag-value').fetcher(function (value, callback) {
+             taginfo.values({
+               debounce: true,
+               key: utilGetSetValue(key),
+               geometry: geometry,
+               query: value
+             }, function (err, data) {
+               if (!err) callback(sort(value, data));
+             });
+           }));
+
+           function sort(value, data) {
+             var sameletter = [];
+             var other = [];
+
+             for (var i = 0; i < data.length; i++) {
+               if (data[i].value.substring(0, value.length) === value) {
+                 sameletter.push(data[i]);
+               } else {
+                 other.push(data[i]);
+               }
+             }
+
+             return sameletter.concat(other);
+           }
+         }
 
-           for (var i = 0; i < entities.length; i++) {
-             var entity = entities[i];
-             if (entity.type !== 'way') continue;
-             if (!filter(entity)) continue;
-             if (context.selectedIDs().indexOf(entity.id) < 0) continue;
-             var nodes = graph.childNodes(entity);
+         function unbind() {
+           var row = select(this);
+           row.selectAll('input.key').call(uiCombobox.off, context);
+           row.selectAll('input.value').call(uiCombobox.off, context);
+         }
 
-             for (var j = 0; j < nodes.length - 1; j++) {
-               var a = nodes[j];
-               var b = nodes[j + 1];
-               var id = [a.id, b.id].sort().join('-');
+         function keyChange(d3_event, d) {
+           if (select(this).attr('readonly')) return;
+           var kOld = d.key; // exit if we are currently about to delete this row anyway - #6366
 
-               if (midpoints[id]) {
-                 midpoints[id].parents.push(entity);
-               } else if (geoVecLength(projection(a.loc), projection(b.loc)) > 40) {
-                 var point = geoVecInterp(a.loc, b.loc, 0.5);
-                 var loc = null;
+           if (_pendingChange && _pendingChange.hasOwnProperty(kOld) && _pendingChange[kOld] === undefined) return;
+           var kNew = context.cleanTagKey(this.value.trim()); // allow no change if the key should be readonly
 
-                 if (extent.intersects(point)) {
-                   loc = point;
-                 } else {
-                   for (var k = 0; k < 4; k++) {
-                     point = geoLineIntersection([a.loc, b.loc], [poly[k], poly[k + 1]]);
+           if (isReadOnly({
+             key: kNew
+           })) {
+             this.value = kOld;
+             return;
+           }
 
-                     if (point && geoVecLength(projection(a.loc), projection(point)) > 20 && geoVecLength(projection(b.loc), projection(point)) > 20) {
-                       loc = point;
-                       break;
-                     }
-                   }
-                 }
+           if (kNew && kNew !== kOld && _tags[kNew] !== undefined) {
+             // new key is already in use, switch focus to the existing row
+             this.value = kOld; // reset the key
 
-                 if (loc) {
-                   midpoints[id] = {
-                     type: 'midpoint',
-                     id: id,
-                     loc: loc,
-                     edge: [a.id, b.id],
-                     parents: [entity]
-                   };
-                 }
+             section.selection().selectAll('.tag-list input.value').each(function (d) {
+               if (d.key === kNew) {
+                 // send focus to that other value combo instead
+                 var input = select(this).node();
+                 input.focus();
+                 input.select();
                }
-             }
+             });
+             return;
            }
 
-           function midpointFilter(d) {
-             if (midpoints[d.id]) return true;
-
-             for (var i = 0; i < d.parents.length; i++) {
-               if (filter(d.parents[i])) {
-                 return true;
-               }
-             }
+           var row = this.parentNode.parentNode;
+           var inputVal = select(row).selectAll('input.value');
+           var vNew = context.cleanTagValue(utilGetSetValue(inputVal));
+           _pendingChange = _pendingChange || {};
 
-             return false;
+           if (kOld) {
+             _pendingChange[kOld] = undefined;
            }
 
-           var groups = drawLayer.selectAll('.midpoint').filter(midpointFilter).data(Object.values(midpoints), function (d) {
-             return d.id;
-           });
-           groups.exit().remove();
-           var enter = groups.enter().insert('g', ':first-child').attr('class', 'midpoint');
-           enter.append('polygon').attr('points', '-6,8 10,0 -6,-8').attr('class', 'shadow');
-           enter.append('polygon').attr('points', '-3,4 5,0 -3,-4').attr('class', 'fill');
-           groups = groups.merge(enter).attr('transform', function (d) {
-             var translate = svgPointTransform(projection);
-             var a = graph.entity(d.edge[0]);
-             var b = graph.entity(d.edge[1]);
-             var angle = geoAngle(a, b, projection) * (180 / Math.PI);
-             return translate(d) + ' rotate(' + angle + ')';
-           }).call(svgTagClasses().tags(function (d) {
-             return d.parents[0].tags;
-           })); // Propagate data bindings.
-
-           groups.select('polygon.shadow');
-           groups.select('polygon.fill'); // Draw touch targets..
+           _pendingChange[kNew] = vNew; // update the ordered key index so this row doesn't change position
 
-           touchLayer.call(drawTargets, graph, Object.values(midpoints), midpointFilter);
-         }
+           var existingKeyIndex = _orderedKeys.indexOf(kOld);
 
-         return drawMidpoints;
-       }
+           if (existingKeyIndex !== -1) _orderedKeys[existingKeyIndex] = kNew;
+           d.key = kNew; // update datum to avoid exit/enter on tag update
 
-       function svgPoints(projection, context) {
-         function markerPath(selection, klass) {
-           selection.attr('class', klass).attr('transform', 'translate(-8, -23)').attr('d', 'M 17,8 C 17,13 11,21 8.5,23.5 C 6,21 0,13 0,8 C 0,4 4,-0.5 8.5,-0.5 C 13,-0.5 17,4 17,8 z');
+           d.value = vNew;
+           this.value = kNew;
+           utilGetSetValue(inputVal, vNew);
+           scheduleChange();
          }
 
-         function sortY(a, b) {
-           return b.loc[1] - a.loc[1];
-         } // Avoid exit/enter if we're just moving stuff around.
-         // The node will get a new version but we only need to run the update selection.
+         function valueChange(d3_event, d) {
+           if (isReadOnly(d)) return; // exit if this is a multiselection and no value was entered
 
+           if (typeof d.value !== 'string' && !this.value) return; // exit if we are currently about to delete this row anyway - #6366
 
-         function fastEntityKey(d) {
-           var mode = context.mode();
-           var isMoving = mode && /^(add|draw|drag|move|rotate)/.test(mode.id);
-           return isMoving ? d.id : osmEntity.key(d);
+           if (_pendingChange && _pendingChange.hasOwnProperty(d.key) && _pendingChange[d.key] === undefined) return;
+           _pendingChange = _pendingChange || {};
+           _pendingChange[d.key] = context.cleanTagValue(this.value);
+           scheduleChange();
          }
 
-         function drawTargets(selection, graph, entities, filter) {
-           var fillClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
-           var getTransform = svgPointTransform(projection).geojson;
-           var activeID = context.activeID();
-           var data = [];
-           entities.forEach(function (node) {
-             if (activeID === node.id) return; // draw no target on the activeID
+         function removeTag(d3_event, d) {
+           if (isReadOnly(d)) return;
 
-             data.push({
-               type: 'Feature',
-               id: node.id,
-               properties: {
-                 target: true,
-                 entity: node
-               },
-               geometry: node.asGeoJSON()
+           if (d.key === '') {
+             // removing the blank row
+             _showBlank = false;
+             section.reRender();
+           } else {
+             // remove the key from the ordered key index
+             _orderedKeys = _orderedKeys.filter(function (key) {
+               return key !== d.key;
              });
-           });
-           var targets = selection.selectAll('.point.target').filter(function (d) {
-             return filter(d.properties.entity);
-           }).data(data, function key(d) {
-             return d.id;
-           }); // exit
+             _pendingChange = _pendingChange || {};
+             _pendingChange[d.key] = undefined;
+             scheduleChange();
+           }
+         }
 
-           targets.exit().remove(); // enter/update
+         function addTag() {
+           // Delay render in case this click is blurring an edited combo.
+           // Without the setTimeout, the `content` render would wipe out the pending tag change.
+           window.setTimeout(function () {
+             _showBlank = true;
+             section.reRender();
+             section.selection().selectAll('.tag-list li:last-child input.key').node().focus();
+           }, 20);
+         }
 
-           targets.enter().append('rect').attr('x', -10).attr('y', -26).attr('width', 20).attr('height', 30).merge(targets).attr('class', function (d) {
-             return 'node point target ' + fillClass + d.id;
-           }).attr('transform', getTransform);
+         function scheduleChange() {
+           // Cache IDs in case the editor is reloaded before the change event is called. - #6028
+           var entityIDs = _entityIDs; // Delay change in case this change is blurring an edited combo. - #5878
+
+           window.setTimeout(function () {
+             if (!_pendingChange) return;
+             dispatch.call('change', this, entityIDs, _pendingChange);
+             _pendingChange = null;
+           }, 10);
          }
 
-         function drawPoints(selection, graph, entities, filter) {
-           var wireframe = context.surface().classed('fill-wireframe');
-           var zoom = geoScaleToZoom(projection.scale());
-           var base = context.history().base(); // Points with a direction will render as vertices at higher zooms..
+         section.state = function (val) {
+           if (!arguments.length) return _state;
 
-           function renderAsPoint(entity) {
-             return entity.geometry(graph) === 'point' && !(zoom >= 18 && entity.directions(graph, projection).length);
-           } // All points will render as vertices in wireframe mode too..
+           if (_state !== val) {
+             _orderedKeys = [];
+             _state = val;
+           }
 
+           return section;
+         };
 
-           var points = wireframe ? [] : entities.filter(renderAsPoint);
-           points.sort(sortY);
-           var drawLayer = selection.selectAll('.layer-osm.points .points-group.points');
-           var touchLayer = selection.selectAll('.layer-touch.points'); // Draw points..
+         section.presets = function (val) {
+           if (!arguments.length) return _presets;
+           _presets = val;
 
-           var groups = drawLayer.selectAll('g.point').filter(filter).data(points, fastEntityKey);
-           groups.exit().remove();
-           var enter = groups.enter().append('g').attr('class', function (d) {
-             return 'node point ' + d.id;
-           }).order();
-           enter.append('path').call(markerPath, 'shadow');
-           enter.append('ellipse').attr('cx', 0.5).attr('cy', 1).attr('rx', 6.5).attr('ry', 3).attr('class', 'stroke');
-           enter.append('path').call(markerPath, 'stroke');
-           enter.append('use').attr('transform', 'translate(-5, -19)').attr('class', 'icon').attr('width', '11px').attr('height', '11px');
-           groups = groups.merge(enter).attr('transform', svgPointTransform(projection)).classed('added', function (d) {
-             return !base.entities[d.id]; // if it doesn't exist in the base graph, it's new
-           }).classed('moved', function (d) {
-             return base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].loc, base.entities[d.id].loc);
-           }).classed('retagged', function (d) {
-             return base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].tags, base.entities[d.id].tags);
-           }).call(svgTagClasses());
-           groups.select('.shadow'); // propagate bound data
+           if (_presets && _presets.length && _presets[0].isFallback()) {
+             section.disclosureExpanded(true); // don't collapse the disclosure if the mapper used the raw tag editor - #1881
+           } else if (!_didInteract) {
+             section.disclosureExpanded(null);
+           }
 
-           groups.select('.stroke'); // propagate bound data
+           return section;
+         };
 
-           groups.select('.icon') // propagate bound data
-           .attr('xlink:href', function (entity) {
-             var preset = _mainPresetIndex.match(entity, graph);
-             var picon = preset && preset.icon;
+         section.tags = function (val) {
+           if (!arguments.length) return _tags;
+           _tags = val;
+           return section;
+         };
 
-             if (!picon) {
-               return '';
-             } else {
-               var isMaki = /^maki-/.test(picon);
-               return '#' + picon + (isMaki ? '-11' : '');
-             }
-           }); // Draw touch targets..
+         section.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs;
 
-           touchLayer.call(drawTargets, graph, points, filter);
-         }
+           if (!_entityIDs || !val || !utilArrayIdentical(_entityIDs, val)) {
+             _entityIDs = val;
+             _orderedKeys = [];
+           }
 
-         return drawPoints;
+           return section;
+         }; // pass an array of regular expressions to test against the tag key
+
+
+         section.readOnlyTags = function (val) {
+           if (!arguments.length) return _readOnlyTags;
+           _readOnlyTags = val;
+           return section;
+         };
+
+         return utilRebind(section, dispatch, 'on');
        }
 
-       function svgTurns(projection, context) {
-         function icon(turn) {
-           var u = turn.u ? '-u' : '';
-           if (turn.no) return '#iD-turn-no' + u;
-           if (turn.only) return '#iD-turn-only' + u;
-           return '#iD-turn-yes' + u;
-         }
+       function uiDataEditor(context) {
+         var dataHeader = uiDataHeader();
+         var rawTagEditor = uiSectionRawTagEditor('custom-data-tag-editor', context).expandedByDefault(true).readOnlyTags([/./]);
 
-         function drawTurns(selection, graph, turns) {
-           function turnTransform(d) {
-             var pxRadius = 50;
-             var toWay = graph.entity(d.to.way);
-             var toPoints = graph.childNodes(toWay).map(function (n) {
-               return n.loc;
-             }).map(projection);
-             var toLength = geoPathLength(toPoints);
-             var mid = toLength / 2; // midpoint of destination way
+         var _datum;
 
-             var toNode = graph.entity(d.to.node);
-             var toVertex = graph.entity(d.to.vertex);
-             var a = geoAngle(toVertex, toNode, projection);
-             var o = projection(toVertex.loc);
-             var r = d.u ? 0 // u-turn: no radius
-             : !toWay.__via ? pxRadius // leaf way: put marker at pxRadius
-             : Math.min(mid, pxRadius); // via way: prefer pxRadius, fallback to mid for very short ways
+         function dataEditor(selection) {
+           var header = selection.selectAll('.header').data([0]);
+           var headerEnter = header.enter().append('div').attr('class', 'header fillL');
+           headerEnter.append('button').attr('class', 'close').on('click', function () {
+             context.enter(modeBrowse(context));
+           }).call(svgIcon('#iD-icon-close'));
+           headerEnter.append('h3').html(_t.html('map_data.title'));
+           var body = selection.selectAll('.body').data([0]);
+           body = body.enter().append('div').attr('class', 'body').merge(body);
+           var editor = body.selectAll('.data-editor').data([0]); // enter/update
 
-             return 'translate(' + (r * Math.cos(a) + o[0]) + ',' + (r * Math.sin(a) + o[1]) + ') ' + 'rotate(' + a * 180 / Math.PI + ')';
-           }
+           editor.enter().append('div').attr('class', 'modal-section data-editor').merge(editor).call(dataHeader.datum(_datum));
+           var rte = body.selectAll('.raw-tag-editor').data([0]); // enter/update
 
-           var drawLayer = selection.selectAll('.layer-osm.points .points-group.turns');
-           var touchLayer = selection.selectAll('.layer-touch.turns'); // Draw turns..
+           rte.enter().append('div').attr('class', 'raw-tag-editor data-editor').merge(rte).call(rawTagEditor.tags(_datum && _datum.properties || {}).state('hover').render).selectAll('textarea.tag-text').attr('readonly', true).classed('readonly', true);
+         }
 
-           var groups = drawLayer.selectAll('g.turn').data(turns, function (d) {
-             return d.key;
-           }); // exit
+         dataEditor.datum = function (val) {
+           if (!arguments.length) return _datum;
+           _datum = val;
+           return this;
+         };
 
-           groups.exit().remove(); // enter
+         return dataEditor;
+       }
 
-           var groupsEnter = groups.enter().append('g').attr('class', function (d) {
-             return 'turn ' + d.key;
-           });
-           var turnsEnter = groupsEnter.filter(function (d) {
-             return !d.u;
-           });
-           turnsEnter.append('rect').attr('transform', 'translate(-22, -12)').attr('width', '44').attr('height', '24');
-           turnsEnter.append('use').attr('transform', 'translate(-22, -12)').attr('width', '44').attr('height', '24');
-           var uEnter = groupsEnter.filter(function (d) {
-             return d.u;
-           });
-           uEnter.append('circle').attr('r', '16');
-           uEnter.append('use').attr('transform', 'translate(-16, -16)').attr('width', '32').attr('height', '32'); // update
+       var pair_1 = pair;
 
-           groups = groups.merge(groupsEnter).attr('opacity', function (d) {
-             return d.direct === false ? '0.7' : null;
-           }).attr('transform', turnTransform);
-           groups.select('use').attr('xlink:href', icon);
-           groups.select('rect'); // propagate bound data
+       function search(input, dims) {
+         if (!dims) dims = 'NSEW';
+         if (typeof input !== 'string') return null;
+         input = input.toUpperCase();
+         var regex = /^[\s\,]*([NSEW])?\s*([\-|\—|\―]?[0-9.]+)[°º˚]?\s*(?:([0-9.]+)['’′‘]\s*)?(?:([0-9.]+)(?:''|"|”|″)\s*)?([NSEW])?/;
+         var m = input.match(regex);
+         if (!m) return null; // no match
 
-           groups.select('circle'); // propagate bound data
-           // Draw touch targets..
+         var matched = m[0]; // extract dimension.. m[1] = leading, m[5] = trailing
 
-           var fillClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
-           groups = touchLayer.selectAll('g.turn').data(turns, function (d) {
-             return d.key;
-           }); // exit
+         var dim;
 
-           groups.exit().remove(); // enter
+         if (m[1] && m[5]) {
+           // if matched both..
+           dim = m[1]; // keep leading
 
-           groupsEnter = groups.enter().append('g').attr('class', function (d) {
-             return 'turn ' + d.key;
-           });
-           turnsEnter = groupsEnter.filter(function (d) {
-             return !d.u;
-           });
-           turnsEnter.append('rect').attr('class', 'target ' + fillClass).attr('transform', 'translate(-22, -12)').attr('width', '44').attr('height', '24');
-           uEnter = groupsEnter.filter(function (d) {
-             return d.u;
-           });
-           uEnter.append('circle').attr('class', 'target ' + fillClass).attr('r', '16'); // update
+           matched = matched.slice(0, -1); // remove trailing dimension from match
+         } else {
+           dim = m[1] || m[5];
+         } // if unrecognized dimension
 
-           groups = groups.merge(groupsEnter).attr('transform', turnTransform);
-           groups.select('rect'); // propagate bound data
 
-           groups.select('circle'); // propagate bound data
+         if (dim && dims.indexOf(dim) === -1) return null; // extract DMS
 
-           return this;
+         var deg = m[2] ? parseFloat(m[2]) : 0;
+         var min = m[3] ? parseFloat(m[3]) / 60 : 0;
+         var sec = m[4] ? parseFloat(m[4]) / 3600 : 0;
+         var sign = deg < 0 ? -1 : 1;
+         if (dim === 'S' || dim === 'W') sign *= -1;
+         return {
+           val: (Math.abs(deg) + min + sec) * sign,
+           dim: dim,
+           matched: matched,
+           remain: input.slice(matched.length)
+         };
+       }
+
+       function pair(input, dims) {
+         input = input.trim();
+         var one = search(input, dims);
+         if (!one) return null;
+         input = one.remain.trim();
+         var two = search(input, dims);
+         if (!two || two.remain) return null;
+
+         if (one.dim) {
+           return swapdim(one.val, two.val, one.dim);
+         } else {
+           return [one.val, two.val];
          }
+       }
 
-         return drawTurns;
+       function swapdim(a, b, dim) {
+         if (dim === 'N' || dim === 'S') return [a, b];
+         if (dim === 'W' || dim === 'E') return [b, a];
        }
 
-       function svgVertices(projection, context) {
-         var radiuses = {
-           //       z16-, z17,   z18+,  w/icon
-           shadow: [6, 7.5, 7.5, 12],
-           stroke: [2.5, 3.5, 3.5, 8],
-           fill: [1, 1.5, 1.5, 1.5]
-         };
+       function uiFeatureList(context) {
+         var _geocodeResults;
 
-         var _currHoverTarget;
+         function featureList(selection) {
+           var header = selection.append('div').attr('class', 'header fillL');
+           header.append('h3').html(_t.html('inspector.feature_list'));
+           var searchWrap = selection.append('div').attr('class', 'search-header');
+           searchWrap.call(svgIcon('#iD-icon-search', 'pre-text'));
+           var search = searchWrap.append('input').attr('placeholder', _t('inspector.search')).attr('type', 'search').call(utilNoAuto).on('keypress', keypress).on('keydown', keydown).on('input', inputevent);
+           var listWrap = selection.append('div').attr('class', 'inspector-body');
+           var list = listWrap.append('div').attr('class', 'feature-list');
+           context.on('exit.feature-list', clearSearch);
+           context.map().on('drawn.feature-list', mapDrawn);
+           context.keybinding().on(uiCmd('⌘F'), focusSearch);
 
-         var _currPersistent = {};
-         var _currHover = {};
-         var _prevHover = {};
-         var _currSelected = {};
-         var _prevSelected = {};
-         var _radii = {};
+           function focusSearch(d3_event) {
+             var mode = context.mode() && context.mode().id;
+             if (mode !== 'browse') return;
+             d3_event.preventDefault();
+             search.node().focus();
+           }
 
-         function sortY(a, b) {
-           return b.loc[1] - a.loc[1];
-         } // Avoid exit/enter if we're just moving stuff around.
-         // The node will get a new version but we only need to run the update selection.
+           function keydown(d3_event) {
+             if (d3_event.keyCode === 27) {
+               // escape
+               search.node().blur();
+             }
+           }
 
+           function keypress(d3_event) {
+             var q = search.property('value'),
+                 items = list.selectAll('.feature-list-item');
 
-         function fastEntityKey(d) {
-           var mode = context.mode();
-           var isMoving = mode && /^(add|draw|drag|move|rotate)/.test(mode.id);
-           return isMoving ? d.id : osmEntity.key(d);
-         }
+             if (d3_event.keyCode === 13 && // ↩ Return
+             q.length && items.size()) {
+               click(d3_event, items.datum());
+             }
+           }
 
-         function draw(selection, graph, vertices, sets, filter) {
-           sets = sets || {
-             selected: {},
-             important: {},
-             hovered: {}
-           };
-           var icons = {};
-           var directions = {};
-           var wireframe = context.surface().classed('fill-wireframe');
-           var zoom = geoScaleToZoom(projection.scale());
-           var z = zoom < 17 ? 0 : zoom < 18 ? 1 : 2;
-           var activeID = context.activeID();
-           var base = context.history().base();
+           function inputevent() {
+             _geocodeResults = undefined;
+             drawList();
+           }
 
-           function getIcon(d) {
-             // always check latest entity, as fastEntityKey avoids enter/exit now
-             var entity = graph.entity(d.id);
-             if (entity.id in icons) return icons[entity.id];
-             icons[entity.id] = entity.hasInterestingTags() && _mainPresetIndex.match(entity, graph).icon;
-             return icons[entity.id];
-           } // memoize directions results, return false for empty arrays (for use in filter)
+           function clearSearch() {
+             search.property('value', '');
+             drawList();
+           }
+
+           function mapDrawn(e) {
+             if (e.full) {
+               drawList();
+             }
+           }
+
+           function features() {
+             var result = [];
+             var graph = context.graph();
+             var visibleCenter = context.map().extent().center();
+             var q = search.property('value').toLowerCase();
+             if (!q) return result;
+             var locationMatch = pair_1(q.toUpperCase()) || q.match(/^(-?\d+\.?\d*)\s+(-?\d+\.?\d*)$/);
 
+             if (locationMatch) {
+               var loc = [parseFloat(locationMatch[0]), parseFloat(locationMatch[1])];
+               result.push({
+                 id: -1,
+                 geometry: 'point',
+                 type: _t('inspector.location'),
+                 name: dmsCoordinatePair([loc[1], loc[0]]),
+                 location: loc
+               });
+             } // A location search takes priority over an ID search
 
-           function getDirections(entity) {
-             if (entity.id in directions) return directions[entity.id];
-             var angles = entity.directions(graph, projection);
-             directions[entity.id] = angles.length ? angles : false;
-             return angles;
-           }
 
-           function updateAttributes(selection) {
-             ['shadow', 'stroke', 'fill'].forEach(function (klass) {
-               var rads = radiuses[klass];
-               selection.selectAll('.' + klass).each(function (entity) {
-                 var i = z && getIcon(entity);
-                 var r = rads[i ? 3 : z]; // slightly increase the size of unconnected endpoints #3775
+             var idMatch = !locationMatch && q.match(/(?:^|\W)(node|way|relation|[nwr])\W?0*([1-9]\d*)(?:\W|$)/i);
 
-                 if (entity.id !== activeID && entity.isEndpoint(graph) && !entity.isConnected(graph)) {
-                   r += 1.5;
-                 }
+             if (idMatch) {
+               var elemType = idMatch[1].charAt(0);
+               var elemId = idMatch[2];
+               result.push({
+                 id: elemType + elemId,
+                 geometry: elemType === 'n' ? 'point' : elemType === 'w' ? 'line' : 'relation',
+                 type: elemType === 'n' ? _t('inspector.node') : elemType === 'w' ? _t('inspector.way') : _t('inspector.relation'),
+                 name: elemId
+               });
+             }
 
-                 if (klass === 'shadow') {
-                   // remember this value, so we don't need to
-                   _radii[entity.id] = r; // recompute it when we draw the touch targets
-                 }
+             var allEntities = graph.entities;
+             var localResults = [];
 
-                 select(this).attr('r', r).attr('visibility', i && klass === 'fill' ? 'hidden' : null);
+             for (var id in allEntities) {
+               var entity = allEntities[id];
+               if (!entity) continue;
+               var name = utilDisplayName(entity) || '';
+               if (name.toLowerCase().indexOf(q) < 0) continue;
+               var matched = _mainPresetIndex.match(entity, graph);
+               var type = matched && matched.name() || utilDisplayType(entity.id);
+               var extent = entity.extent(graph);
+               var distance = extent ? geoSphericalDistance(visibleCenter, extent.center()) : 0;
+               localResults.push({
+                 id: entity.id,
+                 entity: entity,
+                 geometry: entity.geometry(graph),
+                 type: type,
+                 name: name,
+                 distance: distance
                });
+               if (localResults.length > 100) break;
+             }
+
+             localResults = localResults.sort(function byDistance(a, b) {
+               return a.distance - b.distance;
              });
-           }
+             result = result.concat(localResults);
 
-           vertices.sort(sortY);
-           var groups = selection.selectAll('g.vertex').filter(filter).data(vertices, fastEntityKey); // exit
+             (_geocodeResults || []).forEach(function (d) {
+               if (d.osm_type && d.osm_id) {
+                 // some results may be missing these - #1890
+                 // Make a temporary osmEntity so we can preset match
+                 // and better localize the search result - #4725
+                 var id = osmEntity.id.fromOSM(d.osm_type, d.osm_id);
+                 var tags = {};
+                 tags[d["class"]] = d.type;
+                 var attrs = {
+                   id: id,
+                   type: d.osm_type,
+                   tags: tags
+                 };
 
-           groups.exit().remove(); // enter
+                 if (d.osm_type === 'way') {
+                   // for ways, add some fake closed nodes
+                   attrs.nodes = ['a', 'a']; // so that geometry area is possible
+                 }
 
-           var enter = groups.enter().append('g').attr('class', function (d) {
-             return 'node vertex ' + d.id;
-           }).order();
-           enter.append('circle').attr('class', 'shadow');
-           enter.append('circle').attr('class', 'stroke'); // Vertices with tags get a fill.
+                 var tempEntity = osmEntity(attrs);
+                 var tempGraph = coreGraph([tempEntity]);
+                 var matched = _mainPresetIndex.match(tempEntity, tempGraph);
+                 var type = matched && matched.name() || utilDisplayType(id);
+                 result.push({
+                   id: tempEntity.id,
+                   geometry: tempEntity.geometry(tempGraph),
+                   type: type,
+                   name: d.display_name,
+                   extent: new geoExtent([parseFloat(d.boundingbox[3]), parseFloat(d.boundingbox[0])], [parseFloat(d.boundingbox[2]), parseFloat(d.boundingbox[1])])
+                 });
+               }
+             });
 
-           enter.filter(function (d) {
-             return d.hasInterestingTags();
-           }).append('circle').attr('class', 'fill'); // update
+             if (q.match(/^[0-9]+$/)) {
+               // if query is just a number, possibly an OSM ID without a prefix
+               result.push({
+                 id: 'n' + q,
+                 geometry: 'point',
+                 type: _t('inspector.node'),
+                 name: q
+               });
+               result.push({
+                 id: 'w' + q,
+                 geometry: 'line',
+                 type: _t('inspector.way'),
+                 name: q
+               });
+               result.push({
+                 id: 'r' + q,
+                 geometry: 'relation',
+                 type: _t('inspector.relation'),
+                 name: q
+               });
+             }
 
-           groups = groups.merge(enter).attr('transform', svgPointTransform(projection)).classed('sibling', function (d) {
-             return d.id in sets.selected;
-           }).classed('shared', function (d) {
-             return graph.isShared(d);
-           }).classed('endpoint', function (d) {
-             return d.isEndpoint(graph);
-           }).classed('added', function (d) {
-             return !base.entities[d.id]; // if it doesn't exist in the base graph, it's new
-           }).classed('moved', function (d) {
-             return base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].loc, base.entities[d.id].loc);
-           }).classed('retagged', function (d) {
-             return base.entities[d.id] && !fastDeepEqual(graph.entities[d.id].tags, base.entities[d.id].tags);
-           }).call(updateAttributes); // Vertices with icons get a `use`.
+             return result;
+           }
 
-           var iconUse = groups.selectAll('.icon').data(function data(d) {
-             return zoom >= 17 && getIcon(d) ? [d] : [];
-           }, fastEntityKey); // exit
+           function drawList() {
+             var value = search.property('value');
+             var results = features();
+             list.classed('filtered', value.length);
+             var resultsIndicator = list.selectAll('.no-results-item').data([0]).enter().append('button').property('disabled', true).attr('class', 'no-results-item').call(svgIcon('#iD-icon-alert', 'pre-text'));
+             resultsIndicator.append('span').attr('class', 'entity-name');
+             list.selectAll('.no-results-item .entity-name').html(_t.html('geocoder.no_results_worldwide'));
 
-           iconUse.exit().remove(); // enter
+             if (services.geocoder) {
+               list.selectAll('.geocode-item').data([0]).enter().append('button').attr('class', 'geocode-item secondary-action').on('click', geocoderSearch).append('div').attr('class', 'label').append('span').attr('class', 'entity-name').html(_t.html('geocoder.search'));
+             }
 
-           iconUse.enter().append('use').attr('class', 'icon').attr('width', '11px').attr('height', '11px').attr('transform', 'translate(-5.5, -5.5)').attr('xlink:href', function (d) {
-             var picon = getIcon(d);
-             var isMaki = /^maki-/.test(picon);
-             return '#' + picon + (isMaki ? '-11' : '');
-           }); // Vertices with directions get viewfields
+             list.selectAll('.no-results-item').style('display', value.length && !results.length ? 'block' : 'none');
+             list.selectAll('.geocode-item').style('display', value && _geocodeResults === undefined ? 'block' : 'none');
+             list.selectAll('.feature-list-item').data([-1]).remove();
+             var items = list.selectAll('.feature-list-item').data(results, function (d) {
+               return d.id;
+             });
+             var enter = items.enter().insert('button', '.geocode-item').attr('class', 'feature-list-item').on('mouseover', mouseover).on('mouseout', mouseout).on('click', click);
+             var label = enter.append('div').attr('class', 'label');
+             label.each(function (d) {
+               select(this).call(svgIcon('#iD-icon-' + d.geometry, 'pre-text'));
+             });
+             label.append('span').attr('class', 'entity-type').html(function (d) {
+               return d.type;
+             });
+             label.append('span').attr('class', 'entity-name').html(function (d) {
+               return d.name;
+             });
+             enter.style('opacity', 0).transition().style('opacity', 1);
+             items.order();
+             items.exit().remove();
+           }
 
-           var dgroups = groups.selectAll('.viewfieldgroup').data(function data(d) {
-             return zoom >= 18 && getDirections(d) ? [d] : [];
-           }, fastEntityKey); // exit
+           function mouseover(d3_event, d) {
+             if (d.id === -1) return;
+             utilHighlightEntities([d.id], true, context);
+           }
 
-           dgroups.exit().remove(); // enter/update
+           function mouseout(d3_event, d) {
+             if (d.id === -1) return;
+             utilHighlightEntities([d.id], false, context);
+           }
 
-           dgroups = dgroups.enter().insert('g', '.shadow').attr('class', 'viewfieldgroup').merge(dgroups);
-           var viewfields = dgroups.selectAll('.viewfield').data(getDirections, function key(d) {
-             return osmEntity.key(d);
-           }); // exit
+           function click(d3_event, d) {
+             d3_event.preventDefault();
 
-           viewfields.exit().remove(); // enter/update
+             if (d.location) {
+               context.map().centerZoomEase([d.location[1], d.location[0]], 19);
+             } else if (d.entity) {
+               utilHighlightEntities([d.id], false, context);
+               context.enter(modeSelect(context, [d.entity.id]));
+               context.map().zoomToEase(d.entity);
+             } else {
+               // download, zoom to, and select the entity with the given ID
+               context.zoomToEntity(d.id);
+             }
+           }
 
-           viewfields.enter().append('path').attr('class', 'viewfield').attr('d', 'M0,0H0').merge(viewfields).attr('marker-start', 'url(#ideditor-viewfield-marker' + (wireframe ? '-wireframe' : '') + ')').attr('transform', function (d) {
-             return 'rotate(' + d + ')';
-           });
+           function geocoderSearch() {
+             services.geocoder.search(search.property('value'), function (err, resp) {
+               _geocodeResults = resp || [];
+               drawList();
+             });
+           }
          }
 
-         function drawTargets(selection, graph, entities, filter) {
-           var targetClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
-           var nopeClass = context.getDebug('target') ? 'red ' : 'nocolor ';
-           var getTransform = svgPointTransform(projection).geojson;
-           var activeID = context.activeID();
-           var data = {
-             targets: [],
-             nopes: []
-           };
-           entities.forEach(function (node) {
-             if (activeID === node.id) return; // draw no target on the activeID
-
-             var vertexType = svgPassiveVertex(node, graph, activeID);
-
-             if (vertexType !== 0) {
-               // passive or adjacent - allow to connect
-               data.targets.push({
-                 type: 'Feature',
-                 id: node.id,
-                 properties: {
-                   target: true,
-                   entity: node
-                 },
-                 geometry: node.asGeoJSON()
-               });
-             } else {
-               data.nopes.push({
-                 type: 'Feature',
-                 id: node.id + '-nope',
-                 properties: {
-                   nope: true,
-                   target: true,
-                   entity: node
-                 },
-                 geometry: node.asGeoJSON()
-               });
-             }
-           }); // Targets allow hover and vertex snapping
+         return featureList;
+       }
 
-           var targets = selection.selectAll('.vertex.target-allowed').filter(function (d) {
-             return filter(d.properties.entity);
-           }).data(data.targets, function key(d) {
-             return d.id;
-           }); // exit
+       var getOwnPropertyDescriptor$1 = objectGetOwnPropertyDescriptor.f;
 
-           targets.exit().remove(); // enter/update
 
-           targets.enter().append('circle').attr('r', function (d) {
-             return _radii[d.id] || radiuses.shadow[3];
-           }).merge(targets).attr('class', function (d) {
-             return 'node vertex target target-allowed ' + targetClass + d.id;
-           }).attr('transform', getTransform); // NOPE
 
-           var nopes = selection.selectAll('.vertex.target-nope').filter(function (d) {
-             return filter(d.properties.entity);
-           }).data(data.nopes, function key(d) {
-             return d.id;
-           }); // exit
 
-           nopes.exit().remove(); // enter/update
 
-           nopes.enter().append('circle').attr('r', function (d) {
-             return _radii[d.properties.entity.id] || radiuses.shadow[3];
-           }).merge(nopes).attr('class', function (d) {
-             return 'node vertex target target-nope ' + nopeClass + d.id;
-           }).attr('transform', getTransform);
-         } // Points can also render as vertices:
-         // 1. in wireframe mode or
-         // 2. at higher zooms if they have a direction
 
+       // eslint-disable-next-line es/no-string-prototype-startswith -- safe
+       var $startsWith = ''.startsWith;
+       var min$1 = Math.min;
 
-         function renderAsVertex(entity, graph, wireframe, zoom) {
-           var geometry = entity.geometry(graph);
-           return geometry === 'vertex' || geometry === 'point' && (wireframe || zoom >= 18 && entity.directions(graph, projection).length);
-         }
+       var CORRECT_IS_REGEXP_LOGIC$1 = correctIsRegexpLogic('startsWith');
+       // https://github.com/zloirock/core-js/pull/702
+       var MDN_POLYFILL_BUG$1 = !CORRECT_IS_REGEXP_LOGIC$1 && !!function () {
+         var descriptor = getOwnPropertyDescriptor$1(String.prototype, 'startsWith');
+         return descriptor && !descriptor.writable;
+       }();
 
-         function isEditedNode(node, base, head) {
-           var baseNode = base.entities[node.id];
-           var headNode = head.entities[node.id];
-           return !headNode || !baseNode || !fastDeepEqual(headNode.tags, baseNode.tags) || !fastDeepEqual(headNode.loc, baseNode.loc);
+       // `String.prototype.startsWith` method
+       // https://tc39.es/ecma262/#sec-string.prototype.startswith
+       _export({ target: 'String', proto: true, forced: !MDN_POLYFILL_BUG$1 && !CORRECT_IS_REGEXP_LOGIC$1 }, {
+         startsWith: function startsWith(searchString /* , position = 0 */) {
+           var that = String(requireObjectCoercible(this));
+           notARegexp(searchString);
+           var index = toLength(min$1(arguments.length > 1 ? arguments[1] : undefined, that.length));
+           var search = String(searchString);
+           return $startsWith
+             ? $startsWith.call(that, search, index)
+             : that.slice(index, index + search.length) === search;
          }
+       });
 
-         function getSiblingAndChildVertices(ids, graph, wireframe, zoom) {
-           var results = {};
-           var seenIds = {};
-
-           function addChildVertices(entity) {
-             // avoid redundant work and infinite recursion of circular relations
-             if (seenIds[entity.id]) return;
-             seenIds[entity.id] = true;
-             var geometry = entity.geometry(graph);
+       function uiSectionEntityIssues(context) {
+         // Does the user prefer to expand the active issue?  Useful for viewing tag diff.
+         // Expand by default so first timers see it - #6408, #8143
+         var preference = corePreferences('entity-issues.reference.expanded');
 
-             if (!context.features().isHiddenFeature(entity, graph, geometry)) {
-               var i;
+         var _expanded = preference === null ? true : preference === 'true';
 
-               if (entity.type === 'way') {
-                 for (i = 0; i < entity.nodes.length; i++) {
-                   var child = graph.hasEntity(entity.nodes[i]);
+         var _entityIDs = [];
+         var _issues = [];
 
-                   if (child) {
-                     addChildVertices(child);
-                   }
-                 }
-               } else if (entity.type === 'relation') {
-                 for (i = 0; i < entity.members.length; i++) {
-                   var member = graph.hasEntity(entity.members[i].id);
+         var _activeIssueID;
 
-                   if (member) {
-                     addChildVertices(member);
-                   }
-                 }
-               } else if (renderAsVertex(entity, graph, wireframe, zoom)) {
-                 results[entity.id] = entity;
-               }
-             }
-           }
+         var section = uiSection('entity-issues', context).shouldDisplay(function () {
+           return _issues.length > 0;
+         }).label(function () {
+           return _t('inspector.title_count', {
+             title: _t.html('issues.list_title'),
+             count: _issues.length
+           });
+         }).disclosureContent(renderDisclosureContent);
+         context.validator().on('validated.entity_issues', function () {
+           // Refresh on validated events
+           reloadIssues();
+           section.reRender();
+         }).on('focusedIssue.entity_issues', function (issue) {
+           makeActiveIssue(issue.id);
+         });
 
-           ids.forEach(function (id) {
-             var entity = graph.hasEntity(id);
-             if (!entity) return;
+         function reloadIssues() {
+           _issues = context.validator().getSharedEntityIssues(_entityIDs, {
+             includeDisabledRules: true
+           });
+         }
 
-             if (entity.type === 'node') {
-               if (renderAsVertex(entity, graph, wireframe, zoom)) {
-                 results[entity.id] = entity;
-                 graph.parentWays(entity).forEach(function (entity) {
-                   addChildVertices(entity);
-                 });
-               }
-             } else {
-               // way, relation
-               addChildVertices(entity);
-             }
+         function makeActiveIssue(issueID) {
+           _activeIssueID = issueID;
+           section.selection().selectAll('.issue-container').classed('active', function (d) {
+             return d.id === _activeIssueID;
            });
-           return results;
          }
 
-         function drawVertices(selection, graph, entities, filter, extent, fullRedraw) {
-           var wireframe = context.surface().classed('fill-wireframe');
-           var visualDiff = context.surface().classed('highlight-edited');
-           var zoom = geoScaleToZoom(projection.scale());
-           var mode = context.mode();
-           var isMoving = mode && /^(add|draw|drag|move|rotate)/.test(mode.id);
-           var base = context.history().base();
-           var drawLayer = selection.selectAll('.layer-osm.points .points-group.vertices');
-           var touchLayer = selection.selectAll('.layer-touch.points');
+         function renderDisclosureContent(selection) {
+           selection.classed('grouped-items-area', true);
+           _activeIssueID = _issues.length > 0 ? _issues[0].id : null;
+           var containers = selection.selectAll('.issue-container').data(_issues, function (d) {
+             return d.id;
+           }); // Exit
 
-           if (fullRedraw) {
-             _currPersistent = {};
-             _radii = {};
-           } // Collect important vertices from the `entities` list..
-           // (during a partial redraw, it will not contain everything)
+           containers.exit().remove(); // Enter
 
+           var containersEnter = containers.enter().append('div').attr('class', 'issue-container');
+           var itemsEnter = containersEnter.append('div').attr('class', function (d) {
+             return 'issue severity-' + d.severity;
+           }).on('mouseover.highlight', function (d3_event, d) {
+             // don't hover-highlight the selected entity
+             var ids = d.entityIds.filter(function (e) {
+               return _entityIDs.indexOf(e) === -1;
+             });
+             utilHighlightEntities(ids, true, context);
+           }).on('mouseout.highlight', function (d3_event, d) {
+             var ids = d.entityIds.filter(function (e) {
+               return _entityIDs.indexOf(e) === -1;
+             });
+             utilHighlightEntities(ids, false, context);
+           });
+           var labelsEnter = itemsEnter.append('div').attr('class', 'issue-label');
+           var textEnter = labelsEnter.append('button').attr('class', 'issue-text').on('click', function (d3_event, d) {
+             makeActiveIssue(d.id); // expand only the clicked item
 
-           for (var i = 0; i < entities.length; i++) {
-             var entity = entities[i];
-             var geometry = entity.geometry(graph);
-             var keep = false; // a point that looks like a vertex..
+             var extent = d.extent(context.graph());
 
-             if (geometry === 'point' && renderAsVertex(entity, graph, wireframe, zoom)) {
-               _currPersistent[entity.id] = entity;
-               keep = true; // a vertex of some importance..
-             } else if (geometry === 'vertex' && (entity.hasInterestingTags() || entity.isEndpoint(graph) || entity.isConnected(graph) || visualDiff && isEditedNode(entity, base, graph))) {
-               _currPersistent[entity.id] = entity;
-               keep = true;
-             } // whatever this is, it's not a persistent vertex..
+             if (extent) {
+               var setZoom = Math.max(context.map().zoom(), 19);
+               context.map().unobscuredCenterZoomEase(extent.center(), setZoom);
+             }
+           });
+           textEnter.each(function (d) {
+             var iconName = '#iD-icon-' + (d.severity === 'warning' ? 'alert' : 'error');
+             select(this).call(svgIcon(iconName, 'issue-icon'));
+           });
+           textEnter.append('span').attr('class', 'issue-message');
+           var infoButton = labelsEnter.append('button').attr('class', 'issue-info-button').attr('title', _t('icons.information')).call(svgIcon('#iD-icon-inspect'));
+           infoButton.on('click', function (d3_event) {
+             d3_event.stopPropagation();
+             d3_event.preventDefault();
+             this.blur(); // avoid keeping focus on the button - #4641
 
+             var container = select(this.parentNode.parentNode.parentNode);
+             var info = container.selectAll('.issue-info');
+             var isExpanded = info.classed('expanded');
+             _expanded = !isExpanded;
+             corePreferences('entity-issues.reference.expanded', _expanded); // update preference
 
-             if (!keep && !fullRedraw) {
-               delete _currPersistent[entity.id];
+             if (isExpanded) {
+               info.transition().duration(200).style('max-height', '0px').style('opacity', '0').on('end', function () {
+                 info.classed('expanded', false);
+               });
+             } else {
+               info.classed('expanded', true).transition().duration(200).style('max-height', '200px').style('opacity', '1').on('end', function () {
+                 info.style('max-height', null);
+               });
              }
-           } // 3 sets of vertices to consider:
+           });
+           itemsEnter.append('ul').attr('class', 'issue-fix-list');
+           containersEnter.append('div').attr('class', 'issue-info' + (_expanded ? ' expanded' : '')).style('max-height', _expanded ? null : '0').style('opacity', _expanded ? '1' : '0').each(function (d) {
+             if (typeof d.reference === 'function') {
+               select(this).call(d.reference);
+             } else {
+               select(this).html(_t.html('inspector.no_documentation_key'));
+             }
+           }); // Update
 
+           containers = containers.merge(containersEnter).classed('active', function (d) {
+             return d.id === _activeIssueID;
+           });
+           containers.selectAll('.issue-message').html(function (d) {
+             return d.message(context);
+           }); // fixes
 
-           var sets = {
-             persistent: _currPersistent,
-             // persistent = important vertices (render always)
-             selected: _currSelected,
-             // selected + siblings of selected (render always)
-             hovered: _currHover // hovered + siblings of hovered (render only in draw modes)
+           var fixLists = containers.selectAll('.issue-fix-list');
+           var fixes = fixLists.selectAll('.issue-fix-item').data(function (d) {
+             return d.fixes ? d.fixes(context) : [];
+           }, function (fix) {
+             return fix.id;
+           });
+           fixes.exit().remove();
+           var fixesEnter = fixes.enter().append('li').attr('class', 'issue-fix-item');
+           var buttons = fixesEnter.append('button').on('click', function (d3_event, d) {
+             // not all fixes are actionable
+             if (select(this).attr('disabled') || !d.onClick) return; // Don't run another fix for this issue within a second of running one
+             // (Necessary for "Select a feature type" fix. Most fixes should only ever run once)
 
-           };
-           var all = Object.assign({}, isMoving ? _currHover : {}, _currSelected, _currPersistent); // Draw the vertices..
-           // The filter function controls the scope of what objects d3 will touch (exit/enter/update)
-           // Adjust the filter function to expand the scope beyond whatever entities were passed in.
+             if (d.issue.dateLastRanFix && new Date() - d.issue.dateLastRanFix < 1000) return;
+             d.issue.dateLastRanFix = new Date(); // remove hover-highlighting
+
+             utilHighlightEntities(d.issue.entityIds.concat(d.entityIds), false, context);
+             new Promise(function (resolve, reject) {
+               d.onClick(context, resolve, reject);
+
+               if (d.onClick.length <= 1) {
+                 // if the fix doesn't take any completion parameters then consider it resolved
+                 resolve();
+               }
+             }).then(function () {
+               // revalidate whenever the fix has finished running successfully
+               context.validator().validate();
+             });
+           }).on('mouseover.highlight', function (d3_event, d) {
+             utilHighlightEntities(d.entityIds, true, context);
+           }).on('mouseout.highlight', function (d3_event, d) {
+             utilHighlightEntities(d.entityIds, false, context);
+           });
+           buttons.each(function (d) {
+             var iconName = d.icon || 'iD-icon-wrench';
 
-           var filterRendered = function filterRendered(d) {
-             return d.id in _currPersistent || d.id in _currSelected || d.id in _currHover || filter(d);
-           };
+             if (iconName.startsWith('maki')) {
+               iconName += '-15';
+             }
 
-           drawLayer.call(draw, graph, currentVisible(all), sets, filterRendered); // Draw touch targets..
-           // When drawing, render all targets (not just those affected by a partial redraw)
+             select(this).call(svgIcon('#' + iconName, 'fix-icon'));
+           });
+           buttons.append('span').attr('class', 'fix-message').html(function (d) {
+             return d.title;
+           });
+           fixesEnter.merge(fixes).selectAll('button').classed('actionable', function (d) {
+             return d.onClick;
+           }).attr('disabled', function (d) {
+             return d.onClick ? null : 'true';
+           }).attr('title', function (d) {
+             if (d.disabledReason) {
+               return d.disabledReason;
+             }
 
-           var filterTouch = function filterTouch(d) {
-             return isMoving ? true : filterRendered(d);
-           };
+             return null;
+           });
+         }
 
-           touchLayer.call(drawTargets, graph, currentVisible(all), filterTouch);
+         section.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs;
 
-           function currentVisible(which) {
-             return Object.keys(which).map(graph.hasEntity, graph) // the current version of this entity
-             .filter(function (entity) {
-               return entity && entity.intersects(extent, graph);
-             });
+           if (!_entityIDs || !val || !utilArrayIdentical(_entityIDs, val)) {
+             _entityIDs = val;
+             _activeIssueID = null;
+             reloadIssues();
            }
-         } // partial redraw - only update the selected items..
 
+           return section;
+         };
 
-         drawVertices.drawSelected = function (selection, graph, extent) {
-           var wireframe = context.surface().classed('fill-wireframe');
-           var zoom = geoScaleToZoom(projection.scale());
-           _prevSelected = _currSelected || {};
+         return section;
+       }
 
-           if (context.map().isInWideSelection()) {
-             _currSelected = {};
-             context.selectedIDs().forEach(function (id) {
-               var entity = graph.hasEntity(id);
-               if (!entity) return;
+       function uiPresetIcon() {
+         var _preset;
 
-               if (entity.type === 'node') {
-                 if (renderAsVertex(entity, graph, wireframe, zoom)) {
-                   _currSelected[entity.id] = entity;
-                 }
-               }
-             });
-           } else {
-             _currSelected = getSiblingAndChildVertices(context.selectedIDs(), graph, wireframe, zoom);
-           } // note that drawVertices will add `_currSelected` automatically if needed..
+         var _geometry;
 
+         var _sizeClass = 'medium';
 
-           var filter = function filter(d) {
-             return d.id in _prevSelected;
-           };
+         function isSmall() {
+           return _sizeClass === 'small';
+         }
 
-           drawVertices(selection, graph, Object.values(_prevSelected), filter, extent, false);
-         }; // partial redraw - only update the hovered items..
+         function presetIcon(selection) {
+           selection.each(render);
+         }
 
+         function getIcon(p, geom) {
+           if (isSmall() && p.isFallback && p.isFallback()) return 'iD-icon-' + p.id;
+           if (p.icon) return p.icon;
+           if (geom === 'line') return 'iD-other-line';
+           if (geom === 'vertex') return p.isFallback() ? '' : 'temaki-vertex';
+           if (isSmall() && geom === 'point') return '';
+           return 'maki-marker-stroked';
+         }
 
-         drawVertices.drawHover = function (selection, graph, target, extent) {
-           if (target === _currHoverTarget) return; // continue only if something changed
+         function renderPointBorder(container, drawPoint) {
+           var pointBorder = container.selectAll('.preset-icon-point-border').data(drawPoint ? [0] : []);
+           pointBorder.exit().remove();
+           var pointBorderEnter = pointBorder.enter();
+           var w = 40;
+           var h = 40;
+           pointBorderEnter.append('svg').attr('class', 'preset-icon-fill preset-icon-point-border').attr('width', w).attr('height', h).attr('viewBox', "0 0 ".concat(w, " ").concat(h)).append('path').attr('transform', 'translate(11.5, 8)').attr('d', 'M 17,8 C 17,13 11,21 8.5,23.5 C 6,21 0,13 0,8 C 0,4 4,-0.5 8.5,-0.5 C 13,-0.5 17,4 17,8 z');
+           pointBorder = pointBorderEnter.merge(pointBorder);
+         }
 
-           var wireframe = context.surface().classed('fill-wireframe');
-           var zoom = geoScaleToZoom(projection.scale());
-           _prevHover = _currHover || {};
-           _currHoverTarget = target;
-           var entity = target && target.properties && target.properties.entity;
+         function renderCategoryBorder(container, category) {
+           var categoryBorder = container.selectAll('.preset-icon-category-border').data(category ? [0] : []);
+           categoryBorder.exit().remove();
+           var categoryBorderEnter = categoryBorder.enter();
+           var d = 60;
+           var svgEnter = categoryBorderEnter.append('svg').attr('class', 'preset-icon-fill preset-icon-category-border').attr('width', d).attr('height', d).attr('viewBox', "0 0 ".concat(d, " ").concat(d));
+           ['fill', 'stroke'].forEach(function (klass) {
+             svgEnter.append('path').attr('class', "area ".concat(klass)).attr('d', 'M9.5,7.5 L25.5,7.5 L28.5,12.5 L49.5,12.5 C51.709139,12.5 53.5,14.290861 53.5,16.5 L53.5,43.5 C53.5,45.709139 51.709139,47.5 49.5,47.5 L10.5,47.5 C8.290861,47.5 6.5,45.709139 6.5,43.5 L6.5,12.5 L9.5,7.5 Z');
+           });
+           categoryBorder = categoryBorderEnter.merge(categoryBorder);
 
-           if (entity) {
-             _currHover = getSiblingAndChildVertices([entity.id], graph, wireframe, zoom);
-           } else {
-             _currHover = {};
-           } // note that drawVertices will add `_currHover` automatically if needed..
+           if (category) {
+             var tagClasses = svgTagClasses().getClassesString(category.members.collection[0].addTags, '');
+             categoryBorder.selectAll('path.stroke').attr('class', "area stroke ".concat(tagClasses));
+             categoryBorder.selectAll('path.fill').attr('class', "area fill ".concat(tagClasses));
+           }
+         }
 
+         function renderCircleFill(container, drawVertex) {
+           var vertexFill = container.selectAll('.preset-icon-fill-vertex').data(drawVertex ? [0] : []);
+           vertexFill.exit().remove();
+           var vertexFillEnter = vertexFill.enter();
+           var w = 60;
+           var h = 60;
+           var d = 40;
+           vertexFillEnter.append('svg').attr('class', 'preset-icon-fill preset-icon-fill-vertex').attr('width', w).attr('height', h).attr('viewBox', "0 0 ".concat(w, " ").concat(h)).append('circle').attr('cx', w / 2).attr('cy', h / 2).attr('r', d / 2);
+           vertexFill = vertexFillEnter.merge(vertexFill);
+         }
 
-           var filter = function filter(d) {
-             return d.id in _prevHover;
-           };
+         function renderSquareFill(container, drawArea, tagClasses) {
+           var fill = container.selectAll('.preset-icon-fill-area').data(drawArea ? [0] : []);
+           fill.exit().remove();
+           var fillEnter = fill.enter();
+           var d = isSmall() ? 40 : 60;
+           var w = d;
+           var h = d;
+           var l = d * 2 / 3;
+           var c1 = (w - l) / 2;
+           var c2 = c1 + l;
+           fillEnter = fillEnter.append('svg').attr('class', 'preset-icon-fill preset-icon-fill-area').attr('width', w).attr('height', h).attr('viewBox', "0 0 ".concat(w, " ").concat(h));
+           ['fill', 'stroke'].forEach(function (klass) {
+             fillEnter.append('path').attr('d', "M".concat(c1, " ").concat(c1, " L").concat(c1, " ").concat(c2, " L").concat(c2, " ").concat(c2, " L").concat(c2, " ").concat(c1, " Z")).attr('class', "area ".concat(klass));
+           });
+           var rVertex = 2.5;
+           [[c1, c1], [c1, c2], [c2, c2], [c2, c1]].forEach(function (point) {
+             fillEnter.append('circle').attr('class', 'vertex').attr('cx', point[0]).attr('cy', point[1]).attr('r', rVertex);
+           });
 
-           drawVertices(selection, graph, Object.values(_prevHover), filter, extent, false);
-         };
+           if (!isSmall()) {
+             var rMidpoint = 1.25;
+             [[c1, w / 2], [c2, w / 2], [h / 2, c1], [h / 2, c2]].forEach(function (point) {
+               fillEnter.append('circle').attr('class', 'midpoint').attr('cx', point[0]).attr('cy', point[1]).attr('r', rMidpoint);
+             });
+           }
 
-         return drawVertices;
-       }
+           fill = fillEnter.merge(fill);
+           fill.selectAll('path.stroke').attr('class', "area stroke ".concat(tagClasses));
+           fill.selectAll('path.fill').attr('class', "area fill ".concat(tagClasses));
+         }
 
-       function utilBindOnce(target, type, listener, capture) {
-         var typeOnce = type + '.once';
+         function renderLine(container, drawLine, tagClasses) {
+           var line = container.selectAll('.preset-icon-line').data(drawLine ? [0] : []);
+           line.exit().remove();
+           var lineEnter = line.enter();
+           var d = isSmall() ? 40 : 60; // draw the line parametrically
 
-         function one() {
-           target.on(typeOnce, null);
-           listener.apply(this, arguments);
+           var w = d;
+           var h = d;
+           var y = Math.round(d * 0.72);
+           var l = Math.round(d * 0.6);
+           var r = 2.5;
+           var x1 = (w - l) / 2;
+           var x2 = x1 + l;
+           lineEnter = lineEnter.append('svg').attr('class', 'preset-icon-line').attr('width', w).attr('height', h).attr('viewBox', "0 0 ".concat(w, " ").concat(h));
+           ['casing', 'stroke'].forEach(function (klass) {
+             lineEnter.append('path').attr('d', "M".concat(x1, " ").concat(y, " L").concat(x2, " ").concat(y)).attr('class', "line ".concat(klass));
+           });
+           [[x1 - 1, y], [x2 + 1, y]].forEach(function (point) {
+             lineEnter.append('circle').attr('class', 'vertex').attr('cx', point[0]).attr('cy', point[1]).attr('r', r);
+           });
+           line = lineEnter.merge(line);
+           line.selectAll('path.stroke').attr('class', "line stroke ".concat(tagClasses));
+           line.selectAll('path.casing').attr('class', "line casing ".concat(tagClasses));
          }
 
-         target.on(typeOnce, one, capture);
-         return this;
-       }
+         function renderRoute(container, drawRoute, p) {
+           var route = container.selectAll('.preset-icon-route').data(drawRoute ? [0] : []);
+           route.exit().remove();
+           var routeEnter = route.enter();
+           var d = isSmall() ? 40 : 60; // draw the route parametrically
 
-       function defaultFilter$2(d3_event) {
-         return !d3_event.ctrlKey && !d3_event.button;
-       }
+           var w = d;
+           var h = d;
+           var y1 = Math.round(d * 0.80);
+           var y2 = Math.round(d * 0.68);
+           var l = Math.round(d * 0.6);
+           var r = 2;
+           var x1 = (w - l) / 2;
+           var x2 = x1 + l / 3;
+           var x3 = x2 + l / 3;
+           var x4 = x3 + l / 3;
+           routeEnter = routeEnter.append('svg').attr('class', 'preset-icon-route').attr('width', w).attr('height', h).attr('viewBox', "0 0 ".concat(w, " ").concat(h));
+           ['casing', 'stroke'].forEach(function (klass) {
+             routeEnter.append('path').attr('d', "M".concat(x1, " ").concat(y1, " L").concat(x2, " ").concat(y2)).attr('class', "segment0 line ".concat(klass));
+             routeEnter.append('path').attr('d', "M".concat(x2, " ").concat(y2, " L").concat(x3, " ").concat(y1)).attr('class', "segment1 line ".concat(klass));
+             routeEnter.append('path').attr('d', "M".concat(x3, " ").concat(y1, " L").concat(x4, " ").concat(y2)).attr('class', "segment2 line ".concat(klass));
+           });
+           [[x1, y1], [x2, y2], [x3, y1], [x4, y2]].forEach(function (point) {
+             routeEnter.append('circle').attr('class', 'vertex').attr('cx', point[0]).attr('cy', point[1]).attr('r', r);
+           });
+           route = routeEnter.merge(route);
 
-       function defaultExtent$1() {
-         var e = this;
+           if (drawRoute) {
+             var routeType = p.tags.type === 'waterway' ? 'waterway' : p.tags.route;
+             var segmentPresetIDs = routeSegments[routeType];
 
-         if (e instanceof SVGElement) {
-           e = e.ownerSVGElement || e;
+             for (var i in segmentPresetIDs) {
+               var segmentPreset = _mainPresetIndex.item(segmentPresetIDs[i]);
+               var segmentTagClasses = svgTagClasses().getClassesString(segmentPreset.tags, '');
+               route.selectAll("path.stroke.segment".concat(i)).attr('class', "segment".concat(i, " line stroke ").concat(segmentTagClasses));
+               route.selectAll("path.casing.segment".concat(i)).attr('class', "segment".concat(i, " line casing ").concat(segmentTagClasses));
+             }
+           }
+         }
 
-           if (e.hasAttribute('viewBox')) {
-             e = e.viewBox.baseVal;
-             return [[e.x, e.y], [e.x + e.width, e.y + e.height]];
+         function renderSvgIcon(container, picon, geom, isFramed, category, tagClasses) {
+           var isMaki = picon && /^maki-/.test(picon);
+           var isTemaki = picon && /^temaki-/.test(picon);
+           var isFa = picon && /^fa[srb]-/.test(picon);
+           var isiDIcon = picon && !(isMaki || isTemaki || isFa);
+           var icon = container.selectAll('.preset-icon').data(picon ? [0] : []);
+           icon.exit().remove();
+           icon = icon.enter().append('div').attr('class', 'preset-icon').call(svgIcon('')).merge(icon);
+           icon.attr('class', 'preset-icon ' + (geom ? geom + '-geom' : '')).classed('category', category).classed('framed', isFramed).classed('preset-icon-iD', isiDIcon);
+           icon.selectAll('svg').attr('class', 'icon ' + picon + ' ' + (!isiDIcon && geom !== 'line' ? '' : tagClasses));
+           var suffix = '';
+
+           if (isMaki) {
+             suffix = isSmall() && geom === 'point' ? '-11' : '-15';
            }
 
-           return [[0, 0], [e.width.baseVal.value, e.height.baseVal.value]];
+           icon.selectAll('use').attr('href', '#' + picon + suffix);
          }
 
-         return [[0, 0], [e.clientWidth, e.clientHeight]];
-       }
+         function renderImageIcon(container, imageURL) {
+           var imageIcon = container.selectAll('img.image-icon').data(imageURL ? [0] : []);
+           imageIcon.exit().remove();
+           imageIcon = imageIcon.enter().append('img').attr('class', 'image-icon').on('load', function () {
+             return container.classed('showing-img', true);
+           }).on('error', function () {
+             return container.classed('showing-img', false);
+           }).merge(imageIcon);
+           imageIcon.attr('src', imageURL);
+         } // Route icons are drawn with a zigzag annotation underneath:
+         //     o   o
+         //    / \ /
+         //   o   o
+         // This dataset defines the styles that are used to draw the zigzag segments.
 
-       function defaultWheelDelta$1(d3_event) {
-         return -d3_event.deltaY * (d3_event.deltaMode === 1 ? 0.05 : d3_event.deltaMode ? 1 : 0.002);
-       }
 
-       function defaultConstrain$1(transform, extent, translateExtent) {
-         var dx0 = transform.invertX(extent[0][0]) - translateExtent[0][0],
-             dx1 = transform.invertX(extent[1][0]) - translateExtent[1][0],
-             dy0 = transform.invertY(extent[0][1]) - translateExtent[0][1],
-             dy1 = transform.invertY(extent[1][1]) - translateExtent[1][1];
-         return transform.translate(dx1 > dx0 ? (dx0 + dx1) / 2 : Math.min(0, dx0) || Math.max(0, dx1), dy1 > dy0 ? (dy0 + dy1) / 2 : Math.min(0, dy0) || Math.max(0, dy1));
-       }
+         var routeSegments = {
+           bicycle: ['highway/cycleway', 'highway/cycleway', 'highway/cycleway'],
+           bus: ['highway/unclassified', 'highway/secondary', 'highway/primary'],
+           trolleybus: ['highway/unclassified', 'highway/secondary', 'highway/primary'],
+           detour: ['highway/tertiary', 'highway/residential', 'highway/unclassified'],
+           ferry: ['route/ferry', 'route/ferry', 'route/ferry'],
+           foot: ['highway/footway', 'highway/footway', 'highway/footway'],
+           hiking: ['highway/path', 'highway/path', 'highway/path'],
+           horse: ['highway/bridleway', 'highway/bridleway', 'highway/bridleway'],
+           light_rail: ['railway/light_rail', 'railway/light_rail', 'railway/light_rail'],
+           monorail: ['railway/monorail', 'railway/monorail', 'railway/monorail'],
+           mtb: ['highway/path', 'highway/track', 'highway/bridleway'],
+           pipeline: ['man_made/pipeline', 'man_made/pipeline', 'man_made/pipeline'],
+           piste: ['piste/downhill', 'piste/hike', 'piste/nordic'],
+           power: ['power/line', 'power/line', 'power/line'],
+           road: ['highway/secondary', 'highway/primary', 'highway/trunk'],
+           subway: ['railway/subway', 'railway/subway', 'railway/subway'],
+           train: ['railway/rail', 'railway/rail', 'railway/rail'],
+           tram: ['railway/tram', 'railway/tram', 'railway/tram'],
+           waterway: ['waterway/stream', 'waterway/stream', 'waterway/stream']
+         };
 
-       function utilZoomPan() {
-         var filter = defaultFilter$2,
-             extent = defaultExtent$1,
-             constrain = defaultConstrain$1,
-             wheelDelta = defaultWheelDelta$1,
-             scaleExtent = [0, Infinity],
-             translateExtent = [[-Infinity, -Infinity], [Infinity, Infinity]],
-             interpolate = interpolateZoom,
-             dispatch$1 = dispatch('start', 'zoom', 'end'),
-             _wheelDelay = 150,
-             _transform = identity$2,
-             _activeGesture;
+         function render() {
+           var p = _preset.apply(this, arguments);
 
-         function zoom(selection) {
-           selection.on('pointerdown.zoom', pointerdown).on('wheel.zoom', wheeled).style('touch-action', 'none').style('-webkit-tap-highlight-color', 'rgba(0,0,0,0)');
-           select(window).on('pointermove.zoompan', pointermove).on('pointerup.zoompan pointercancel.zoompan', pointerup);
-         }
+           var geom = _geometry ? _geometry.apply(this, arguments) : null;
 
-         zoom.transform = function (collection, transform, point) {
-           var selection = collection.selection ? collection.selection() : collection;
+           if (geom === 'relation' && p.tags && (p.tags.type === 'route' && p.tags.route && routeSegments[p.tags.route] || p.tags.type === 'waterway')) {
+             geom = 'route';
+           }
 
-           if (collection !== selection) {
-             schedule(collection, transform, point);
-           } else {
-             selection.interrupt().each(function () {
-               gesture(this, arguments).start(null).zoom(null, null, typeof transform === 'function' ? transform.apply(this, arguments) : transform).end(null);
-             });
+           var showThirdPartyIcons = corePreferences('preferences.privacy.thirdpartyicons') || 'true';
+           var isFallback = isSmall() && p.isFallback && p.isFallback();
+           var imageURL = showThirdPartyIcons === 'true' && p.imageURL;
+           var picon = getIcon(p, geom);
+           var isCategory = !p.setTags;
+           var drawPoint = picon && geom === 'point' && isSmall() && !isFallback;
+           var drawVertex = picon !== null && geom === 'vertex' && (!isSmall() || !isFallback);
+           var drawLine = picon && geom === 'line' && !isFallback && !isCategory;
+           var drawArea = picon && geom === 'area' && !isFallback && !isCategory;
+           var drawRoute = picon && geom === 'route';
+           var isFramed = drawVertex || drawArea || drawLine || drawRoute || isCategory;
+           var tags = !isCategory ? p.setTags({}, geom) : {};
+
+           for (var k in tags) {
+             if (tags[k] === '*') {
+               tags[k] = 'yes';
+             }
            }
-         };
 
-         zoom.scaleBy = function (selection, k, p) {
-           zoom.scaleTo(selection, function () {
-             var k0 = _transform.k,
-                 k1 = typeof k === 'function' ? k.apply(this, arguments) : k;
-             return k0 * k1;
-           }, p);
-         };
+           var tagClasses = svgTagClasses().getClassesString(tags, '');
+           var selection = select(this);
+           var container = selection.selectAll('.preset-icon-container').data([0]);
+           container = container.enter().append('div').attr('class', "preset-icon-container ".concat(_sizeClass)).merge(container);
+           container.classed('showing-img', !!imageURL).classed('fallback', isFallback);
+           renderCategoryBorder(container, isCategory && p);
+           renderPointBorder(container, drawPoint);
+           renderCircleFill(container, drawVertex);
+           renderSquareFill(container, drawArea, tagClasses);
+           renderLine(container, drawLine, tagClasses);
+           renderRoute(container, drawRoute, p);
+           renderSvgIcon(container, picon, geom, isFramed, isCategory, tagClasses);
+           renderImageIcon(container, imageURL);
+         }
 
-         zoom.scaleTo = function (selection, k, p) {
-           zoom.transform(selection, function () {
-             var e = extent.apply(this, arguments),
-                 t0 = _transform,
-                 p0 = !p ? centroid(e) : typeof p === 'function' ? p.apply(this, arguments) : p,
-                 p1 = t0.invert(p0),
-                 k1 = typeof k === 'function' ? k.apply(this, arguments) : k;
-             return constrain(translate(scale(t0, k1), p0, p1), e, translateExtent);
-           }, p);
+         presetIcon.preset = function (val) {
+           if (!arguments.length) return _preset;
+           _preset = utilFunctor(val);
+           return presetIcon;
          };
 
-         zoom.translateBy = function (selection, x, y) {
-           zoom.transform(selection, function () {
-             return constrain(_transform.translate(typeof x === 'function' ? x.apply(this, arguments) : x, typeof y === 'function' ? y.apply(this, arguments) : y), extent.apply(this, arguments), translateExtent);
-           });
+         presetIcon.geometry = function (val) {
+           if (!arguments.length) return _geometry;
+           _geometry = utilFunctor(val);
+           return presetIcon;
          };
 
-         zoom.translateTo = function (selection, x, y, p) {
-           zoom.transform(selection, function () {
-             var e = extent.apply(this, arguments),
-                 t = _transform,
-                 p0 = !p ? centroid(e) : typeof p === 'function' ? p.apply(this, arguments) : p;
-             return constrain(identity$2.translate(p0[0], p0[1]).scale(t.k).translate(typeof x === 'function' ? -x.apply(this, arguments) : -x, typeof y === 'function' ? -y.apply(this, arguments) : -y), e, translateExtent);
-           }, p);
+         presetIcon.sizeClass = function (val) {
+           if (!arguments.length) return _sizeClass;
+           _sizeClass = val;
+           return presetIcon;
          };
 
-         function scale(transform, k) {
-           k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], k));
-           return k === transform.k ? transform : new Transform(k, transform.x, transform.y);
-         }
-
-         function translate(transform, p0, p1) {
-           var x = p0[0] - p1[0] * transform.k,
-               y = p0[1] - p1[1] * transform.k;
-           return x === transform.x && y === transform.y ? transform : new Transform(transform.k, x, y);
-         }
+         return presetIcon;
+       }
 
-         function centroid(extent) {
-           return [(+extent[0][0] + +extent[1][0]) / 2, (+extent[0][1] + +extent[1][1]) / 2];
-         }
+       function uiSectionFeatureType(context) {
+         var dispatch = dispatch$8('choose');
+         var _entityIDs = [];
+         var _presets = [];
 
-         function schedule(transition, transform, point) {
-           transition.on('start.zoom', function () {
-             gesture(this, arguments).start(null);
-           }).on('interrupt.zoom end.zoom', function () {
-             gesture(this, arguments).end(null);
-           }).tween('zoom', function () {
-             var that = this,
-                 args = arguments,
-                 g = gesture(that, args),
-                 e = extent.apply(that, args),
-                 p = !point ? centroid(e) : typeof point === 'function' ? point.apply(that, args) : point,
-                 w = Math.max(e[1][0] - e[0][0], e[1][1] - e[0][1]),
-                 a = _transform,
-                 b = typeof transform === 'function' ? transform.apply(that, args) : transform,
-                 i = interpolate(a.invert(p).concat(w / a.k), b.invert(p).concat(w / b.k));
-             return function (t) {
-               if (t === 1) t = b; // Avoid rounding error on end.
-               else {
-                   var l = i(t),
-                       k = w / l[2];
-                   t = new Transform(k, p[0] - l[0] * k, p[1] - l[1] * k);
-                 }
-               g.zoom(null, null, t);
-             };
-           });
-         }
+         var _tagReference;
 
-         function gesture(that, args, clean) {
-           return !clean && _activeGesture || new Gesture(that, args);
-         }
+         var section = uiSection('feature-type', context).label(_t.html('inspector.feature_type')).disclosureContent(renderDisclosureContent);
 
-         function Gesture(that, args) {
-           this.that = that;
-           this.args = args;
-           this.active = 0;
-           this.extent = extent.apply(that, args);
-         }
+         function renderDisclosureContent(selection) {
+           selection.classed('preset-list-item', true);
+           selection.classed('mixed-types', _presets.length > 1);
+           var presetButtonWrap = selection.selectAll('.preset-list-button-wrap').data([0]).enter().append('div').attr('class', 'preset-list-button-wrap');
+           var presetButton = presetButtonWrap.append('button').attr('class', 'preset-list-button preset-reset').call(uiTooltip().title(_t.html('inspector.back_tooltip')).placement('bottom'));
+           presetButton.append('div').attr('class', 'preset-icon-container');
+           presetButton.append('div').attr('class', 'label').append('div').attr('class', 'label-inner');
+           presetButtonWrap.append('div').attr('class', 'accessory-buttons');
+           var tagReferenceBodyWrap = selection.selectAll('.tag-reference-body-wrap').data([0]);
+           tagReferenceBodyWrap = tagReferenceBodyWrap.enter().append('div').attr('class', 'tag-reference-body-wrap').merge(tagReferenceBodyWrap); // update header
 
-         Gesture.prototype = {
-           start: function start(d3_event) {
-             if (++this.active === 1) {
-               _activeGesture = this;
-               dispatch$1.call('start', this, d3_event);
-             }
+           if (_tagReference) {
+             selection.selectAll('.preset-list-button-wrap .accessory-buttons').style('display', _presets.length === 1 ? null : 'none').call(_tagReference.button);
+             tagReferenceBodyWrap.style('display', _presets.length === 1 ? null : 'none').call(_tagReference.body);
+           }
 
-             return this;
-           },
-           zoom: function zoom(d3_event, key, transform) {
-             if (this.mouse && key !== 'mouse') this.mouse[1] = transform.invert(this.mouse[0]);
-             if (this.pointer0 && key !== 'touch') this.pointer0[1] = transform.invert(this.pointer0[0]);
-             if (this.pointer1 && key !== 'touch') this.pointer1[1] = transform.invert(this.pointer1[0]);
-             _transform = transform;
-             dispatch$1.call('zoom', this, d3_event, key, transform);
-             return this;
-           },
-           end: function end(d3_event) {
-             if (--this.active === 0) {
-               _activeGesture = null;
-               dispatch$1.call('end', this, d3_event);
-             }
+           selection.selectAll('.preset-reset').on('click', function () {
+             dispatch.call('choose', this, _presets);
+           }).on('pointerdown pointerup mousedown mouseup', function (d3_event) {
+             d3_event.preventDefault();
+             d3_event.stopPropagation();
+           });
+           var geometries = entityGeometries();
+           selection.select('.preset-list-item button').call(uiPresetIcon().geometry(_presets.length === 1 ? geometries.length === 1 && geometries[0] : null).preset(_presets.length === 1 ? _presets[0] : _mainPresetIndex.item('point')));
+           var names = _presets.length === 1 ? [_presets[0].nameLabel(), _presets[0].subtitleLabel()].filter(Boolean) : [_t('inspector.multiple_types')];
+           var label = selection.select('.label-inner');
+           var nameparts = label.selectAll('.namepart').data(names, function (d) {
+             return d;
+           });
+           nameparts.exit().remove();
+           nameparts.enter().append('div').attr('class', 'namepart').html(function (d) {
+             return d;
+           });
+         }
 
-             return this;
-           }
+         section.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs;
+           _entityIDs = val;
+           return section;
          };
 
-         function wheeled(d3_event) {
-           if (!filter.apply(this, arguments)) return;
-           var g = gesture(this, arguments),
-               t = _transform,
-               k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], t.k * Math.pow(2, wheelDelta.apply(this, arguments)))),
-               p = utilFastMouse(this)(d3_event); // If the mouse is in the same location as before, reuse it.
-           // If there were recent wheel events, reset the wheel idle timeout.
-
-           if (g.wheel) {
-             if (g.mouse[0][0] !== p[0] || g.mouse[0][1] !== p[1]) {
-               g.mouse[1] = t.invert(g.mouse[0] = p);
-             }
+         section.presets = function (val) {
+           if (!arguments.length) return _presets; // don't reload the same preset
 
-             clearTimeout(g.wheel); // Otherwise, capture the mouse point and location at the start.
-           } else {
-             g.mouse = [p, t.invert(p)];
-             interrupt(this);
-             g.start(d3_event);
+           if (!utilArrayIdentical(val, _presets)) {
+             _presets = val;
+
+             if (_presets.length === 1) {
+               _tagReference = uiTagReference(_presets[0].reference()).showing(false);
+             }
            }
 
-           d3_event.preventDefault();
-           d3_event.stopImmediatePropagation();
-           g.wheel = setTimeout(wheelidled, _wheelDelay);
-           g.zoom(d3_event, 'mouse', constrain(translate(scale(t, k), g.mouse[0], g.mouse[1]), g.extent, translateExtent));
+           return section;
+         };
 
-           function wheelidled() {
-             g.wheel = null;
-             g.end(d3_event);
-           }
-         }
+         function entityGeometries() {
+           var counts = {};
 
-         var _downPointerIDs = new Set();
+           for (var i in _entityIDs) {
+             var geometry = context.graph().geometry(_entityIDs[i]);
+             if (!counts[geometry]) counts[geometry] = 0;
+             counts[geometry] += 1;
+           }
 
-         var _pointerLocGetter;
+           return Object.keys(counts).sort(function (geom1, geom2) {
+             return counts[geom2] - counts[geom1];
+           });
+         }
 
-         function pointerdown(d3_event) {
-           _downPointerIDs.add(d3_event.pointerId);
+         return utilRebind(section, dispatch, 'on');
+       }
 
-           if (!filter.apply(this, arguments)) return;
-           var g = gesture(this, arguments, _downPointerIDs.size === 1);
-           var started;
-           d3_event.stopImmediatePropagation();
-           _pointerLocGetter = utilFastMouse(this);
+       // It borrows some code from uiHelp
 
-           var loc = _pointerLocGetter(d3_event);
+       function uiFieldHelp(context, fieldName) {
+         var fieldHelp = {};
 
-           var p = [loc, _transform.invert(loc), d3_event.pointerId];
+         var _inspector = select(null);
 
-           if (!g.pointer0) {
-             g.pointer0 = p;
-             started = true;
-           } else if (!g.pointer1 && g.pointer0[2] !== p[2]) {
-             g.pointer1 = p;
-           }
+         var _wrap = select(null);
 
-           if (started) {
-             interrupt(this);
-             g.start(d3_event);
-           }
-         }
+         var _body = select(null);
 
-         function pointermove(d3_event) {
-           if (!_downPointerIDs.has(d3_event.pointerId)) return;
-           if (!_activeGesture || !_pointerLocGetter) return;
-           var g = gesture(this, arguments);
-           var isPointer0 = g.pointer0 && g.pointer0[2] === d3_event.pointerId;
-           var isPointer1 = !isPointer0 && g.pointer1 && g.pointer1[2] === d3_event.pointerId;
+         var fieldHelpKeys = {
+           restrictions: [['about', ['about', 'from_via_to', 'maxdist', 'maxvia']], ['inspecting', ['about', 'from_shadow', 'allow_shadow', 'restrict_shadow', 'only_shadow', 'restricted', 'only']], ['modifying', ['about', 'indicators', 'allow_turn', 'restrict_turn', 'only_turn']], ['tips', ['simple', 'simple_example', 'indirect', 'indirect_example', 'indirect_noedit']]]
+         };
+         var fieldHelpHeadings = {};
+         var replacements = {
+           distField: _t.html('restriction.controls.distance'),
+           viaField: _t.html('restriction.controls.via'),
+           fromShadow: icon('#iD-turn-shadow', 'inline shadow from'),
+           allowShadow: icon('#iD-turn-shadow', 'inline shadow allow'),
+           restrictShadow: icon('#iD-turn-shadow', 'inline shadow restrict'),
+           onlyShadow: icon('#iD-turn-shadow', 'inline shadow only'),
+           allowTurn: icon('#iD-turn-yes', 'inline turn'),
+           restrictTurn: icon('#iD-turn-no', 'inline turn'),
+           onlyTurn: icon('#iD-turn-only', 'inline turn')
+         }; // For each section, squash all the texts into a single markdown document
 
-           if ((isPointer0 || isPointer1) && 'buttons' in d3_event && !d3_event.buttons) {
-             // The pointer went up without ending the gesture somehow, e.g.
-             // a down mouse was moved off the map and released. End it here.
-             if (g.pointer0) _downPointerIDs["delete"](g.pointer0[2]);
-             if (g.pointer1) _downPointerIDs["delete"](g.pointer1[2]);
-             g.end(d3_event);
-             return;
-           }
+         var docs = fieldHelpKeys[fieldName].map(function (key) {
+           var helpkey = 'help.field.' + fieldName + '.' + key[0];
+           var text = key[1].reduce(function (all, part) {
+             var subkey = helpkey + '.' + part;
+             var depth = fieldHelpHeadings[subkey]; // is this subkey a heading?
 
-           d3_event.preventDefault();
-           d3_event.stopImmediatePropagation();
+             var hhh = depth ? Array(depth + 1).join('#') + ' ' : ''; // if so, prepend with some ##'s
 
-           var loc = _pointerLocGetter(d3_event);
+             return all + hhh + _t.html(subkey, replacements) + '\n\n';
+           }, '');
+           return {
+             key: helpkey,
+             title: _t.html(helpkey + '.title'),
+             html: marked_1(text.trim())
+           };
+         });
 
-           var t, p, l;
-           if (isPointer0) g.pointer0[0] = loc;else if (isPointer1) g.pointer1[0] = loc;
-           t = _transform;
+         function show() {
+           updatePosition();
 
-           if (g.pointer1) {
-             var p0 = g.pointer0[0],
-                 l0 = g.pointer0[1],
-                 p1 = g.pointer1[0],
-                 l1 = g.pointer1[1],
-                 dp = (dp = p1[0] - p0[0]) * dp + (dp = p1[1] - p0[1]) * dp,
-                 dl = (dl = l1[0] - l0[0]) * dl + (dl = l1[1] - l0[1]) * dl;
-             t = scale(t, Math.sqrt(dp / dl));
-             p = [(p0[0] + p1[0]) / 2, (p0[1] + p1[1]) / 2];
-             l = [(l0[0] + l1[0]) / 2, (l0[1] + l1[1]) / 2];
-           } else if (g.pointer0) {
-             p = g.pointer0[0];
-             l = g.pointer0[1];
-           } else return;
+           _body.classed('hide', false).style('opacity', '0').transition().duration(200).style('opacity', '1');
+         }
 
-           g.zoom(d3_event, 'touch', constrain(translate(t, p, l), g.extent, translateExtent));
+         function hide() {
+           _body.classed('hide', true).transition().duration(200).style('opacity', '0').on('end', function () {
+             _body.classed('hide', true);
+           });
          }
 
-         function pointerup(d3_event) {
-           if (!_downPointerIDs.has(d3_event.pointerId)) return;
+         function clickHelp(index) {
+           var d = docs[index];
+           var tkeys = fieldHelpKeys[fieldName][index][1];
 
-           _downPointerIDs["delete"](d3_event.pointerId);
+           _body.selectAll('.field-help-nav-item').classed('active', function (d, i) {
+             return i === index;
+           });
 
-           if (!_activeGesture) return;
-           var g = gesture(this, arguments);
-           d3_event.stopImmediatePropagation();
-           if (g.pointer0 && g.pointer0[2] === d3_event.pointerId) delete g.pointer0;else if (g.pointer1 && g.pointer1[2] === d3_event.pointerId) delete g.pointer1;
+           var content = _body.selectAll('.field-help-content').html(d.html); // class the paragraphs so we can find and style them
 
-           if (g.pointer1 && !g.pointer0) {
-             g.pointer0 = g.pointer1;
-             delete g.pointer1;
-           }
 
-           if (g.pointer0) g.pointer0[1] = _transform.invert(g.pointer0[0]);else {
-             g.end(d3_event);
+           content.selectAll('p').attr('class', function (d, i) {
+             return tkeys[i];
+           }); // insert special content for certain help sections
+
+           if (d.key === 'help.field.restrictions.inspecting') {
+             content.insert('img', 'p.from_shadow').attr('class', 'field-help-image cf').attr('src', context.imagePath('tr_inspect.gif'));
+           } else if (d.key === 'help.field.restrictions.modifying') {
+             content.insert('img', 'p.allow_turn').attr('class', 'field-help-image cf').attr('src', context.imagePath('tr_modify.gif'));
            }
          }
 
-         zoom.wheelDelta = function (_) {
-           return arguments.length ? (wheelDelta = utilFunctor(+_), zoom) : wheelDelta;
-         };
+         fieldHelp.button = function (selection) {
+           if (_body.empty()) return;
+           var button = selection.selectAll('.field-help-button').data([0]); // enter/update
 
-         zoom.filter = function (_) {
-           return arguments.length ? (filter = utilFunctor(!!_), zoom) : filter;
-         };
+           button.enter().append('button').attr('class', 'field-help-button').call(svgIcon('#iD-icon-help')).merge(button).on('click', function (d3_event) {
+             d3_event.stopPropagation();
+             d3_event.preventDefault();
 
-         zoom.extent = function (_) {
-           return arguments.length ? (extent = utilFunctor([[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]]), zoom) : extent;
+             if (_body.classed('hide')) {
+               show();
+             } else {
+               hide();
+             }
+           });
          };
 
-         zoom.scaleExtent = function (_) {
-           return arguments.length ? (scaleExtent[0] = +_[0], scaleExtent[1] = +_[1], zoom) : [scaleExtent[0], scaleExtent[1]];
-         };
+         function updatePosition() {
+           var wrap = _wrap.node();
 
-         zoom.translateExtent = function (_) {
-           return arguments.length ? (translateExtent[0][0] = +_[0][0], translateExtent[1][0] = +_[1][0], translateExtent[0][1] = +_[0][1], translateExtent[1][1] = +_[1][1], zoom) : [[translateExtent[0][0], translateExtent[0][1]], [translateExtent[1][0], translateExtent[1][1]]];
-         };
+           var inspector = _inspector.node();
 
-         zoom.constrain = function (_) {
-           return arguments.length ? (constrain = _, zoom) : constrain;
-         };
+           var wRect = wrap.getBoundingClientRect();
+           var iRect = inspector.getBoundingClientRect();
 
-         zoom.interpolate = function (_) {
-           return arguments.length ? (interpolate = _, zoom) : interpolate;
-         };
+           _body.style('top', wRect.top + inspector.scrollTop - iRect.top + 'px');
+         }
 
-         zoom._transform = function (_) {
-           return arguments.length ? (_transform = _, zoom) : _transform;
+         fieldHelp.body = function (selection) {
+           // This control expects the field to have a form-field-input-wrap div
+           _wrap = selection.selectAll('.form-field-input-wrap');
+           if (_wrap.empty()) return; // absolute position relative to the inspector, so it "floats" above the fields
+
+           _inspector = context.container().select('.sidebar .entity-editor-pane .inspector-body');
+           if (_inspector.empty()) return;
+           _body = _inspector.selectAll('.field-help-body').data([0]);
+
+           var enter = _body.enter().append('div').attr('class', 'field-help-body hide'); // initially hidden
+
+
+           var titleEnter = enter.append('div').attr('class', 'field-help-title cf');
+           titleEnter.append('h2').attr('class', _mainLocalizer.textDirection() === 'rtl' ? 'fr' : 'fl').html(_t.html('help.field.' + fieldName + '.title'));
+           titleEnter.append('button').attr('class', 'fr close').on('click', function (d3_event) {
+             d3_event.stopPropagation();
+             d3_event.preventDefault();
+             hide();
+           }).call(svgIcon('#iD-icon-close'));
+           var navEnter = enter.append('div').attr('class', 'field-help-nav cf');
+           var titles = docs.map(function (d) {
+             return d.title;
+           });
+           navEnter.selectAll('.field-help-nav-item').data(titles).enter().append('div').attr('class', 'field-help-nav-item').html(function (d) {
+             return d;
+           }).on('click', function (d3_event, d) {
+             d3_event.stopPropagation();
+             d3_event.preventDefault();
+             clickHelp(titles.indexOf(d));
+           });
+           enter.append('div').attr('class', 'field-help-content');
+           _body = _body.merge(enter);
+           clickHelp(0);
          };
 
-         return utilRebind(zoom, dispatch$1, 'on');
+         return fieldHelp;
        }
 
-       // if pointer events are supported. Falls back to default `dblclick` event.
-
-       function utilDoubleUp() {
-         var dispatch$1 = dispatch('doubleUp');
-         var _maxTimespan = 500; // milliseconds
+       function uiFieldCheck(field, context) {
+         var dispatch = dispatch$8('change');
+         var options = field.options;
+         var values = [];
+         var texts = [];
 
-         var _maxDistance = 20; // web pixels; be somewhat generous to account for touch devices
+         var _tags;
 
-         var _pointer; // object representing the pointer that could trigger double up
+         var input = select(null);
+         var text = select(null);
+         var label = select(null);
+         var reverser = select(null);
 
+         var _impliedYes;
 
-         function pointerIsValidFor(loc) {
-           // second pointerup must occur within a small timeframe after the first pointerdown
-           return new Date().getTime() - _pointer.startTime <= _maxTimespan && // all pointer events must occur within a small distance of the first pointerdown
-           geoVecLength(_pointer.startLoc, loc) <= _maxDistance;
-         }
+         var _entityIDs = [];
 
-         function pointerdown(d3_event) {
-           // ignore right-click
-           if (d3_event.ctrlKey || d3_event.button === 2) return;
-           var loc = [d3_event.clientX, d3_event.clientY]; // Don't rely on pointerId here since it can change between pointerdown
-           // events on touch devices
+         var _value;
 
-           if (_pointer && !pointerIsValidFor(loc)) {
-             // if this pointer is no longer valid, clear it so another can be started
-             _pointer = undefined;
+         if (options) {
+           for (var i in options) {
+             var v = options[i];
+             values.push(v === 'undefined' ? undefined : v);
+             texts.push(field.t.html('options.' + v, {
+               'default': v
+             }));
            }
+         } else {
+           values = [undefined, 'yes'];
+           texts = [_t.html('inspector.unknown'), _t.html('inspector.check.yes')];
 
-           if (!_pointer) {
-             _pointer = {
-               startLoc: loc,
-               startTime: new Date().getTime(),
-               upCount: 0,
-               pointerId: d3_event.pointerId
-             };
-           } else {
-             // double down
-             _pointer.pointerId = d3_event.pointerId;
+           if (field.type !== 'defaultCheck') {
+             values.push('no');
+             texts.push(_t.html('inspector.check.no'));
            }
-         }
-
-         function pointerup(d3_event) {
-           // ignore right-click
-           if (d3_event.ctrlKey || d3_event.button === 2) return;
-           if (!_pointer || _pointer.pointerId !== d3_event.pointerId) return;
-           _pointer.upCount += 1;
+         } // Checks tags to see whether an undefined value is "Assumed to be Yes"
 
-           if (_pointer.upCount === 2) {
-             // double up!
-             var loc = [d3_event.clientX, d3_event.clientY];
 
-             if (pointerIsValidFor(loc)) {
-               var locInThis = utilFastMouse(this)(d3_event);
-               dispatch$1.call('doubleUp', this, d3_event, locInThis);
-             } // clear the pointer info in any case
+         function checkImpliedYes() {
+           _impliedYes = field.id === 'oneway_yes'; // hack: pretend `oneway` field is a `oneway_yes` field
+           // where implied oneway tag exists (e.g. `junction=roundabout`) #2220, #1841
 
+           if (field.id === 'oneway') {
+             var entity = context.entity(_entityIDs[0]);
 
-             _pointer = undefined;
+             for (var key in entity.tags) {
+               if (key in osmOneWayTags && entity.tags[key] in osmOneWayTags[key]) {
+                 _impliedYes = true;
+                 texts[0] = _t.html('_tagging.presets.fields.oneway_yes.options.undefined');
+                 break;
+               }
+             }
            }
          }
 
-         function doubleUp(selection) {
-           if ('PointerEvent' in window) {
-             // dblclick isn't well supported on touch devices so manually use
-             // pointer events if they're available
-             selection.on('pointerdown.doubleUp', pointerdown).on('pointerup.doubleUp', pointerup);
-           } else {
-             // fallback to dblclick
-             selection.on('dblclick.doubleUp', function (d3_event) {
-               dispatch$1.call('doubleUp', this, d3_event, utilFastMouse(this)(d3_event));
-             });
-           }
+         function reverserHidden() {
+           if (!context.container().select('div.inspector-hover').empty()) return true;
+           return !(_value === 'yes' || _impliedYes && !_value);
          }
 
-         doubleUp.off = function (selection) {
-           selection.on('pointerdown.doubleUp', null).on('pointerup.doubleUp', null).on('dblclick.doubleUp', null);
-         };
-
-         return utilRebind(doubleUp, dispatch$1, 'on');
-       }
-
-       var TILESIZE = 256;
-       var minZoom = 2;
-       var maxZoom = 24;
-       var kMin = geoZoomToScale(minZoom, TILESIZE);
-       var kMax = geoZoomToScale(maxZoom, TILESIZE);
-
-       function clamp(num, min, max) {
-         return Math.max(min, Math.min(num, max));
-       }
-
-       function rendererMap(context) {
-         var dispatch$1 = dispatch('move', 'drawn', 'crossEditableZoom', 'hitMinZoom', 'changeHighlighting', 'changeAreaFill');
-         var projection = context.projection;
-         var curtainProjection = context.curtainProjection;
-         var drawLayers;
-         var drawPoints;
-         var drawVertices;
-         var drawLines;
-         var drawAreas;
-         var drawMidpoints;
-         var drawLabels;
-
-         var _selection = select(null);
-
-         var supersurface = select(null);
-         var wrapper = select(null);
-         var surface = select(null);
-         var _dimensions = [1, 1];
-         var _dblClickZoomEnabled = true;
-         var _redrawEnabled = true;
-
-         var _gestureTransformStart;
+         function reverserSetText(selection) {
+           var entity = _entityIDs.length && context.hasEntity(_entityIDs[0]);
+           if (reverserHidden() || !entity) return selection;
+           var first = entity.first();
+           var last = entity.isClosed() ? entity.nodes[entity.nodes.length - 2] : entity.last();
+           var pseudoDirection = first < last;
+           var icon = pseudoDirection ? '#iD-icon-forward' : '#iD-icon-backward';
+           selection.selectAll('.reverser-span').html(_t.html('inspector.check.reverser')).call(svgIcon(icon, 'inline'));
+           return selection;
+         }
 
-         var _transformStart = projection.transform();
+         var check = function check(selection) {
+           checkImpliedYes();
+           label = selection.selectAll('.form-field-input-wrap').data([0]);
+           var enter = label.enter().append('label').attr('class', 'form-field-input-wrap form-field-input-check');
+           enter.append('input').property('indeterminate', field.type !== 'defaultCheck').attr('type', 'checkbox').attr('id', field.domId);
+           enter.append('span').html(texts[0]).attr('class', 'value');
 
-         var _transformLast;
+           if (field.type === 'onewayCheck') {
+             enter.append('button').attr('class', 'reverser' + (reverserHidden() ? ' hide' : '')).append('span').attr('class', 'reverser-span');
+           }
 
-         var _isTransformed = false;
-         var _minzoom = 0;
+           label = label.merge(enter);
+           input = label.selectAll('input');
+           text = label.selectAll('span.value');
+           input.on('click', function (d3_event) {
+             d3_event.stopPropagation();
+             var t = {};
 
-         var _getMouseCoords;
+             if (Array.isArray(_tags[field.key])) {
+               if (values.indexOf('yes') !== -1) {
+                 t[field.key] = 'yes';
+               } else {
+                 t[field.key] = values[0];
+               }
+             } else {
+               t[field.key] = values[(values.indexOf(_value) + 1) % values.length];
+             } // Don't cycle through `alternating` or `reversible` states - #4970
+             // (They are supported as translated strings, but should not toggle with clicks)
 
-         var _lastPointerEvent;
 
-         var _lastWithinEditableZoom; // whether a pointerdown event started the zoom
+             if (t[field.key] === 'reversible' || t[field.key] === 'alternating') {
+               t[field.key] = values[0];
+             }
 
+             dispatch.call('change', this, t);
+           });
 
-         var _pointerDown = false; // use pointer events on supported platforms; fallback to mouse events
+           if (field.type === 'onewayCheck') {
+             reverser = label.selectAll('.reverser');
+             reverser.call(reverserSetText).on('click', function (d3_event) {
+               d3_event.preventDefault();
+               d3_event.stopPropagation();
+               context.perform(function (graph) {
+                 for (var i in _entityIDs) {
+                   graph = actionReverse(_entityIDs[i])(graph);
+                 }
 
-         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse'; // use pointer event interaction if supported; fallback to touch/mouse events in d3-zoom
+                 return graph;
+               }, _t('operations.reverse.annotation.line', {
+                 n: 1
+               })); // must manually revalidate since no 'change' event was called
 
+               context.validator().validate();
+               select(this).call(reverserSetText);
+             });
+           }
+         };
 
-         var _zoomerPannerFunction = 'PointerEvent' in window ? utilZoomPan : d3_zoom;
+         check.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs;
+           _entityIDs = val;
+           return check;
+         };
 
-         var _zoomerPanner = _zoomerPannerFunction().scaleExtent([kMin, kMax]).interpolate(interpolate).filter(zoomEventFilter).on('zoom.map', zoomPan).on('start.map', function (d3_event) {
-           _pointerDown = d3_event && (d3_event.type === 'pointerdown' || d3_event.sourceEvent && d3_event.sourceEvent.type === 'pointerdown');
-         }).on('end.map', function () {
-           _pointerDown = false;
-         });
+         check.tags = function (tags) {
+           _tags = tags;
 
-         var _doubleUpHandler = utilDoubleUp();
+           function isChecked(val) {
+             return val !== 'no' && val !== '' && val !== undefined && val !== null;
+           }
 
-         var scheduleRedraw = throttle(redraw, 750); // var isRedrawScheduled = false;
-         // var pendingRedrawCall;
-         // function scheduleRedraw() {
-         //     // Only schedule the redraw if one has not already been set.
-         //     if (isRedrawScheduled) return;
-         //     isRedrawScheduled = true;
-         //     var that = this;
-         //     var args = arguments;
-         //     pendingRedrawCall = window.requestIdleCallback(function () {
-         //         // Reset the boolean so future redraws can be set.
-         //         isRedrawScheduled = false;
-         //         redraw.apply(that, args);
-         //     }, { timeout: 1400 });
-         // }
+           function textFor(val) {
+             if (val === '') val = undefined;
+             var index = values.indexOf(val);
+             return index !== -1 ? texts[index] : '"' + val + '"';
+           }
 
+           checkImpliedYes();
+           var isMixed = Array.isArray(tags[field.key]);
+           _value = !isMixed && tags[field.key] && tags[field.key].toLowerCase();
 
-         function cancelPendingRedraw() {
-           scheduleRedraw.cancel(); // isRedrawScheduled = false;
-           // window.cancelIdleCallback(pendingRedrawCall);
-         }
+           if (field.type === 'onewayCheck' && (_value === '1' || _value === '-1')) {
+             _value = 'yes';
+           }
 
-         function map(selection) {
-           _selection = selection;
-           context.on('change.map', immediateRedraw);
-           var osm = context.connection();
+           input.property('indeterminate', isMixed || field.type !== 'defaultCheck' && !_value).property('checked', isChecked(_value));
+           text.html(isMixed ? _t.html('inspector.multiple_values') : textFor(_value)).classed('mixed', isMixed);
+           label.classed('set', !!_value);
 
-           if (osm) {
-             osm.on('change.map', immediateRedraw);
+           if (field.type === 'onewayCheck') {
+             reverser.classed('hide', reverserHidden()).call(reverserSetText);
            }
+         };
 
-           function didUndoOrRedo(targetTransform) {
-             var mode = context.mode().id;
-             if (mode !== 'browse' && mode !== 'select') return;
+         check.focus = function () {
+           input.node().focus();
+         };
 
-             if (targetTransform) {
-               map.transformEase(targetTransform);
-             }
-           }
+         return utilRebind(check, dispatch, 'on');
+       }
 
-           context.history().on('merge.map', function () {
-             scheduleRedraw();
-           }).on('change.map', immediateRedraw).on('undone.map', function (stack, fromStack) {
-             didUndoOrRedo(fromStack.transform);
-           }).on('redone.map', function (stack) {
-             didUndoOrRedo(stack.transform);
-           });
-           context.background().on('change.map', immediateRedraw);
-           context.features().on('redraw.map', immediateRedraw);
-           drawLayers.on('change.map', function () {
-             context.background().updateImagery();
-             immediateRedraw();
-           });
-           selection.on('wheel.map mousewheel.map', function (d3_event) {
-             // disable swipe-to-navigate browser pages on trackpad/magic mouse – #5552
-             d3_event.preventDefault();
-           }).call(_zoomerPanner).call(_zoomerPanner.transform, projection.transform()).on('dblclick.zoom', null); // override d3-zoom dblclick handling
+       function uiFieldCombo(field, context) {
+         var dispatch = dispatch$8('change');
 
-           map.supersurface = supersurface = selection.append('div').attr('class', 'supersurface').call(utilSetTransform, 0, 0); // Need a wrapper div because Opera can't cope with an absolutely positioned
-           // SVG element: http://bl.ocks.org/jfirebaugh/6fbfbd922552bf776c16
+         var _isMulti = field.type === 'multiCombo' || field.type === 'manyCombo';
 
-           wrapper = supersurface.append('div').attr('class', 'layer layer-data');
-           map.surface = surface = wrapper.call(drawLayers).selectAll('.surface');
-           surface.call(drawLabels.observe).call(_doubleUpHandler).on(_pointerPrefix + 'down.zoom', function (d3_event) {
-             _lastPointerEvent = d3_event;
+         var _isNetwork = field.type === 'networkCombo';
 
-             if (d3_event.button === 2) {
-               d3_event.stopPropagation();
-             }
-           }, true).on(_pointerPrefix + 'up.zoom', function (d3_event) {
-             _lastPointerEvent = d3_event;
+         var _isSemi = field.type === 'semiCombo';
 
-             if (resetTransform()) {
-               immediateRedraw();
-             }
-           }).on(_pointerPrefix + 'move.map', function (d3_event) {
-             _lastPointerEvent = d3_event;
-           }).on(_pointerPrefix + 'over.vertices', function (d3_event) {
-             if (map.editableDataEnabled() && !_isTransformed) {
-               var hover = d3_event.target.__data__;
-               surface.call(drawVertices.drawHover, context.graph(), hover, map.extent());
-               dispatch$1.call('drawn', this, {
-                 full: false
-               });
-             }
-           }).on(_pointerPrefix + 'out.vertices', function (d3_event) {
-             if (map.editableDataEnabled() && !_isTransformed) {
-               var hover = d3_event.relatedTarget && d3_event.relatedTarget.__data__;
-               surface.call(drawVertices.drawHover, context.graph(), hover, map.extent());
-               dispatch$1.call('drawn', this, {
-                 full: false
-               });
-             }
-           });
-           var detected = utilDetect(); // only WebKit supports gesture events
+         var _optarray = field.options;
 
-           if ('GestureEvent' in window && // Listening for gesture events on iOS 13.4+ breaks double-tapping,
-           // but we only need to do this on desktop Safari anyway. – #7694
-           !detected.isMobileWebKit) {
-             // Desktop Safari sends gesture events for multitouch trackpad pinches.
-             // We can listen for these and translate them into map zooms.
-             surface.on('gesturestart.surface', function (d3_event) {
-               d3_event.preventDefault();
-               _gestureTransformStart = projection.transform();
-             }).on('gesturechange.surface', gestureChange);
-           } // must call after surface init
+         var _showTagInfoSuggestions = field.type !== 'manyCombo' && field.autoSuggestions !== false;
 
+         var _allowCustomValues = field.type !== 'manyCombo' && field.customValues !== false;
 
-           updateAreaFill();
+         var _snake_case = field.snake_case || field.snake_case === undefined;
 
-           _doubleUpHandler.on('doubleUp.map', function (d3_event, p0) {
-             if (!_dblClickZoomEnabled) return; // don't zoom if targeting something other than the map itself
+         var _combobox = uiCombobox(context, 'combo-' + field.safeid).caseSensitive(field.caseSensitive).minItems(_isMulti || _isSemi ? 1 : 2);
 
-             if (_typeof(d3_event.target.__data__) === 'object' && // or area fills
-             !select(d3_event.target).classed('fill')) return;
-             var zoomOut = d3_event.shiftKey;
-             var t = projection.transform();
-             var p1 = t.invert(p0);
-             t = t.scale(zoomOut ? 0.5 : 2);
-             t.x = p0[0] - p1[0] * t.k;
-             t.y = p0[1] - p1[1] * t.k;
-             map.transformEase(t);
-           });
+         var _container = select(null);
 
-           context.on('enter.map', function () {
-             if (!map.editableDataEnabled(true
-             /* skip zoom check */
-             )) return; // redraw immediately any objects affected by a change in selectedIDs.
+         var _inputWrap = select(null);
 
-             var graph = context.graph();
-             var selectedAndParents = {};
-             context.selectedIDs().forEach(function (id) {
-               var entity = graph.hasEntity(id);
+         var _input = select(null);
 
-               if (entity) {
-                 selectedAndParents[entity.id] = entity;
+         var _comboData = [];
+         var _multiData = [];
+         var _entityIDs = [];
 
-                 if (entity.type === 'node') {
-                   graph.parentWays(entity).forEach(function (parent) {
-                     selectedAndParents[parent.id] = parent;
-                   });
-                 }
-               }
-             });
-             var data = Object.values(selectedAndParents);
+         var _tags;
 
-             var filter = function filter(d) {
-               return d.id in selectedAndParents;
-             };
+         var _countryCode;
 
-             data = context.features().filter(data, graph);
-             surface.call(drawVertices.drawSelected, graph, map.extent()).call(drawLines, graph, data, filter).call(drawAreas, graph, data, filter).call(drawMidpoints, graph, data, filter, map.trimmedExtent());
-             dispatch$1.call('drawn', this, {
-               full: false
-             }); // redraw everything else later
+         var _staticPlaceholder; // initialize deprecated tags array
 
-             scheduleRedraw();
-           });
-           map.dimensions(utilGetDimensions(selection));
+
+         var _dataDeprecated = [];
+         _mainFileFetcher.get('deprecated').then(function (d) {
+           _dataDeprecated = d;
+         })["catch"](function () {
+           /* ignore */
+         }); // ensure multiCombo field.key ends with a ':'
+
+         if (_isMulti && field.key && /[^:]$/.test(field.key)) {
+           field.key += ':';
          }
 
-         function zoomEventFilter(d3_event) {
-           // Fix for #2151, (see also d3/d3-zoom#60, d3/d3-brush#18)
-           // Intercept `mousedown` and check if there is an orphaned zoom gesture.
-           // This can happen if a previous `mousedown` occurred without a `mouseup`.
-           // If we detect this, dispatch `mouseup` to complete the orphaned gesture,
-           // so that d3-zoom won't stop propagation of new `mousedown` events.
-           if (d3_event.type === 'mousedown') {
-             var hasOrphan = false;
-             var listeners = window.__on;
+         function snake(s) {
+           return s.replace(/\s+/g, '_').toLowerCase();
+         }
 
-             for (var i = 0; i < listeners.length; i++) {
-               var listener = listeners[i];
+         function clean(s) {
+           return s.split(';').map(function (s) {
+             return s.trim();
+           }).join(';');
+         } // returns the tag value for a display value
+         // (for multiCombo, dval should be the key suffix, not the entire key)
 
-               if (listener.name === 'zoom' && listener.type === 'mouseup') {
-                 hasOrphan = true;
-                 break;
-               }
-             }
 
-             if (hasOrphan) {
-               var event = window.CustomEvent;
+         function tagValue(dval) {
+           dval = clean(dval || '');
 
-               if (event) {
-                 event = new event('mouseup');
-               } else {
-                 event = window.document.createEvent('Event');
-                 event.initEvent('mouseup', false, false);
-               } // Event needs to be dispatched with an event.view property.
+           var found = _comboData.find(function (o) {
+             return o.key && clean(o.value) === dval;
+           });
 
+           if (found) return found.key;
 
-               event.view = window;
-               window.dispatchEvent(event);
-             }
+           if (field.type === 'typeCombo' && !dval) {
+             return 'yes';
            }
 
-           return d3_event.button !== 2; // ignore right clicks
-         }
+           return (_snake_case ? snake(dval) : dval) || undefined;
+         } // returns the display value for a tag value
+         // (for multiCombo, tval should be the key suffix, not the entire key)
 
-         function pxCenter() {
-           return [_dimensions[0] / 2, _dimensions[1] / 2];
-         }
 
-         function drawEditable(difference, extent) {
-           var mode = context.mode();
-           var graph = context.graph();
-           var features = context.features();
-           var all = context.history().intersects(map.extent());
-           var fullRedraw = false;
-           var data;
-           var set;
-           var filter;
-           var applyFeatureLayerFilters = true;
+         function displayValue(tval) {
+           tval = tval || '';
 
-           if (map.isInWideSelection()) {
-             data = [];
-             utilEntityAndDeepMemberIDs(mode.selectedIDs(), context.graph()).forEach(function (id) {
-               var entity = context.hasEntity(id);
-               if (entity) data.push(entity);
+           if (field.hasTextForStringId('options.' + tval)) {
+             return field.t('options.' + tval, {
+               "default": tval
              });
-             fullRedraw = true;
-             filter = utilFunctor(true); // selected features should always be visible, so we can skip filtering
+           }
 
-             applyFeatureLayerFilters = false;
-           } else if (difference) {
-             var complete = difference.complete(map.extent());
-             data = Object.values(complete).filter(Boolean);
-             set = new Set(Object.keys(complete));
+           if (field.type === 'typeCombo' && tval.toLowerCase() === 'yes') {
+             return '';
+           }
 
-             filter = function filter(d) {
-               return set.has(d.id);
-             };
+           return tval;
+         } // Compute the difference between arrays of objects by `value` property
+         //
+         // objectDifference([{value:1}, {value:2}, {value:3}], [{value:2}])
+         // > [{value:1}, {value:3}]
+         //
 
-             features.clear(data);
-           } else {
-             // force a full redraw if gatherStats detects that a feature
-             // should be auto-hidden (e.g. points or buildings)..
-             if (features.gatherStats(all, graph, _dimensions)) {
-               extent = undefined;
-             }
 
-             if (extent) {
-               data = context.history().intersects(map.extent().intersection(extent));
-               set = new Set(data.map(function (entity) {
-                 return entity.id;
-               }));
+         function objectDifference(a, b) {
+           return a.filter(function (d1) {
+             return !b.some(function (d2) {
+               return !d2.isMixed && d1.value === d2.value;
+             });
+           });
+         }
 
-               filter = function filter(d) {
-                 return set.has(d.id);
-               };
-             } else {
-               data = all;
-               fullRedraw = true;
-               filter = utilFunctor(true);
-             }
+         function initCombo(selection, attachTo) {
+           if (!_allowCustomValues) {
+             selection.attr('readonly', 'readonly');
            }
 
-           if (applyFeatureLayerFilters) {
-             data = features.filter(data, graph);
+           if (_showTagInfoSuggestions && services.taginfo) {
+             selection.call(_combobox.fetcher(setTaginfoValues), attachTo);
+             setTaginfoValues('', setPlaceholder);
            } else {
-             context.features().resetStats();
-           }
-
-           if (mode && mode.id === 'select') {
-             // update selected vertices - the user might have just double-clicked a way,
-             // creating a new vertex, triggering a partial redraw without a mode change
-             surface.call(drawVertices.drawSelected, graph, map.extent());
+             selection.call(_combobox, attachTo);
+             setStaticValues(setPlaceholder);
            }
+         }
 
-           surface.call(drawVertices, graph, data, filter, map.extent(), fullRedraw).call(drawLines, graph, data, filter).call(drawAreas, graph, data, filter).call(drawMidpoints, graph, data, filter, map.trimmedExtent()).call(drawLabels, graph, data, filter, _dimensions, fullRedraw).call(drawPoints, graph, data, filter);
-           dispatch$1.call('drawn', this, {
-             full: true
+         function setStaticValues(callback) {
+           if (!_optarray) return;
+           _comboData = _optarray.map(function (v) {
+             return {
+               key: v,
+               value: field.t('options.' + v, {
+                 "default": v
+               }),
+               title: v,
+               display: field.t.html('options.' + v, {
+                 "default": v
+               }),
+               klass: field.hasTextForStringId('options.' + v) ? '' : 'raw-option'
+             };
            });
+
+           _combobox.data(objectDifference(_comboData, _multiData));
+
+           if (callback) callback(_comboData);
          }
 
-         map.init = function () {
-           drawLayers = svgLayers(projection, context);
-           drawPoints = svgPoints(projection, context);
-           drawVertices = svgVertices(projection, context);
-           drawLines = svgLines(projection, context);
-           drawAreas = svgAreas(projection, context);
-           drawMidpoints = svgMidpoints(projection, context);
-           drawLabels = svgLabels(projection, context);
-         };
+         function setTaginfoValues(q, callback) {
+           var fn = _isMulti ? 'multikeys' : 'values';
+           var query = (_isMulti ? field.key : '') + q;
+           var hasCountryPrefix = _isNetwork && _countryCode && _countryCode.indexOf(q.toLowerCase()) === 0;
 
-         function editOff() {
-           context.features().resetStats();
-           surface.selectAll('.layer-osm *').remove();
-           surface.selectAll('.layer-touch:not(.markers) *').remove();
-           var allowed = {
-             'browse': true,
-             'save': true,
-             'select-note': true,
-             'select-data': true,
-             'select-error': true
+           if (hasCountryPrefix) {
+             query = _countryCode + ':';
+           }
+
+           var params = {
+             debounce: q !== '',
+             key: field.key,
+             query: query
            };
-           var mode = context.mode();
 
-           if (mode && !allowed[mode.id]) {
-             context.enter(modeBrowse(context));
+           if (_entityIDs.length) {
+             params.geometry = context.graph().geometry(_entityIDs[0]);
            }
 
-           dispatch$1.call('drawn', this, {
-             full: true
+           services.taginfo[fn](params, function (err, data) {
+             if (err) return;
+             data = data.filter(function (d) {
+               if (field.type === 'typeCombo' && d.value === 'yes') {
+                 // don't show the fallback value
+                 return false;
+               } // don't show values with very low usage
+
+
+               return !d.count || d.count > 10;
+             });
+             var deprecatedValues = osmEntity.deprecatedTagValuesByKey(_dataDeprecated)[field.key];
+
+             if (deprecatedValues) {
+               // don't suggest deprecated tag values
+               data = data.filter(function (d) {
+                 return deprecatedValues.indexOf(d.value) === -1;
+               });
+             }
+
+             if (hasCountryPrefix) {
+               data = data.filter(function (d) {
+                 return d.value.toLowerCase().indexOf(_countryCode + ':') === 0;
+               });
+             } // hide the caret if there are no suggestions
+
+
+             _container.classed('empty-combobox', data.length === 0);
+
+             _comboData = data.map(function (d) {
+               var k = d.value;
+               if (_isMulti) k = k.replace(field.key, '');
+               var label = field.t('options.' + k, {
+                 "default": k
+               });
+               return {
+                 key: k,
+                 value: label,
+                 display: field.t.html('options.' + k, {
+                   "default": k
+                 }),
+                 title: d.title || label,
+                 klass: field.hasTextForStringId('options.' + k) ? '' : 'raw-option'
+               };
+             });
+             _comboData = objectDifference(_comboData, _multiData);
+             if (callback) callback(_comboData);
            });
          }
 
-         function gestureChange(d3_event) {
-           // Remap Safari gesture events to wheel events - #5492
-           // We want these disabled most places, but enabled for zoom/unzoom on map surface
-           // https://developer.mozilla.org/en-US/docs/Web/API/GestureEvent
-           var e = d3_event;
-           e.preventDefault();
-           var props = {
-             deltaMode: 0,
-             // dummy values to ignore in zoomPan
-             deltaY: 1,
-             // dummy values to ignore in zoomPan
-             clientX: e.clientX,
-             clientY: e.clientY,
-             screenX: e.screenX,
-             screenY: e.screenY,
-             x: e.x,
-             y: e.y
-           };
-           var e2 = new WheelEvent('wheel', props);
-           e2._scale = e.scale; // preserve the original scale
+         function setPlaceholder(values) {
+           if (_isMulti || _isSemi) {
+             _staticPlaceholder = field.placeholder() || _t('inspector.add');
+           } else {
+             var vals = values.map(function (d) {
+               return d.value;
+             }).filter(function (s) {
+               return s.length < 20;
+             });
+             var placeholders = vals.length > 1 ? vals : values.map(function (d) {
+               return d.key;
+             });
+             _staticPlaceholder = field.placeholder() || placeholders.slice(0, 3).join(', ');
+           }
 
-           e2._rotation = e.rotation; // preserve the original rotation
+           if (!/(…|\.\.\.)$/.test(_staticPlaceholder)) {
+             _staticPlaceholder += '…';
+           }
 
-           _selection.node().dispatchEvent(e2);
-         }
+           var ph;
 
-         function zoomPan(event, key, transform) {
-           var source = event && event.sourceEvent || event;
-           var eventTransform = transform || event && event.transform;
-           var x = eventTransform.x;
-           var y = eventTransform.y;
-           var k = eventTransform.k; // Special handling of 'wheel' events:
-           // They might be triggered by the user scrolling the mouse wheel,
-           // or 2-finger pinch/zoom gestures, the transform may need adjustment.
+           if (!_isMulti && !_isSemi && _tags && Array.isArray(_tags[field.key])) {
+             ph = _t('inspector.multiple_values');
+           } else {
+             ph = _staticPlaceholder;
+           }
 
-           if (source && source.type === 'wheel') {
-             // assume that the gesture is already handled by pointer events
-             if (_pointerDown) return;
-             var detected = utilDetect();
-             var dX = source.deltaX;
-             var dY = source.deltaY;
-             var x2 = x;
-             var y2 = y;
-             var k2 = k;
-             var t0, p0, p1; // Normalize mousewheel scroll speed (Firefox) - #3029
-             // If wheel delta is provided in LINE units, recalculate it in PIXEL units
-             // We are essentially redoing the calculations that occur here:
-             //   https://github.com/d3/d3-zoom/blob/78563a8348aa4133b07cac92e2595c2227ca7cd7/src/zoom.js#L203
-             // See this for more info:
-             //   https://github.com/basilfx/normalize-wheel/blob/master/src/normalizeWheel.js
+           _container.selectAll('input').attr('placeholder', ph);
+         }
 
-             if (source.deltaMode === 1
-             /* LINE */
-             ) {
-                 // Convert from lines to pixels, more if the user is scrolling fast.
-                 // (I made up the exp function to roughly match Firefox to what Chrome does)
-                 // These numbers should be floats, because integers are treated as pan gesture below.
-                 var lines = Math.abs(source.deltaY);
-                 var sign = source.deltaY > 0 ? 1 : -1;
-                 dY = sign * clamp(Math.exp((lines - 1) * 0.75) * 4.000244140625, 4.000244140625, // min
-                 350.000244140625 // max
-                 ); // On Firefox Windows and Linux we always get +/- the scroll line amount (default 3)
-                 // There doesn't seem to be any scroll acceleration.
-                 // This multiplier increases the speed a little bit - #5512
+         function change() {
+           var t = {};
+           var val;
 
-                 if (detected.os !== 'mac') {
-                   dY *= 5;
-                 } // recalculate x2,y2,k2
+           if (_isMulti || _isSemi) {
+             val = tagValue(utilGetSetValue(_input).replace(/,/g, ';')) || '';
 
+             _container.classed('active', false);
 
-                 t0 = _isTransformed ? _transformLast : _transformStart;
-                 p0 = _getMouseCoords(source);
-                 p1 = t0.invert(p0);
-                 k2 = t0.k * Math.pow(2, -dY / 500);
-                 k2 = clamp(k2, kMin, kMax);
-                 x2 = p0[0] - p1[0] * k2;
-                 y2 = p0[1] - p1[1] * k2; // 2 finger map pinch zooming (Safari) - #5492
-                 // These are fake `wheel` events we made from Safari `gesturechange` events..
-               } else if (source._scale) {
-               // recalculate x2,y2,k2
-               t0 = _gestureTransformStart;
-               p0 = _getMouseCoords(source);
-               p1 = t0.invert(p0);
-               k2 = t0.k * source._scale;
-               k2 = clamp(k2, kMin, kMax);
-               x2 = p0[0] - p1[0] * k2;
-               y2 = p0[1] - p1[1] * k2; // 2 finger map pinch zooming (all browsers except Safari) - #5492
-               // Pinch zooming via the `wheel` event will always have:
-               // - `ctrlKey = true`
-               // - `deltaY` is not round integer pixels (ignore `deltaX`)
-             } else if (source.ctrlKey && !isInteger(dY)) {
-               dY *= 6; // slightly scale up whatever the browser gave us
-               // recalculate x2,y2,k2
+             utilGetSetValue(_input, '');
+             var vals = val.split(';').filter(Boolean);
+             if (!vals.length) return;
 
-               t0 = _isTransformed ? _transformLast : _transformStart;
-               p0 = _getMouseCoords(source);
-               p1 = t0.invert(p0);
-               k2 = t0.k * Math.pow(2, -dY / 500);
-               k2 = clamp(k2, kMin, kMax);
-               x2 = p0[0] - p1[0] * k2;
-               y2 = p0[1] - p1[1] * k2; // Trackpad scroll zooming with shift or alt/option key down
-             } else if ((source.altKey || source.shiftKey) && isInteger(dY)) {
-               // recalculate x2,y2,k2
-               t0 = _isTransformed ? _transformLast : _transformStart;
-               p0 = _getMouseCoords(source);
-               p1 = t0.invert(p0);
-               k2 = t0.k * Math.pow(2, -dY / 500);
-               k2 = clamp(k2, kMin, kMax);
-               x2 = p0[0] - p1[0] * k2;
-               y2 = p0[1] - p1[1] * k2; // 2 finger map panning (Mac only, all browsers) - #5492, #5512
-               // Panning via the `wheel` event will always have:
-               // - `ctrlKey = false`
-               // - `deltaX`,`deltaY` are round integer pixels
-             } else if (detected.os === 'mac' && !source.ctrlKey && isInteger(dX) && isInteger(dY)) {
-               p1 = projection.translate();
-               x2 = p1[0] - dX;
-               y2 = p1[1] - dY;
-               k2 = projection.scale();
-               k2 = clamp(k2, kMin, kMax);
-             } // something changed - replace the event transform
+             if (_isMulti) {
+               utilArrayUniq(vals).forEach(function (v) {
+                 var key = (field.key || '') + v;
 
+                 if (_tags) {
+                   // don't set a multicombo value to 'yes' if it already has a non-'no' value
+                   // e.g. `language:de=main`
+                   var old = _tags[key];
+                   if (typeof old === 'string' && old.toLowerCase() !== 'no') return;
+                 }
 
-             if (x2 !== x || y2 !== y || k2 !== k) {
-               x = x2;
-               y = y2;
-               k = k2;
-               eventTransform = identity$2.translate(x2, y2).scale(k2);
+                 key = context.cleanTagKey(key);
+                 field.keys.push(key);
+                 t[key] = 'yes';
+               });
+             } else if (_isSemi) {
+               var arr = _multiData.map(function (d) {
+                 return d.key;
+               });
 
-               if (_zoomerPanner._transform) {
-                 // utilZoomPan interface
-                 _zoomerPanner._transform(eventTransform);
-               } else {
-                 // d3_zoom interface
-                 _selection.node().__zoom = eventTransform;
-               }
+               arr = arr.concat(vals);
+               t[field.key] = context.cleanTagValue(utilArrayUniq(arr).filter(Boolean).join(';'));
              }
+
+             window.setTimeout(function () {
+               _input.node().focus();
+             }, 10);
+           } else {
+             var rawValue = utilGetSetValue(_input); // don't override multiple values with blank string
+
+             if (!rawValue && Array.isArray(_tags[field.key])) return;
+             val = context.cleanTagValue(tagValue(rawValue));
+             t[field.key] = val || undefined;
+           }
+
+           dispatch.call('change', this, t);
+         }
+
+         function removeMultikey(d3_event, d) {
+           d3_event.preventDefault();
+           d3_event.stopPropagation();
+           var t = {};
+
+           if (_isMulti) {
+             t[d.key] = undefined;
+           } else if (_isSemi) {
+             var arr = _multiData.map(function (md) {
+               return md.key === d.key ? null : md.key;
+             }).filter(Boolean);
+
+             arr = utilArrayUniq(arr);
+             t[field.key] = arr.length ? arr.join(';') : undefined;
            }
 
-           if (_transformStart.x === x && _transformStart.y === y && _transformStart.k === k) {
-             return; // no change
-           }
+           dispatch.call('change', this, t);
+         }
 
-           var withinEditableZoom = map.withinEditableZoom();
+         function combo(selection) {
+           _container = selection.selectAll('.form-field-input-wrap').data([0]);
+           var type = _isMulti || _isSemi ? 'multicombo' : 'combo';
+           _container = _container.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + type).merge(_container);
 
-           if (_lastWithinEditableZoom !== withinEditableZoom) {
-             if (_lastWithinEditableZoom !== undefined) {
-               // notify that the map zoomed in or out over the editable zoom threshold
-               dispatch$1.call('crossEditableZoom', this, withinEditableZoom);
+           if (_isMulti || _isSemi) {
+             _container = _container.selectAll('.chiplist').data([0]);
+             var listClass = 'chiplist'; // Use a separate line for each value in the Destinations and Via fields
+             // to mimic highway exit signs
+
+             if (field.key === 'destination' || field.key === 'via') {
+               listClass += ' full-line-chips';
              }
 
-             _lastWithinEditableZoom = withinEditableZoom;
+             _container = _container.enter().append('ul').attr('class', listClass).on('click', function () {
+               window.setTimeout(function () {
+                 _input.node().focus();
+               }, 10);
+             }).merge(_container);
+             _inputWrap = _container.selectAll('.input-wrap').data([0]);
+             _inputWrap = _inputWrap.enter().append('li').attr('class', 'input-wrap').merge(_inputWrap);
+             _input = _inputWrap.selectAll('input').data([0]);
+           } else {
+             _input = _container.selectAll('input').data([0]);
            }
 
-           if (geoScaleToZoom(k, TILESIZE) < _minzoom) {
-             surface.interrupt();
-             dispatch$1.call('hitMinZoom', this, map);
-             setCenterZoom(map.center(), context.minEditableZoom(), 0, true);
-             scheduleRedraw();
-             dispatch$1.call('move', this, map);
-             return;
+           _input = _input.enter().append('input').attr('type', 'text').attr('id', field.domId).call(utilNoAuto).call(initCombo, selection).merge(_input);
+
+           if (_isNetwork) {
+             var extent = combinedEntityExtent();
+             var countryCode = extent && iso1A2Code(extent.center());
+             _countryCode = countryCode && countryCode.toLowerCase();
            }
 
-           projection.transform(eventTransform);
-           var scale = k / _transformStart.k;
-           var tX = (x / scale - _transformStart.x) * scale;
-           var tY = (y / scale - _transformStart.y) * scale;
+           _input.on('change', change).on('blur', change);
 
-           if (context.inIntro()) {
-             curtainProjection.transform({
-               x: x - tX,
-               y: y - tY,
-               k: k
-             });
-           }
+           _input.on('keydown.field', function (d3_event) {
+             switch (d3_event.keyCode) {
+               case 13:
+                 // ↩ Return
+                 _input.node().blur(); // blurring also enters the value
 
-           if (source) {
-             _lastPointerEvent = event;
-           }
 
-           _isTransformed = true;
-           _transformLast = eventTransform;
-           utilSetTransform(supersurface, tX, tY, scale);
-           scheduleRedraw();
-           dispatch$1.call('move', this, map);
+                 d3_event.stopPropagation();
+                 break;
+             }
+           });
 
-           function isInteger(val) {
-             return typeof val === 'number' && isFinite(val) && Math.floor(val) === val;
+           if (_isMulti || _isSemi) {
+             _combobox.on('accept', function () {
+               _input.node().blur();
+
+               _input.node().focus();
+             });
+
+             _input.on('focus', function () {
+               _container.classed('active', true);
+             });
            }
          }
 
-         function resetTransform() {
-           if (!_isTransformed) return false;
-           utilSetTransform(supersurface, 0, 0);
-           _isTransformed = false;
+         combo.tags = function (tags) {
+           _tags = tags;
 
-           if (context.inIntro()) {
-             curtainProjection.transform(projection.transform());
-           }
+           if (_isMulti || _isSemi) {
+             _multiData = [];
+             var maxLength;
 
-           return true;
-         }
+             if (_isMulti) {
+               // Build _multiData array containing keys already set..
+               for (var k in tags) {
+                 if (field.key && k.indexOf(field.key) !== 0) continue;
+                 if (!field.key && field.keys.indexOf(k) === -1) continue;
+                 var v = tags[k];
+                 if (!v || typeof v === 'string' && v.toLowerCase() === 'no') continue;
+                 var suffix = field.key ? k.substr(field.key.length) : k;
 
-         function redraw(difference, extent) {
-           if (surface.empty() || !_redrawEnabled) return; // If we are in the middle of a zoom/pan, we can't do differenced redraws.
-           // It would result in artifacts where differenced entities are redrawn with
-           // one transform and unchanged entities with another.
+                 _multiData.push({
+                   key: k,
+                   value: displayValue(suffix),
+                   isMixed: Array.isArray(v)
+                 });
+               }
 
-           if (resetTransform()) {
-             difference = extent = undefined;
-           }
+               if (field.key) {
+                 // Set keys for form-field modified (needed for undo and reset buttons)..
+                 field.keys = _multiData.map(function (d) {
+                   return d.key;
+                 }); // limit the input length so it fits after prepending the key prefix
 
-           var zoom = map.zoom();
-           var z = String(~~zoom);
+                 maxLength = context.maxCharsForTagKey() - utilUnicodeCharsCount(field.key);
+               } else {
+                 maxLength = context.maxCharsForTagKey();
+               }
+             } else if (_isSemi) {
+               var allValues = [];
+               var commonValues;
 
-           if (surface.attr('data-zoom') !== z) {
-             surface.attr('data-zoom', z);
-           } // class surface as `lowzoom` around z17-z18.5 (based on latitude)
+               if (Array.isArray(tags[field.key])) {
+                 tags[field.key].forEach(function (tagVal) {
+                   var thisVals = utilArrayUniq((tagVal || '').split(';')).filter(Boolean);
+                   allValues = allValues.concat(thisVals);
 
+                   if (!commonValues) {
+                     commonValues = thisVals;
+                   } else {
+                     commonValues = commonValues.filter(function (value) {
+                       return thisVals.includes(value);
+                     });
+                   }
+                 });
+                 allValues = utilArrayUniq(allValues).filter(Boolean);
+               } else {
+                 allValues = utilArrayUniq((tags[field.key] || '').split(';')).filter(Boolean);
+                 commonValues = allValues;
+               }
 
-           var lat = map.center()[1];
-           var lowzoom = linear$2().domain([-60, 0, 60]).range([17, 18.5, 17]).clamp(true);
-           surface.classed('low-zoom', zoom <= lowzoom(lat));
+               _multiData = allValues.map(function (v) {
+                 return {
+                   key: v,
+                   value: displayValue(v),
+                   isMixed: !commonValues.includes(v)
+                 };
+               });
+               var currLength = utilUnicodeCharsCount(commonValues.join(';')); // limit the input length to the remaining available characters
 
-           if (!difference) {
-             supersurface.call(context.background());
-             wrapper.call(drawLayers);
-           } // OSM
+               maxLength = context.maxCharsForTagValue() - currLength;
 
+               if (currLength > 0) {
+                 // account for the separator if a new value will be appended to existing
+                 maxLength -= 1;
+               }
+             } // a negative maxlength doesn't make sense
 
-           if (map.editableDataEnabled() || map.isInWideSelection()) {
-             context.loadTiles(projection);
-             drawEditable(difference, extent);
-           } else {
-             editOff();
-           }
 
-           _transformStart = projection.transform();
-           return map;
-         }
+             maxLength = Math.max(0, maxLength);
+             var allowDragAndDrop = _isSemi // only semiCombo values are ordered
+             && !Array.isArray(tags[field.key]); // Exclude existing multikeys from combo options..
 
-         var immediateRedraw = function immediateRedraw(difference, extent) {
-           if (!difference && !extent) cancelPendingRedraw();
-           redraw(difference, extent);
-         };
+             var available = objectDifference(_comboData, _multiData);
 
-         map.lastPointerEvent = function () {
-           return _lastPointerEvent;
-         };
+             _combobox.data(available); // Hide 'Add' button if this field uses fixed set of
+             // options and they're all currently used,
+             // or if the field is already at its character limit
 
-         map.mouse = function (d3_event) {
-           var event = _lastPointerEvent || d3_event;
 
-           if (event) {
-             var s;
+             var hideAdd = !_allowCustomValues && !available.length || maxLength <= 0;
 
-             while (s = event.sourceEvent) {
-               event = s;
+             _container.selectAll('.chiplist .input-wrap').style('display', hideAdd ? 'none' : null); // Render chips
+
+
+             var chips = _container.selectAll('.chip').data(_multiData);
+
+             chips.exit().remove();
+             var enter = chips.enter().insert('li', '.input-wrap').attr('class', 'chip');
+             enter.append('span');
+             enter.append('a');
+             chips = chips.merge(enter).order().classed('raw-value', function (d) {
+               var k = d.key;
+               if (_isMulti) k = k.replace(field.key, '');
+               return !field.hasTextForStringId('options.' + k);
+             }).classed('draggable', allowDragAndDrop).classed('mixed', function (d) {
+               return d.isMixed;
+             }).attr('title', function (d) {
+               return d.isMixed ? _t('inspector.unshared_value_tooltip') : null;
+             });
+
+             if (allowDragAndDrop) {
+               registerDragAndDrop(chips);
              }
 
-             return _getMouseCoords(event);
+             chips.select('span').html(function (d) {
+               return d.value;
+             });
+             chips.select('a').attr('href', '#').on('click', removeMultikey).attr('class', 'remove').html('×');
+           } else {
+             var isMixed = Array.isArray(tags[field.key]);
+             var mixedValues = isMixed && tags[field.key].map(function (val) {
+               return displayValue(val);
+             }).filter(Boolean);
+             var showsValue = !isMixed && tags[field.key] && !(field.type === 'typeCombo' && tags[field.key] === 'yes');
+             var isRawValue = showsValue && !field.hasTextForStringId('options.' + tags[field.key]);
+             var isKnownValue = showsValue && !isRawValue;
+             var isReadOnly = !_allowCustomValues || isKnownValue;
+             utilGetSetValue(_input, !isMixed ? displayValue(tags[field.key]) : '').classed('raw-value', isRawValue).classed('known-value', isKnownValue).attr('readonly', isReadOnly ? 'readonly' : undefined).attr('title', isMixed ? mixedValues.join('\n') : undefined).attr('placeholder', isMixed ? _t('inspector.multiple_values') : _staticPlaceholder || '').classed('mixed', isMixed).on('keydown.deleteCapture', function (d3_event) {
+               if (isReadOnly && isKnownValue && (d3_event.keyCode === utilKeybinding.keyCodes['⌫'] || d3_event.keyCode === utilKeybinding.keyCodes['⌦'])) {
+                 d3_event.preventDefault();
+                 d3_event.stopPropagation();
+                 var t = {};
+                 t[field.key] = undefined;
+                 dispatch.call('change', this, t);
+               }
+             });
            }
+         };
 
-           return null;
-         }; // returns Lng/Lat
+         function registerDragAndDrop(selection) {
+           // allow drag and drop re-ordering of chips
+           var dragOrigin, targetIndex;
+           selection.call(d3_drag().on('start', function (d3_event) {
+             dragOrigin = {
+               x: d3_event.x,
+               y: d3_event.y
+             };
+             targetIndex = null;
+           }).on('drag', function (d3_event) {
+             var x = d3_event.x - dragOrigin.x,
+                 y = d3_event.y - dragOrigin.y;
+             if (!select(this).classed('dragging') && // don't display drag until dragging beyond a distance threshold
+             Math.sqrt(Math.pow(x, 2) + Math.pow(y, 2)) <= 5) return;
+             var index = selection.nodes().indexOf(this);
+             select(this).classed('dragging', true);
+             targetIndex = null;
+             var targetIndexOffsetTop = null;
+             var draggedTagWidth = select(this).node().offsetWidth;
 
+             if (field.key === 'destination' || field.key === 'via') {
+               // meaning tags are full width
+               _container.selectAll('.chip').style('transform', function (d2, index2) {
+                 var node = select(this).node();
 
-         map.mouseCoordinates = function () {
-           var coord = map.mouse() || pxCenter();
-           return projection.invert(coord);
-         };
+                 if (index === index2) {
+                   return 'translate(' + x + 'px, ' + y + 'px)'; // move the dragged tag up the order
+                 } else if (index2 > index && d3_event.y > node.offsetTop) {
+                   if (targetIndex === null || index2 > targetIndex) {
+                     targetIndex = index2;
+                   }
 
-         map.dblclickZoomEnable = function (val) {
-           if (!arguments.length) return _dblClickZoomEnabled;
-           _dblClickZoomEnabled = val;
-           return map;
-         };
+                   return 'translateY(-100%)'; // move the dragged tag down the order
+                 } else if (index2 < index && d3_event.y < node.offsetTop + node.offsetHeight) {
+                   if (targetIndex === null || index2 < targetIndex) {
+                     targetIndex = index2;
+                   }
 
-         map.redrawEnable = function (val) {
-           if (!arguments.length) return _redrawEnabled;
-           _redrawEnabled = val;
-           return map;
-         };
+                   return 'translateY(100%)';
+                 }
 
-         map.isTransformed = function () {
-           return _isTransformed;
-         };
+                 return null;
+               });
+             } else {
+               _container.selectAll('.chip').each(function (d2, index2) {
+                 var node = select(this).node(); // check the cursor is in the bounding box
 
-         function setTransform(t2, duration, force) {
-           var t = projection.transform();
-           if (!force && t2.k === t.k && t2.x === t.x && t2.y === t.y) return false;
+                 if (index !== index2 && d3_event.x < node.offsetLeft + node.offsetWidth + 5 && d3_event.x > node.offsetLeft && d3_event.y < node.offsetTop + node.offsetHeight && d3_event.y > node.offsetTop) {
+                   targetIndex = index2;
+                   targetIndexOffsetTop = node.offsetTop;
+                 }
+               }).style('transform', function (d2, index2) {
+                 var node = select(this).node();
 
-           if (duration) {
-             _selection.transition().duration(duration).on('start', function () {
-               map.startEase();
-             }).call(_zoomerPanner.transform, identity$2.translate(t2.x, t2.y).scale(t2.k));
-           } else {
-             projection.transform(t2);
-             _transformStart = t2;
+                 if (index === index2) {
+                   return 'translate(' + x + 'px, ' + y + 'px)';
+                 } // only translate tags in the same row
 
-             _selection.call(_zoomerPanner.transform, _transformStart);
-           }
 
-           return true;
-         }
+                 if (node.offsetTop === targetIndexOffsetTop) {
+                   if (index2 < index && index2 >= targetIndex) {
+                     return 'translateX(' + draggedTagWidth + 'px)';
+                   } else if (index2 > index && index2 <= targetIndex) {
+                     return 'translateX(-' + draggedTagWidth + 'px)';
+                   }
+                 }
 
-         function setCenterZoom(loc2, z2, duration, force) {
-           var c = map.center();
-           var z = map.zoom();
-           if (loc2[0] === c[0] && loc2[1] === c[1] && z2 === z && !force) return false;
-           var proj = geoRawMercator().transform(projection.transform()); // copy projection
+                 return null;
+               });
+             }
+           }).on('end', function () {
+             if (!select(this).classed('dragging')) {
+               return;
+             }
 
-           var k2 = clamp(geoZoomToScale(z2, TILESIZE), kMin, kMax);
-           proj.scale(k2);
-           var t = proj.translate();
-           var point = proj(loc2);
-           var center = pxCenter();
-           t[0] += center[0] - point[0];
-           t[1] += center[1] - point[1];
-           return setTransform(identity$2.translate(t[0], t[1]).scale(k2), duration, force);
-         }
+             var index = selection.nodes().indexOf(this);
+             select(this).classed('dragging', false);
 
-         map.pan = function (delta, duration) {
-           var t = projection.translate();
-           var k = projection.scale();
-           t[0] += delta[0];
-           t[1] += delta[1];
+             _container.selectAll('.chip').style('transform', null);
 
-           if (duration) {
-             _selection.transition().duration(duration).on('start', function () {
-               map.startEase();
-             }).call(_zoomerPanner.transform, identity$2.translate(t[0], t[1]).scale(k));
-           } else {
-             projection.translate(t);
-             _transformStart = projection.transform();
+             if (typeof targetIndex === 'number') {
+               var element = _multiData[index];
 
-             _selection.call(_zoomerPanner.transform, _transformStart);
+               _multiData.splice(index, 1);
 
-             dispatch$1.call('move', this, map);
-             immediateRedraw();
-           }
+               _multiData.splice(targetIndex, 0, element);
 
-           return map;
-         };
+               var t = {};
 
-         map.dimensions = function (val) {
-           if (!arguments.length) return _dimensions;
-           _dimensions = val;
-           drawLayers.dimensions(_dimensions);
-           context.background().dimensions(_dimensions);
-           projection.clipExtent([[0, 0], _dimensions]);
-           _getMouseCoords = utilFastMouse(supersurface.node());
-           scheduleRedraw();
-           return map;
-         };
+               if (_multiData.length) {
+                 t[field.key] = _multiData.map(function (element) {
+                   return element.key;
+                 }).join(';');
+               } else {
+                 t[field.key] = undefined;
+               }
 
-         function zoomIn(delta) {
-           setCenterZoom(map.center(), ~~map.zoom() + delta, 250, true);
-         }
+               dispatch.call('change', this, t);
+             }
 
-         function zoomOut(delta) {
-           setCenterZoom(map.center(), ~~map.zoom() - delta, 250, true);
+             dragOrigin = undefined;
+             targetIndex = undefined;
+           }));
          }
 
-         map.zoomIn = function () {
-           zoomIn(1);
+         combo.focus = function () {
+           _input.node().focus();
          };
 
-         map.zoomInFurther = function () {
-           zoomIn(4);
+         combo.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs;
+           _entityIDs = val;
+           return combo;
          };
 
-         map.canZoomIn = function () {
-           return map.zoom() < maxZoom;
-         };
+         function combinedEntityExtent() {
+           return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+         }
 
-         map.zoomOut = function () {
-           zoomOut(1);
-         };
+         return utilRebind(combo, dispatch, 'on');
+       }
 
-         map.zoomOutFurther = function () {
-           zoomOut(4);
-         };
+       function uiFieldText(field, context) {
+         var dispatch = dispatch$8('change');
+         var input = select(null);
+         var outlinkButton = select(null);
+         var _entityIDs = [];
 
-         map.canZoomOut = function () {
-           return map.zoom() > minZoom;
-         };
+         var _tags;
 
-         map.center = function (loc2) {
-           if (!arguments.length) {
-             return projection.invert(pxCenter());
-           }
+         var _phoneFormats = {};
 
-           if (setCenterZoom(loc2, map.zoom())) {
-             dispatch$1.call('move', this, map);
-           }
+         if (field.type === 'tel') {
+           _mainFileFetcher.get('phone_formats').then(function (d) {
+             _phoneFormats = d;
+             updatePhonePlaceholder();
+           })["catch"](function () {
+             /* ignore */
+           });
+         }
 
-           scheduleRedraw();
-           return map;
-         };
+         function calcLocked() {
+           // Protect certain fields that have a companion `*:wikidata` value
+           var isLocked = (field.id === 'brand' || field.id === 'network' || field.id === 'operator' || field.id === 'flag') && _entityIDs.length && _entityIDs.some(function (entityID) {
+             var entity = context.graph().hasEntity(entityID);
+             if (!entity) return false; // Features linked to Wikidata are likely important and should be protected
 
-         map.unobscuredCenterZoomEase = function (loc, zoom) {
-           var offset = map.unobscuredOffsetPx();
-           var proj = geoRawMercator().transform(projection.transform()); // copy projection
-           // use the target zoom to calculate the offset center
+             if (entity.tags.wikidata) return true;
+             var preset = _mainPresetIndex.match(entity, context.graph());
+             var isSuggestion = preset && preset.suggestion; // Lock the field if there is a value and a companion `*:wikidata` value
 
-           proj.scale(geoZoomToScale(zoom, TILESIZE));
-           var locPx = proj(loc);
-           var offsetLocPx = [locPx[0] + offset[0], locPx[1] + offset[1]];
-           var offsetLoc = proj.invert(offsetLocPx);
-           map.centerZoomEase(offsetLoc, zoom);
-         };
+             var which = field.id; // 'brand', 'network', 'operator', 'flag'
 
-         map.unobscuredOffsetPx = function () {
-           var openPane = context.container().select('.map-panes .map-pane.shown');
+             return isSuggestion && !!entity.tags[which] && !!entity.tags[which + ':wikidata'];
+           });
 
-           if (!openPane.empty()) {
-             return [openPane.node().offsetWidth / 2, 0];
-           }
+           field.locked(isLocked);
+         }
 
-           return [0, 0];
-         };
+         function i(selection) {
+           calcLocked();
+           var isLocked = field.locked();
+           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
+           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
+           input = wrap.selectAll('input').data([0]);
+           input = input.enter().append('input').attr('type', field.type === 'identifier' || field.type === 'roadheight' ? 'text' : field.type).attr('id', field.domId).classed(field.type, true).call(utilNoAuto).merge(input);
+           input.classed('disabled', !!isLocked).attr('readonly', isLocked || null).on('input', change(true)).on('blur', change()).on('change', change());
 
-         map.zoom = function (z2) {
-           if (!arguments.length) {
-             return Math.max(geoScaleToZoom(projection.scale(), TILESIZE), 0);
+           if (field.type === 'tel') {
+             updatePhonePlaceholder();
+           } else if (field.type === 'number') {
+             var rtl = _mainLocalizer.textDirection() === 'rtl';
+             input.attr('type', 'text');
+             var inc = field.increment;
+             var buttons = wrap.selectAll('.increment, .decrement').data(rtl ? [inc, -inc] : [-inc, inc]);
+             buttons.enter().append('button').attr('class', function (d) {
+               var which = d > 0 ? 'increment' : 'decrement';
+               return 'form-field-button ' + which;
+             }).merge(buttons).on('click', function (d3_event, d) {
+               d3_event.preventDefault();
+               var raw_vals = input.node().value || '0';
+               var vals = raw_vals.split(';');
+               vals = vals.map(function (v) {
+                 var num = parseFloat(v.trim(), 10);
+                 return isFinite(num) ? clamped(num + d) : v.trim();
+               });
+               input.node().value = vals.join(';');
+               change()();
+             });
+           } else if (field.type === 'identifier' && field.urlFormat && field.pattern) {
+             input.attr('type', 'text');
+             outlinkButton = wrap.selectAll('.foreign-id-permalink').data([0]);
+             outlinkButton.enter().append('button').call(svgIcon('#iD-icon-out-link')).attr('class', 'form-field-button foreign-id-permalink').attr('title', function () {
+               var domainResults = /^https?:\/\/(.{1,}?)\//.exec(field.urlFormat);
+
+               if (domainResults.length >= 2 && domainResults[1]) {
+                 var domain = domainResults[1];
+                 return _t('icons.view_on', {
+                   domain: domain
+                 });
+               }
+
+               return '';
+             }).on('click', function (d3_event) {
+               d3_event.preventDefault();
+               var value = validIdentifierValueForLink();
+
+               if (value) {
+                 var url = field.urlFormat.replace(/{value}/, encodeURIComponent(value));
+                 window.open(url, '_blank');
+               }
+             }).merge(outlinkButton);
            }
+         }
 
-           if (z2 < _minzoom) {
-             surface.interrupt();
-             dispatch$1.call('hitMinZoom', this, map);
-             z2 = context.minEditableZoom();
-           }
+         function updatePhonePlaceholder() {
+           if (input.empty() || !Object.keys(_phoneFormats).length) return;
+           var extent = combinedEntityExtent();
+           var countryCode = extent && iso1A2Code(extent.center());
 
-           if (setCenterZoom(map.center(), z2)) {
-             dispatch$1.call('move', this, map);
-           }
+           var format = countryCode && _phoneFormats[countryCode.toLowerCase()];
 
-           scheduleRedraw();
-           return map;
-         };
+           if (format) input.attr('placeholder', format);
+         }
 
-         map.centerZoom = function (loc2, z2) {
-           if (setCenterZoom(loc2, z2)) {
-             dispatch$1.call('move', this, map);
+         function validIdentifierValueForLink() {
+           if (field.type === 'identifier' && field.pattern) {
+             var value = utilGetSetValue(input).trim().split(';')[0];
+             return value && value.match(new RegExp(field.pattern));
            }
 
-           scheduleRedraw();
-           return map;
-         };
-
-         map.zoomTo = function (entity) {
-           var extent = entity.extent(context.graph());
-           if (!isFinite(extent.area())) return map;
-           var z2 = clamp(map.trimmedExtentZoom(extent), 0, 20);
-           return map.centerZoom(extent.center(), z2);
-         };
+           return null;
+         } // clamp number to min/max
 
-         map.centerEase = function (loc2, duration) {
-           duration = duration || 250;
-           setCenterZoom(loc2, map.zoom(), duration);
-           return map;
-         };
 
-         map.zoomEase = function (z2, duration) {
-           duration = duration || 250;
-           setCenterZoom(map.center(), z2, duration, false);
-           return map;
-         };
+         function clamped(num) {
+           if (field.minValue !== undefined) {
+             num = Math.max(num, field.minValue);
+           }
 
-         map.centerZoomEase = function (loc2, z2, duration) {
-           duration = duration || 250;
-           setCenterZoom(loc2, z2, duration, false);
-           return map;
-         };
+           if (field.maxValue !== undefined) {
+             num = Math.min(num, field.maxValue);
+           }
 
-         map.transformEase = function (t2, duration) {
-           duration = duration || 250;
-           setTransform(t2, duration, false
-           /* don't force */
-           );
-           return map;
-         };
+           return num;
+         }
 
-         map.zoomToEase = function (obj, duration) {
-           var extent;
+         function change(onInput) {
+           return function () {
+             var t = {};
+             var val = utilGetSetValue(input);
+             if (!onInput) val = context.cleanTagValue(val); // don't override multiple values with blank string
 
-           if (Array.isArray(obj)) {
-             obj.forEach(function (entity) {
-               var entityExtent = entity.extent(context.graph());
+             if (!val && Array.isArray(_tags[field.key])) return;
 
-               if (!extent) {
-                 extent = entityExtent;
-               } else {
-                 extent = extent.extend(entityExtent);
+             if (!onInput) {
+               if (field.type === 'number' && val) {
+                 var vals = val.split(';');
+                 vals = vals.map(function (v) {
+                   var num = parseFloat(v.trim(), 10);
+                   return isFinite(num) ? clamped(num) : v.trim();
+                 });
+                 val = vals.join(';');
                }
-             });
-           } else {
-             extent = obj.extent(context.graph());
-           }
 
-           if (!isFinite(extent.area())) return map;
-           var z2 = clamp(map.trimmedExtentZoom(extent), 0, 20);
-           return map.centerZoomEase(extent.center(), z2, duration);
-         };
-
-         map.startEase = function () {
-           utilBindOnce(surface, _pointerPrefix + 'down.ease', function () {
-             map.cancelEase();
-           });
-           return map;
-         };
+               utilGetSetValue(input, val);
+             }
 
-         map.cancelEase = function () {
-           _selection.interrupt();
+             t[field.key] = val || undefined;
+             dispatch.call('change', this, t, onInput);
+           };
+         }
 
-           return map;
+         i.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs;
+           _entityIDs = val;
+           return i;
          };
 
-         map.extent = function (val) {
-           if (!arguments.length) {
-             return new geoExtent(projection.invert([0, _dimensions[1]]), projection.invert([_dimensions[0], 0]));
-           } else {
-             var extent = geoExtent(val);
-             map.centerZoom(extent.center(), map.extentZoom(extent));
-           }
-         };
+         i.tags = function (tags) {
+           _tags = tags;
+           var isMixed = Array.isArray(tags[field.key]);
+           utilGetSetValue(input, !isMixed && tags[field.key] ? tags[field.key] : '').attr('title', isMixed ? tags[field.key].filter(Boolean).join('\n') : undefined).attr('placeholder', isMixed ? _t('inspector.multiple_values') : field.placeholder() || _t('inspector.unknown')).classed('mixed', isMixed);
 
-         map.trimmedExtent = function (val) {
-           if (!arguments.length) {
-             var headerY = 71;
-             var footerY = 30;
-             var pad = 10;
-             return new geoExtent(projection.invert([pad, _dimensions[1] - footerY - pad]), projection.invert([_dimensions[0] - pad, headerY + pad]));
-           } else {
-             var extent = geoExtent(val);
-             map.centerZoom(extent.center(), map.trimmedExtentZoom(extent));
+           if (outlinkButton && !outlinkButton.empty()) {
+             var disabled = !validIdentifierValueForLink();
+             outlinkButton.classed('disabled', disabled);
            }
          };
 
-         function calcExtentZoom(extent, dim) {
-           var tl = projection([extent[0][0], extent[1][1]]);
-           var br = projection([extent[1][0], extent[0][1]]); // Calculate maximum zoom that fits extent
+         i.focus = function () {
+           var node = input.node();
+           if (node) node.focus();
+         };
 
-           var hFactor = (br[0] - tl[0]) / dim[0];
-           var vFactor = (br[1] - tl[1]) / dim[1];
-           var hZoomDiff = Math.log(Math.abs(hFactor)) / Math.LN2;
-           var vZoomDiff = Math.log(Math.abs(vFactor)) / Math.LN2;
-           var newZoom = map.zoom() - Math.max(hZoomDiff, vZoomDiff);
-           return newZoom;
+         function combinedEntityExtent() {
+           return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
          }
 
-         map.extentZoom = function (val) {
-           return calcExtentZoom(geoExtent(val), _dimensions);
-         };
-
-         map.trimmedExtentZoom = function (val) {
-           var trimY = 120;
-           var trimX = 40;
-           var trimmed = [_dimensions[0] - trimX, _dimensions[1] - trimY];
-           return calcExtentZoom(geoExtent(val), trimmed);
-         };
+         return utilRebind(i, dispatch, 'on');
+       }
 
-         map.withinEditableZoom = function () {
-           return map.zoom() >= context.minEditableZoom();
-         };
+       function uiFieldAccess(field, context) {
+         var dispatch = dispatch$8('change');
+         var items = select(null);
 
-         map.isInWideSelection = function () {
-           return !map.withinEditableZoom() && context.selectedIDs().length;
-         };
+         var _tags;
 
-         map.editableDataEnabled = function (skipZoomCheck) {
-           var layer = context.layers().layer('osm');
-           if (!layer || !layer.enabled()) return false;
-           return skipZoomCheck || map.withinEditableZoom();
-         };
+         function access(selection) {
+           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
+           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
+           var list = wrap.selectAll('ul').data([0]);
+           list = list.enter().append('ul').attr('class', 'rows').merge(list);
+           items = list.selectAll('li').data(field.keys); // Enter
 
-         map.notesEditable = function () {
-           var layer = context.layers().layer('notes');
-           if (!layer || !layer.enabled()) return false;
-           return map.withinEditableZoom();
-         };
+           var enter = items.enter().append('li').attr('class', function (d) {
+             return 'labeled-input preset-access-' + d;
+           });
+           enter.append('span').attr('class', 'label preset-label-access').attr('for', function (d) {
+             return 'preset-input-access-' + d;
+           }).html(function (d) {
+             return field.t.html('types.' + d);
+           });
+           enter.append('div').attr('class', 'preset-input-access-wrap').append('input').attr('type', 'text').attr('class', function (d) {
+             return 'preset-input-access preset-input-access-' + d;
+           }).call(utilNoAuto).each(function (d) {
+             select(this).call(uiCombobox(context, 'access-' + d).data(access.options(d)));
+           }); // Update
 
-         map.minzoom = function (val) {
-           if (!arguments.length) return _minzoom;
-           _minzoom = val;
-           return map;
-         };
+           items = items.merge(enter);
+           wrap.selectAll('.preset-input-access').on('change', change).on('blur', change);
+         }
 
-         map.toggleHighlightEdited = function () {
-           surface.classed('highlight-edited', !surface.classed('highlight-edited'));
-           map.pan([0, 0]); // trigger a redraw
+         function change(d3_event, d) {
+           var tag = {};
+           var value = context.cleanTagValue(utilGetSetValue(select(this))); // don't override multiple values with blank string
 
-           dispatch$1.call('changeHighlighting', this);
-         };
+           if (!value && typeof _tags[d] !== 'string') return;
+           tag[d] = value || undefined;
+           dispatch.call('change', this, tag);
+         }
 
-         map.areaFillOptions = ['wireframe', 'partial', 'full'];
+         access.options = function (type) {
+           var options = ['no', 'permissive', 'private', 'permit', 'destination'];
 
-         map.activeAreaFill = function (val) {
-           if (!arguments.length) return corePreferences('area-fill') || 'partial';
-           corePreferences('area-fill', val);
+           if (type !== 'access') {
+             options.unshift('yes');
+             options.push('designated');
 
-           if (val !== 'wireframe') {
-             corePreferences('area-fill-toggle', val);
+             if (type === 'bicycle') {
+               options.push('dismount');
+             }
            }
 
-           updateAreaFill();
-           map.pan([0, 0]); // trigger a redraw
+           return options.map(function (option) {
+             return {
+               title: field.t('options.' + option + '.description'),
+               value: option
+             };
+           });
+         };
 
-           dispatch$1.call('changeAreaFill', this);
-           return map;
+         var placeholdersByHighway = {
+           footway: {
+             foot: 'designated',
+             motor_vehicle: 'no'
+           },
+           steps: {
+             foot: 'yes',
+             motor_vehicle: 'no',
+             bicycle: 'no',
+             horse: 'no'
+           },
+           pedestrian: {
+             foot: 'yes',
+             motor_vehicle: 'no'
+           },
+           cycleway: {
+             motor_vehicle: 'no',
+             bicycle: 'designated'
+           },
+           bridleway: {
+             motor_vehicle: 'no',
+             horse: 'designated'
+           },
+           path: {
+             foot: 'yes',
+             motor_vehicle: 'no',
+             bicycle: 'yes',
+             horse: 'yes'
+           },
+           motorway: {
+             foot: 'no',
+             motor_vehicle: 'yes',
+             bicycle: 'no',
+             horse: 'no'
+           },
+           trunk: {
+             motor_vehicle: 'yes'
+           },
+           primary: {
+             foot: 'yes',
+             motor_vehicle: 'yes',
+             bicycle: 'yes',
+             horse: 'yes'
+           },
+           secondary: {
+             foot: 'yes',
+             motor_vehicle: 'yes',
+             bicycle: 'yes',
+             horse: 'yes'
+           },
+           tertiary: {
+             foot: 'yes',
+             motor_vehicle: 'yes',
+             bicycle: 'yes',
+             horse: 'yes'
+           },
+           residential: {
+             foot: 'yes',
+             motor_vehicle: 'yes',
+             bicycle: 'yes',
+             horse: 'yes'
+           },
+           unclassified: {
+             foot: 'yes',
+             motor_vehicle: 'yes',
+             bicycle: 'yes',
+             horse: 'yes'
+           },
+           service: {
+             foot: 'yes',
+             motor_vehicle: 'yes',
+             bicycle: 'yes',
+             horse: 'yes'
+           },
+           motorway_link: {
+             foot: 'no',
+             motor_vehicle: 'yes',
+             bicycle: 'no',
+             horse: 'no'
+           },
+           trunk_link: {
+             motor_vehicle: 'yes'
+           },
+           primary_link: {
+             foot: 'yes',
+             motor_vehicle: 'yes',
+             bicycle: 'yes',
+             horse: 'yes'
+           },
+           secondary_link: {
+             foot: 'yes',
+             motor_vehicle: 'yes',
+             bicycle: 'yes',
+             horse: 'yes'
+           },
+           tertiary_link: {
+             foot: 'yes',
+             motor_vehicle: 'yes',
+             bicycle: 'yes',
+             horse: 'yes'
+           }
          };
 
-         map.toggleWireframe = function () {
-           var activeFill = map.activeAreaFill();
+         access.tags = function (tags) {
+           _tags = tags;
+           utilGetSetValue(items.selectAll('.preset-input-access'), function (d) {
+             return typeof tags[d] === 'string' ? tags[d] : '';
+           }).classed('mixed', function (d) {
+             return tags[d] && Array.isArray(tags[d]);
+           }).attr('title', function (d) {
+             return tags[d] && Array.isArray(tags[d]) && tags[d].filter(Boolean).join('\n');
+           }).attr('placeholder', function (d) {
+             if (tags[d] && Array.isArray(tags[d])) {
+               return _t('inspector.multiple_values');
+             }
 
-           if (activeFill === 'wireframe') {
-             activeFill = corePreferences('area-fill-toggle') || 'partial';
-           } else {
-             activeFill = 'wireframe';
-           }
+             if (d === 'access') {
+               return 'yes';
+             }
 
-           map.activeAreaFill(activeFill);
-         };
+             if (tags.access && typeof tags.access === 'string') {
+               return tags.access;
+             }
 
-         function updateAreaFill() {
-           var activeFill = map.activeAreaFill();
-           map.areaFillOptions.forEach(function (opt) {
-             surface.classed('fill-' + opt, Boolean(opt === activeFill));
-           });
-         }
+             if (tags.highway) {
+               if (typeof tags.highway === 'string') {
+                 if (placeholdersByHighway[tags.highway] && placeholdersByHighway[tags.highway][d]) {
+                   return placeholdersByHighway[tags.highway][d];
+                 }
+               } else {
+                 var impliedAccesses = tags.highway.filter(Boolean).map(function (highwayVal) {
+                   return placeholdersByHighway[highwayVal] && placeholdersByHighway[highwayVal][d];
+                 }).filter(Boolean);
 
-         map.layers = function () {
-           return drawLayers;
+                 if (impliedAccesses.length === tags.highway.length && new Set(impliedAccesses).size === 1) {
+                   // if all the highway values have the same implied access for this type then use that
+                   return impliedAccesses[0];
+                 }
+               }
+             }
+
+             return field.placeholder();
+           });
          };
 
-         map.doubleUpHandler = function () {
-           return _doubleUpHandler;
+         access.focus = function () {
+           items.selectAll('.preset-input-access').node().focus();
          };
 
-         return utilRebind(map, dispatch$1, 'on');
+         return utilRebind(access, dispatch, 'on');
        }
 
-       function rendererPhotos(context) {
-         var dispatch$1 = dispatch('change');
-         var _layerIDs = ['streetside', 'mapillary', 'mapillary-map-features', 'mapillary-signs', 'openstreetcam'];
-         var _allPhotoTypes = ['flat', 'panoramic'];
+       function uiFieldAddress(field, context) {
+         var dispatch = dispatch$8('change');
 
-         var _shownPhotoTypes = _allPhotoTypes.slice(); // shallow copy
+         var _selection = select(null);
 
+         var _wrap = select(null);
 
-         var _dateFilters = ['fromDate', 'toDate'];
+         var addrField = _mainPresetIndex.field('address'); // needed for placeholder strings
 
-         var _fromDate;
+         var _entityIDs = [];
 
-         var _toDate;
+         var _tags;
 
-         var _usernames;
+         var _countryCode;
 
-         function photos() {}
+         var _addressFormats = [{
+           format: [['housenumber', 'street'], ['city', 'postcode']]
+         }];
+         _mainFileFetcher.get('address_formats').then(function (d) {
+           _addressFormats = d;
 
-         function updateStorage() {
-           if (window.mocha) return;
-           var hash = utilStringQs(window.location.hash);
-           var enabled = context.layers().all().filter(function (d) {
-             return _layerIDs.indexOf(d.id) !== -1 && d.layer && d.layer.supported() && d.layer.enabled();
-           }).map(function (d) {
-             return d.id;
+           if (!_selection.empty()) {
+             _selection.call(address);
+           }
+         })["catch"](function () {
+           /* ignore */
+         });
+
+         function getNearStreets() {
+           var extent = combinedEntityExtent();
+           var l = extent.center();
+           var box = geoExtent(l).padByMeters(200);
+           var streets = context.history().intersects(box).filter(isAddressable).map(function (d) {
+             var loc = context.projection([(extent[0][0] + extent[1][0]) / 2, (extent[0][1] + extent[1][1]) / 2]);
+             var choice = geoChooseEdge(context.graph().childNodes(d), loc, context.projection);
+             return {
+               title: d.tags.name,
+               value: d.tags.name,
+               dist: choice.distance
+             };
+           }).sort(function (a, b) {
+             return a.dist - b.dist;
            });
+           return utilArrayUniqBy(streets, 'value');
 
-           if (enabled.length) {
-             hash.photo_overlay = enabled.join(',');
-           } else {
-             delete hash.photo_overlay;
+           function isAddressable(d) {
+             return d.tags.highway && d.tags.name && d.type === 'way';
            }
-
-           window.location.replace('#' + utilQsString(hash, true));
          }
 
-         photos.overlayLayerIDs = function () {
-           return _layerIDs;
-         };
+         function getNearCities() {
+           var extent = combinedEntityExtent();
+           var l = extent.center();
+           var box = geoExtent(l).padByMeters(200);
+           var cities = context.history().intersects(box).filter(isAddressable).map(function (d) {
+             return {
+               title: d.tags['addr:city'] || d.tags.name,
+               value: d.tags['addr:city'] || d.tags.name,
+               dist: geoSphericalDistance(d.extent(context.graph()).center(), l)
+             };
+           }).sort(function (a, b) {
+             return a.dist - b.dist;
+           });
+           return utilArrayUniqBy(cities, 'value');
 
-         photos.allPhotoTypes = function () {
-           return _allPhotoTypes;
-         };
+           function isAddressable(d) {
+             if (d.tags.name) {
+               if (d.tags.admin_level === '8' && d.tags.boundary === 'administrative') return true;
+               if (d.tags.border_type === 'city') return true;
+               if (d.tags.place === 'city' || d.tags.place === 'town' || d.tags.place === 'village') return true;
+             }
 
-         photos.dateFilters = function () {
-           return _dateFilters;
-         };
+             if (d.tags['addr:city']) return true;
+             return false;
+           }
+         }
 
-         photos.dateFilterValue = function (val) {
-           return val === _dateFilters[0] ? _fromDate : _toDate;
-         };
+         function getNearValues(key) {
+           var extent = combinedEntityExtent();
+           var l = extent.center();
+           var box = geoExtent(l).padByMeters(200);
+           var results = context.history().intersects(box).filter(function hasTag(d) {
+             return _entityIDs.indexOf(d.id) === -1 && d.tags[key];
+           }).map(function (d) {
+             return {
+               title: d.tags[key],
+               value: d.tags[key],
+               dist: geoSphericalDistance(d.extent(context.graph()).center(), l)
+             };
+           }).sort(function (a, b) {
+             return a.dist - b.dist;
+           });
+           return utilArrayUniqBy(results, 'value');
+         }
 
-         photos.setDateFilter = function (type, val, updateUrl) {
-           // validate the date
-           var date = val && new Date(val);
+         function updateForCountryCode() {
+           if (!_countryCode) return;
+           var addressFormat;
 
-           if (date && !isNaN(date)) {
-             val = date.toISOString().substr(0, 10);
-           } else {
-             val = null;
-           }
+           for (var i = 0; i < _addressFormats.length; i++) {
+             var format = _addressFormats[i];
 
-           if (type === _dateFilters[0]) {
-             _fromDate = val;
+             if (!format.countryCodes) {
+               addressFormat = format; // choose the default format, keep going
+             } else if (format.countryCodes.indexOf(_countryCode) !== -1) {
+               addressFormat = format; // choose the country format, stop here
 
-             if (_fromDate && _toDate && new Date(_toDate) < new Date(_fromDate)) {
-               _toDate = _fromDate;
+               break;
              }
            }
 
-           if (type === _dateFilters[1]) {
-             _toDate = val;
+           var dropdowns = addressFormat.dropdowns || ['city', 'county', 'country', 'district', 'hamlet', 'neighbourhood', 'place', 'postcode', 'province', 'quarter', 'state', 'street', 'subdistrict', 'suburb'];
+           var widths = addressFormat.widths || {
+             housenumber: 1 / 3,
+             street: 2 / 3,
+             city: 2 / 3,
+             state: 1 / 4,
+             postcode: 1 / 3
+           };
 
-             if (_fromDate && _toDate && new Date(_toDate) < new Date(_fromDate)) {
-               _fromDate = _toDate;
-             }
+           function row(r) {
+             // Normalize widths.
+             var total = r.reduce(function (sum, key) {
+               return sum + (widths[key] || 0.5);
+             }, 0);
+             return r.map(function (key) {
+               return {
+                 id: key,
+                 width: (widths[key] || 0.5) / total
+               };
+             });
            }
 
-           dispatch$1.call('change', this);
-
-           if (updateUrl) {
-             var rangeString;
-
-             if (_fromDate || _toDate) {
-               rangeString = (_fromDate || '') + '_' + (_toDate || '');
-             }
-
-             setUrlFilterValue('photo_dates', rangeString);
-           }
-         };
+           var rows = _wrap.selectAll('.addr-row').data(addressFormat.format, function (d) {
+             return d.toString();
+           });
 
-         photos.setUsernameFilter = function (val, updateUrl) {
-           if (val && typeof val === 'string') val = val.replace(/;/g, ',').split(',');
+           rows.exit().remove();
+           rows.enter().append('div').attr('class', 'addr-row').selectAll('input').data(row).enter().append('input').property('type', 'text').call(updatePlaceholder).attr('class', function (d) {
+             return 'addr-' + d.id;
+           }).call(utilNoAuto).each(addDropdown).style('width', function (d) {
+             return d.width * 100 + '%';
+           });
 
-           if (val) {
-             val = val.map(function (d) {
-               return d.trim();
-             }).filter(Boolean);
+           function addDropdown(d) {
+             if (dropdowns.indexOf(d.id) === -1) return; // not a dropdown
 
-             if (!val.length) {
-               val = null;
-             }
+             var nearValues = d.id === 'street' ? getNearStreets : d.id === 'city' ? getNearCities : getNearValues;
+             select(this).call(uiCombobox(context, 'address-' + d.id).minItems(1).caseSensitive(true).fetcher(function (value, callback) {
+               callback(nearValues('addr:' + d.id));
+             }));
            }
 
-           _usernames = val;
-           dispatch$1.call('change', this);
+           _wrap.selectAll('input').on('blur', change()).on('change', change());
 
-           if (updateUrl) {
-             var hashString;
+           _wrap.selectAll('input:not(.combobox-input)').on('input', change(true));
 
-             if (_usernames) {
-               hashString = _usernames.join(',');
-             }
+           if (_tags) updateTags(_tags);
+         }
 
-             setUrlFilterValue('photo_username', hashString);
-           }
-         };
+         function address(selection) {
+           _selection = selection;
+           _wrap = selection.selectAll('.form-field-input-wrap').data([0]);
+           _wrap = _wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(_wrap);
+           var extent = combinedEntityExtent();
 
-         function setUrlFilterValue(property, val) {
-           if (!window.mocha) {
-             var hash = utilStringQs(window.location.hash);
+           if (extent) {
+             var countryCode;
 
-             if (val) {
-               if (hash[property] === val) return;
-               hash[property] = val;
+             if (context.inIntro()) {
+               // localize the address format for the walkthrough
+               countryCode = _t('intro.graph.countrycode');
              } else {
-               if (!(property in hash)) return;
-               delete hash[property];
+               var center = extent.center();
+               countryCode = iso1A2Code(center);
              }
 
-             window.location.replace('#' + utilQsString(hash, true));
+             if (countryCode) {
+               _countryCode = countryCode.toLowerCase();
+               updateForCountryCode();
+             }
            }
          }
 
-         function showsLayer(id) {
-           var layer = context.layers().layer(id);
-           return layer && layer.supported() && layer.enabled();
-         }
+         function change(onInput) {
+           return function () {
+             var tags = {};
 
-         photos.shouldFilterByDate = function () {
-           return showsLayer('mapillary') || showsLayer('openstreetcam') || showsLayer('streetside');
-         };
+             _wrap.selectAll('input').each(function (subfield) {
+               var key = field.key + ':' + subfield.id;
+               var value = this.value;
+               if (!onInput) value = context.cleanTagValue(value); // don't override multiple values with blank string
 
-         photos.shouldFilterByPhotoType = function () {
-           return showsLayer('mapillary') || showsLayer('streetside') && showsLayer('openstreetcam');
-         };
+               if (Array.isArray(_tags[key]) && !value) return;
+               tags[key] = value || undefined;
+             });
 
-         photos.shouldFilterByUsername = function () {
-           return showsLayer('mapillary') || showsLayer('openstreetcam') || showsLayer('streetside');
-         };
+             dispatch.call('change', this, tags, onInput);
+           };
+         }
 
-         photos.showsPhotoType = function (val) {
-           if (!photos.shouldFilterByPhotoType()) return true;
-           return _shownPhotoTypes.indexOf(val) !== -1;
-         };
+         function updatePlaceholder(inputSelection) {
+           return inputSelection.attr('placeholder', function (subfield) {
+             if (_tags && Array.isArray(_tags[field.key + ':' + subfield.id])) {
+               return _t('inspector.multiple_values');
+             }
 
-         photos.showsFlat = function () {
-           return photos.showsPhotoType('flat');
-         };
+             if (_countryCode) {
+               var localkey = subfield.id + '!' + _countryCode;
+               var tkey = addrField.hasTextForStringId('placeholders.' + localkey) ? localkey : subfield.id;
+               return addrField.t('placeholders.' + tkey);
+             }
+           });
+         }
 
-         photos.showsPanoramic = function () {
-           return photos.showsPhotoType('panoramic');
-         };
+         function updateTags(tags) {
+           utilGetSetValue(_wrap.selectAll('input'), function (subfield) {
+             var val = tags[field.key + ':' + subfield.id];
+             return typeof val === 'string' ? val : '';
+           }).attr('title', function (subfield) {
+             var val = tags[field.key + ':' + subfield.id];
+             return val && Array.isArray(val) && val.filter(Boolean).join('\n');
+           }).classed('mixed', function (subfield) {
+             return Array.isArray(tags[field.key + ':' + subfield.id]);
+           }).call(updatePlaceholder);
+         }
 
-         photos.fromDate = function () {
-           return _fromDate;
-         };
+         function combinedEntityExtent() {
+           return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+         }
 
-         photos.toDate = function () {
-           return _toDate;
+         address.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs;
+           _entityIDs = val;
+           return address;
          };
 
-         photos.togglePhotoType = function (val) {
-           var index = _shownPhotoTypes.indexOf(val);
+         address.tags = function (tags) {
+           _tags = tags;
+           updateTags(tags);
+         };
 
-           if (index !== -1) {
-             _shownPhotoTypes.splice(index, 1);
-           } else {
-             _shownPhotoTypes.push(val);
-           }
+         address.focus = function () {
+           var node = _wrap.selectAll('input').node();
 
-           dispatch$1.call('change', this);
-           return photos;
+           if (node) node.focus();
          };
 
-         photos.usernames = function () {
-           return _usernames;
-         };
+         return utilRebind(address, dispatch, 'on');
+       }
 
-         photos.init = function () {
-           var hash = utilStringQs(window.location.hash);
+       function uiFieldCycleway(field, context) {
+         var dispatch = dispatch$8('change');
+         var items = select(null);
+         var wrap = select(null);
 
-           if (hash.photo_dates) {
-             // expect format like `photo_dates=2019-01-01_2020-12-31`, but allow a couple different separators
-             var parts = /^(.*)[–_](.*)$/g.exec(hash.photo_dates.trim());
-             this.setDateFilter('fromDate', parts && parts.length >= 2 && parts[1], false);
-             this.setDateFilter('toDate', parts && parts.length >= 3 && parts[2], false);
-           }
+         var _tags;
 
-           if (hash.photo_username) {
-             this.setUsernameFilter(hash.photo_username, false);
+         function cycleway(selection) {
+           function stripcolon(s) {
+             return s.replace(':', '');
            }
 
-           if (hash.photo_overlay) {
-             // support enabling photo layers by default via a URL parameter, e.g. `photo_overlay=openstreetcam;mapillary;streetside`
-             var hashOverlayIDs = hash.photo_overlay.replace(/;/g, ',').split(',');
-             hashOverlayIDs.forEach(function (id) {
-               var layer = _layerIDs.indexOf(id) !== -1 && context.layers().layer(id);
-               if (layer && !layer.enabled()) layer.enabled(true);
-             });
-           }
+           wrap = selection.selectAll('.form-field-input-wrap').data([0]);
+           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
+           var div = wrap.selectAll('ul').data([0]);
+           div = div.enter().append('ul').attr('class', 'rows').merge(div);
+           var keys = ['cycleway:left', 'cycleway:right'];
+           items = div.selectAll('li').data(keys);
+           var enter = items.enter().append('li').attr('class', function (d) {
+             return 'labeled-input preset-cycleway-' + stripcolon(d);
+           });
+           enter.append('span').attr('class', 'label preset-label-cycleway').attr('for', function (d) {
+             return 'preset-input-cycleway-' + stripcolon(d);
+           }).html(function (d) {
+             return field.t.html('types.' + d);
+           });
+           enter.append('div').attr('class', 'preset-input-cycleway-wrap').append('input').attr('type', 'text').attr('class', function (d) {
+             return 'preset-input-cycleway preset-input-' + stripcolon(d);
+           }).call(utilNoAuto).each(function (d) {
+             select(this).call(uiCombobox(context, 'cycleway-' + stripcolon(d)).data(cycleway.options(d)));
+           });
+           items = items.merge(enter); // Update
 
-           if (hash.photo) {
-             // support opening a photo via a URL parameter, e.g. `photo=mapillary-fztgSDtLpa08ohPZFZjeRQ`
-             var photoIds = hash.photo.replace(/;/g, ',').split(',');
-             var photoId = photoIds.length && photoIds[0].trim();
-             var results = /(.*)\/(.*)/g.exec(photoId);
+           wrap.selectAll('.preset-input-cycleway').on('change', change).on('blur', change);
+         }
 
-             if (results && results.length >= 3) {
-               var serviceId = results[1];
-               var photoKey = results[2];
-               var service = services[serviceId];
+         function change(d3_event, key) {
+           var newValue = context.cleanTagValue(utilGetSetValue(select(this))); // don't override multiple values with blank string
 
-               if (service && service.ensureViewerLoaded) {
-                 // if we're showing a photo then make sure its layer is enabled too
-                 var layer = _layerIDs.indexOf(serviceId) !== -1 && context.layers().layer(serviceId);
-                 if (layer && !layer.enabled()) layer.enabled(true);
-                 var baselineTime = Date.now();
-                 service.on('loadedImages.rendererPhotos', function () {
-                   // don't open the viewer if too much time has elapsed
-                   if (Date.now() - baselineTime > 45000) {
-                     service.on('loadedImages.rendererPhotos', null);
-                     return;
-                   }
+           if (!newValue && (Array.isArray(_tags.cycleway) || Array.isArray(_tags[key]))) return;
 
-                   if (!service.cachedImage(photoKey)) return;
-                   service.on('loadedImages.rendererPhotos', null);
-                   service.ensureViewerLoaded(context).then(function () {
-                     service.selectImage(context, photoKey).showViewer(context);
-                   });
-                 });
-               }
-             }
+           if (newValue === 'none' || newValue === '') {
+             newValue = undefined;
            }
 
-           context.layers().on('change.rendererPhotos', updateStorage);
-         };
-
-         return utilRebind(photos, dispatch$1, 'on');
-       }
-
-       function uiAccount(context) {
-         var osm = context.connection();
-
-         function update(selection) {
-           if (!osm) return;
+           var otherKey = key === 'cycleway:left' ? 'cycleway:right' : 'cycleway:left';
+           var otherValue = typeof _tags.cycleway === 'string' ? _tags.cycleway : _tags[otherKey];
 
-           if (!osm.authenticated()) {
-             selection.selectAll('.userLink, .logoutLink').classed('hide', true);
-             return;
+           if (otherValue && Array.isArray(otherValue)) {
+             // we must always have an explicit value for comparison
+             otherValue = otherValue[0];
            }
 
-           osm.userDetails(function (err, details) {
-             var userLink = selection.select('.userLink'),
-                 logoutLink = selection.select('.logoutLink');
-             userLink.html('');
-             logoutLink.html('');
-             if (err || !details) return;
-             selection.selectAll('.userLink, .logoutLink').classed('hide', false); // Link
-
-             var userLinkA = userLink.append('a').attr('href', osm.userURL(details.display_name)).attr('target', '_blank'); // Add thumbnail or dont
+           if (otherValue === 'none' || otherValue === '') {
+             otherValue = undefined;
+           }
 
-             if (details.image_url) {
-               userLinkA.append('img').attr('class', 'icon pre-text user-icon').attr('src', details.image_url);
-             } else {
-               userLinkA.call(svgIcon('#iD-icon-avatar', 'pre-text light'));
-             } // Add user name
+           var tag = {}; // If the left and right tags match, use the cycleway tag to tag both
+           // sides the same way
 
+           if (newValue === otherValue) {
+             tag = {
+               cycleway: newValue,
+               'cycleway:left': undefined,
+               'cycleway:right': undefined
+             };
+           } else {
+             // Always set both left and right as changing one can affect the other
+             tag = {
+               cycleway: undefined
+             };
+             tag[key] = newValue;
+             tag[otherKey] = otherValue;
+           }
 
-             userLinkA.append('span').attr('class', 'label').html(details.display_name);
-             logoutLink.append('a').attr('class', 'logout').attr('href', '#').html(_t.html('logout')).on('click.logout', function (d3_event) {
-               d3_event.preventDefault();
-               osm.logout();
-             });
-           });
+           dispatch.call('change', this, tag);
          }
 
-         return function (selection) {
-           selection.append('li').attr('class', 'userLink').classed('hide', true);
-           selection.append('li').attr('class', 'logoutLink').classed('hide', true);
-
-           if (osm) {
-             osm.on('change.account', function () {
-               update(selection);
-             });
-             update(selection);
-           }
+         cycleway.options = function () {
+           return field.options.map(function (option) {
+             return {
+               title: field.t('options.' + option + '.description'),
+               value: option
+             };
+           });
          };
-       }
 
-       function uiAttribution(context) {
-         var _selection = select(null);
+         cycleway.tags = function (tags) {
+           _tags = tags; // If cycleway is set, use that instead of individual values
 
-         function render(selection, data, klass) {
-           var div = selection.selectAll(".".concat(klass)).data([0]);
-           div = div.enter().append('div').attr('class', klass).merge(div);
-           var attributions = div.selectAll('.attribution').data(data, function (d) {
-             return d.id;
-           });
-           attributions.exit().remove();
-           attributions = attributions.enter().append('span').attr('class', 'attribution').each(function (d, i, nodes) {
-             var attribution = select(nodes[i]);
+           var commonValue = typeof tags.cycleway === 'string' && tags.cycleway;
+           utilGetSetValue(items.selectAll('.preset-input-cycleway'), function (d) {
+             if (commonValue) return commonValue;
+             return !tags.cycleway && typeof tags[d] === 'string' ? tags[d] : '';
+           }).attr('title', function (d) {
+             if (Array.isArray(tags.cycleway) || Array.isArray(tags[d])) {
+               var vals = [];
 
-             if (d.terms_html) {
-               attribution.html(d.terms_html);
-               return;
-             }
+               if (Array.isArray(tags.cycleway)) {
+                 vals = vals.concat(tags.cycleway);
+               }
 
-             if (d.terms_url) {
-               attribution = attribution.append('a').attr('href', d.terms_url).attr('target', '_blank');
-             }
+               if (Array.isArray(tags[d])) {
+                 vals = vals.concat(tags[d]);
+               }
 
-             var sourceID = d.id.replace(/\./g, '<TX_DOT>');
-             var terms_text = _t("imagery.".concat(sourceID, ".attribution.text"), {
-               "default": d.terms_text || d.id || d.name()
-             });
+               return vals.filter(Boolean).join('\n');
+             }
 
-             if (d.icon && !d.overlay) {
-               attribution.append('img').attr('class', 'source-image').attr('src', d.icon);
+             return null;
+           }).attr('placeholder', function (d) {
+             if (Array.isArray(tags.cycleway) || Array.isArray(tags[d])) {
+               return _t('inspector.multiple_values');
              }
 
-             attribution.append('span').attr('class', 'attribution-text').html(terms_text);
-           }).merge(attributions);
-           var copyright = attributions.selectAll('.copyright-notice').data(function (d) {
-             var notice = d.copyrightNotices(context.map().zoom(), context.map().extent());
-             return notice ? [notice] : [];
+             return field.placeholder();
+           }).classed('mixed', function (d) {
+             return Array.isArray(tags.cycleway) || Array.isArray(tags[d]);
            });
-           copyright.exit().remove();
-           copyright = copyright.enter().append('span').attr('class', 'copyright-notice').merge(copyright);
-           copyright.html(String);
-         }
-
-         function update() {
-           var baselayer = context.background().baseLayerSource();
-
-           _selection.call(render, baselayer ? [baselayer] : [], 'base-layer-attribution');
-
-           var z = context.map().zoom();
-           var overlays = context.background().overlayLayerSources() || [];
-
-           _selection.call(render, overlays.filter(function (s) {
-             return s.validZoom(z);
-           }), 'overlay-layer-attribution');
-         }
+         };
 
-         return function (selection) {
-           _selection = selection;
-           context.background().on('change.attribution', update);
-           context.map().on('move.attribution', throttle(update, 400, {
-             leading: false
-           }));
-           update();
+         cycleway.focus = function () {
+           var node = wrap.selectAll('input').node();
+           if (node) node.focus();
          };
+
+         return utilRebind(cycleway, dispatch, 'on');
        }
 
-       function uiContributors(context) {
-         var osm = context.connection(),
-             debouncedUpdate = debounce(function () {
-           update();
-         }, 1000),
-             limit = 4,
-             hidden = false,
-             wrap = select(null);
+       function uiFieldLanes(field, context) {
+         var dispatch = dispatch$8('change');
+         var LANE_WIDTH = 40;
+         var LANE_HEIGHT = 200;
+         var _entityIDs = [];
 
-         function update() {
-           if (!osm) return;
-           var users = {},
-               entities = context.history().intersects(context.map().extent());
-           entities.forEach(function (entity) {
-             if (entity && entity.user) users[entity.user] = true;
-           });
-           var u = Object.keys(users),
-               subset = u.slice(0, u.length > limit ? limit - 1 : limit);
-           wrap.html('').call(svgIcon('#iD-icon-nearby', 'pre-text light'));
-           var userList = select(document.createElement('span'));
-           userList.selectAll().data(subset).enter().append('a').attr('class', 'user-link').attr('href', function (d) {
-             return osm.userURL(d);
-           }).attr('target', '_blank').html(String);
+         function lanes(selection) {
+           var lanesData = context.entity(_entityIDs[0]).lanes();
 
-           if (u.length > limit) {
-             var count = select(document.createElement('span'));
-             var othersNum = u.length - limit + 1;
-             count.append('a').attr('target', '_blank').attr('href', function () {
-               return osm.changesetsURL(context.map().center(), context.map().zoom());
-             }).html(othersNum);
-             wrap.append('span').html(_t.html('contributors.truncated_list', {
-               n: othersNum,
-               users: userList.html(),
-               count: count.html()
-             }));
-           } else {
-             wrap.append('span').html(_t.html('contributors.list', {
-               users: userList.html()
-             }));
+           if (!context.container().select('.inspector-wrap.inspector-hidden').empty() || !selection.node().parentNode) {
+             selection.call(lanes.off);
+             return;
            }
 
-           if (!u.length) {
-             hidden = true;
-             wrap.transition().style('opacity', 0);
-           } else if (hidden) {
-             wrap.transition().style('opacity', 1);
-           }
+           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
+           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
+           var surface = wrap.selectAll('.surface').data([0]);
+           var d = utilGetDimensions(wrap);
+           var freeSpace = d[0] - lanesData.lanes.length * LANE_WIDTH * 1.5 + LANE_WIDTH * 0.5;
+           surface = surface.enter().append('svg').attr('width', d[0]).attr('height', 300).attr('class', 'surface').merge(surface);
+           var lanesSelection = surface.selectAll('.lanes').data([0]);
+           lanesSelection = lanesSelection.enter().append('g').attr('class', 'lanes').merge(lanesSelection);
+           lanesSelection.attr('transform', function () {
+             return 'translate(' + freeSpace / 2 + ', 0)';
+           });
+           var lane = lanesSelection.selectAll('.lane').data(lanesData.lanes);
+           lane.exit().remove();
+           var enter = lane.enter().append('g').attr('class', 'lane');
+           enter.append('g').append('rect').attr('y', 50).attr('width', LANE_WIDTH).attr('height', LANE_HEIGHT);
+           enter.append('g').attr('class', 'forward').append('text').attr('y', 40).attr('x', 14).html('▲');
+           enter.append('g').attr('class', 'bothways').append('text').attr('y', 40).attr('x', 14).html('▲▼');
+           enter.append('g').attr('class', 'backward').append('text').attr('y', 40).attr('x', 14).html('▼');
+           lane = lane.merge(enter);
+           lane.attr('transform', function (d) {
+             return 'translate(' + LANE_WIDTH * d.index * 1.5 + ', 0)';
+           });
+           lane.select('.forward').style('visibility', function (d) {
+             return d.direction === 'forward' ? 'visible' : 'hidden';
+           });
+           lane.select('.bothways').style('visibility', function (d) {
+             return d.direction === 'bothways' ? 'visible' : 'hidden';
+           });
+           lane.select('.backward').style('visibility', function (d) {
+             return d.direction === 'backward' ? 'visible' : 'hidden';
+           });
          }
 
-         return function (selection) {
-           if (!osm) return;
-           wrap = selection;
-           update();
-           osm.on('loaded.contributors', debouncedUpdate);
-           context.map().on('move.contributors', debouncedUpdate);
+         lanes.entityIDs = function (val) {
+           _entityIDs = val;
          };
+
+         lanes.tags = function () {};
+
+         lanes.focus = function () {};
+
+         lanes.off = function () {};
+
+         return utilRebind(lanes, dispatch, 'on');
        }
+       uiFieldLanes.supportsMultiselection = false;
 
-       var _popoverID = 0;
-       function uiPopover(klass) {
-         var _id = _popoverID++;
+       var _languagesArray = [];
+       function uiFieldLocalized(field, context) {
+         var dispatch = dispatch$8('change', 'input');
+         var wikipedia = services.wikipedia;
+         var input = select(null);
+         var localizedInputs = select(null);
 
-         var _anchorSelection = select(null);
+         var _countryCode;
 
-         var popover = function popover(selection) {
-           _anchorSelection = selection;
-           selection.each(setup);
-         };
+         var _tags; // A concern here in switching to async data means that _languagesArray will not
+         // be available the first time through, so things like the fetchers and
+         // the language() function will not work immediately.
 
-         var _animation = utilFunctor(false);
 
-         var _placement = utilFunctor('top'); // top, bottom, left, right
+         _mainFileFetcher.get('languages').then(loadLanguagesArray)["catch"](function () {
+           /* ignore */
+         });
+         var _territoryLanguages = {};
+         _mainFileFetcher.get('territory_languages').then(function (d) {
+           _territoryLanguages = d;
+         })["catch"](function () {
+           /* ignore */
+         }); // reuse these combos
 
+         var langCombo = uiCombobox(context, 'localized-lang').fetcher(fetchLanguages).minItems(0);
 
-         var _alignment = utilFunctor('center'); // leading, center, trailing
+         var _selection = select(null);
 
+         var _multilingual = [];
 
-         var _scrollContainer = utilFunctor(select(null));
+         var _buttonTip = uiTooltip().title(_t.html('translate.translate')).placement('left');
 
-         var _content;
+         var _wikiTitles;
 
-         var _displayType = utilFunctor('');
+         var _entityIDs = [];
 
-         var _hasArrow = utilFunctor(true); // use pointer events on supported platforms; fallback to mouse events
+         function loadLanguagesArray(dataLanguages) {
+           if (_languagesArray.length !== 0) return; // some conversion is needed to ensure correct OSM tags are used
 
+           var replacements = {
+             sr: 'sr-Cyrl',
+             // in OSM, `sr` implies Cyrillic
+             'sr-Cyrl': false // `sr-Cyrl` isn't used in OSM
 
-         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
+           };
 
-         popover.displayType = function (val) {
-           if (arguments.length) {
-             _displayType = utilFunctor(val);
-             return popover;
-           } else {
-             return _displayType;
-           }
-         };
+           for (var code in dataLanguages) {
+             if (replacements[code] === false) continue;
+             var metaCode = code;
+             if (replacements[code]) metaCode = replacements[code];
 
-         popover.hasArrow = function (val) {
-           if (arguments.length) {
-             _hasArrow = utilFunctor(val);
-             return popover;
-           } else {
-             return _hasArrow;
+             _languagesArray.push({
+               localName: _mainLocalizer.languageName(metaCode, {
+                 localOnly: true
+               }),
+               nativeName: dataLanguages[metaCode].nativeName,
+               code: code,
+               label: _mainLocalizer.languageName(metaCode)
+             });
            }
-         };
+         }
 
-         popover.placement = function (val) {
-           if (arguments.length) {
-             _placement = utilFunctor(val);
-             return popover;
-           } else {
-             return _placement;
-           }
-         };
+         function calcLocked() {
+           // Protect name field for suggestion presets that don't display a brand/operator field
+           var isLocked = field.id === 'name' && _entityIDs.length && _entityIDs.some(function (entityID) {
+             var entity = context.graph().hasEntity(entityID);
+             if (!entity) return false; // Features linked to Wikidata are likely important and should be protected
 
-         popover.alignment = function (val) {
-           if (arguments.length) {
-             _alignment = utilFunctor(val);
-             return popover;
-           } else {
-             return _alignment;
-           }
-         };
+             if (entity.tags.wikidata) return true; // Assume the name has already been confirmed if its source has been researched
 
-         popover.scrollContainer = function (val) {
-           if (arguments.length) {
-             _scrollContainer = utilFunctor(val);
-             return popover;
-           } else {
-             return _scrollContainer;
-           }
-         };
+             if (entity.tags['name:etymology:wikidata']) return true; // Lock the `name` if this is a suggestion preset that assigns the name,
+             // and the preset does not display a `brand` or `operator` field.
+             // (For presets like hotels, car dealerships, post offices, the `name` should remain editable)
+             // see also similar logic in `outdated_tags.js`
 
-         popover.content = function (val) {
-           if (arguments.length) {
-             _content = val;
-             return popover;
-           } else {
-             return _content;
-           }
-         };
+             var preset = _mainPresetIndex.match(entity, context.graph());
 
-         popover.isShown = function () {
-           var popoverSelection = _anchorSelection.select('.popover-' + _id);
+             if (preset) {
+               var isSuggestion = preset.suggestion;
+               var fields = preset.fields();
+               var showsBrandField = fields.some(function (d) {
+                 return d.id === 'brand';
+               });
+               var showsOperatorField = fields.some(function (d) {
+                 return d.id === 'operator';
+               });
+               var setsName = preset.addTags.name;
+               var setsBrandWikidata = preset.addTags['brand:wikidata'];
+               var setsOperatorWikidata = preset.addTags['operator:wikidata'];
+               return isSuggestion && setsName && (setsBrandWikidata && !showsBrandField || setsOperatorWikidata && !showsOperatorField);
+             }
 
-           return !popoverSelection.empty() && popoverSelection.classed('in');
-         };
+             return false;
+           });
 
-         popover.show = function () {
-           _anchorSelection.each(show);
-         };
+           field.locked(isLocked);
+         } // update _multilingual, maintaining the existing order
 
-         popover.updateContent = function () {
-           _anchorSelection.each(updateContent);
-         };
 
-         popover.hide = function () {
-           _anchorSelection.each(hide);
-         };
+         function calcMultilingual(tags) {
+           var existingLangsOrdered = _multilingual.map(function (item) {
+             return item.lang;
+           });
 
-         popover.toggle = function () {
-           _anchorSelection.each(toggle);
-         };
+           var existingLangs = new Set(existingLangsOrdered.filter(Boolean));
 
-         popover.destroy = function (selection, selector) {
-           // by default, just destroy the current popover
-           selector = selector || '.popover-' + _id;
-           selection.on(_pointerPrefix + 'enter.popover', null).on(_pointerPrefix + 'leave.popover', null).on(_pointerPrefix + 'up.popover', null).on(_pointerPrefix + 'down.popover', null).on('click.popover', null).attr('title', function () {
-             return this.getAttribute('data-original-title') || this.getAttribute('title');
-           }).attr('data-original-title', null).selectAll(selector).remove();
-         };
+           for (var k in tags) {
+             var m = k.match(/^(.*):(.*)$/);
 
-         popover.destroyAny = function (selection) {
-           selection.call(popover.destroy, '.popover');
-         };
+             if (m && m[1] === field.key && m[2]) {
+               var item = {
+                 lang: m[2],
+                 value: tags[k]
+               };
 
-         function setup() {
-           var anchor = select(this);
+               if (existingLangs.has(item.lang)) {
+                 // update the value
+                 _multilingual[existingLangsOrdered.indexOf(item.lang)].value = item.value;
+                 existingLangs["delete"](item.lang);
+               } else {
+                 _multilingual.push(item);
+               }
+             }
+           } // Don't remove items based on deleted tags, since this makes the UI
+           // disappear unexpectedly when clearing values - #8164
 
-           var animate = _animation.apply(this, arguments);
 
-           var popoverSelection = anchor.selectAll('.popover-' + _id).data([0]);
-           var enter = popoverSelection.enter().append('div').attr('class', 'popover popover-' + _id + ' ' + (klass ? klass : '')).classed('arrowed', _hasArrow.apply(this, arguments));
-           enter.append('div').attr('class', 'popover-arrow');
-           enter.append('div').attr('class', 'popover-inner');
-           popoverSelection = enter.merge(popoverSelection);
+           _multilingual.forEach(function (item) {
+             if (item.lang && existingLangs.has(item.lang)) {
+               item.value = '';
+             }
+           });
+         }
 
-           if (animate) {
-             popoverSelection.classed('fade', true);
-           }
+         function localized(selection) {
+           _selection = selection;
+           calcLocked();
+           var isLocked = field.locked();
+           var wrap = selection.selectAll('.form-field-input-wrap').data([0]); // enter/update
 
-           var display = _displayType.apply(this, arguments);
+           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
+           input = wrap.selectAll('.localized-main').data([0]); // enter/update
 
-           if (display === 'hover') {
-             var _lastNonMouseEnterTime;
+           input = input.enter().append('input').attr('type', 'text').attr('id', field.domId).attr('class', 'localized-main').call(utilNoAuto).merge(input);
+           input.classed('disabled', !!isLocked).attr('readonly', isLocked || null).on('input', change(true)).on('blur', change()).on('change', change());
+           var translateButton = wrap.selectAll('.localized-add').data([0]);
+           translateButton = translateButton.enter().append('button').attr('class', 'localized-add form-field-button').call(svgIcon('#iD-icon-plus')).merge(translateButton);
+           translateButton.classed('disabled', !!isLocked).call(isLocked ? _buttonTip.destroy : _buttonTip).on('click', addNew);
 
-             anchor.on(_pointerPrefix + 'enter.popover', function (d3_event) {
-               if (d3_event.pointerType) {
-                 if (d3_event.pointerType !== 'mouse') {
-                   _lastNonMouseEnterTime = d3_event.timeStamp; // only allow hover behavior for mouse input
+           if (_tags && !_multilingual.length) {
+             calcMultilingual(_tags);
+           }
 
-                   return;
-                 } else if (_lastNonMouseEnterTime && d3_event.timeStamp - _lastNonMouseEnterTime < 1500) {
-                   // HACK: iOS 13.4 sends an erroneous `mouse` type pointerenter
-                   // event for non-mouse interactions right after sending
-                   // the correct type pointerenter event. Workaround by discarding
-                   // any mouse event that occurs immediately after a non-mouse event.
-                   return;
-                 }
-               } // don't show if buttons are pressed, e.g. during click and drag of map
+           localizedInputs = selection.selectAll('.localized-multilingual').data([0]);
+           localizedInputs = localizedInputs.enter().append('div').attr('class', 'localized-multilingual').merge(localizedInputs);
+           localizedInputs.call(renderMultilingual);
+           localizedInputs.selectAll('button, input').classed('disabled', !!isLocked).attr('readonly', isLocked || null);
 
+           function addNew(d3_event) {
+             d3_event.preventDefault();
+             if (field.locked()) return;
+             var defaultLang = _mainLocalizer.languageCode().toLowerCase();
 
-               if (d3_event.buttons !== 0) return;
-               show.apply(this, arguments);
-             }).on(_pointerPrefix + 'leave.popover', function () {
-               hide.apply(this, arguments);
-             }) // show on focus too for better keyboard navigation support
-             .on('focus.popover', function () {
-               show.apply(this, arguments);
-             }).on('blur.popover', function () {
-               hide.apply(this, arguments);
+             var langExists = _multilingual.find(function (datum) {
+               return datum.lang === defaultLang;
              });
-           } else if (display === 'clickFocus') {
-             anchor.on(_pointerPrefix + 'down.popover', function (d3_event) {
-               d3_event.preventDefault();
-               d3_event.stopPropagation();
-             }).on(_pointerPrefix + 'up.popover', function (d3_event) {
-               d3_event.preventDefault();
-               d3_event.stopPropagation();
-             }).on('click.popover', toggle);
-             popoverSelection // This attribute lets the popover take focus
-             .attr('tabindex', 0).on('blur.popover', function () {
-               anchor.each(function () {
-                 hide.apply(this, arguments);
+
+             var isLangEn = defaultLang.indexOf('en') > -1;
+
+             if (isLangEn || langExists) {
+               defaultLang = '';
+               langExists = _multilingual.find(function (datum) {
+                 return datum.lang === defaultLang;
                });
-             });
-           }
-         }
+             }
 
-         function show() {
-           var anchor = select(this);
-           var popoverSelection = anchor.selectAll('.popover-' + _id);
+             if (!langExists) {
+               // prepend the value so it appears at the top
+               _multilingual.unshift({
+                 lang: defaultLang,
+                 value: ''
+               });
 
-           if (popoverSelection.empty()) {
-             // popover was removed somehow, put it back
-             anchor.call(popover.destroy);
-             anchor.each(setup);
-             popoverSelection = anchor.selectAll('.popover-' + _id);
+               localizedInputs.call(renderMultilingual);
+             }
            }
 
-           popoverSelection.classed('in', true);
+           function change(onInput) {
+             return function (d3_event) {
+               if (field.locked()) {
+                 d3_event.preventDefault();
+                 return;
+               }
 
-           var displayType = _displayType.apply(this, arguments);
+               var val = utilGetSetValue(select(this));
+               if (!onInput) val = context.cleanTagValue(val); // don't override multiple values with blank string
 
-           if (displayType === 'clickFocus') {
-             anchor.classed('active', true);
-             popoverSelection.node().focus();
+               if (!val && Array.isArray(_tags[field.key])) return;
+               var t = {};
+               t[field.key] = val || undefined;
+               dispatch.call('change', this, t, onInput);
+             };
            }
+         }
 
-           anchor.each(updateContent);
+         function key(lang) {
+           return field.key + ':' + lang;
          }
 
-         function updateContent() {
-           var anchor = select(this);
+         function changeLang(d3_event, d) {
+           var tags = {}; // make sure unrecognized suffixes are lowercase - #7156
 
-           if (_content) {
-             anchor.selectAll('.popover-' + _id + ' > .popover-inner').call(_content.apply(this, arguments));
+           var lang = utilGetSetValue(select(this)).toLowerCase();
+
+           var language = _languagesArray.find(function (d) {
+             return d.label.toLowerCase() === lang || d.localName && d.localName.toLowerCase() === lang || d.nativeName && d.nativeName.toLowerCase() === lang;
+           });
+
+           if (language) lang = language.code;
+
+           if (d.lang && d.lang !== lang) {
+             tags[key(d.lang)] = undefined;
            }
 
-           updatePosition.apply(this, arguments); // hack: update multiple times to fix instances where the absolute offset is
-           // set before the dynamic popover size is calculated by the browser
+           var newKey = lang && context.cleanTagKey(key(lang));
+           var value = utilGetSetValue(select(this.parentNode).selectAll('.localized-value'));
 
-           updatePosition.apply(this, arguments);
-           updatePosition.apply(this, arguments);
+           if (newKey && value) {
+             tags[newKey] = value;
+           } else if (newKey && _wikiTitles && _wikiTitles[d.lang]) {
+             tags[newKey] = _wikiTitles[d.lang];
+           }
+
+           d.lang = lang;
+           dispatch.call('change', this, tags);
          }
 
-         function updatePosition() {
-           var anchor = select(this);
-           var popoverSelection = anchor.selectAll('.popover-' + _id);
+         function changeValue(d3_event, d) {
+           if (!d.lang) return;
+           var value = context.cleanTagValue(utilGetSetValue(select(this))) || undefined; // don't override multiple values with blank string
 
-           var scrollContainer = _scrollContainer && _scrollContainer.apply(this, arguments);
+           if (!value && Array.isArray(d.value)) return;
+           var t = {};
+           t[key(d.lang)] = value;
+           d.value = value;
+           dispatch.call('change', this, t);
+         }
 
-           var scrollNode = scrollContainer && !scrollContainer.empty() && scrollContainer.node();
-           var scrollLeft = scrollNode ? scrollNode.scrollLeft : 0;
-           var scrollTop = scrollNode ? scrollNode.scrollTop : 0;
+         function fetchLanguages(value, cb) {
+           var v = value.toLowerCase(); // show the user's language first
 
-           var placement = _placement.apply(this, arguments);
+           var langCodes = [_mainLocalizer.localeCode(), _mainLocalizer.languageCode()];
 
-           popoverSelection.classed('left', false).classed('right', false).classed('top', false).classed('bottom', false).classed(placement, true);
+           if (_countryCode && _territoryLanguages[_countryCode]) {
+             langCodes = langCodes.concat(_territoryLanguages[_countryCode]);
+           }
 
-           var alignment = _alignment.apply(this, arguments);
+           var langItems = [];
+           langCodes.forEach(function (code) {
+             var langItem = _languagesArray.find(function (item) {
+               return item.code === code;
+             });
 
-           var alignFactor = 0.5;
+             if (langItem) langItems.push(langItem);
+           });
+           langItems = utilArrayUniq(langItems.concat(_languagesArray));
+           cb(langItems.filter(function (d) {
+             return d.label.toLowerCase().indexOf(v) >= 0 || d.localName && d.localName.toLowerCase().indexOf(v) >= 0 || d.nativeName && d.nativeName.toLowerCase().indexOf(v) >= 0 || d.code.toLowerCase().indexOf(v) >= 0;
+           }).map(function (d) {
+             return {
+               value: d.label
+             };
+           }));
+         }
 
-           if (alignment === 'leading') {
-             alignFactor = 0;
-           } else if (alignment === 'trailing') {
-             alignFactor = 1;
-           }
+         function renderMultilingual(selection) {
+           var entries = selection.selectAll('div.entry').data(_multilingual, function (d) {
+             return d.lang;
+           });
+           entries.exit().style('top', '0').style('max-height', '240px').transition().duration(200).style('opacity', '0').style('max-height', '0px').remove();
+           var entriesEnter = entries.enter().append('div').attr('class', 'entry').each(function (_, index) {
+             var wrap = select(this);
+             var domId = utilUniqueDomId(index);
+             var label = wrap.append('label').attr('class', 'field-label').attr('for', domId);
+             var text = label.append('span').attr('class', 'label-text');
+             text.append('span').attr('class', 'label-textvalue').html(_t.html('translate.localized_translation_label'));
+             text.append('span').attr('class', 'label-textannotation');
+             label.append('button').attr('class', 'remove-icon-multilingual').on('click', function (d3_event, d) {
+               if (field.locked()) return;
+               d3_event.preventDefault(); // remove the UI item manually
 
-           var anchorFrame = getFrame(anchor.node());
-           var popoverFrame = getFrame(popoverSelection.node());
-           var position;
+               _multilingual.splice(_multilingual.indexOf(d), 1);
 
-           switch (placement) {
-             case 'top':
-               position = {
-                 x: anchorFrame.x + (anchorFrame.w - popoverFrame.w) * alignFactor,
-                 y: anchorFrame.y - popoverFrame.h
-               };
-               break;
+               var langKey = d.lang && key(d.lang);
 
-             case 'bottom':
-               position = {
-                 x: anchorFrame.x + (anchorFrame.w - popoverFrame.w) * alignFactor,
-                 y: anchorFrame.y + anchorFrame.h
-               };
-               break;
+               if (langKey && langKey in _tags) {
+                 delete _tags[langKey]; // remove from entity tags
 
-             case 'left':
-               position = {
-                 x: anchorFrame.x - popoverFrame.w,
-                 y: anchorFrame.y + (anchorFrame.h - popoverFrame.h) * alignFactor
-               };
-               break;
+                 var t = {};
+                 t[langKey] = undefined;
+                 dispatch.call('change', this, t);
+                 return;
+               }
 
-             case 'right':
-               position = {
-                 x: anchorFrame.x + anchorFrame.w,
-                 y: anchorFrame.y + (anchorFrame.h - popoverFrame.h) * alignFactor
-               };
-               break;
+               renderMultilingual(selection);
+             }).call(svgIcon('#iD-operation-delete'));
+             wrap.append('input').attr('class', 'localized-lang').attr('id', domId).attr('type', 'text').attr('placeholder', _t('translate.localized_translation_language')).on('blur', changeLang).on('change', changeLang).call(langCombo);
+             wrap.append('input').attr('type', 'text').attr('class', 'localized-value').on('blur', changeValue).on('change', changeValue);
+           });
+           entriesEnter.style('margin-top', '0px').style('max-height', '0px').style('opacity', '0').transition().duration(200).style('margin-top', '10px').style('max-height', '240px').style('opacity', '1').on('end', function () {
+             select(this).style('max-height', '').style('overflow', 'visible');
+           });
+           entries = entries.merge(entriesEnter);
+           entries.order(); // allow removing the entry UIs even if there isn't a tag to remove
+
+           entries.classed('present', true);
+           utilGetSetValue(entries.select('.localized-lang'), function (d) {
+             var langItem = _languagesArray.find(function (item) {
+               return item.code === d.lang;
+             });
+
+             if (langItem) return langItem.label;
+             return d.lang;
+           });
+           utilGetSetValue(entries.select('.localized-value'), function (d) {
+             return typeof d.value === 'string' ? d.value : '';
+           }).attr('title', function (d) {
+             return Array.isArray(d.value) ? d.value.filter(Boolean).join('\n') : null;
+           }).attr('placeholder', function (d) {
+             return Array.isArray(d.value) ? _t('inspector.multiple_values') : _t('translate.localized_translation_name');
+           }).classed('mixed', function (d) {
+             return Array.isArray(d.value);
+           });
+         }
+
+         localized.tags = function (tags) {
+           _tags = tags; // Fetch translations from wikipedia
+
+           if (typeof tags.wikipedia === 'string' && !_wikiTitles) {
+             _wikiTitles = {};
+             var wm = tags.wikipedia.match(/([^:]+):(.+)/);
+
+             if (wm && wm[0] && wm[1]) {
+               wikipedia.translations(wm[1], wm[2], function (err, d) {
+                 if (err || !d) return;
+                 _wikiTitles = d;
+               });
+             }
            }
 
-           if (position) {
-             if (scrollNode && (placement === 'top' || placement === 'bottom')) {
-               var initialPosX = position.x;
+           var isMixed = Array.isArray(tags[field.key]);
+           utilGetSetValue(input, typeof tags[field.key] === 'string' ? tags[field.key] : '').attr('title', isMixed ? tags[field.key].filter(Boolean).join('\n') : undefined).attr('placeholder', isMixed ? _t('inspector.multiple_values') : field.placeholder()).classed('mixed', isMixed);
+           calcMultilingual(tags);
 
-               if (position.x + popoverFrame.w > scrollNode.offsetWidth - 10) {
-                 position.x = scrollNode.offsetWidth - 10 - popoverFrame.w;
-               } else if (position.x < 10) {
-                 position.x = 10;
-               }
+           _selection.call(localized);
+         };
+
+         localized.focus = function () {
+           input.node().focus();
+         };
+
+         localized.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs;
+           _entityIDs = val;
+           _multilingual = [];
+           loadCountryCode();
+           return localized;
+         };
+
+         function loadCountryCode() {
+           var extent = combinedEntityExtent();
+           var countryCode = extent && iso1A2Code(extent.center());
+           _countryCode = countryCode && countryCode.toLowerCase();
+         }
 
-               var arrow = anchor.selectAll('.popover-' + _id + ' > .popover-arrow'); // keep the arrow centered on the button, or as close as possible
+         function combinedEntityExtent() {
+           return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+         }
 
-               var arrowPosX = Math.min(Math.max(popoverFrame.w / 2 - (position.x - initialPosX), 10), popoverFrame.w - 10);
-               arrow.style('left', ~~arrowPosX + 'px');
-             }
+         return utilRebind(localized, dispatch, 'on');
+       }
 
-             popoverSelection.style('left', ~~position.x + 'px').style('top', ~~position.y + 'px');
-           } else {
-             popoverSelection.style('left', null).style('top', null);
-           }
+       function uiFieldRoadspeed(field, context) {
+         var dispatch = dispatch$8('change');
+         var unitInput = select(null);
+         var input = select(null);
+         var _entityIDs = [];
 
-           function getFrame(node) {
-             var positionStyle = select(node).style('position');
+         var _tags;
 
-             if (positionStyle === 'absolute' || positionStyle === 'static') {
-               return {
-                 x: node.offsetLeft - scrollLeft,
-                 y: node.offsetTop - scrollTop,
-                 w: node.offsetWidth,
-                 h: node.offsetHeight
-               };
-             } else {
-               return {
-                 x: 0,
-                 y: 0,
-                 w: node.offsetWidth,
-                 h: node.offsetHeight
-               };
-             }
-           }
-         }
+         var _isImperial;
 
-         function hide() {
-           var anchor = select(this);
+         var speedCombo = uiCombobox(context, 'roadspeed');
+         var unitCombo = uiCombobox(context, 'roadspeed-unit').data(['km/h', 'mph'].map(comboValues));
+         var metricValues = [20, 30, 40, 50, 60, 70, 80, 90, 100, 110, 120];
+         var imperialValues = [5, 10, 15, 20, 25, 30, 35, 40, 45, 50, 55, 60, 65, 70, 75, 80];
 
-           if (_displayType.apply(this, arguments) === 'clickFocus') {
-             anchor.classed('active', false);
-           }
+         function roadspeed(selection) {
+           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
+           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
+           input = wrap.selectAll('input.roadspeed-number').data([0]);
+           input = input.enter().append('input').attr('type', 'text').attr('class', 'roadspeed-number').attr('id', field.domId).call(utilNoAuto).call(speedCombo).merge(input);
+           input.on('change', change).on('blur', change);
+           var loc = combinedEntityExtent().center();
+           _isImperial = roadSpeedUnit(loc) === 'mph';
+           unitInput = wrap.selectAll('input.roadspeed-unit').data([0]);
+           unitInput = unitInput.enter().append('input').attr('type', 'text').attr('class', 'roadspeed-unit').call(unitCombo).merge(unitInput);
+           unitInput.on('blur', changeUnits).on('change', changeUnits);
 
-           anchor.selectAll('.popover-' + _id).classed('in', false);
+           function changeUnits() {
+             _isImperial = utilGetSetValue(unitInput) === 'mph';
+             utilGetSetValue(unitInput, _isImperial ? 'mph' : 'km/h');
+             setUnitSuggestions();
+             change();
+           }
          }
 
-         function toggle() {
-           if (select(this).select('.popover-' + _id).classed('in')) {
-             hide.apply(this, arguments);
-           } else {
-             show.apply(this, arguments);
-           }
+         function setUnitSuggestions() {
+           speedCombo.data((_isImperial ? imperialValues : metricValues).map(comboValues));
+           utilGetSetValue(unitInput, _isImperial ? 'mph' : 'km/h');
          }
 
-         return popover;
-       }
+         function comboValues(d) {
+           return {
+             value: d.toString(),
+             title: d.toString()
+           };
+         }
 
-       function uiTooltip(klass) {
-         var tooltip = uiPopover((klass || '') + ' tooltip').displayType('hover');
+         function change() {
+           var tag = {};
+           var value = utilGetSetValue(input).trim(); // don't override multiple values with blank string
 
-         var _title = function _title() {
-           var title = this.getAttribute('data-original-title');
+           if (!value && Array.isArray(_tags[field.key])) return;
 
-           if (title) {
-             return title;
+           if (!value) {
+             tag[field.key] = undefined;
+           } else if (isNaN(value) || !_isImperial) {
+             tag[field.key] = context.cleanTagValue(value);
            } else {
-             title = this.getAttribute('title');
-             this.removeAttribute('title');
-             this.setAttribute('data-original-title', title);
+             tag[field.key] = context.cleanTagValue(value + ' mph');
            }
 
-           return title;
-         };
+           dispatch.call('change', this, tag);
+         }
 
-         var _heading = utilFunctor(null);
+         roadspeed.tags = function (tags) {
+           _tags = tags;
+           var value = tags[field.key];
+           var isMixed = Array.isArray(value);
 
-         var _keys = utilFunctor(null);
+           if (!isMixed) {
+             if (value && value.indexOf('mph') >= 0) {
+               value = parseInt(value, 10).toString();
+               _isImperial = true;
+             } else if (value) {
+               _isImperial = false;
+             }
+           }
 
-         tooltip.title = function (val) {
-           if (!arguments.length) return _title;
-           _title = utilFunctor(val);
-           return tooltip;
+           setUnitSuggestions();
+           utilGetSetValue(input, typeof value === 'string' ? value : '').attr('title', isMixed ? value.filter(Boolean).join('\n') : null).attr('placeholder', isMixed ? _t('inspector.multiple_values') : field.placeholder()).classed('mixed', isMixed);
          };
 
-         tooltip.heading = function (val) {
-           if (!arguments.length) return _heading;
-           _heading = utilFunctor(val);
-           return tooltip;
+         roadspeed.focus = function () {
+           input.node().focus();
          };
 
-         tooltip.keys = function (val) {
-           if (!arguments.length) return _keys;
-           _keys = utilFunctor(val);
-           return tooltip;
+         roadspeed.entityIDs = function (val) {
+           _entityIDs = val;
          };
 
-         tooltip.content(function () {
-           var heading = _heading.apply(this, arguments);
-
-           var text = _title.apply(this, arguments);
-
-           var keys = _keys.apply(this, arguments);
+         function combinedEntityExtent() {
+           return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+         }
 
-           return function (selection) {
-             var headingSelect = selection.selectAll('.tooltip-heading').data(heading ? [heading] : []);
-             headingSelect.exit().remove();
-             headingSelect.enter().append('div').attr('class', 'tooltip-heading').merge(headingSelect).html(heading);
-             var textSelect = selection.selectAll('.tooltip-text').data(text ? [text] : []);
-             textSelect.exit().remove();
-             textSelect.enter().append('div').attr('class', 'tooltip-text').merge(textSelect).html(text);
-             var keyhintWrap = selection.selectAll('.keyhint-wrap').data(keys && keys.length ? [0] : []);
-             keyhintWrap.exit().remove();
-             var keyhintWrapEnter = keyhintWrap.enter().append('div').attr('class', 'keyhint-wrap');
-             keyhintWrapEnter.append('span').html(_t.html('tooltip_keyhint'));
-             keyhintWrap = keyhintWrapEnter.merge(keyhintWrap);
-             keyhintWrap.selectAll('kbd.shortcut').data(keys && keys.length ? keys : []).enter().append('kbd').attr('class', 'shortcut').html(function (d) {
-               return d;
-             });
-           };
-         });
-         return tooltip;
+         return utilRebind(roadspeed, dispatch, 'on');
        }
 
-       function uiEditMenu(context) {
-         var dispatch$1 = dispatch('toggled');
-
-         var _menu = select(null);
-
-         var _operations = []; // the position the menu should be displayed relative to
-
-         var _anchorLoc = [0, 0];
-         var _anchorLocLonLat = [0, 0]; // a string indicating how the menu was opened
-
-         var _triggerType = '';
-         var _vpTopMargin = 85; // viewport top margin
-
-         var _vpBottomMargin = 45; // viewport bottom margin
+       function uiFieldRadio(field, context) {
+         var dispatch = dispatch$8('change');
+         var placeholder = select(null);
+         var wrap = select(null);
+         var labels = select(null);
+         var radios = select(null);
+         var radioData = (field.options || field.keys).slice(); // shallow copy
 
-         var _vpSideMargin = 35; // viewport side margin
+         var typeField;
+         var layerField;
+         var _oldType = {};
+         var _entityIDs = [];
 
-         var _menuTop = false;
+         function selectedKey() {
+           var node = wrap.selectAll('.form-field-input-radio label.active input');
+           return !node.empty() && node.datum();
+         }
 
-         var _menuHeight;
+         function radio(selection) {
+           selection.classed('preset-radio', true);
+           wrap = selection.selectAll('.form-field-input-wrap').data([0]);
+           var enter = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-radio');
+           enter.append('span').attr('class', 'placeholder');
+           wrap = wrap.merge(enter);
+           placeholder = wrap.selectAll('.placeholder');
+           labels = wrap.selectAll('label').data(radioData);
+           enter = labels.enter().append('label');
+           enter.append('input').attr('type', 'radio').attr('name', field.id).attr('value', function (d) {
+             return field.t('options.' + d, {
+               'default': d
+             });
+           }).attr('checked', false);
+           enter.append('span').html(function (d) {
+             return field.t.html('options.' + d, {
+               'default': d
+             });
+           });
+           labels = labels.merge(enter);
+           radios = labels.selectAll('input').on('change', changeRadio);
+         }
 
-         var _menuWidth; // hardcode these values to make menu positioning easier
+         function structureExtras(selection, tags) {
+           var selected = selectedKey() || tags.layer !== undefined;
+           var type = _mainPresetIndex.field(selected);
+           var layer = _mainPresetIndex.field('layer');
+           var showLayer = selected === 'bridge' || selected === 'tunnel' || tags.layer !== undefined;
+           var extrasWrap = selection.selectAll('.structure-extras-wrap').data(selected ? [0] : []);
+           extrasWrap.exit().remove();
+           extrasWrap = extrasWrap.enter().append('div').attr('class', 'structure-extras-wrap').merge(extrasWrap);
+           var list = extrasWrap.selectAll('ul').data([0]);
+           list = list.enter().append('ul').attr('class', 'rows').merge(list); // Type
 
+           if (type) {
+             if (!typeField || typeField.id !== selected) {
+               typeField = uiField(context, type, _entityIDs, {
+                 wrap: false
+               }).on('change', changeType);
+             }
 
-         var _verticalPadding = 4; // see also `.edit-menu .tooltip` CSS; include margin
+             typeField.tags(tags);
+           } else {
+             typeField = null;
+           }
 
-         var _tooltipWidth = 210; // offset the menu slightly from the target location
+           var typeItem = list.selectAll('.structure-type-item').data(typeField ? [typeField] : [], function (d) {
+             return d.id;
+           }); // Exit
 
-         var _menuSideMargin = 10;
-         var _tooltips = [];
+           typeItem.exit().remove(); // Enter
 
-         var editMenu = function editMenu(selection) {
-           var isTouchMenu = _triggerType.includes('touch') || _triggerType.includes('pen');
+           var typeEnter = typeItem.enter().insert('li', ':first-child').attr('class', 'labeled-input structure-type-item');
+           typeEnter.append('span').attr('class', 'label structure-label-type').attr('for', 'preset-input-' + selected).html(_t.html('inspector.radio.structure.type'));
+           typeEnter.append('div').attr('class', 'structure-input-type-wrap'); // Update
 
-           var ops = _operations.filter(function (op) {
-             return !isTouchMenu || !op.mouseOnly;
-           });
+           typeItem = typeItem.merge(typeEnter);
 
-           if (!ops.length) return;
-           _tooltips = []; // Position the menu above the anchor for stylus and finger input
-           // since the mapper's hand likely obscures the screen below the anchor
+           if (typeField) {
+             typeItem.selectAll('.structure-input-type-wrap').call(typeField.render);
+           } // Layer
 
-           _menuTop = isTouchMenu; // Show labels for touch input since there aren't hover tooltips
 
-           var showLabels = isTouchMenu;
-           var buttonHeight = showLabels ? 32 : 34;
+           if (layer && showLayer) {
+             if (!layerField) {
+               layerField = uiField(context, layer, _entityIDs, {
+                 wrap: false
+               }).on('change', changeLayer);
+             }
 
-           if (showLabels) {
-             // Get a general idea of the width based on the length of the label
-             _menuWidth = 52 + Math.min(120, 6 * Math.max.apply(Math, ops.map(function (op) {
-               return op.title.length;
-             })));
+             layerField.tags(tags);
+             field.keys = utilArrayUnion(field.keys, ['layer']);
            } else {
-             _menuWidth = 44;
+             layerField = null;
+             field.keys = field.keys.filter(function (k) {
+               return k !== 'layer';
+             });
            }
 
-           _menuHeight = _verticalPadding * 2 + ops.length * buttonHeight;
-           _menu = selection.append('div').attr('class', 'edit-menu').classed('touch-menu', isTouchMenu).style('padding', _verticalPadding + 'px 0');
-
-           var buttons = _menu.selectAll('.edit-menu-item').data(ops); // enter
-
-
-           var buttonsEnter = buttons.enter().append('button').attr('class', function (d) {
-             return 'edit-menu-item edit-menu-item-' + d.id;
-           }).style('height', buttonHeight + 'px').on('click', click) // don't listen for `mouseup` because we only care about non-mouse pointer types
-           .on('pointerup', pointerup).on('pointerdown mousedown', function pointerdown(d3_event) {
-             // don't let button presses also act as map input - #1869
-             d3_event.stopPropagation();
-           }).on('mouseenter.highlight', function (d3_event, d) {
-             if (!d.relatedEntityIds || select(this).classed('disabled')) return;
-             utilHighlightEntities(d.relatedEntityIds(), true, context);
-           }).on('mouseleave.highlight', function (d3_event, d) {
-             if (!d.relatedEntityIds) return;
-             utilHighlightEntities(d.relatedEntityIds(), false, context);
-           });
-           buttonsEnter.each(function (d) {
-             var tooltip = uiTooltip().heading(d.title).title(d.tooltip()).keys([d.keys[0]]);
+           var layerItem = list.selectAll('.structure-layer-item').data(layerField ? [layerField] : []); // Exit
 
-             _tooltips.push(tooltip);
+           layerItem.exit().remove(); // Enter
 
-             select(this).call(tooltip).append('div').attr('class', 'icon-wrap').call(svgIcon('#iD-operation-' + d.id, 'operation'));
-           });
+           var layerEnter = layerItem.enter().append('li').attr('class', 'labeled-input structure-layer-item');
+           layerEnter.append('span').attr('class', 'label structure-label-layer').attr('for', 'preset-input-layer').html(_t.html('inspector.radio.structure.layer'));
+           layerEnter.append('div').attr('class', 'structure-input-layer-wrap'); // Update
 
-           if (showLabels) {
-             buttonsEnter.append('span').attr('class', 'label').html(function (d) {
-               return d.title;
-             });
-           } // update
+           layerItem = layerItem.merge(layerEnter);
 
+           if (layerField) {
+             layerItem.selectAll('.structure-input-layer-wrap').call(layerField.render);
+           }
+         }
 
-           buttonsEnter.merge(buttons).classed('disabled', function (d) {
-             return d.disabled();
-           });
-           updatePosition();
-           var initialScale = context.projection.scale();
-           context.map().on('move.edit-menu', function () {
-             if (initialScale !== context.projection.scale()) {
-               editMenu.close();
-             }
-           }).on('drawn.edit-menu', function (info) {
-             if (info.full) updatePosition();
-           });
-           var lastPointerUpType; // `pointerup` is always called before `click`
+         function changeType(t, onInput) {
+           var key = selectedKey();
+           if (!key) return;
+           var val = t[key];
 
-           function pointerup(d3_event) {
-             lastPointerUpType = d3_event.pointerType;
+           if (val !== 'no') {
+             _oldType[key] = val;
            }
 
-           function click(d3_event, operation) {
-             d3_event.stopPropagation();
+           if (field.type === 'structureRadio') {
+             // remove layer if it should not be set
+             if (val === 'no' || key !== 'bridge' && key !== 'tunnel' || key === 'tunnel' && val === 'building_passage') {
+               t.layer = undefined;
+             } // add layer if it should be set
 
-             if (operation.relatedEntityIds) {
-               utilHighlightEntities(operation.relatedEntityIds(), false, context);
-             }
 
-             if (operation.disabled()) {
-               if (lastPointerUpType === 'touch' || lastPointerUpType === 'pen') {
-                 // there are no tooltips for touch interactions so flash feedback instead
-                 context.ui().flash.duration(4000).iconName('#iD-operation-' + operation.id).iconClass('operation disabled').label(operation.tooltip)();
-               }
-             } else {
-               if (lastPointerUpType === 'touch' || lastPointerUpType === 'pen') {
-                 context.ui().flash.duration(2000).iconName('#iD-operation-' + operation.id).iconClass('operation').label(operation.annotation() || operation.title)();
+             if (t.layer === undefined) {
+               if (key === 'bridge' && val !== 'no') {
+                 t.layer = '1';
                }
 
-               operation();
-               editMenu.close();
+               if (key === 'tunnel' && val !== 'no' && val !== 'building_passage') {
+                 t.layer = '-1';
+               }
              }
+           }
 
-             lastPointerUpType = null;
+           dispatch.call('change', this, t, onInput);
+         }
+
+         function changeLayer(t, onInput) {
+           if (t.layer === '0') {
+             t.layer = undefined;
            }
 
-           dispatch$1.call('toggled', this, true);
-         };
+           dispatch.call('change', this, t, onInput);
+         }
 
-         function updatePosition() {
-           if (!_menu || _menu.empty()) return;
-           var anchorLoc = context.projection(_anchorLocLonLat);
-           var viewport = context.surfaceRect();
+         function changeRadio() {
+           var t = {};
+           var activeKey;
 
-           if (anchorLoc[0] < 0 || anchorLoc[0] > viewport.width || anchorLoc[1] < 0 || anchorLoc[1] > viewport.height) {
-             // close the menu if it's gone offscreen
-             editMenu.close();
-             return;
+           if (field.key) {
+             t[field.key] = undefined;
            }
 
-           var menuLeft = displayOnLeft(viewport);
-           var offset = [0, 0];
-           offset[0] = menuLeft ? -1 * (_menuSideMargin + _menuWidth) : _menuSideMargin;
+           radios.each(function (d) {
+             var active = select(this).property('checked');
+             if (active) activeKey = d;
 
-           if (_menuTop) {
-             if (anchorLoc[1] - _menuHeight < _vpTopMargin) {
-               // menu is near top viewport edge, shift downward
-               offset[1] = -anchorLoc[1] + _vpTopMargin;
+             if (field.key) {
+               if (active) t[field.key] = d;
              } else {
-               offset[1] = -_menuHeight;
+               var val = _oldType[activeKey] || 'yes';
+               t[d] = active ? val : undefined;
              }
-           } else {
-             if (anchorLoc[1] + _menuHeight > viewport.height - _vpBottomMargin) {
-               // menu is near bottom viewport edge, shift upwards
-               offset[1] = -anchorLoc[1] - _menuHeight + viewport.height - _vpBottomMargin;
+           });
+
+           if (field.type === 'structureRadio') {
+             if (activeKey === 'bridge') {
+               t.layer = '1';
+             } else if (activeKey === 'tunnel' && t.tunnel !== 'building_passage') {
+               t.layer = '-1';
              } else {
-               offset[1] = 0;
+               t.layer = undefined;
              }
            }
 
-           var origin = geoVecAdd(anchorLoc, offset);
-
-           _menu.style('left', origin[0] + 'px').style('top', origin[1] + 'px');
+           dispatch.call('change', this, t);
+         }
 
-           var tooltipSide = tooltipPosition(viewport, menuLeft);
+         radio.tags = function (tags) {
+           radios.property('checked', function (d) {
+             if (field.key) {
+               return tags[field.key] === d;
+             }
 
-           _tooltips.forEach(function (tooltip) {
-             tooltip.placement(tooltipSide);
+             return !!(typeof tags[d] === 'string' && tags[d].toLowerCase() !== 'no');
            });
 
-           function displayOnLeft(viewport) {
-             if (_mainLocalizer.textDirection() === 'ltr') {
-               if (anchorLoc[0] + _menuSideMargin + _menuWidth > viewport.width - _vpSideMargin) {
-                 // right menu would be too close to the right viewport edge, go left
-                 return true;
-               } // prefer right menu
-
-
-               return false;
-             } else {
-               // rtl
-               if (anchorLoc[0] - _menuSideMargin - _menuWidth < _vpSideMargin) {
-                 // left menu would be too close to the left viewport edge, go right
-                 return false;
-               } // prefer left menu
-
-
-               return true;
+           function isMixed(d) {
+             if (field.key) {
+               return Array.isArray(tags[field.key]) && tags[field.key].includes(d);
              }
-           }
-
-           function tooltipPosition(viewport, menuLeft) {
-             if (_mainLocalizer.textDirection() === 'ltr') {
-               if (menuLeft) {
-                 // if there's not room for a right-side menu then there definitely
-                 // isn't room for right-side tooltips
-                 return 'left';
-               }
-
-               if (anchorLoc[0] + _menuSideMargin + _menuWidth + _tooltipWidth > viewport.width - _vpSideMargin) {
-                 // right tooltips would be too close to the right viewport edge, go left
-                 return 'left';
-               } // prefer right tooltips
 
+             return Array.isArray(tags[d]);
+           }
 
-               return 'right';
-             } else {
-               // rtl
-               if (!menuLeft) {
-                 return 'right';
-               }
-
-               if (anchorLoc[0] - _menuSideMargin - _menuWidth - _tooltipWidth < _vpSideMargin) {
-                 // left tooltips would be too close to the left viewport edge, go right
-                 return 'right';
-               } // prefer left tooltips
+           labels.classed('active', function (d) {
+             if (field.key) {
+               return Array.isArray(tags[field.key]) && tags[field.key].includes(d) || tags[field.key] === d;
+             }
 
+             return Array.isArray(tags[d]) || !!(tags[d] && tags[d].toLowerCase() !== 'no');
+           }).classed('mixed', isMixed).attr('title', function (d) {
+             return isMixed(d) ? _t('inspector.unshared_value_tooltip') : null;
+           });
+           var selection = radios.filter(function () {
+             return this.checked;
+           });
 
-               return 'left';
-             }
+           if (selection.empty()) {
+             placeholder.html(_t.html('inspector.none'));
+           } else {
+             placeholder.html(selection.attr('value'));
+             _oldType[selection.datum()] = tags[selection.datum()];
            }
-         }
 
-         editMenu.close = function () {
-           context.map().on('move.edit-menu', null).on('drawn.edit-menu', null);
-
-           _menu.remove();
+           if (field.type === 'structureRadio') {
+             // For waterways without a tunnel tag, set 'culvert' as
+             // the _oldType to default to if the user picks 'tunnel'
+             if (!!tags.waterway && !_oldType.tunnel) {
+               _oldType.tunnel = 'culvert';
+             }
 
-           _tooltips = [];
-           dispatch$1.call('toggled', this, false);
+             wrap.call(structureExtras, tags);
+           }
          };
 
-         editMenu.anchorLoc = function (val) {
-           if (!arguments.length) return _anchorLoc;
-           _anchorLoc = val;
-           _anchorLocLonLat = context.projection.invert(_anchorLoc);
-           return editMenu;
+         radio.focus = function () {
+           radios.node().focus();
          };
 
-         editMenu.triggerType = function (val) {
-           if (!arguments.length) return _triggerType;
-           _triggerType = val;
-           return editMenu;
+         radio.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs;
+           _entityIDs = val;
+           _oldType = {};
+           return radio;
          };
 
-         editMenu.operations = function (val) {
-           if (!arguments.length) return _operations;
-           _operations = val;
-           return editMenu;
+         radio.isAllowed = function () {
+           return _entityIDs.length === 1;
          };
 
-         return utilRebind(editMenu, dispatch$1, 'on');
+         return utilRebind(radio, dispatch, 'on');
        }
 
-       function uiFeatureInfo(context) {
-         function update(selection) {
-           var features = context.features();
-           var stats = features.stats();
-           var count = 0;
-           var hiddenList = features.hidden().map(function (k) {
-             if (stats[k]) {
-               count += stats[k];
-               return _t('inspector.title_count', {
-                 title: _t.html('feature.' + k + '.description'),
-                 count: stats[k]
-               });
-             }
+       function uiFieldRestrictions(field, context) {
+         var dispatch = dispatch$8('change');
+         var breathe = behaviorBreathe();
+         corePreferences('turn-restriction-via-way', null); // remove old key
 
-             return null;
-           }).filter(Boolean);
-           selection.html('');
+         var storedViaWay = corePreferences('turn-restriction-via-way0'); // use new key #6922
 
-           if (hiddenList.length) {
-             var tooltipBehavior = uiTooltip().placement('top').title(function () {
-               return hiddenList.join('<br/>');
-             });
-             selection.append('a').attr('class', 'chip').attr('href', '#').html(_t.html('feature_info.hidden_warning', {
-               count: count
-             })).call(tooltipBehavior).on('click', function (d3_event) {
-               tooltipBehavior.hide();
-               d3_event.preventDefault(); // open the Map Data pane
+         var storedDistance = corePreferences('turn-restriction-distance');
 
-               context.ui().togglePanes(context.container().select('.map-panes .map-data-pane'));
-             });
-           }
+         var _maxViaWay = storedViaWay !== null ? +storedViaWay : 0;
 
-           selection.classed('hide', !hiddenList.length);
-         }
+         var _maxDistance = storedDistance ? +storedDistance : 30;
 
-         return function (selection) {
-           update(selection);
-           context.features().on('change.feature_info', function () {
-             update(selection);
-           });
-         };
-       }
+         var _initialized = false;
 
-       function uiFlash(context) {
-         var _flashTimer;
+         var _parent = select(null); // the entire field
 
-         var _duration = 2000;
-         var _iconName = '#iD-icon-no';
-         var _iconClass = 'disabled';
-         var _label = '';
 
-         function flash() {
-           if (_flashTimer) {
-             _flashTimer.stop();
-           }
+         var _container = select(null); // just the map
 
-           context.container().select('.main-footer-wrap').classed('footer-hide', true).classed('footer-show', false);
-           context.container().select('.flash-wrap').classed('footer-hide', false).classed('footer-show', true);
-           var content = context.container().select('.flash-wrap').selectAll('.flash-content').data([0]); // Enter
 
-           var contentEnter = content.enter().append('div').attr('class', 'flash-content');
-           var iconEnter = contentEnter.append('svg').attr('class', 'flash-icon icon').append('g').attr('transform', 'translate(10,10)');
-           iconEnter.append('circle').attr('r', 9);
-           iconEnter.append('use').attr('transform', 'translate(-7,-7)').attr('width', '14').attr('height', '14');
-           contentEnter.append('div').attr('class', 'flash-text'); // Update
+         var _oldTurns;
 
-           content = content.merge(contentEnter);
-           content.selectAll('.flash-icon').attr('class', 'icon flash-icon ' + (_iconClass || ''));
-           content.selectAll('.flash-icon use').attr('xlink:href', _iconName);
-           content.selectAll('.flash-text').attr('class', 'flash-text').html(_label);
-           _flashTimer = d3_timeout(function () {
-             _flashTimer = null;
-             context.container().select('.main-footer-wrap').classed('footer-hide', false).classed('footer-show', true);
-             context.container().select('.flash-wrap').classed('footer-hide', true).classed('footer-show', false);
-           }, _duration);
-           return content;
-         }
+         var _graph;
 
-         flash.duration = function (_) {
-           if (!arguments.length) return _duration;
-           _duration = _;
-           return flash;
-         };
+         var _vertexID;
 
-         flash.label = function (_) {
-           if (!arguments.length) return _label;
-           _label = _;
-           return flash;
-         };
+         var _intersection;
 
-         flash.iconName = function (_) {
-           if (!arguments.length) return _iconName;
-           _iconName = _;
-           return flash;
-         };
+         var _fromWayID;
 
-         flash.iconClass = function (_) {
-           if (!arguments.length) return _iconClass;
-           _iconClass = _;
-           return flash;
-         };
+         var _lastXPos;
 
-         return flash;
-       }
+         function restrictions(selection) {
+           _parent = selection; // try to reuse the intersection, but always rebuild it if the graph has changed
 
-       function uiFullScreen(context) {
-         var element = context.container().node(); // var button = d3_select(null);
+           if (_vertexID && (context.graph() !== _graph || !_intersection)) {
+             _graph = context.graph();
+             _intersection = osmIntersection(_graph, _vertexID, _maxDistance);
+           } // It's possible for there to be no actual intersection here.
+           // for example, a vertex of two `highway=path`
+           // In this case, hide the field.
 
-         function getFullScreenFn() {
-           if (element.requestFullscreen) {
-             return element.requestFullscreen;
-           } else if (element.msRequestFullscreen) {
-             return element.msRequestFullscreen;
-           } else if (element.mozRequestFullScreen) {
-             return element.mozRequestFullScreen;
-           } else if (element.webkitRequestFullscreen) {
-             return element.webkitRequestFullscreen;
-           }
-         }
 
-         function getExitFullScreenFn() {
-           if (document.exitFullscreen) {
-             return document.exitFullscreen;
-           } else if (document.msExitFullscreen) {
-             return document.msExitFullscreen;
-           } else if (document.mozCancelFullScreen) {
-             return document.mozCancelFullScreen;
-           } else if (document.webkitExitFullscreen) {
-             return document.webkitExitFullscreen;
+           var isOK = _intersection && _intersection.vertices.length && // has vertices
+           _intersection.vertices // has the vertex that the user selected
+           .filter(function (vertex) {
+             return vertex.id === _vertexID;
+           }).length && _intersection.ways.length > 2 && // has more than 2 ways
+           _intersection.ways // has more than 1 TO way
+           .filter(function (way) {
+             return way.__to;
+           }).length > 1; // Also hide in the case where
+
+           select(selection.node().parentNode).classed('hide', !isOK); // if form field is hidden or has detached from dom, clean up.
+
+           if (!isOK || !context.container().select('.inspector-wrap.inspector-hidden').empty() || !selection.node().parentNode || !selection.node().parentNode.parentNode) {
+             selection.call(restrictions.off);
+             return;
            }
-         }
 
-         function isFullScreen() {
-           return document.fullscreenElement || document.mozFullScreenElement || document.webkitFullscreenElement || document.msFullscreenElement;
-         }
+           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
+           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
+           var container = wrap.selectAll('.restriction-container').data([0]); // enter
 
-         function isSupported() {
-           return !!getFullScreenFn();
-         }
+           var containerEnter = container.enter().append('div').attr('class', 'restriction-container');
+           containerEnter.append('div').attr('class', 'restriction-help'); // update
 
-         function fullScreen(d3_event) {
-           d3_event.preventDefault();
+           _container = containerEnter.merge(container).call(renderViewer);
+           var controls = wrap.selectAll('.restriction-controls').data([0]); // enter/update
 
-           if (!isFullScreen()) {
-             // button.classed('active', true);
-             getFullScreenFn().apply(element);
-           } else {
-             // button.classed('active', false);
-             getExitFullScreenFn().apply(document);
-           }
+           controls.enter().append('div').attr('class', 'restriction-controls-container').append('div').attr('class', 'restriction-controls').merge(controls).call(renderControls);
          }
 
-         return function () {
-           // selection) {
-           if (!isSupported()) return; // button = selection.append('button')
-           //     .attr('title', t('full_screen'))
-           //     .on('click', fullScreen)
-           //     .call(tooltip);
-           // button.append('span')
-           //     .attr('class', 'icon full-screen');
+         function renderControls(selection) {
+           var distControl = selection.selectAll('.restriction-distance').data([0]);
+           distControl.exit().remove();
+           var distControlEnter = distControl.enter().append('div').attr('class', 'restriction-control restriction-distance');
+           distControlEnter.append('span').attr('class', 'restriction-control-label restriction-distance-label').html(_t.html('restriction.controls.distance') + ':');
+           distControlEnter.append('input').attr('class', 'restriction-distance-input').attr('type', 'range').attr('min', '20').attr('max', '50').attr('step', '5');
+           distControlEnter.append('span').attr('class', 'restriction-distance-text'); // update
 
-           var detected = utilDetect();
-           var keys = detected.os === 'mac' ? [uiCmd('⌃⌘F'), 'f11'] : ['f11'];
-           context.keybinding().on(keys, fullScreen);
-         };
-       }
+           selection.selectAll('.restriction-distance-input').property('value', _maxDistance).on('input', function () {
+             var val = select(this).property('value');
+             _maxDistance = +val;
+             _intersection = null;
 
-       function uiGeolocate(context) {
-         var _geolocationOptions = {
-           // prioritize speed and power usage over precision
-           enableHighAccuracy: false,
-           // don't hang indefinitely getting the location
-           timeout: 6000 // 6sec
+             _container.selectAll('.layer-osm .layer-turns *').remove();
 
-         };
+             corePreferences('turn-restriction-distance', _maxDistance);
 
-         var _locating = uiLoading(context).message(_t.html('geolocate.locating')).blocking(true);
+             _parent.call(restrictions);
+           });
+           selection.selectAll('.restriction-distance-text').html(displayMaxDistance(_maxDistance));
+           var viaControl = selection.selectAll('.restriction-via-way').data([0]);
+           viaControl.exit().remove();
+           var viaControlEnter = viaControl.enter().append('div').attr('class', 'restriction-control restriction-via-way');
+           viaControlEnter.append('span').attr('class', 'restriction-control-label restriction-via-way-label').html(_t.html('restriction.controls.via') + ':');
+           viaControlEnter.append('input').attr('class', 'restriction-via-way-input').attr('type', 'range').attr('min', '0').attr('max', '2').attr('step', '1');
+           viaControlEnter.append('span').attr('class', 'restriction-via-way-text'); // update
 
-         var _layer = context.layers().layer('geolocate');
+           selection.selectAll('.restriction-via-way-input').property('value', _maxViaWay).on('input', function () {
+             var val = select(this).property('value');
+             _maxViaWay = +val;
 
-         var _position;
+             _container.selectAll('.layer-osm .layer-turns *').remove();
 
-         var _extent;
+             corePreferences('turn-restriction-via-way0', _maxViaWay);
 
-         var _timeoutID;
+             _parent.call(restrictions);
+           });
+           selection.selectAll('.restriction-via-way-text').html(displayMaxVia(_maxViaWay));
+         }
 
-         var _button = select(null);
+         function renderViewer(selection) {
+           if (!_intersection) return;
+           var vgraph = _intersection.graph;
+           var filter = utilFunctor(true);
+           var projection = geoRawMercator(); // Reflow warning: `utilGetDimensions` calls `getBoundingClientRect`
+           // Instead of asking the restriction-container for its dimensions,
+           //  we can ask the .sidebar, which can have its dimensions cached.
+           // width: calc as sidebar - padding
+           // height: hardcoded (from `80_app.css`)
+           // var d = utilGetDimensions(selection);
 
-         function click() {
-           if (context.inIntro()) return;
+           var sdims = utilGetDimensions(context.container().select('.sidebar'));
+           var d = [sdims[0] - 50, 370];
+           var c = geoVecScale(d, 0.5);
+           var z = 22;
+           projection.scale(geoZoomToScale(z)); // Calculate extent of all key vertices
 
-           if (!_layer.enabled() && !_locating.isShown()) {
-             // This timeout ensures that we still call finish() even if
-             // the user declines to share their location in Firefox
-             _timeoutID = setTimeout(error, 10000
-             /* 10sec */
-             );
-             context.container().call(_locating); // get the latest position even if we already have one
+           var extent = geoExtent();
 
-             navigator.geolocation.getCurrentPosition(success, error, _geolocationOptions);
-           } else {
-             _locating.close();
+           for (var i = 0; i < _intersection.vertices.length; i++) {
+             extent._extend(_intersection.vertices[i].extent());
+           } // If this is a large intersection, adjust zoom to fit extent
 
-             _layer.enabled(null, false);
 
-             updateButtonState();
+           if (_intersection.vertices.length > 1) {
+             var padding = 180; // in z22 pixels
+
+             var tl = projection([extent[0][0], extent[1][1]]);
+             var br = projection([extent[1][0], extent[0][1]]);
+             var hFactor = (br[0] - tl[0]) / (d[0] - padding);
+             var vFactor = (br[1] - tl[1]) / (d[1] - padding);
+             var hZoomDiff = Math.log(Math.abs(hFactor)) / Math.LN2;
+             var vZoomDiff = Math.log(Math.abs(vFactor)) / Math.LN2;
+             z = z - Math.max(hZoomDiff, vZoomDiff);
+             projection.scale(geoZoomToScale(z));
            }
-         }
 
-         function zoomTo() {
-           context.enter(modeBrowse(context));
-           var map = context.map();
+           var padTop = 35; // reserve top space for hint text
 
-           _layer.enabled(_position, true);
+           var extentCenter = projection(extent.center());
+           extentCenter[1] = extentCenter[1] - padTop;
+           projection.translate(geoVecSubtract(c, extentCenter)).clipExtent([[0, 0], d]);
+           var drawLayers = svgLayers(projection, context).only(['osm', 'touch']).dimensions(d);
+           var drawVertices = svgVertices(projection, context);
+           var drawLines = svgLines(projection, context);
+           var drawTurns = svgTurns(projection, context);
+           var firstTime = selection.selectAll('.surface').empty();
+           selection.call(drawLayers);
+           var surface = selection.selectAll('.surface').classed('tr', true);
 
-           updateButtonState();
-           map.centerZoomEase(_extent.center(), Math.min(20, map.extentZoom(_extent)));
-         }
+           if (firstTime) {
+             _initialized = true;
+             surface.call(breathe);
+           } // This can happen if we've lowered the detail while a FROM way
+           // is selected, and that way is no longer part of the intersection.
 
-         function success(geolocation) {
-           _position = geolocation;
-           var coords = _position.coords;
-           _extent = geoExtent([coords.longitude, coords.latitude]).padByMeters(coords.accuracy);
-           zoomTo();
-           finish();
-         }
 
-         function error() {
-           if (_position) {
-             // use the position from a previous call if we have one
-             zoomTo();
-           } else {
-             context.ui().flash.label(_t.html('geolocate.location_unavailable')).iconName('#iD-icon-geolocate')();
+           if (_fromWayID && !vgraph.hasEntity(_fromWayID)) {
+             _fromWayID = null;
+             _oldTurns = null;
            }
 
-           finish();
-         }
+           surface.call(utilSetDimensions, d).call(drawVertices, vgraph, _intersection.vertices, filter, extent, z).call(drawLines, vgraph, _intersection.ways, filter).call(drawTurns, vgraph, _intersection.turns(_fromWayID, _maxViaWay));
+           surface.on('click.restrictions', click).on('mouseover.restrictions', mouseover);
+           surface.selectAll('.selected').classed('selected', false);
+           surface.selectAll('.related').classed('related', false);
+           var way;
 
-         function finish() {
-           _locating.close(); // unblock ui
+           if (_fromWayID) {
+             way = vgraph.entity(_fromWayID);
+             surface.selectAll('.' + _fromWayID).classed('selected', true).classed('related', true);
+           }
 
+           document.addEventListener('resizeWindow', function () {
+             utilSetDimensions(_container, null);
+             redraw(1);
+           }, false);
+           updateHints(null);
 
-           if (_timeoutID) {
-             clearTimeout(_timeoutID);
-           }
+           function click(d3_event) {
+             surface.call(breathe.off).call(breathe);
+             var datum = d3_event.target.__data__;
+             var entity = datum && datum.properties && datum.properties.entity;
 
-           _timeoutID = undefined;
-         }
+             if (entity) {
+               datum = entity;
+             }
 
-         function updateButtonState() {
-           _button.classed('active', _layer.enabled());
-         }
+             if (datum instanceof osmWay && (datum.__from || datum.__via)) {
+               _fromWayID = datum.id;
+               _oldTurns = null;
+               redraw();
+             } else if (datum instanceof osmTurn) {
+               var actions, extraActions, turns, i;
+               var restrictionType = osmInferRestriction(vgraph, datum, projection);
 
-         return function (selection) {
-           if (!navigator.geolocation || !navigator.geolocation.getCurrentPosition) return;
-           _button = selection.append('button').on('click', click).call(svgIcon('#iD-icon-geolocate', 'light')).call(uiTooltip().placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left').title(_t.html('geolocate.title')).keys([_t('geolocate.key')]));
-           context.keybinding().on(_t('geolocate.key'), click);
-         };
-       }
+               if (datum.restrictionID && !datum.direct) {
+                 return;
+               } else if (datum.restrictionID && !datum.only) {
+                 // NO -> ONLY
+                 var seen = {};
+                 var datumOnly = JSON.parse(JSON.stringify(datum)); // deep clone the datum
 
-       function uiPanelBackground(context) {
-         var background = context.background();
-         var _currSourceName = null;
-         var _metadata = {};
-         var _metadataKeys = ['zoom', 'vintage', 'source', 'description', 'resolution', 'accuracy'];
+                 datumOnly.only = true; // but change this property
 
-         var debouncedRedraw = debounce(redraw, 250);
+                 restrictionType = restrictionType.replace(/^no/, 'only'); // Adding an ONLY restriction should destroy all other direct restrictions from the FROM towards the VIA.
+                 // We will remember them in _oldTurns, and restore them if the user clicks again.
 
-         function redraw(selection) {
-           var source = background.baseLayerSource();
-           if (!source) return;
-           var isDG = source.id.match(/^DigitalGlobe/i) !== null;
-           var sourceLabel = source.label();
+                 turns = _intersection.turns(_fromWayID, 2);
+                 extraActions = [];
+                 _oldTurns = [];
 
-           if (_currSourceName !== sourceLabel) {
-             _currSourceName = sourceLabel;
-             _metadata = {};
-           }
+                 for (i = 0; i < turns.length; i++) {
+                   var turn = turns[i];
+                   if (seen[turn.restrictionID]) continue; // avoid deleting the turn twice (#4968, #4928)
 
-           selection.html('');
-           var list = selection.append('ul').attr('class', 'background-info');
-           list.append('li').html(_currSourceName);
+                   if (turn.direct && turn.path[1] === datum.path[1]) {
+                     seen[turns[i].restrictionID] = true;
+                     turn.restrictionType = osmInferRestriction(vgraph, turn, projection);
 
-           _metadataKeys.forEach(function (k) {
-             // DigitalGlobe vintage is available in raster layers for now.
-             if (isDG && k === 'vintage') return;
-             list.append('li').attr('class', 'background-info-list-' + k).classed('hide', !_metadata[k]).html(_t.html('info_panels.background.' + k) + ':').append('span').attr('class', 'background-info-span-' + k).html(_metadata[k]);
-           });
+                     _oldTurns.push(turn);
 
-           debouncedGetMetadata(selection);
-           var toggleTiles = context.getDebug('tile') ? 'hide_tiles' : 'show_tiles';
-           selection.append('a').html(_t.html('info_panels.background.' + toggleTiles)).attr('href', '#').attr('class', 'button button-toggle-tiles').on('click', function (d3_event) {
-             d3_event.preventDefault();
-             context.setDebug('tile', !context.getDebug('tile'));
-             selection.call(redraw);
-           });
+                     extraActions.push(actionUnrestrictTurn(turn));
+                   }
+                 }
 
-           if (isDG) {
-             var key = source.id + '-vintage';
-             var sourceVintage = context.background().findSource(key);
-             var showsVintage = context.background().showsLayer(sourceVintage);
-             var toggleVintage = showsVintage ? 'hide_vintage' : 'show_vintage';
-             selection.append('a').html(_t.html('info_panels.background.' + toggleVintage)).attr('href', '#').attr('class', 'button button-toggle-vintage').on('click', function (d3_event) {
-               d3_event.preventDefault();
-               context.background().toggleOverlayLayer(sourceVintage);
-               selection.call(redraw);
-             });
-           } // disable if necessary
+                 actions = _intersection.actions.concat(extraActions, [actionRestrictTurn(datumOnly, restrictionType), _t('operations.restriction.annotation.create')]);
+               } else if (datum.restrictionID) {
+                 // ONLY -> Allowed
+                 // Restore whatever restrictions we might have destroyed by cycling thru the ONLY state.
+                 // This relies on the assumption that the intersection was already split up when we
+                 // performed the previous action (NO -> ONLY), so the IDs in _oldTurns shouldn't have changed.
+                 turns = _oldTurns || [];
+                 extraActions = [];
 
+                 for (i = 0; i < turns.length; i++) {
+                   if (turns[i].key !== datum.key) {
+                     extraActions.push(actionRestrictTurn(turns[i], turns[i].restrictionType));
+                   }
+                 }
 
-           ['DigitalGlobe-Premium', 'DigitalGlobe-Standard'].forEach(function (layerId) {
-             if (source.id !== layerId) {
-               var key = layerId + '-vintage';
-               var sourceVintage = context.background().findSource(key);
+                 _oldTurns = null;
+                 actions = _intersection.actions.concat(extraActions, [actionUnrestrictTurn(datum), _t('operations.restriction.annotation.delete')]);
+               } else {
+                 // Allowed -> NO
+                 actions = _intersection.actions.concat([actionRestrictTurn(datum, restrictionType), _t('operations.restriction.annotation.create')]);
+               }
 
-               if (context.background().showsLayer(sourceVintage)) {
-                 context.background().toggleOverlayLayer(sourceVintage);
+               context.perform.apply(context, actions); // At this point the datum will be changed, but will have same key..
+               // Refresh it and update the help..
+
+               var s = surface.selectAll('.' + datum.key);
+               datum = s.empty() ? null : s.datum();
+               updateHints(datum);
+             } else {
+               _fromWayID = null;
+               _oldTurns = null;
+               redraw();
+             }
+           }
+
+           function mouseover(d3_event) {
+             var datum = d3_event.target.__data__;
+             updateHints(datum);
+           }
+
+           _lastXPos = _lastXPos || sdims[0];
+
+           function redraw(minChange) {
+             var xPos = -1;
+
+             if (minChange) {
+               xPos = utilGetDimensions(context.container().select('.sidebar'))[0];
+             }
+
+             if (!minChange || minChange && Math.abs(xPos - _lastXPos) >= minChange) {
+               if (context.hasEntity(_vertexID)) {
+                 _lastXPos = xPos;
+
+                 _container.call(renderViewer);
                }
              }
-           });
-         }
+           }
 
-         var debouncedGetMetadata = debounce(getMetadata, 250);
+           function highlightPathsFrom(wayID) {
+             surface.selectAll('.related').classed('related', false).classed('allow', false).classed('restrict', false).classed('only', false);
+             surface.selectAll('.' + wayID).classed('related', true);
 
-         function getMetadata(selection) {
-           var tile = context.container().select('.layer-background img.tile-center'); // tile near viewport center
+             if (wayID) {
+               var turns = _intersection.turns(wayID, _maxViaWay);
 
-           if (tile.empty()) return;
-           var sourceName = _currSourceName;
-           var d = tile.datum();
-           var zoom = d && d.length >= 3 && d[2] || Math.floor(context.map().zoom());
-           var center = context.map().center(); // update zoom
+               for (var i = 0; i < turns.length; i++) {
+                 var turn = turns[i];
+                 var ids = [turn.to.way];
+                 var klass = turn.no ? 'restrict' : turn.only ? 'only' : 'allow';
 
-           _metadata.zoom = String(zoom);
-           selection.selectAll('.background-info-list-zoom').classed('hide', false).selectAll('.background-info-span-zoom').html(_metadata.zoom);
-           if (!d || !d.length >= 3) return;
-           background.baseLayerSource().getMetadata(center, d, function (err, result) {
-             if (err || _currSourceName !== sourceName) return; // update vintage
+                 if (turn.only || turns.length === 1) {
+                   if (turn.via.ways) {
+                     ids = ids.concat(turn.via.ways);
+                   }
+                 } else if (turn.to.way === wayID) {
+                   continue;
+                 }
 
-             var vintage = result.vintage;
-             _metadata.vintage = vintage && vintage.range || _t('info_panels.background.unknown');
-             selection.selectAll('.background-info-list-vintage').classed('hide', false).selectAll('.background-info-span-vintage').html(_metadata.vintage); // update other _metadata
+                 surface.selectAll(utilEntitySelector(ids)).classed('related', true).classed('allow', klass === 'allow').classed('restrict', klass === 'restrict').classed('only', klass === 'only');
+               }
+             }
+           }
 
-             _metadataKeys.forEach(function (k) {
-               if (k === 'zoom' || k === 'vintage') return; // done already
+           function updateHints(datum) {
+             var help = _container.selectAll('.restriction-help').html('');
 
-               var val = result[k];
-               _metadata[k] = val;
-               selection.selectAll('.background-info-list-' + k).classed('hide', !val).selectAll('.background-info-span-' + k).html(val);
+             var placeholders = {};
+             ['from', 'via', 'to'].forEach(function (k) {
+               placeholders[k] = '<span class="qualifier">' + _t('restriction.help.' + k) + '</span>';
              });
-           });
-         }
+             var entity = datum && datum.properties && datum.properties.entity;
 
-         var panel = function panel(selection) {
-           selection.call(redraw);
-           context.map().on('drawn.info-background', function () {
-             selection.call(debouncedRedraw);
-           }).on('move.info-background', function () {
-             selection.call(debouncedGetMetadata);
-           });
-         };
+             if (entity) {
+               datum = entity;
+             }
 
-         panel.off = function () {
-           context.map().on('drawn.info-background', null).on('move.info-background', null);
-         };
+             if (_fromWayID) {
+               way = vgraph.entity(_fromWayID);
+               surface.selectAll('.' + _fromWayID).classed('selected', true).classed('related', true);
+             } // Hovering a way
 
-         panel.id = 'background';
-         panel.label = _t.html('info_panels.background.title');
-         panel.key = _t('info_panels.background.key');
-         return panel;
-       }
 
-       function uiPanelHistory(context) {
-         var osm;
+             if (datum instanceof osmWay && datum.__from) {
+               way = datum;
+               highlightPathsFrom(_fromWayID ? null : way.id);
+               surface.selectAll('.' + way.id).classed('related', true);
+               var clickSelect = !_fromWayID || _fromWayID !== way.id;
+               help.append('div') // "Click to select FROM {fromName}." / "FROM {fromName}"
+               .html(_t.html('restriction.help.' + (clickSelect ? 'select_from_name' : 'from_name'), {
+                 from: placeholders.from,
+                 fromName: displayName(way.id, vgraph)
+               })); // Hovering a turn arrow
+             } else if (datum instanceof osmTurn) {
+               var restrictionType = osmInferRestriction(vgraph, datum, projection);
+               var turnType = restrictionType.replace(/^(only|no)\_/, '');
+               var indirect = datum.direct === false ? _t.html('restriction.help.indirect') : '';
+               var klass, turnText, nextText;
 
-         function displayTimestamp(timestamp) {
-           if (!timestamp) return _t('info_panels.history.unknown');
-           var options = {
-             day: 'numeric',
-             month: 'short',
-             year: 'numeric',
-             hour: 'numeric',
-             minute: 'numeric',
-             second: 'numeric'
-           };
-           var d = new Date(timestamp);
-           if (isNaN(d.getTime())) return _t('info_panels.history.unknown');
-           return d.toLocaleString(_mainLocalizer.localeCode(), options);
-         }
+               if (datum.no) {
+                 klass = 'restrict';
+                 turnText = _t.html('restriction.help.turn.no_' + turnType, {
+                   indirect: indirect
+                 });
+                 nextText = _t.html('restriction.help.turn.only_' + turnType, {
+                   indirect: ''
+                 });
+               } else if (datum.only) {
+                 klass = 'only';
+                 turnText = _t.html('restriction.help.turn.only_' + turnType, {
+                   indirect: indirect
+                 });
+                 nextText = _t.html('restriction.help.turn.allowed_' + turnType, {
+                   indirect: ''
+                 });
+               } else {
+                 klass = 'allow';
+                 turnText = _t.html('restriction.help.turn.allowed_' + turnType, {
+                   indirect: indirect
+                 });
+                 nextText = _t.html('restriction.help.turn.no_' + turnType, {
+                   indirect: ''
+                 });
+               }
 
-         function displayUser(selection, userName) {
-           if (!userName) {
-             selection.append('span').html(_t.html('info_panels.history.unknown'));
-             return;
-           }
+               help.append('div') // "NO Right Turn (indirect)"
+               .attr('class', 'qualifier ' + klass).html(turnText);
+               help.append('div') // "FROM {fromName} TO {toName}"
+               .html(_t.html('restriction.help.from_name_to_name', {
+                 from: placeholders.from,
+                 fromName: displayName(datum.from.way, vgraph),
+                 to: placeholders.to,
+                 toName: displayName(datum.to.way, vgraph)
+               }));
 
-           selection.append('span').attr('class', 'user-name').html(userName);
-           var links = selection.append('div').attr('class', 'links');
+               if (datum.via.ways && datum.via.ways.length) {
+                 var names = [];
 
-           if (osm) {
-             links.append('a').attr('class', 'user-osm-link').attr('href', osm.userURL(userName)).attr('target', '_blank').html('OSM');
-           }
+                 for (var i = 0; i < datum.via.ways.length; i++) {
+                   var prev = names[names.length - 1];
+                   var curr = displayName(datum.via.ways[i], vgraph);
 
-           links.append('a').attr('class', 'user-hdyc-link').attr('href', 'https://hdyc.neis-one.org/?' + userName).attr('target', '_blank').attr('tabindex', -1).html('HDYC');
-         }
+                   if (!prev || curr !== prev) {
+                     // collapse identical names
+                     names.push(curr);
+                   }
+                 }
 
-         function displayChangeset(selection, changeset) {
-           if (!changeset) {
-             selection.append('span').html(_t.html('info_panels.history.unknown'));
-             return;
-           }
+                 help.append('div') // "VIA {viaNames}"
+                 .html(_t.html('restriction.help.via_names', {
+                   via: placeholders.via,
+                   viaNames: names.join(', ')
+                 }));
+               }
 
-           selection.append('span').attr('class', 'changeset-id').html(changeset);
-           var links = selection.append('div').attr('class', 'links');
+               if (!indirect) {
+                 help.append('div') // Click for "No Right Turn"
+                 .html(_t.html('restriction.help.toggle', {
+                   turn: nextText.trim()
+                 }));
+               }
 
-           if (osm) {
-             links.append('a').attr('class', 'changeset-osm-link').attr('href', osm.changesetURL(changeset)).attr('target', '_blank').html('OSM');
-           }
+               highlightPathsFrom(null);
+               var alongIDs = datum.path.slice();
+               surface.selectAll(utilEntitySelector(alongIDs)).classed('related', true).classed('allow', klass === 'allow').classed('restrict', klass === 'restrict').classed('only', klass === 'only'); // Hovering empty surface
+             } else {
+               highlightPathsFrom(null);
 
-           links.append('a').attr('class', 'changeset-osmcha-link').attr('href', 'https://osmcha.org/changesets/' + changeset).attr('target', '_blank').html('OSMCha');
-           links.append('a').attr('class', 'changeset-achavi-link').attr('href', 'https://overpass-api.de/achavi/?changeset=' + changeset).attr('target', '_blank').html('Achavi');
+               if (_fromWayID) {
+                 help.append('div') // "FROM {fromName}"
+                 .html(_t.html('restriction.help.from_name', {
+                   from: placeholders.from,
+                   fromName: displayName(_fromWayID, vgraph)
+                 }));
+               } else {
+                 help.append('div') // "Click to select a FROM segment."
+                 .html(_t.html('restriction.help.select_from', {
+                   from: placeholders.from
+                 }));
+               }
+             }
+           }
          }
 
-         function redraw(selection) {
-           var selectedNoteID = context.selectedNoteID();
-           osm = context.connection();
-           var selected, note, entity;
+         function displayMaxDistance(maxDist) {
+           var isImperial = !_mainLocalizer.usesMetric();
+           var opts;
 
-           if (selectedNoteID && osm) {
-             // selected 1 note
-             selected = [_t('note.note') + ' ' + selectedNoteID];
-             note = osm.getNote(selectedNoteID);
+           if (isImperial) {
+             var distToFeet = {
+               // imprecise conversion for prettier display
+               20: 70,
+               25: 85,
+               30: 100,
+               35: 115,
+               40: 130,
+               45: 145,
+               50: 160
+             }[maxDist];
+             opts = {
+               distance: _t('units.feet', {
+                 quantity: distToFeet
+               })
+             };
            } else {
-             // selected 1..n entities
-             selected = context.selectedIDs().filter(function (e) {
-               return context.hasEntity(e);
-             });
-
-             if (selected.length) {
-               entity = context.entity(selected[0]);
-             }
+             opts = {
+               distance: _t('units.meters', {
+                 quantity: maxDist
+               })
+             };
            }
 
-           var singular = selected.length === 1 ? selected[0] : null;
-           selection.html('');
-           selection.append('h4').attr('class', 'history-heading').html(singular || _t.html('info_panels.selected', {
-             n: selected.length
-           }));
-           if (!singular) return;
+           return _t.html('restriction.controls.distance_up_to', opts);
+         }
 
-           if (entity) {
-             selection.call(redrawEntity, entity);
-           } else if (note) {
-             selection.call(redrawNote, note);
-           }
+         function displayMaxVia(maxVia) {
+           return maxVia === 0 ? _t.html('restriction.controls.via_node_only') : maxVia === 1 ? _t.html('restriction.controls.via_up_to_one') : _t.html('restriction.controls.via_up_to_two');
          }
 
-         function redrawNote(selection, note) {
-           if (!note || note.isNew()) {
-             selection.append('div').html(_t.html('info_panels.history.note_no_history'));
-             return;
-           }
+         function displayName(entityID, graph) {
+           var entity = graph.entity(entityID);
+           var name = utilDisplayName(entity) || '';
+           var matched = _mainPresetIndex.match(entity, graph);
+           var type = matched && matched.name() || utilDisplayType(entity.id);
+           return name || type;
+         }
 
-           var list = selection.append('ul');
-           list.append('li').html(_t.html('info_panels.history.note_comments') + ':').append('span').html(note.comments.length);
+         restrictions.entityIDs = function (val) {
+           _intersection = null;
+           _fromWayID = null;
+           _oldTurns = null;
+           _vertexID = val[0];
+         };
 
-           if (note.comments.length) {
-             list.append('li').html(_t.html('info_panels.history.note_created_date') + ':').append('span').html(displayTimestamp(note.comments[0].date));
-             list.append('li').html(_t.html('info_panels.history.note_created_user') + ':').call(displayUser, note.comments[0].user);
-           }
+         restrictions.tags = function () {};
 
-           if (osm) {
-             selection.append('a').attr('class', 'view-history-on-osm').attr('target', '_blank').attr('href', osm.noteURL(note)).call(svgIcon('#iD-icon-out-link', 'inline')).append('span').html(_t.html('info_panels.history.note_link_text'));
-           }
-         }
+         restrictions.focus = function () {};
 
-         function redrawEntity(selection, entity) {
-           if (!entity || entity.isNew()) {
-             selection.append('div').html(_t.html('info_panels.history.no_history'));
-             return;
-           }
+         restrictions.off = function (selection) {
+           if (!_initialized) return;
+           selection.selectAll('.surface').call(breathe.off).on('click.restrictions', null).on('mouseover.restrictions', null);
+           select(window).on('resize.restrictions', null);
+         };
 
-           var links = selection.append('div').attr('class', 'links');
+         return utilRebind(restrictions, dispatch, 'on');
+       }
+       uiFieldRestrictions.supportsMultiselection = false;
 
-           if (osm) {
-             links.append('a').attr('class', 'view-history-on-osm').attr('href', osm.historyURL(entity)).attr('target', '_blank').attr('title', _t('info_panels.history.link_text')).html('OSM');
-           }
+       function uiFieldTextarea(field, context) {
+         var dispatch = dispatch$8('change');
+         var input = select(null);
 
-           links.append('a').attr('class', 'pewu-history-viewer-link').attr('href', 'https://pewu.github.io/osm-history/#/' + entity.type + '/' + entity.osmId()).attr('target', '_blank').attr('tabindex', -1).html('PeWu');
-           var list = selection.append('ul');
-           list.append('li').html(_t.html('info_panels.history.version') + ':').append('span').html(entity.version);
-           list.append('li').html(_t.html('info_panels.history.last_edit') + ':').append('span').html(displayTimestamp(entity.timestamp));
-           list.append('li').html(_t.html('info_panels.history.edited_by') + ':').call(displayUser, entity.user);
-           list.append('li').html(_t.html('info_panels.history.changeset') + ':').call(displayChangeset, entity.changeset);
+         var _tags;
+
+         function textarea(selection) {
+           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
+           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
+           input = wrap.selectAll('textarea').data([0]);
+           input = input.enter().append('textarea').attr('id', field.domId).call(utilNoAuto).on('input', change(true)).on('blur', change()).on('change', change()).merge(input);
          }
 
-         var panel = function panel(selection) {
-           selection.call(redraw);
-           context.map().on('drawn.info-history', function () {
-             selection.call(redraw);
-           });
-           context.on('enter.info-history', function () {
-             selection.call(redraw);
-           });
+         function change(onInput) {
+           return function () {
+             var val = utilGetSetValue(input);
+             if (!onInput) val = context.cleanTagValue(val); // don't override multiple values with blank string
+
+             if (!val && Array.isArray(_tags[field.key])) return;
+             var t = {};
+             t[field.key] = val || undefined;
+             dispatch.call('change', this, t, onInput);
+           };
+         }
+
+         textarea.tags = function (tags) {
+           _tags = tags;
+           var isMixed = Array.isArray(tags[field.key]);
+           utilGetSetValue(input, !isMixed && tags[field.key] ? tags[field.key] : '').attr('title', isMixed ? tags[field.key].filter(Boolean).join('\n') : undefined).attr('placeholder', isMixed ? _t('inspector.multiple_values') : field.placeholder() || _t('inspector.unknown')).classed('mixed', isMixed);
          };
 
-         panel.off = function () {
-           context.map().on('drawn.info-history', null);
-           context.on('enter.info-history', null);
+         textarea.focus = function () {
+           input.node().focus();
          };
 
-         panel.id = 'history';
-         panel.label = _t.html('info_panels.history.title');
-         panel.key = _t('info_panels.history.key');
-         return panel;
+         return utilRebind(textarea, dispatch, 'on');
        }
 
-       var OSM_PRECISION = 7;
-       /**
-        * Returns a localized representation of the given length measurement.
-        *
-        * @param {Number} m area in meters
-        * @param {Boolean} isImperial true for U.S. customary units; false for metric
-        */
+       var getOwnPropertyDescriptor = objectGetOwnPropertyDescriptor.f;
 
-       function displayLength(m, isImperial) {
-         var d = m * (isImperial ? 3.28084 : 1);
-         var unit;
 
-         if (isImperial) {
-           if (d >= 5280) {
-             d /= 5280;
-             unit = 'miles';
-           } else {
-             unit = 'feet';
-           }
-         } else {
-           if (d >= 1000) {
-             d /= 1000;
-             unit = 'kilometers';
-           } else {
-             unit = 'meters';
-           }
-         }
 
-         return _t('units.' + unit, {
-           quantity: d.toLocaleString(_mainLocalizer.localeCode(), {
-             maximumSignificantDigits: 4
-           })
-         });
-       }
-       /**
-        * Returns a localized representation of the given area measurement.
-        *
-        * @param {Number} m2 area in square meters
-        * @param {Boolean} isImperial true for U.S. customary units; false for metric
-        */
 
-       function displayArea(m2, isImperial) {
-         var locale = _mainLocalizer.localeCode();
-         var d = m2 * (isImperial ? 10.7639111056 : 1);
-         var d1, d2, area;
-         var unit1 = '';
-         var unit2 = '';
 
-         if (isImperial) {
-           if (d >= 6969600) {
-             // > 0.25mi² show mi²
-             d1 = d / 27878400;
-             unit1 = 'square_miles';
-           } else {
-             d1 = d;
-             unit1 = 'square_feet';
-           }
 
-           if (d > 4356 && d < 43560000) {
-             // 0.1 - 1000 acres
-             d2 = d / 43560;
-             unit2 = 'acres';
-           }
-         } else {
-           if (d >= 250000) {
-             // > 0.25km² show km²
-             d1 = d / 1000000;
-             unit1 = 'square_kilometers';
-           } else {
-             d1 = d;
-             unit1 = 'square_meters';
-           }
+       // eslint-disable-next-line es/no-string-prototype-endswith -- safe
+       var $endsWith = ''.endsWith;
+       var min = Math.min;
 
-           if (d > 1000 && d < 10000000) {
-             // 0.1 - 1000 hectares
-             d2 = d / 10000;
-             unit2 = 'hectares';
-           }
+       var CORRECT_IS_REGEXP_LOGIC = correctIsRegexpLogic('endsWith');
+       // https://github.com/zloirock/core-js/pull/702
+       var MDN_POLYFILL_BUG = !CORRECT_IS_REGEXP_LOGIC && !!function () {
+         var descriptor = getOwnPropertyDescriptor(String.prototype, 'endsWith');
+         return descriptor && !descriptor.writable;
+       }();
+
+       // `String.prototype.endsWith` method
+       // https://tc39.es/ecma262/#sec-string.prototype.endswith
+       _export({ target: 'String', proto: true, forced: !MDN_POLYFILL_BUG && !CORRECT_IS_REGEXP_LOGIC }, {
+         endsWith: function endsWith(searchString /* , endPosition = @length */) {
+           var that = String(requireObjectCoercible(this));
+           notARegexp(searchString);
+           var endPosition = arguments.length > 1 ? arguments[1] : undefined;
+           var len = toLength(that.length);
+           var end = endPosition === undefined ? len : min(toLength(endPosition), len);
+           var search = String(searchString);
+           return $endsWith
+             ? $endsWith.call(that, search, end)
+             : that.slice(end - search.length, end) === search;
          }
+       });
 
-         area = _t('units.' + unit1, {
-           quantity: d1.toLocaleString(locale, {
-             maximumSignificantDigits: 4
-           })
-         });
+       function uiFieldWikidata(field, context) {
+         var wikidata = services.wikidata;
+         var dispatch = dispatch$8('change');
 
-         if (d2) {
-           return _t('units.area_pair', {
-             area1: area,
-             area2: _t('units.' + unit2, {
-               quantity: d2.toLocaleString(locale, {
-                 maximumSignificantDigits: 2
-               })
-             })
-           });
-         } else {
-           return area;
-         }
-       }
+         var _selection = select(null);
 
-       function wrap$2(x, min, max) {
-         var d = max - min;
-         return ((x - min) % d + d) % d + min;
-       }
+         var _searchInput = select(null);
 
-       function clamp$1(x, min, max) {
-         return Math.max(min, Math.min(x, max));
-       }
+         var _qid = null;
+         var _wikidataEntity = null;
+         var _wikiURL = '';
+         var _entityIDs = [];
 
-       function displayCoordinate(deg, pos, neg) {
-         var locale = _mainLocalizer.localeCode();
-         var min = (Math.abs(deg) - Math.floor(Math.abs(deg))) * 60;
-         var sec = (min - Math.floor(min)) * 60;
-         var displayDegrees = _t('units.arcdegrees', {
-           quantity: Math.floor(Math.abs(deg)).toLocaleString(locale)
-         });
-         var displayCoordinate;
+         var _wikipediaKey = field.keys && field.keys.find(function (key) {
+           return key.includes('wikipedia');
+         }),
+             _hintKey = field.key === 'wikidata' ? 'name' : field.key.split(':')[0];
 
-         if (Math.floor(sec) > 0) {
-           displayCoordinate = displayDegrees + _t('units.arcminutes', {
-             quantity: Math.floor(min).toLocaleString(locale)
-           }) + _t('units.arcseconds', {
-             quantity: Math.round(sec).toLocaleString(locale)
-           });
-         } else if (Math.floor(min) > 0) {
-           displayCoordinate = displayDegrees + _t('units.arcminutes', {
-             quantity: Math.round(min).toLocaleString(locale)
+         var combobox = uiCombobox(context, 'combo-' + field.safeid).caseSensitive(true).minItems(1);
+
+         function wiki(selection) {
+           _selection = selection;
+           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
+           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
+           var list = wrap.selectAll('ul').data([0]);
+           list = list.enter().append('ul').attr('class', 'rows').merge(list);
+           var searchRow = list.selectAll('li.wikidata-search').data([0]);
+           var searchRowEnter = searchRow.enter().append('li').attr('class', 'wikidata-search');
+           searchRowEnter.append('input').attr('type', 'text').attr('id', field.domId).style('flex', '1').call(utilNoAuto).on('focus', function () {
+             var node = select(this).node();
+             node.setSelectionRange(0, node.value.length);
+           }).on('blur', function () {
+             setLabelForEntity();
+           }).call(combobox.fetcher(fetchWikidataItems));
+           combobox.on('accept', function (d) {
+             if (d) {
+               _qid = d.id;
+               change();
+             }
+           }).on('cancel', function () {
+             setLabelForEntity();
            });
-         } else {
-           displayCoordinate = _t('units.arcdegrees', {
-             quantity: Math.round(Math.abs(deg)).toLocaleString(locale)
+           searchRowEnter.append('button').attr('class', 'form-field-button wiki-link').attr('title', _t('icons.view_on', {
+             domain: 'wikidata.org'
+           })).call(svgIcon('#iD-icon-out-link')).on('click', function (d3_event) {
+             d3_event.preventDefault();
+             if (_wikiURL) window.open(_wikiURL, '_blank');
            });
-         }
+           searchRow = searchRow.merge(searchRowEnter);
+           _searchInput = searchRow.select('input');
+           var wikidataProperties = ['description', 'identifier'];
+           var items = list.selectAll('li.labeled-input').data(wikidataProperties); // Enter
 
-         if (deg === 0) {
-           return displayCoordinate;
-         } else {
-           return _t('units.coordinate', {
-             coordinate: displayCoordinate,
-             direction: _t('units.' + (deg > 0 ? pos : neg))
+           var enter = items.enter().append('li').attr('class', function (d) {
+             return 'labeled-input preset-wikidata-' + d;
+           });
+           enter.append('span').attr('class', 'label').html(function (d) {
+             return _t.html('wikidata.' + d);
+           });
+           enter.append('input').attr('type', 'text').call(utilNoAuto).classed('disabled', 'true').attr('readonly', 'true');
+           enter.append('button').attr('class', 'form-field-button').attr('title', _t('icons.copy')).call(svgIcon('#iD-operation-copy')).on('click', function (d3_event) {
+             d3_event.preventDefault();
+             select(this.parentNode).select('input').node().select();
+             document.execCommand('copy');
            });
          }
-       }
-       /**
-        * Returns given coordinate pair in degree-minute-second format.
-        *
-        * @param {Array<Number>} coord longitude and latitude
-        */
 
+         function fetchWikidataItems(q, callback) {
+           if (!q && _hintKey) {
+             // other tags may be good search terms
+             for (var i in _entityIDs) {
+               var entity = context.hasEntity(_entityIDs[i]);
 
-       function dmsCoordinatePair(coord) {
-         return _t('units.coordinate_pair', {
-           latitude: displayCoordinate(clamp$1(coord[1], -90, 90), 'north', 'south'),
-           longitude: displayCoordinate(wrap$2(coord[0], -180, 180), 'east', 'west')
-         });
-       }
-       /**
-        * Returns the given coordinate pair in decimal format.
-        * note: unlocalized to avoid comma ambiguity - see #4765
-        *
-        * @param {Array<Number>} coord longitude and latitude
-        */
-
-       function decimalCoordinatePair(coord) {
-         return _t('units.coordinate_pair', {
-           latitude: clamp$1(coord[1], -90, 90).toFixed(OSM_PRECISION),
-           longitude: wrap$2(coord[0], -180, 180).toFixed(OSM_PRECISION)
-         });
-       }
+               if (entity.tags[_hintKey]) {
+                 q = entity.tags[_hintKey];
+                 break;
+               }
+             }
+           }
 
-       function uiPanelLocation(context) {
-         var currLocation = '';
+           wikidata.itemsForSearchQuery(q, function (err, data) {
+             if (err) return;
 
-         function redraw(selection) {
-           selection.html('');
-           var list = selection.append('ul'); // Mouse coordinates
+             for (var i in data) {
+               data[i].value = data[i].label + ' (' + data[i].id + ')';
+               data[i].title = data[i].description;
+             }
 
-           var coord = context.map().mouseCoordinates();
+             if (callback) callback(data);
+           });
+         }
 
-           if (coord.some(isNaN)) {
-             coord = context.map().center();
-           }
+         function change() {
+           var syncTags = {};
+           syncTags[field.key] = _qid;
+           dispatch.call('change', this, syncTags); // attempt asynchronous update of wikidata tag..
 
-           list.append('li').html(dmsCoordinatePair(coord)).append('li').html(decimalCoordinatePair(coord)); // Location Info
+           var initGraph = context.graph();
+           var initEntityIDs = _entityIDs;
+           wikidata.entityByQID(_qid, function (err, entity) {
+             if (err) return; // If graph has changed, we can't apply this update.
 
-           selection.append('div').attr('class', 'location-info').html(currLocation || ' ');
-           debouncedGetLocation(selection, coord);
-         }
+             if (context.graph() !== initGraph) return;
+             if (!entity.sitelinks) return;
+             var langs = wikidata.languagesToQuery(); // use the label and description languages as fallbacks
 
-         var debouncedGetLocation = debounce(getLocation, 250);
+             ['labels', 'descriptions'].forEach(function (key) {
+               if (!entity[key]) return;
+               var valueLangs = Object.keys(entity[key]);
+               if (valueLangs.length === 0) return;
+               var valueLang = valueLangs[0];
 
-         function getLocation(selection, coord) {
-           if (!services.geocoder) {
-             currLocation = _t('info_panels.location.unknown_location');
-             selection.selectAll('.location-info').html(currLocation);
-           } else {
-             services.geocoder.reverse(coord, function (err, result) {
-               currLocation = result ? result.display_name : _t('info_panels.location.unknown_location');
-               selection.selectAll('.location-info').html(currLocation);
+               if (langs.indexOf(valueLang) === -1) {
+                 langs.push(valueLang);
+               }
              });
-           }
-         }
-
-         var panel = function panel(selection) {
-           selection.call(redraw);
-           context.surface().on(('PointerEvent' in window ? 'pointer' : 'mouse') + 'move.info-location', function () {
-             selection.call(redraw);
-           });
-         };
+             var newWikipediaValue;
 
-         panel.off = function () {
-           context.surface().on('.info-location', null);
-         };
+             if (_wikipediaKey) {
+               var foundPreferred;
 
-         panel.id = 'location';
-         panel.label = _t.html('info_panels.location.title');
-         panel.key = _t('info_panels.location.key');
-         return panel;
-       }
+               for (var i in langs) {
+                 var lang = langs[i];
+                 var siteID = lang.replace('-', '_') + 'wiki';
 
-       function uiPanelMeasurement(context) {
-         function radiansToMeters(r) {
-           // using WGS84 authalic radius (6371007.1809 m)
-           return r * 6371007.1809;
-         }
+                 if (entity.sitelinks[siteID]) {
+                   foundPreferred = true;
+                   newWikipediaValue = lang + ':' + entity.sitelinks[siteID].title; // use the first match
 
-         function steradiansToSqmeters(r) {
-           // http://gis.stackexchange.com/a/124857/40446
-           return r / (4 * Math.PI) * 510065621724000;
-         }
+                   break;
+                 }
+               }
 
-         function toLineString(feature) {
-           if (feature.type === 'LineString') return feature;
-           var result = {
-             type: 'LineString',
-             coordinates: []
-           };
+               if (!foundPreferred) {
+                 // No wikipedia sites available in the user's language or the fallback languages,
+                 // default to any wikipedia sitelink
+                 var wikiSiteKeys = Object.keys(entity.sitelinks).filter(function (site) {
+                   return site.endsWith('wiki');
+                 });
 
-           if (feature.type === 'Polygon') {
-             result.coordinates = feature.coordinates[0];
-           } else if (feature.type === 'MultiPolygon') {
-             result.coordinates = feature.coordinates[0][0];
-           }
+                 if (wikiSiteKeys.length === 0) {
+                   // if no wikipedia pages are linked to this wikidata entity, delete that tag
+                   newWikipediaValue = null;
+                 } else {
+                   var wikiLang = wikiSiteKeys[0].slice(0, -4).replace('_', '-');
+                   var wikiTitle = entity.sitelinks[wikiSiteKeys[0]].title;
+                   newWikipediaValue = wikiLang + ':' + wikiTitle;
+                 }
+               }
+             }
 
-           return result;
-         }
+             if (newWikipediaValue) {
+               newWikipediaValue = context.cleanTagValue(newWikipediaValue);
+             }
 
-         var _isImperial = !_mainLocalizer.usesMetric();
+             if (typeof newWikipediaValue === 'undefined') return;
+             var actions = initEntityIDs.map(function (entityID) {
+               var entity = context.hasEntity(entityID);
+               if (!entity) return null;
+               var currTags = Object.assign({}, entity.tags); // shallow copy
 
-         function redraw(selection) {
-           var graph = context.graph();
-           var selectedNoteID = context.selectedNoteID();
-           var osm = services.osm;
-           var localeCode = _mainLocalizer.localeCode();
-           var heading;
-           var center, location, centroid;
-           var closed, geometry;
-           var totalNodeCount,
-               length = 0,
-               area = 0,
-               distance;
+               if (newWikipediaValue === null) {
+                 if (!currTags[_wikipediaKey]) return null;
+                 delete currTags[_wikipediaKey];
+               } else {
+                 currTags[_wikipediaKey] = newWikipediaValue;
+               }
 
-           if (selectedNoteID && osm) {
-             // selected 1 note
-             var note = osm.getNote(selectedNoteID);
-             heading = _t('note.note') + ' ' + selectedNoteID;
-             location = note.loc;
-             geometry = 'note';
-           } else {
-             // selected 1..n entities
-             var selectedIDs = context.selectedIDs().filter(function (id) {
-               return context.hasEntity(id);
-             });
-             var selected = selectedIDs.map(function (id) {
-               return context.entity(id);
-             });
-             heading = selected.length === 1 ? selected[0].id : _t('info_panels.selected', {
-               n: selected.length
-             });
+               return actionChangeTags(entityID, currTags);
+             }).filter(Boolean);
+             if (!actions.length) return; // Coalesce the update of wikidata tag into the previous tag change
 
-             if (selected.length) {
-               var extent = geoExtent();
+             context.overwrite(function actionUpdateWikipediaTags(graph) {
+               actions.forEach(function (action) {
+                 graph = action(graph);
+               });
+               return graph;
+             }, context.history().undoAnnotation()); // do not dispatch.call('change') here, because entity_editor
+             // changeTags() is not intended to be called asynchronously
+           });
+         }
 
-               for (var i in selected) {
-                 var entity = selected[i];
+         function setLabelForEntity() {
+           var label = '';
 
-                 extent._extend(entity.extent(graph));
+           if (_wikidataEntity) {
+             label = entityPropertyForDisplay(_wikidataEntity, 'labels');
 
-                 geometry = entity.geometry(graph);
+             if (label.length === 0) {
+               label = _wikidataEntity.id.toString();
+             }
+           }
 
-                 if (geometry === 'line' || geometry === 'area') {
-                   closed = entity.type === 'relation' || entity.isClosed() && !entity.isDegenerate();
-                   var feature = entity.asGeoJSON(graph);
-                   length += radiansToMeters(d3_geoLength(toLineString(feature))); // d3_geoCentroid is wrong for counterclockwise-wound polygons, so wind them clockwise
+           utilGetSetValue(_searchInput, label);
+         }
 
-                   centroid = d3_geoCentroid(geojsonRewind(Object.assign({}, feature), true));
+         wiki.tags = function (tags) {
+           var isMixed = Array.isArray(tags[field.key]);
 
-                   if (closed) {
-                     area += steradiansToSqmeters(entity.area(graph));
-                   }
-                 }
-               }
+           _searchInput.attr('title', isMixed ? tags[field.key].filter(Boolean).join('\n') : null).attr('placeholder', isMixed ? _t('inspector.multiple_values') : '').classed('mixed', isMixed);
 
-               if (selected.length > 1) {
-                 geometry = null;
-                 closed = null;
-                 centroid = null;
-               }
+           _qid = typeof tags[field.key] === 'string' && tags[field.key] || '';
 
-               if (selected.length === 2 && selected[0].type === 'node' && selected[1].type === 'node') {
-                 distance = geoSphericalDistance(selected[0].loc, selected[1].loc);
-               }
+           if (!/^Q[0-9]*$/.test(_qid)) {
+             // not a proper QID
+             unrecognized();
+             return;
+           } // QID value in correct format
 
-               if (selected.length === 1 && selected[0].type === 'node') {
-                 location = selected[0].loc;
-               } else {
-                 totalNodeCount = utilGetAllNodes(selectedIDs, context.graph()).length;
-               }
 
-               if (!location && !centroid) {
-                 center = extent.center();
-               }
+           _wikiURL = 'https://wikidata.org/wiki/' + _qid;
+           wikidata.entityByQID(_qid, function (err, entity) {
+             if (err) {
+               unrecognized();
+               return;
              }
-           }
 
-           selection.html('');
+             _wikidataEntity = entity;
+             setLabelForEntity();
+             var description = entityPropertyForDisplay(entity, 'descriptions');
 
-           if (heading) {
-             selection.append('h4').attr('class', 'measurement-heading').html(heading);
-           }
+             _selection.select('button.wiki-link').classed('disabled', false);
 
-           var list = selection.append('ul');
-           var coordItem;
+             _selection.select('.preset-wikidata-description').style('display', function () {
+               return description.length > 0 ? 'flex' : 'none';
+             }).select('input').attr('value', description);
 
-           if (geometry) {
-             list.append('li').html(_t.html('info_panels.measurement.geometry') + ':').append('span').html(closed ? _t('info_panels.measurement.closed_' + geometry) : _t('geometry.' + geometry));
-           }
+             _selection.select('.preset-wikidata-identifier').style('display', function () {
+               return entity.id ? 'flex' : 'none';
+             }).select('input').attr('value', entity.id);
+           }); // not a proper QID
 
-           if (totalNodeCount) {
-             list.append('li').html(_t.html('info_panels.measurement.node_count') + ':').append('span').html(totalNodeCount.toLocaleString(localeCode));
-           }
+           function unrecognized() {
+             _wikidataEntity = null;
+             setLabelForEntity();
 
-           if (area) {
-             list.append('li').html(_t.html('info_panels.measurement.area') + ':').append('span').html(displayArea(area, _isImperial));
-           }
+             _selection.select('.preset-wikidata-description').style('display', 'none');
 
-           if (length) {
-             list.append('li').html(_t.html('info_panels.measurement.' + (closed ? 'perimeter' : 'length')) + ':').append('span').html(displayLength(length, _isImperial));
-           }
+             _selection.select('.preset-wikidata-identifier').style('display', 'none');
 
-           if (typeof distance === 'number') {
-             list.append('li').html(_t.html('info_panels.measurement.distance') + ':').append('span').html(displayLength(distance, _isImperial));
-           }
+             _selection.select('button.wiki-link').classed('disabled', true);
 
-           if (location) {
-             coordItem = list.append('li').html(_t.html('info_panels.measurement.location') + ':');
-             coordItem.append('span').html(dmsCoordinatePair(location));
-             coordItem.append('span').html(decimalCoordinatePair(location));
+             if (_qid && _qid !== '') {
+               _wikiURL = 'https://wikidata.org/wiki/Special:Search?search=' + _qid;
+             } else {
+               _wikiURL = '';
+             }
            }
+         };
 
-           if (centroid) {
-             coordItem = list.append('li').html(_t.html('info_panels.measurement.centroid') + ':');
-             coordItem.append('span').html(dmsCoordinatePair(centroid));
-             coordItem.append('span').html(decimalCoordinatePair(centroid));
-           }
+         function entityPropertyForDisplay(wikidataEntity, propKey) {
+           if (!wikidataEntity[propKey]) return '';
+           var propObj = wikidataEntity[propKey];
+           var langKeys = Object.keys(propObj);
+           if (langKeys.length === 0) return ''; // sorted by priority, since we want to show the user's language first if possible
 
-           if (center) {
-             coordItem = list.append('li').html(_t.html('info_panels.measurement.center') + ':');
-             coordItem.append('span').html(dmsCoordinatePair(center));
-             coordItem.append('span').html(decimalCoordinatePair(center));
-           }
+           var langs = wikidata.languagesToQuery();
 
-           if (length || area || typeof distance === 'number') {
-             var toggle = _isImperial ? 'imperial' : 'metric';
-             selection.append('a').html(_t.html('info_panels.measurement.' + toggle)).attr('href', '#').attr('class', 'button button-toggle-units').on('click', function (d3_event) {
-               d3_event.preventDefault();
-               _isImperial = !_isImperial;
-               selection.call(redraw);
-             });
-           }
+           for (var i in langs) {
+             var lang = langs[i];
+             var valueObj = propObj[lang];
+             if (valueObj && valueObj.value && valueObj.value.length > 0) return valueObj.value;
+           } // default to any available value
+
+
+           return propObj[langKeys[0]].value;
          }
 
-         var panel = function panel(selection) {
-           selection.call(redraw);
-           context.map().on('drawn.info-measurement', function () {
-             selection.call(redraw);
-           });
-           context.on('enter.info-measurement', function () {
-             selection.call(redraw);
-           });
+         wiki.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs;
+           _entityIDs = val;
+           return wiki;
          };
 
-         panel.off = function () {
-           context.map().on('drawn.info-measurement', null);
-           context.on('enter.info-measurement', null);
+         wiki.focus = function () {
+           _searchInput.node().focus();
          };
 
-         panel.id = 'measurement';
-         panel.label = _t.html('info_panels.measurement.title');
-         panel.key = _t('info_panels.measurement.key');
-         return panel;
+         return utilRebind(wiki, dispatch, 'on');
        }
 
-       var uiInfoPanels = {
-         background: uiPanelBackground,
-         history: uiPanelHistory,
-         location: uiPanelLocation,
-         measurement: uiPanelMeasurement
-       };
+       function uiFieldWikipedia(field, context) {
+         var _arguments = arguments;
+         var dispatch = dispatch$8('change');
+         var wikipedia = services.wikipedia;
+         var wikidata = services.wikidata;
 
-       function uiInfo(context) {
-         var ids = Object.keys(uiInfoPanels);
-         var wasActive = ['measurement'];
-         var panels = {};
-         var active = {}; // create panels
+         var _langInput = select(null);
 
-         ids.forEach(function (k) {
-           if (!panels[k]) {
-             panels[k] = uiInfoPanels[k](context);
-             active[k] = false;
-           }
-         });
+         var _titleInput = select(null);
 
-         function info(selection) {
-           function redraw() {
-             var activeids = ids.filter(function (k) {
-               return active[k];
-             }).sort();
-             var containers = infoPanels.selectAll('.panel-container').data(activeids, function (k) {
-               return k;
-             });
-             containers.exit().style('opacity', 1).transition().duration(200).style('opacity', 0).on('end', function (d) {
-               select(this).call(panels[d].off).remove();
-             });
-             var enter = containers.enter().append('div').attr('class', function (d) {
-               return 'fillD2 panel-container panel-container-' + d;
-             });
-             enter.style('opacity', 0).transition().duration(200).style('opacity', 1);
-             var title = enter.append('div').attr('class', 'panel-title fillD2');
-             title.append('h3').html(function (d) {
-               return panels[d].label;
-             });
-             title.append('button').attr('class', 'close').on('click', function (d3_event, d) {
-               d3_event.stopImmediatePropagation();
-               d3_event.preventDefault();
-               info.toggle(d);
-             }).call(svgIcon('#iD-icon-close'));
-             enter.append('div').attr('class', function (d) {
-               return 'panel-content panel-content-' + d;
-             }); // redraw the panels
+         var _wikiURL = '';
 
-             infoPanels.selectAll('.panel-content').each(function (d) {
-               select(this).call(panels[d]);
-             });
-           }
+         var _entityIDs;
 
-           info.toggle = function (which) {
-             var activeids = ids.filter(function (k) {
-               return active[k];
-             });
+         var _tags;
 
-             if (which) {
-               // toggle one
-               active[which] = !active[which];
+         var _dataWikipedia = [];
+         _mainFileFetcher.get('wmf_sitematrix').then(function (d) {
+           _dataWikipedia = d;
+           if (_tags) updateForTags(_tags);
+         })["catch"](function () {
+           /* ignore */
+         });
+         var langCombo = uiCombobox(context, 'wikipedia-lang').fetcher(function (value, callback) {
+           var v = value.toLowerCase();
+           callback(_dataWikipedia.filter(function (d) {
+             return d[0].toLowerCase().indexOf(v) >= 0 || d[1].toLowerCase().indexOf(v) >= 0 || d[2].toLowerCase().indexOf(v) >= 0;
+           }).map(function (d) {
+             return {
+               value: d[1]
+             };
+           }));
+         });
+         var titleCombo = uiCombobox(context, 'wikipedia-title').fetcher(function (value, callback) {
+           if (!value) {
+             value = '';
 
-               if (activeids.length === 1 && activeids[0] === which) {
-                 // none active anymore
-                 wasActive = [which];
-               }
+             for (var i in _entityIDs) {
+               var entity = context.hasEntity(_entityIDs[i]);
 
-               context.container().select('.' + which + '-panel-toggle-item').classed('active', active[which]).select('input').property('checked', active[which]);
-             } else {
-               // toggle all
-               if (activeids.length) {
-                 wasActive = activeids;
-                 activeids.forEach(function (k) {
-                   active[k] = false;
-                 });
-               } else {
-                 wasActive.forEach(function (k) {
-                   active[k] = true;
-                 });
+               if (entity.tags.name) {
+                 value = entity.tags.name;
+                 break;
                }
              }
+           }
 
-             redraw();
-           };
+           var searchfn = value.length > 7 ? wikipedia.search : wikipedia.suggestions;
+           searchfn(language()[2], value, function (query, data) {
+             callback(data.map(function (d) {
+               return {
+                 value: d
+               };
+             }));
+           });
+         });
 
-           var infoPanels = selection.selectAll('.info-panels').data([0]);
-           infoPanels = infoPanels.enter().append('div').attr('class', 'info-panels').merge(infoPanels);
-           redraw();
-           context.keybinding().on(uiCmd('⌘' + _t('info_panels.key')), function (d3_event) {
-             d3_event.stopImmediatePropagation();
+         function wiki(selection) {
+           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
+           wrap = wrap.enter().append('div').attr('class', "form-field-input-wrap form-field-input-".concat(field.type)).merge(wrap);
+           var langContainer = wrap.selectAll('.wiki-lang-container').data([0]);
+           langContainer = langContainer.enter().append('div').attr('class', 'wiki-lang-container').merge(langContainer);
+           _langInput = langContainer.selectAll('input.wiki-lang').data([0]);
+           _langInput = _langInput.enter().append('input').attr('type', 'text').attr('class', 'wiki-lang').attr('placeholder', _t('translate.localized_translation_language')).call(utilNoAuto).call(langCombo).merge(_langInput);
+
+           _langInput.on('blur', changeLang).on('change', changeLang);
+
+           var titleContainer = wrap.selectAll('.wiki-title-container').data([0]);
+           titleContainer = titleContainer.enter().append('div').attr('class', 'wiki-title-container').merge(titleContainer);
+           _titleInput = titleContainer.selectAll('input.wiki-title').data([0]);
+           _titleInput = _titleInput.enter().append('input').attr('type', 'text').attr('class', 'wiki-title').attr('id', field.domId).call(utilNoAuto).call(titleCombo).merge(_titleInput);
+
+           _titleInput.on('blur', function () {
+             change(true);
+           }).on('change', function () {
+             change(false);
+           });
+
+           var link = titleContainer.selectAll('.wiki-link').data([0]);
+           link = link.enter().append('button').attr('class', 'form-field-button wiki-link').attr('title', _t('icons.view_on', {
+             domain: 'wikipedia.org'
+           })).call(svgIcon('#iD-icon-out-link')).merge(link);
+           link.on('click', function (d3_event) {
              d3_event.preventDefault();
-             info.toggle();
-           });
-           ids.forEach(function (k) {
-             var key = _t('info_panels.' + k + '.key', {
-               "default": null
-             });
-             if (!key) return;
-             context.keybinding().on(uiCmd('⌘⇧' + key), function (d3_event) {
-               d3_event.stopImmediatePropagation();
-               d3_event.preventDefault();
-               info.toggle(k);
-             });
+             if (_wikiURL) window.open(_wikiURL, '_blank');
            });
          }
 
-         return info;
-       }
+         function defaultLanguageInfo(skipEnglishFallback) {
+           var langCode = _mainLocalizer.languageCode().toLowerCase();
 
-       function pointBox(loc, context) {
-         var rect = context.surfaceRect();
-         var point = context.curtainProjection(loc);
-         return {
-           left: point[0] + rect.left - 40,
-           top: point[1] + rect.top - 60,
-           width: 80,
-           height: 90
-         };
-       }
-       function pad(locOrBox, padding, context) {
-         var box;
+           for (var i in _dataWikipedia) {
+             var d = _dataWikipedia[i]; // default to the language of iD's current locale
 
-         if (locOrBox instanceof Array) {
-           var rect = context.surfaceRect();
-           var point = context.curtainProjection(locOrBox);
-           box = {
-             left: point[0] + rect.left,
-             top: point[1] + rect.top
-           };
-         } else {
-           box = locOrBox;
+             if (d[2] === langCode) return d;
+           } // fallback to English
+
+
+           return skipEnglishFallback ? ['', '', ''] : ['English', 'English', 'en'];
          }
 
-         return {
-           left: box.left - padding,
-           top: box.top - padding,
-           width: (box.width || 0) + 2 * padding,
-           height: (box.width || 0) + 2 * padding
-         };
-       }
-       function icon(name, svgklass, useklass) {
-         return '<svg class="icon ' + (svgklass || '') + '">' + '<use xlink:href="' + name + '"' + (useklass ? ' class="' + useklass + '"' : '') + '></use></svg>';
-       }
-       var helpStringReplacements; // Returns the localized HTML element for `id` with a standardized set of icon, key, and
-       // label replacements suitable for tutorials and documentation. Optionally supplemented
-       // with custom `replacements`
+         function language(skipEnglishFallback) {
+           var value = utilGetSetValue(_langInput).toLowerCase();
 
-       function helpHtml(id, replacements) {
-         // only load these the first time
-         if (!helpStringReplacements) helpStringReplacements = {
-           // insert icons corresponding to various UI elements
-           point_icon: icon('#iD-icon-point', 'inline'),
-           line_icon: icon('#iD-icon-line', 'inline'),
-           area_icon: icon('#iD-icon-area', 'inline'),
-           note_icon: icon('#iD-icon-note', 'inline add-note'),
-           plus: icon('#iD-icon-plus', 'inline'),
-           minus: icon('#iD-icon-minus', 'inline'),
-           layers_icon: icon('#iD-icon-layers', 'inline'),
-           data_icon: icon('#iD-icon-data', 'inline'),
-           inspect: icon('#iD-icon-inspect', 'inline'),
-           help_icon: icon('#iD-icon-help', 'inline'),
-           undo_icon: icon(_mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-redo' : '#iD-icon-undo', 'inline'),
-           redo_icon: icon(_mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-undo' : '#iD-icon-redo', 'inline'),
-           save_icon: icon('#iD-icon-save', 'inline'),
-           // operation icons
-           circularize_icon: icon('#iD-operation-circularize', 'inline operation'),
-           continue_icon: icon('#iD-operation-continue', 'inline operation'),
-           copy_icon: icon('#iD-operation-copy', 'inline operation'),
-           delete_icon: icon('#iD-operation-delete', 'inline operation'),
-           disconnect_icon: icon('#iD-operation-disconnect', 'inline operation'),
-           downgrade_icon: icon('#iD-operation-downgrade', 'inline operation'),
-           extract_icon: icon('#iD-operation-extract', 'inline operation'),
-           merge_icon: icon('#iD-operation-merge', 'inline operation'),
-           move_icon: icon('#iD-operation-move', 'inline operation'),
-           orthogonalize_icon: icon('#iD-operation-orthogonalize', 'inline operation'),
-           paste_icon: icon('#iD-operation-paste', 'inline operation'),
-           reflect_long_icon: icon('#iD-operation-reflect-long', 'inline operation'),
-           reflect_short_icon: icon('#iD-operation-reflect-short', 'inline operation'),
-           reverse_icon: icon('#iD-operation-reverse', 'inline operation'),
-           rotate_icon: icon('#iD-operation-rotate', 'inline operation'),
-           split_icon: icon('#iD-operation-split', 'inline operation'),
-           straighten_icon: icon('#iD-operation-straighten', 'inline operation'),
-           // interaction icons
-           leftclick: icon('#iD-walkthrough-mouse-left', 'inline operation'),
-           rightclick: icon('#iD-walkthrough-mouse-right', 'inline operation'),
-           mousewheel_icon: icon('#iD-walkthrough-mousewheel', 'inline operation'),
-           tap_icon: icon('#iD-walkthrough-tap', 'inline operation'),
-           doubletap_icon: icon('#iD-walkthrough-doubletap', 'inline operation'),
-           longpress_icon: icon('#iD-walkthrough-longpress', 'inline operation'),
-           touchdrag_icon: icon('#iD-walkthrough-touchdrag', 'inline operation'),
-           pinch_icon: icon('#iD-walkthrough-pinch-apart', 'inline operation'),
-           // insert keys; may be localized and platform-dependent
-           shift: uiCmd.display('⇧'),
-           alt: uiCmd.display('⌥'),
-           "return": uiCmd.display('↵'),
-           esc: _t.html('shortcuts.key.esc'),
-           space: _t.html('shortcuts.key.space'),
-           add_note_key: _t.html('modes.add_note.key'),
-           help_key: _t.html('help.key'),
-           shortcuts_key: _t.html('shortcuts.toggle.key'),
-           // reference localized UI labels directly so that they'll always match
-           save: _t.html('save.title'),
-           undo: _t.html('undo.title'),
-           redo: _t.html('redo.title'),
-           upload: _t.html('commit.save'),
-           point: _t.html('modes.add_point.title'),
-           line: _t.html('modes.add_line.title'),
-           area: _t.html('modes.add_area.title'),
-           note: _t.html('modes.add_note.label'),
-           circularize: _t.html('operations.circularize.title'),
-           "continue": _t.html('operations.continue.title'),
-           copy: _t.html('operations.copy.title'),
-           "delete": _t.html('operations.delete.title'),
-           disconnect: _t.html('operations.disconnect.title'),
-           downgrade: _t.html('operations.downgrade.title'),
-           extract: _t.html('operations.extract.title'),
-           merge: _t.html('operations.merge.title'),
-           move: _t.html('operations.move.title'),
-           orthogonalize: _t.html('operations.orthogonalize.title'),
-           paste: _t.html('operations.paste.title'),
-           reflect_long: _t.html('operations.reflect.title.long'),
-           reflect_short: _t.html('operations.reflect.title.short'),
-           reverse: _t.html('operations.reverse.title'),
-           rotate: _t.html('operations.rotate.title'),
-           split: _t.html('operations.split.title'),
-           straighten: _t.html('operations.straighten.title'),
-           map_data: _t.html('map_data.title'),
-           osm_notes: _t.html('map_data.layers.notes.title'),
-           fields: _t.html('inspector.fields'),
-           tags: _t.html('inspector.tags'),
-           relations: _t.html('inspector.relations'),
-           new_relation: _t.html('inspector.new_relation'),
-           turn_restrictions: _t.html('presets.fields.restrictions.label'),
-           background_settings: _t.html('background.description'),
-           imagery_offset: _t.html('background.fix_misalignment'),
-           start_the_walkthrough: _t.html('splash.walkthrough'),
-           help: _t.html('help.title'),
-           ok: _t.html('intro.ok')
-         };
-         var reps;
+           for (var i in _dataWikipedia) {
+             var d = _dataWikipedia[i]; // return the language already set in the UI, if supported
 
-         if (replacements) {
-           reps = Object.assign(replacements, helpStringReplacements);
-         } else {
-           reps = helpStringReplacements;
+             if (d[0].toLowerCase() === value || d[1].toLowerCase() === value || d[2] === value) return d;
+           } // fallback to English
+
+
+           return defaultLanguageInfo(skipEnglishFallback);
          }
 
-         return _t.html(id, reps) // use keyboard key styling for shortcuts
-         .replace(/\`(.*?)\`/g, '<kbd>$1</kbd>');
-       }
+         function changeLang() {
+           utilGetSetValue(_langInput, language()[1]);
+           change(true);
+         }
 
-       function slugify(text) {
-         return text.toString().toLowerCase().replace(/\s+/g, '-') // Replace spaces with -
-         .replace(/[^\w\-]+/g, '') // Remove all non-word chars
-         .replace(/\-\-+/g, '-') // Replace multiple - with single -
-         .replace(/^-+/, '') // Trim - from start of text
-         .replace(/-+$/, ''); // Trim - from end of text
-       } // console warning for missing walkthrough names
+         function change(skipWikidata) {
+           var value = utilGetSetValue(_titleInput);
+           var m = value.match(/https?:\/\/([-a-z]+)\.wikipedia\.org\/(?:wiki|\1-[-a-z]+)\/([^#]+)(?:#(.+))?/);
 
+           var langInfo = m && _dataWikipedia.find(function (d) {
+             return m[1] === d[2];
+           });
 
-       var missingStrings = {};
+           var syncTags = {};
 
-       function checkKey(key, text) {
-         if (_t(key, {
-           "default": undefined
-         }) === undefined) {
-           if (missingStrings.hasOwnProperty(key)) return; // warn once
+           if (langInfo) {
+             var nativeLangName = langInfo[1]; // Normalize title http://www.mediawiki.org/wiki/API:Query#Title_normalization
 
-           missingStrings[key] = text;
-           var missing = key + ': ' + text;
-           if (typeof console !== 'undefined') console.log(missing); // eslint-disable-line
-         }
-       }
+             value = decodeURIComponent(m[2]).replace(/_/g, ' ');
 
-       function localize(obj) {
-         var key; // Assign name if entity has one..
+             if (m[3]) {
+               var anchor; // try {
+               // leave this out for now - #6232
+               // Best-effort `anchordecode:` implementation
+               // anchor = decodeURIComponent(m[3].replace(/\.([0-9A-F]{2})/g, '%$1'));
+               // } catch (e) {
 
-         var name = obj.tags && obj.tags.name;
+               anchor = decodeURIComponent(m[3]); // }
 
-         if (name) {
-           key = 'intro.graph.name.' + slugify(name);
-           obj.tags.name = _t(key, {
-             "default": name
-           });
-           checkKey(key, name);
-         } // Assign street name if entity has one..
+               value += '#' + anchor.replace(/_/g, ' ');
+             }
 
+             value = value.slice(0, 1).toUpperCase() + value.slice(1);
+             utilGetSetValue(_langInput, nativeLangName);
+             utilGetSetValue(_titleInput, value);
+           }
 
-         var street = obj.tags && obj.tags['addr:street'];
+           if (value) {
+             syncTags.wikipedia = context.cleanTagValue(language()[2] + ':' + value);
+           } else {
+             syncTags.wikipedia = undefined;
+           }
 
-         if (street) {
-           key = 'intro.graph.name.' + slugify(street);
-           obj.tags['addr:street'] = _t(key, {
-             "default": street
-           });
-           checkKey(key, street); // Add address details common across walkthrough..
+           dispatch.call('change', this, syncTags);
+           if (skipWikidata || !value || !language()[2]) return; // attempt asynchronous update of wikidata tag..
 
-           var addrTags = ['block_number', 'city', 'county', 'district', 'hamlet', 'neighbourhood', 'postcode', 'province', 'quarter', 'state', 'subdistrict', 'suburb'];
-           addrTags.forEach(function (k) {
-             var key = 'intro.graph.' + k;
-             var tag = 'addr:' + k;
-             var val = obj.tags && obj.tags[tag];
-             var str = _t(key, {
-               "default": val
+           var initGraph = context.graph();
+           var initEntityIDs = _entityIDs;
+           wikidata.itemsByTitle(language()[2], value, function (err, data) {
+             if (err || !data || !Object.keys(data).length) return; // If graph has changed, we can't apply this update.
+
+             if (context.graph() !== initGraph) return;
+             var qids = Object.keys(data);
+             var value = qids && qids.find(function (id) {
+               return id.match(/^Q\d+$/);
              });
+             var actions = initEntityIDs.map(function (entityID) {
+               var entity = context.entity(entityID).tags;
+               var currTags = Object.assign({}, entity); // shallow copy
 
-             if (str) {
-               if (str.match(/^<.*>$/) !== null) {
-                 delete obj.tags[tag];
-               } else {
-                 obj.tags[tag] = str;
+               if (currTags.wikidata !== value) {
+                 currTags.wikidata = value;
+                 return actionChangeTags(entityID, currTags);
                }
-             }
+
+               return null;
+             }).filter(Boolean);
+             if (!actions.length) return; // Coalesce the update of wikidata tag into the previous tag change
+
+             context.overwrite(function actionUpdateWikidataTags(graph) {
+               actions.forEach(function (action) {
+                 graph = action(graph);
+               });
+               return graph;
+             }, context.history().undoAnnotation()); // do not dispatch.call('change') here, because entity_editor
+             // changeTags() is not intended to be called asynchronously
            });
          }
 
-         return obj;
-       } // Used to detect squareness.. some duplicataion of code from actionOrthogonalize.
-
-       function isMostlySquare(points) {
-         // note: uses 15 here instead of the 12 from actionOrthogonalize because
-         // actionOrthogonalize can actually straighten some larger angles as it iterates
-         var threshold = 15; // degrees within right or straight
+         wiki.tags = function (tags) {
+           _tags = tags;
+           updateForTags(tags);
+         };
 
-         var lowerBound = Math.cos((90 - threshold) * Math.PI / 180); // near right
+         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 upperBound = Math.cos(threshold * Math.PI / 180); // near straight
+           var m = value.match(/([^:]+):([^#]+)(?:#(.+))?/);
+           var tagLang = m && m[1];
+           var tagArticleTitle = m && m[2];
+           var anchor = m && m[3];
 
-         for (var i = 0; i < points.length; i++) {
-           var a = points[(i - 1 + points.length) % points.length];
-           var origin = points[i];
-           var b = points[(i + 1) % points.length];
-           var dotp = geoVecNormalizedDot(a, b, origin);
-           var mag = Math.abs(dotp);
+           var tagLangInfo = tagLang && _dataWikipedia.find(function (d) {
+             return tagLang === d[2];
+           }); // value in correct format
 
-           if (mag > lowerBound && mag < upperBound) {
-             return false;
-           }
-         }
 
-         return true;
-       }
-       function selectMenuItem(context, operation) {
-         return context.container().select('.edit-menu .edit-menu-item-' + operation);
-       }
-       function transitionTime(point1, point2) {
-         var distance = geoSphericalDistance(point1, point2);
-         if (distance === 0) return 0;else if (distance < 80) return 500;else return 1000;
-       }
+           if (tagLangInfo) {
+             var nativeLangName = tagLangInfo[1];
+             utilGetSetValue(_langInput, nativeLangName);
+             utilGetSetValue(_titleInput, tagArticleTitle + (anchor ? '#' + anchor : ''));
 
-       function uiCurtain(containerNode) {
-         var surface = select(null),
-             tooltip = select(null),
-             darkness = select(null);
+             if (anchor) {
+               try {
+                 // Best-effort `anchorencode:` implementation
+                 anchor = encodeURIComponent(anchor.replace(/ /g, '_')).replace(/%/g, '.');
+               } catch (e) {
+                 anchor = anchor.replace(/ /g, '_');
+               }
+             }
 
-         function curtain(selection) {
-           surface = selection.append('svg').attr('class', 'curtain').style('top', 0).style('left', 0);
-           darkness = surface.append('path').attr('x', 0).attr('y', 0).attr('class', 'curtain-darkness');
-           select(window).on('resize.curtain', resize);
-           tooltip = selection.append('div').attr('class', 'tooltip');
-           tooltip.append('div').attr('class', 'popover-arrow');
-           tooltip.append('div').attr('class', 'popover-inner');
-           resize();
+             _wikiURL = 'https://' + tagLang + '.wikipedia.org/wiki/' + tagArticleTitle.replace(/ /g, '_') + (anchor ? '#' + anchor : ''); // unrecognized value format
+           } else {
+             utilGetSetValue(_titleInput, value);
 
-           function resize() {
-             surface.attr('width', containerNode.clientWidth).attr('height', containerNode.clientHeight);
-             curtain.cut(darkness.datum());
+             if (value && value !== '') {
+               utilGetSetValue(_langInput, '');
+               var defaultLangInfo = defaultLanguageInfo();
+               _wikiURL = "https://".concat(defaultLangInfo[2], ".wikipedia.org/w/index.php?fulltext=1&search=").concat(value);
+             } else {
+               var shownOrDefaultLangInfo = language(true
+               /* skipEnglishFallback */
+               );
+               utilGetSetValue(_langInput, shownOrDefaultLangInfo[1]);
+               _wikiURL = '';
+             }
            }
          }
-         /**
-          * Reveal cuts the curtain to highlight the given box,
-          * and shows a tooltip with instructions next to the box.
-          *
-          * @param  {String|ClientRect} [box]   box used to cut the curtain
-          * @param  {String}    [text]          text for a tooltip
-          * @param  {Object}    [options]
-          * @param  {string}    [options.tooltipClass]    optional class to add to the tooltip
-          * @param  {integer}   [options.duration]        transition time in milliseconds
-          * @param  {string}    [options.buttonText]      if set, create a button with this text label
-          * @param  {function}  [options.buttonCallback]  if set, the callback for the button
-          * @param  {function}  [options.padding]         extra margin in px to put around bbox
-          * @param  {String|ClientRect} [options.tooltipBox]  box for tooltip position, if different from box for the curtain
-          */
 
+         wiki.entityIDs = function (val) {
+           if (!_arguments.length) return _entityIDs;
+           _entityIDs = val;
+           return wiki;
+         };
 
-         curtain.reveal = function (box, html, options) {
-           options = options || {};
+         wiki.focus = function () {
+           _titleInput.node().focus();
+         };
 
-           if (typeof box === 'string') {
-             box = select(box).node();
-           }
+         return utilRebind(wiki, dispatch, 'on');
+       }
+       uiFieldWikipedia.supportsMultiselection = false;
 
-           if (box && box.getBoundingClientRect) {
-             box = copyBox(box.getBoundingClientRect());
-             var containerRect = containerNode.getBoundingClientRect();
-             box.top -= containerRect.top;
-             box.left -= containerRect.left;
-           }
+       var uiFields = {
+         access: uiFieldAccess,
+         address: uiFieldAddress,
+         check: uiFieldCheck,
+         combo: uiFieldCombo,
+         cycleway: uiFieldCycleway,
+         defaultCheck: uiFieldCheck,
+         email: uiFieldText,
+         identifier: uiFieldText,
+         lanes: uiFieldLanes,
+         localized: uiFieldLocalized,
+         roadspeed: uiFieldRoadspeed,
+         roadheight: uiFieldText,
+         manyCombo: uiFieldCombo,
+         multiCombo: uiFieldCombo,
+         networkCombo: uiFieldCombo,
+         number: uiFieldText,
+         onewayCheck: uiFieldCheck,
+         radio: uiFieldRadio,
+         restrictions: uiFieldRestrictions,
+         semiCombo: uiFieldCombo,
+         structureRadio: uiFieldRadio,
+         tel: uiFieldText,
+         text: uiFieldText,
+         textarea: uiFieldTextarea,
+         typeCombo: uiFieldCombo,
+         url: uiFieldText,
+         wikidata: uiFieldWikidata,
+         wikipedia: uiFieldWikipedia
+       };
 
-           if (box && options.padding) {
-             box.top -= options.padding;
-             box.left -= options.padding;
-             box.bottom += options.padding;
-             box.right += options.padding;
-             box.height += options.padding * 2;
-             box.width += options.padding * 2;
-           }
+       function uiField(context, presetField, entityIDs, options) {
+         options = Object.assign({
+           show: true,
+           wrap: true,
+           remove: true,
+           revert: true,
+           info: true
+         }, options);
+         var dispatch = dispatch$8('change', 'revert');
+         var field = Object.assign({}, presetField); // shallow copy
 
-           var tooltipBox;
+         field.domId = utilUniqueDomId('form-field-' + field.safeid);
+         var _show = options.show;
+         var _state = '';
+         var _tags = {};
 
-           if (options.tooltipBox) {
-             tooltipBox = options.tooltipBox;
+         var _entityExtent;
 
-             if (typeof tooltipBox === 'string') {
-               tooltipBox = select(tooltipBox).node();
-             }
+         if (entityIDs && entityIDs.length) {
+           _entityExtent = entityIDs.reduce(function (extent, entityID) {
+             var entity = context.graph().entity(entityID);
+             return extent.extend(entity.extent(context.graph()));
+           }, geoExtent());
+         }
 
-             if (tooltipBox && tooltipBox.getBoundingClientRect) {
-               tooltipBox = copyBox(tooltipBox.getBoundingClientRect());
-             }
-           } else {
-             tooltipBox = box;
-           }
+         var _locked = false;
 
-           if (tooltipBox && html) {
-             if (html.indexOf('**') !== -1) {
-               if (html.indexOf('<span') === 0) {
-                 html = html.replace(/^(<span.*?>)(.+?)(\*\*)/, '$1<span>$2</span>$3');
-               } else {
-                 html = html.replace(/^(.+?)(\*\*)/, '<span>$1</span>$2');
-               } // pseudo markdown bold text for the instruction section..
+         var _lockedTip = uiTooltip().title(_t.html('inspector.lock.suggestion', {
+           label: field.label
+         })).placement('bottom');
 
+         field.keys = field.keys || [field.key]; // only create the fields that are actually being shown
 
-               html = html.replace(/\*\*(.*?)\*\*/g, '<span class="instruction">$1</span>');
-             }
+         if (_show && !field.impl) {
+           createField();
+         } // Creates the field.. This is done lazily,
+         // once we know that the field will be shown.
 
-             html = html.replace(/\*(.*?)\*/g, '<em>$1</em>'); // emphasis
 
-             html = html.replace(/\{br\}/g, '<br/><br/>'); // linebreak
+         function createField() {
+           field.impl = uiFields[field.type](field, context).on('change', function (t, onInput) {
+             dispatch.call('change', field, t, onInput);
+           });
 
-             if (options.buttonText && options.buttonCallback) {
-               html += '<div class="button-section">' + '<button href="#" class="button action">' + options.buttonText + '</button></div>';
+           if (entityIDs) {
+             field.entityIDs = entityIDs; // if this field cares about the entities, pass them along
+
+             if (field.impl.entityIDs) {
+               field.impl.entityIDs(entityIDs);
              }
+           }
+         }
 
-             var classes = 'curtain-tooltip popover tooltip arrowed in ' + (options.tooltipClass || '');
-             tooltip.classed(classes, true).selectAll('.popover-inner').html(html);
+         function isModified() {
+           if (!entityIDs || !entityIDs.length) return false;
+           return entityIDs.some(function (entityID) {
+             var original = context.graph().base().entities[entityID];
+             var latest = context.graph().entity(entityID);
+             return field.keys.some(function (key) {
+               return original ? latest.tags[key] !== original.tags[key] : latest.tags[key];
+             });
+           });
+         }
 
-             if (options.buttonText && options.buttonCallback) {
-               var button = tooltip.selectAll('.button-section .button.action');
-               button.on('click', function (d3_event) {
-                 d3_event.preventDefault();
-                 options.buttonCallback();
-               });
-             }
+         function tagsContainFieldKey() {
+           return field.keys.some(function (key) {
+             if (field.type === 'multiCombo') {
+               for (var tagKey in _tags) {
+                 if (tagKey.indexOf(key) === 0) {
+                   return true;
+                 }
+               }
 
-             var tip = copyBox(tooltip.node().getBoundingClientRect()),
-                 w = containerNode.clientWidth,
-                 h = containerNode.clientHeight,
-                 tooltipWidth = 200,
-                 tooltipArrow = 5,
-                 side,
-                 pos; // hack: this will have bottom placement,
-             // so need to reserve extra space for the tooltip illustration.
+               return false;
+             }
 
-             if (options.tooltipClass === 'intro-mouse') {
-               tip.height += 80;
-             } // trim box dimensions to just the portion that fits in the container..
+             return _tags[key] !== undefined;
+           });
+         }
 
+         function revert(d3_event, d) {
+           d3_event.stopPropagation();
+           d3_event.preventDefault();
+           if (!entityIDs || _locked) return;
+           dispatch.call('revert', d, d.keys);
+         }
 
-             if (tooltipBox.top + tooltipBox.height > h) {
-               tooltipBox.height -= tooltipBox.top + tooltipBox.height - h;
-             }
+         function remove(d3_event, d) {
+           d3_event.stopPropagation();
+           d3_event.preventDefault();
+           if (_locked) return;
+           var t = {};
+           d.keys.forEach(function (key) {
+             t[key] = undefined;
+           });
+           dispatch.call('change', d, t);
+         }
 
-             if (tooltipBox.left + tooltipBox.width > w) {
-               tooltipBox.width -= tooltipBox.left + tooltipBox.width - w;
-             } // determine tooltip placement..
+         field.render = function (selection) {
+           var container = selection.selectAll('.form-field').data([field]); // Enter
 
+           var enter = container.enter().append('div').attr('class', function (d) {
+             return 'form-field form-field-' + d.safeid;
+           }).classed('nowrap', !options.wrap);
 
-             if (tooltipBox.top + tooltipBox.height < 100) {
-               // tooltip below box..
-               side = 'bottom';
-               pos = [tooltipBox.left + tooltipBox.width / 2 - tip.width / 2, tooltipBox.top + tooltipBox.height];
-             } else if (tooltipBox.top > h - 140) {
-               // tooltip above box..
-               side = 'top';
-               pos = [tooltipBox.left + tooltipBox.width / 2 - tip.width / 2, tooltipBox.top - tip.height];
-             } else {
-               // tooltip to the side of the tooltipBox..
-               var tipY = tooltipBox.top + tooltipBox.height / 2 - tip.height / 2;
+           if (options.wrap) {
+             var labelEnter = enter.append('label').attr('class', 'field-label').attr('for', function (d) {
+               return d.domId;
+             });
+             var textEnter = labelEnter.append('span').attr('class', 'label-text');
+             textEnter.append('span').attr('class', 'label-textvalue').html(function (d) {
+               return d.label();
+             });
+             textEnter.append('span').attr('class', 'label-textannotation');
 
-               if (_mainLocalizer.textDirection() === 'rtl') {
-                 if (tooltipBox.left - tooltipWidth - tooltipArrow < 70) {
-                   side = 'right';
-                   pos = [tooltipBox.left + tooltipBox.width + tooltipArrow, tipY];
-                 } else {
-                   side = 'left';
-                   pos = [tooltipBox.left - tooltipWidth - tooltipArrow, tipY];
-                 }
-               } else {
-                 if (tooltipBox.left + tooltipBox.width + tooltipArrow + tooltipWidth > w - 70) {
-                   side = 'left';
-                   pos = [tooltipBox.left - tooltipWidth - tooltipArrow, tipY];
-                 } else {
-                   side = 'right';
-                   pos = [tooltipBox.left + tooltipBox.width + tooltipArrow, tipY];
-                 }
-               }
+             if (options.remove) {
+               labelEnter.append('button').attr('class', 'remove-icon').attr('title', _t('icons.remove')).call(svgIcon('#iD-operation-delete'));
              }
 
-             if (options.duration !== 0 || !tooltip.classed(side)) {
-               tooltip.call(uiToggle(true));
+             if (options.revert) {
+               labelEnter.append('button').attr('class', 'modified-icon').attr('title', _t('icons.undo')).call(svgIcon(_mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-redo' : '#iD-icon-undo'));
              }
+           } // Update
 
-             tooltip.style('top', pos[1] + 'px').style('left', pos[0] + 'px').attr('class', classes + ' ' + side); // shift popover-inner if it is very close to the top or bottom edge
-             // (doesn't affect the placement of the popover-arrow)
 
-             var shiftY = 0;
+           container = container.merge(enter);
+           container.select('.field-label > .remove-icon') // propagate bound data
+           .on('click', remove);
+           container.select('.field-label > .modified-icon') // propagate bound data
+           .on('click', revert);
+           container.each(function (d) {
+             var selection = select(this);
 
-             if (side === 'left' || side === 'right') {
-               if (pos[1] < 60) {
-                 shiftY = 60 - pos[1];
-               } else if (pos[1] + tip.height > h - 100) {
-                 shiftY = h - pos[1] - tip.height - 100;
-               }
+             if (!d.impl) {
+               createField();
              }
 
-             tooltip.selectAll('.popover-inner').style('top', shiftY + 'px');
-           } else {
-             tooltip.classed('in', false).call(uiToggle(false));
-           }
-
-           curtain.cut(box, options.duration);
-           return tooltip;
-         };
+             var reference, help; // instantiate field help
 
-         curtain.cut = function (datum, duration) {
-           darkness.datum(datum).interrupt();
-           var selection;
+             if (options.wrap && field.type === 'restrictions') {
+               help = uiFieldHelp(context, 'restrictions');
+             } // instantiate tag reference
 
-           if (duration === 0) {
-             selection = darkness;
-           } else {
-             selection = darkness.transition().duration(duration || 600).ease(linear$1);
-           }
 
-           selection.attr('d', function (d) {
-             var containerWidth = containerNode.clientWidth;
-             var containerHeight = containerNode.clientHeight;
-             var string = 'M 0,0 L 0,' + containerHeight + ' L ' + containerWidth + ',' + containerHeight + 'L' + containerWidth + ',0 Z';
-             if (!d) return string;
-             return string + 'M' + d.left + ',' + d.top + 'L' + d.left + ',' + (d.top + d.height) + 'L' + (d.left + d.width) + ',' + (d.top + d.height) + 'L' + (d.left + d.width) + ',' + d.top + 'Z';
-           });
-         };
+             if (options.wrap && options.info) {
+               var referenceKey = d.key || '';
 
-         curtain.remove = function () {
-           surface.remove();
-           tooltip.remove();
-           select(window).on('resize.curtain', null);
-         }; // ClientRects are immutable, so copy them to an object,
-         // in case we need to trim the height/width.
+               if (d.type === 'multiCombo') {
+                 // lookup key without the trailing ':'
+                 referenceKey = referenceKey.replace(/:$/, '');
+               }
 
+               reference = uiTagReference(d.reference || {
+                 key: referenceKey
+               });
 
-         function copyBox(src) {
-           return {
-             top: src.top,
-             right: src.right,
-             bottom: src.bottom,
-             left: src.left,
-             width: src.width,
-             height: src.height
-           };
-         }
+               if (_state === 'hover') {
+                 reference.showing(false);
+               }
+             }
 
-         return curtain;
-       }
+             selection.call(d.impl); // add field help components
 
-       function uiIntroWelcome(context, reveal) {
-         var dispatch$1 = dispatch('done');
-         var chapter = {
-           title: 'intro.welcome.title'
-         };
+             if (help) {
+               selection.call(help.body).select('.field-label').call(help.button);
+             } // add tag reference components
 
-         function welcome() {
-           context.map().centerZoom([-85.63591, 41.94285], 19);
-           reveal('.intro-nav-wrap .chapter-welcome', helpHtml('intro.welcome.welcome'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: practice
-           });
-         }
 
-         function practice() {
-           reveal('.intro-nav-wrap .chapter-welcome', helpHtml('intro.welcome.practice'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: words
-           });
-         }
+             if (reference) {
+               selection.call(reference.body).select('.field-label').call(reference.button);
+             }
 
-         function words() {
-           reveal('.intro-nav-wrap .chapter-welcome', helpHtml('intro.welcome.words'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: chapters
+             d.impl.tags(_tags);
            });
-         }
+           container.classed('locked', _locked).classed('modified', isModified()).classed('present', tagsContainFieldKey()); // show a tip and lock icon if the field is locked
 
-         function chapters() {
-           dispatch$1.call('done');
-           reveal('.intro-nav-wrap .chapter-navigation', helpHtml('intro.welcome.chapters', {
-             next: _t('intro.navigation.title')
-           }));
-         }
+           var annotation = container.selectAll('.field-label .label-textannotation');
+           var icon = annotation.selectAll('.icon').data(_locked ? [0] : []);
+           icon.exit().remove();
+           icon.enter().append('svg').attr('class', 'icon').append('use').attr('xlink:href', '#fas-lock');
+           container.call(_locked ? _lockedTip : _lockedTip.destroy);
+         };
 
-         chapter.enter = function () {
-           welcome();
+         field.state = function (val) {
+           if (!arguments.length) return _state;
+           _state = val;
+           return field;
          };
 
-         chapter.exit = function () {
-           context.container().select('.curtain-tooltip.intro-mouse').selectAll('.counter').remove();
+         field.tags = function (val) {
+           if (!arguments.length) return _tags;
+           _tags = val;
+
+           if (tagsContainFieldKey() && !_show) {
+             // always show a field if it has a value to display
+             _show = true;
+
+             if (!field.impl) {
+               createField();
+             }
+           }
+
+           return field;
          };
 
-         chapter.restart = function () {
-           chapter.exit();
-           chapter.enter();
+         field.locked = function (val) {
+           if (!arguments.length) return _locked;
+           _locked = val;
+           return field;
          };
 
-         return utilRebind(chapter, dispatch$1, 'on');
-       }
+         field.show = function () {
+           _show = true;
 
-       function uiIntroNavigation(context, reveal) {
-         var dispatch$1 = dispatch('done');
-         var timeouts = [];
-         var hallId = 'n2061';
-         var townHall = [-85.63591, 41.94285];
-         var springStreetId = 'w397';
-         var springStreetEndId = 'n1834';
-         var springStreet = [-85.63582, 41.94255];
-         var onewayField = _mainPresetIndex.field('oneway');
-         var maxspeedField = _mainPresetIndex.field('maxspeed');
-         var chapter = {
-           title: 'intro.navigation.title'
-         };
+           if (!field.impl) {
+             createField();
+           }
 
-         function timeout(f, t) {
-           timeouts.push(window.setTimeout(f, t));
-         }
+           if (field["default"] && field.key && _tags[field.key] !== field["default"]) {
+             var t = {};
+             t[field.key] = field["default"];
+             dispatch.call('change', this, t);
+           }
+         }; // A shown field has a visible UI, a non-shown field is in the 'Add field' dropdown
 
-         function eventCancel(d3_event) {
-           d3_event.stopPropagation();
-           d3_event.preventDefault();
-         }
 
-         function isTownHallSelected() {
-           var ids = context.selectedIDs();
-           return ids.length === 1 && ids[0] === hallId;
-         }
+         field.isShown = function () {
+           return _show;
+         }; // An allowed field can appear in the UI or in the 'Add field' dropdown.
+         // A non-allowed field is hidden from the user altogether
 
-         function dragMap() {
-           context.enter(modeBrowse(context));
-           context.history().reset('initial');
-           var msec = transitionTime(townHall, context.map().center());
 
-           if (msec) {
-             reveal(null, null, {
-               duration: 0
-             });
+         field.isAllowed = function () {
+           if (entityIDs && entityIDs.length > 1 && uiFields[field.type].supportsMultiselection === false) return false;
+           if (field.geometry && !entityIDs.every(function (entityID) {
+             return field.matchGeometry(context.graph().geometry(entityID));
+           })) return false;
+
+           if (entityIDs && _entityExtent && field.locationSetID) {
+             // is field allowed in this location?
+             var validLocations = _mainLocations.locationsAt(_entityExtent.center());
+             if (!validLocations[field.locationSetID]) return false;
            }
 
-           context.map().centerZoomEase(townHall, 19, msec);
-           timeout(function () {
-             var centerStart = context.map().center();
-             var textId = context.lastPointerType() === 'mouse' ? 'drag' : 'drag_touch';
-             var dragString = helpHtml('intro.navigation.map_info') + '{br}' + helpHtml('intro.navigation.' + textId);
-             reveal('.surface', dragString);
-             context.map().on('drawn.intro', function () {
-               reveal('.surface', dragString, {
-                 duration: 0
-               });
-             });
-             context.map().on('move.intro', function () {
-               var centerNow = context.map().center();
+           var prerequisiteTag = field.prerequisiteTag;
 
-               if (centerStart[0] !== centerNow[0] || centerStart[1] !== centerNow[1]) {
-                 context.map().on('move.intro', null);
-                 timeout(function () {
-                   continueTo(zoomMap);
-                 }, 3000);
+           if (entityIDs && !tagsContainFieldKey() && // ignore tagging prerequisites if a value is already present
+           prerequisiteTag) {
+             if (!entityIDs.every(function (entityID) {
+               var entity = context.graph().entity(entityID);
+
+               if (prerequisiteTag.key) {
+                 var value = entity.tags[prerequisiteTag.key];
+                 if (!value) return false;
+
+                 if (prerequisiteTag.valueNot) {
+                   return prerequisiteTag.valueNot !== value;
+                 }
+
+                 if (prerequisiteTag.value) {
+                   return prerequisiteTag.value === value;
+                 }
+               } else if (prerequisiteTag.keyNot) {
+                 if (entity.tags[prerequisiteTag.keyNot]) return false;
                }
-             });
-           }, msec + 100);
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             nextStep();
+               return true;
+             })) return false;
            }
-         }
 
-         function zoomMap() {
-           var zoomStart = context.map().zoom();
-           var textId = context.lastPointerType() === 'mouse' ? 'zoom' : 'zoom_touch';
-           var zoomString = helpHtml('intro.navigation.' + textId);
-           reveal('.surface', zoomString);
-           context.map().on('drawn.intro', function () {
-             reveal('.surface', zoomString, {
-               duration: 0
-             });
-           });
-           context.map().on('move.intro', function () {
-             if (context.map().zoom() !== zoomStart) {
-               context.map().on('move.intro', null);
-               timeout(function () {
-                 continueTo(features);
-               }, 3000);
-             }
-           });
+           return true;
+         };
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             nextStep();
+         field.focus = function () {
+           if (field.impl) {
+             field.impl.focus();
            }
-         }
+         };
 
-         function features() {
-           var onClick = function onClick() {
-             continueTo(pointsLinesAreas);
-           };
+         return utilRebind(field, dispatch, 'on');
+       }
 
-           reveal('.surface', helpHtml('intro.navigation.features'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: onClick
-           });
-           context.map().on('drawn.intro', function () {
-             reveal('.surface', helpHtml('intro.navigation.features'), {
-               duration: 0,
-               buttonText: _t.html('intro.ok'),
-               buttonCallback: onClick
-             });
+       function uiFormFields(context) {
+         var moreCombo = uiCombobox(context, 'more-fields').minItems(1);
+         var _fieldsArr = [];
+         var _lastPlaceholder = '';
+         var _state = '';
+         var _klass = '';
+
+         function formFields(selection) {
+           var allowedFields = _fieldsArr.filter(function (field) {
+             return field.isAllowed();
            });
 
-           function continueTo(nextStep) {
-             context.map().on('drawn.intro', null);
-             nextStep();
-           }
-         }
+           var shown = allowedFields.filter(function (field) {
+             return field.isShown();
+           });
+           var notShown = allowedFields.filter(function (field) {
+             return !field.isShown();
+           });
+           var container = selection.selectAll('.form-fields-container').data([0]);
+           container = container.enter().append('div').attr('class', 'form-fields-container ' + (_klass || '')).merge(container);
+           var fields = container.selectAll('.wrap-form-field').data(shown, function (d) {
+             return d.id + (d.entityIDs ? d.entityIDs.join() : '');
+           });
+           fields.exit().remove(); // Enter
 
-         function pointsLinesAreas() {
-           var onClick = function onClick() {
-             continueTo(nodesWays);
-           };
+           var enter = fields.enter().append('div').attr('class', function (d) {
+             return 'wrap-form-field wrap-form-field-' + d.safeid;
+           }); // Update
 
-           reveal('.surface', helpHtml('intro.navigation.points_lines_areas'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: onClick
+           fields = fields.merge(enter);
+           fields.order().each(function (d) {
+             select(this).call(d.render);
            });
-           context.map().on('drawn.intro', function () {
-             reveal('.surface', helpHtml('intro.navigation.points_lines_areas'), {
-               duration: 0,
-               buttonText: _t.html('intro.ok'),
-               buttonCallback: onClick
-             });
+           var titles = [];
+           var moreFields = notShown.map(function (field) {
+             var title = field.title();
+             titles.push(title);
+             var terms = field.terms();
+             if (field.key) terms.push(field.key);
+             if (field.keys) terms = terms.concat(field.keys);
+             return {
+               display: field.label(),
+               value: title,
+               title: title,
+               field: field,
+               terms: terms
+             };
            });
+           var placeholder = titles.slice(0, 3).join(', ') + (titles.length > 3 ? '…' : '');
+           var more = selection.selectAll('.more-fields').data(_state === 'hover' || moreFields.length === 0 ? [] : [0]);
+           more.exit().remove();
+           var moreEnter = more.enter().append('div').attr('class', 'more-fields').append('label');
+           moreEnter.append('span').html(_t.html('inspector.add_fields'));
+           more = moreEnter.merge(more);
+           var input = more.selectAll('.value').data([0]);
+           input.exit().remove();
+           input = input.enter().append('input').attr('class', 'value').attr('type', 'text').attr('placeholder', placeholder).call(utilNoAuto).merge(input);
+           input.call(utilGetSetValue, '').call(moreCombo.data(moreFields).on('accept', function (d) {
+             if (!d) return; // user entered something that was not matched
 
-           function continueTo(nextStep) {
-             context.map().on('drawn.intro', null);
-             nextStep();
+             var field = d.field;
+             field.show();
+             selection.call(formFields); // rerender
+
+             field.focus();
+           })); // avoid updating placeholder excessively (triggers style recalc)
+
+           if (_lastPlaceholder !== placeholder) {
+             input.attr('placeholder', placeholder);
+             _lastPlaceholder = placeholder;
            }
          }
 
-         function nodesWays() {
-           var onClick = function onClick() {
-             continueTo(clickTownHall);
-           };
+         formFields.fieldsArr = function (val) {
+           if (!arguments.length) return _fieldsArr;
+           _fieldsArr = val || [];
+           return formFields;
+         };
 
-           reveal('.surface', helpHtml('intro.navigation.nodes_ways'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: onClick
-           });
-           context.map().on('drawn.intro', function () {
-             reveal('.surface', helpHtml('intro.navigation.nodes_ways'), {
-               duration: 0,
-               buttonText: _t.html('intro.ok'),
-               buttonCallback: onClick
-             });
-           });
+         formFields.state = function (val) {
+           if (!arguments.length) return _state;
+           _state = val;
+           return formFields;
+         };
 
-           function continueTo(nextStep) {
-             context.map().on('drawn.intro', null);
-             nextStep();
-           }
-         }
+         formFields.klass = function (val) {
+           if (!arguments.length) return _klass;
+           _klass = val;
+           return formFields;
+         };
 
-         function clickTownHall() {
-           context.enter(modeBrowse(context));
-           context.history().reset('initial');
-           var entity = context.hasEntity(hallId);
-           if (!entity) return;
-           reveal(null, null, {
-             duration: 0
-           });
-           context.map().centerZoomEase(entity.loc, 19, 500);
-           timeout(function () {
-             var entity = context.hasEntity(hallId);
-             if (!entity) return;
-             var box = pointBox(entity.loc, context);
-             var textId = context.lastPointerType() === 'mouse' ? 'click_townhall' : 'tap_townhall';
-             reveal(box, helpHtml('intro.navigation.' + textId));
-             context.map().on('move.intro drawn.intro', function () {
-               var entity = context.hasEntity(hallId);
-               if (!entity) return;
-               var box = pointBox(entity.loc, context);
-               reveal(box, helpHtml('intro.navigation.' + textId), {
-                 duration: 0
-               });
-             });
-             context.on('enter.intro', function () {
-               if (isTownHallSelected()) continueTo(selectedTownHall);
-             });
-           }, 550); // after centerZoomEase
+         return formFields;
+       }
 
-           context.history().on('change.intro', function () {
-             if (!context.hasEntity(hallId)) {
-               continueTo(clickTownHall);
-             }
-           });
+       function uiSectionPresetFields(context) {
+         var section = uiSection('preset-fields', context).label(_t.html('inspector.fields')).disclosureContent(renderDisclosureContent);
+         var dispatch = dispatch$8('change', 'revert');
+         var formFields = uiFormFields(context);
 
-           function continueTo(nextStep) {
-             context.on('enter.intro', null);
-             context.map().on('move.intro drawn.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
-           }
-         }
+         var _state;
 
-         function selectedTownHall() {
-           if (!isTownHallSelected()) return clickTownHall();
-           var entity = context.hasEntity(hallId);
-           if (!entity) return clickTownHall();
-           var box = pointBox(entity.loc, context);
+         var _fieldsArr;
 
-           var onClick = function onClick() {
-             continueTo(editorTownHall);
-           };
+         var _presets = [];
+
+         var _tags;
+
+         var _entityIDs;
+
+         function renderDisclosureContent(selection) {
+           if (!_fieldsArr) {
+             var graph = context.graph();
+             var geometries = Object.keys(_entityIDs.reduce(function (geoms, entityID) {
+               geoms[graph.entity(entityID).geometry(graph)] = true;
+               return geoms;
+             }, {}));
+             var presetsManager = _mainPresetIndex;
+             var allFields = [];
+             var allMoreFields = [];
+             var sharedTotalFields;
+
+             _presets.forEach(function (preset) {
+               var fields = preset.fields();
+               var moreFields = preset.moreFields();
+               allFields = utilArrayUnion(allFields, fields);
+               allMoreFields = utilArrayUnion(allMoreFields, moreFields);
+
+               if (!sharedTotalFields) {
+                 sharedTotalFields = utilArrayUnion(fields, moreFields);
+               } else {
+                 sharedTotalFields = sharedTotalFields.filter(function (field) {
+                   return fields.indexOf(field) !== -1 || moreFields.indexOf(field) !== -1;
+                 });
+               }
+             });
+
+             var sharedFields = allFields.filter(function (field) {
+               return sharedTotalFields.indexOf(field) !== -1;
+             });
+             var sharedMoreFields = allMoreFields.filter(function (field) {
+               return sharedTotalFields.indexOf(field) !== -1;
+             });
+             _fieldsArr = [];
+             sharedFields.forEach(function (field) {
+               if (field.matchAllGeometry(geometries)) {
+                 _fieldsArr.push(uiField(context, field, _entityIDs));
+               }
+             });
+             var singularEntity = _entityIDs.length === 1 && graph.hasEntity(_entityIDs[0]);
+
+             if (singularEntity && singularEntity.isHighwayIntersection(graph) && presetsManager.field('restrictions')) {
+               _fieldsArr.push(uiField(context, presetsManager.field('restrictions'), _entityIDs));
+             }
 
-           reveal(box, helpHtml('intro.navigation.selected_townhall'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: onClick
-           });
-           context.map().on('move.intro drawn.intro', function () {
-             var entity = context.hasEntity(hallId);
-             if (!entity) return;
-             var box = pointBox(entity.loc, context);
-             reveal(box, helpHtml('intro.navigation.selected_townhall'), {
-               duration: 0,
-               buttonText: _t.html('intro.ok'),
-               buttonCallback: onClick
+             var additionalFields = utilArrayUnion(sharedMoreFields, presetsManager.universal());
+             additionalFields.sort(function (field1, field2) {
+               return field1.label().localeCompare(field2.label(), _mainLocalizer.localeCode());
              });
+             additionalFields.forEach(function (field) {
+               if (sharedFields.indexOf(field) === -1 && field.matchAllGeometry(geometries)) {
+                 _fieldsArr.push(uiField(context, field, _entityIDs, {
+                   show: false
+                 }));
+               }
+             });
+
+             _fieldsArr.forEach(function (field) {
+               field.on('change', function (t, onInput) {
+                 dispatch.call('change', field, _entityIDs, t, onInput);
+               }).on('revert', function (keys) {
+                 dispatch.call('revert', field, keys);
+               });
+             });
+           }
+
+           _fieldsArr.forEach(function (field) {
+             field.state(_state).tags(_tags);
            });
-           context.history().on('change.intro', function () {
-             if (!context.hasEntity(hallId)) {
-               continueTo(clickTownHall);
+
+           selection.call(formFields.fieldsArr(_fieldsArr).state(_state).klass('grouped-items-area'));
+           selection.selectAll('.wrap-form-field input').on('keydown', function (d3_event) {
+             // if user presses enter, and combobox is not active, accept edits..
+             if (d3_event.keyCode === 13 && // ↩ Return
+             context.container().select('.combobox').empty()) {
+               context.enter(modeBrowse(context));
              }
            });
+         }
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
+         section.presets = function (val) {
+           if (!arguments.length) return _presets;
+
+           if (!_presets || !val || !utilArrayIdentical(_presets, val)) {
+             _presets = val;
+             _fieldsArr = null;
            }
-         }
 
-         function editorTownHall() {
-           if (!isTownHallSelected()) return clickTownHall(); // disallow scrolling
+           return section;
+         };
 
-           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
+         section.state = function (val) {
+           if (!arguments.length) return _state;
+           _state = val;
+           return section;
+         };
 
-           var onClick = function onClick() {
-             continueTo(presetTownHall);
-           };
+         section.tags = function (val) {
+           if (!arguments.length) return _tags;
+           _tags = val; // Don't reset _fieldsArr here.
 
-           reveal('.entity-editor-pane', helpHtml('intro.navigation.editor_townhall'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: onClick
-           });
-           context.on('exit.intro', function () {
-             continueTo(clickTownHall);
-           });
-           context.history().on('change.intro', function () {
-             if (!context.hasEntity(hallId)) {
-               continueTo(clickTownHall);
-             }
-           });
+           return section;
+         };
 
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             context.history().on('change.intro', null);
-             context.container().select('.inspector-wrap').on('wheel.intro', null);
-             nextStep();
+         section.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs;
+
+           if (!val || !_entityIDs || !utilArrayIdentical(_entityIDs, val)) {
+             _entityIDs = val;
+             _fieldsArr = null;
            }
-         }
 
-         function presetTownHall() {
-           if (!isTownHallSelected()) return clickTownHall(); // reset pane, in case user happened to change it..
+           return section;
+         };
 
-           context.container().select('.inspector-wrap .panewrap').style('right', '0%'); // disallow scrolling
+         return utilRebind(section, dispatch, 'on');
+       }
 
-           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel); // preset match, in case the user happened to change it.
+       function uiSectionRawMemberEditor(context) {
+         var section = uiSection('raw-member-editor', context).shouldDisplay(function () {
+           if (!_entityIDs || _entityIDs.length !== 1) return false;
+           var entity = context.hasEntity(_entityIDs[0]);
+           return entity && entity.type === 'relation';
+         }).label(function () {
+           var entity = context.hasEntity(_entityIDs[0]);
+           if (!entity) return '';
+           var gt = entity.members.length > _maxMembers ? '>' : '';
+           var count = gt + entity.members.slice(0, _maxMembers).length;
+           return _t('inspector.title_count', {
+             title: _t.html('inspector.members'),
+             count: count
+           });
+         }).disclosureContent(renderDisclosureContent);
+         var taginfo = services.taginfo;
 
-           var entity = context.entity(context.selectedIDs()[0]);
-           var preset = _mainPresetIndex.match(entity, context.graph());
+         var _entityIDs;
 
-           var onClick = function onClick() {
-             continueTo(fieldsTownHall);
-           };
+         var _maxMembers = 1000;
 
-           reveal('.entity-editor-pane .section-feature-type', helpHtml('intro.navigation.preset_townhall', {
-             preset: preset.name()
-           }), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: onClick
-           });
-           context.on('exit.intro', function () {
-             continueTo(clickTownHall);
-           });
-           context.history().on('change.intro', function () {
-             if (!context.hasEntity(hallId)) {
-               continueTo(clickTownHall);
-             }
+         function downloadMember(d3_event, d) {
+           d3_event.preventDefault(); // display the loading indicator
+
+           select(this.parentNode).classed('tag-reference-loading', true);
+           context.loadEntity(d.id, function () {
+             section.reRender();
            });
+         }
 
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             context.history().on('change.intro', null);
-             context.container().select('.inspector-wrap').on('wheel.intro', null);
-             nextStep();
-           }
+         function zoomToMember(d3_event, d) {
+           d3_event.preventDefault();
+           var entity = context.entity(d.id);
+           context.map().zoomToEase(entity); // highlight the feature in case it wasn't previously on-screen
+
+           utilHighlightEntities([d.id], true, context);
          }
 
-         function fieldsTownHall() {
-           if (!isTownHallSelected()) return clickTownHall(); // reset pane, in case user happened to change it..
+         function selectMember(d3_event, d) {
+           d3_event.preventDefault(); // remove the hover-highlight styling
 
-           context.container().select('.inspector-wrap .panewrap').style('right', '0%'); // disallow scrolling
+           utilHighlightEntities([d.id], false, context);
+           var entity = context.entity(d.id);
+           var mapExtent = context.map().extent();
 
-           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
+           if (!entity.intersects(mapExtent, context.graph())) {
+             // zoom to the entity if its extent is not visible now
+             context.map().zoomToEase(entity);
+           }
 
-           var onClick = function onClick() {
-             continueTo(closeTownHall);
-           };
+           context.enter(modeSelect(context, [d.id]));
+         }
 
-           reveal('.entity-editor-pane .section-preset-fields', helpHtml('intro.navigation.fields_townhall'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: onClick
-           });
-           context.on('exit.intro', function () {
-             continueTo(clickTownHall);
-           });
-           context.history().on('change.intro', function () {
-             if (!context.hasEntity(hallId)) {
-               continueTo(clickTownHall);
-             }
-           });
+         function changeRole(d3_event, d) {
+           var oldRole = d.role;
+           var newRole = context.cleanRelationRole(select(this).property('value'));
 
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             context.history().on('change.intro', null);
-             context.container().select('.inspector-wrap').on('wheel.intro', null);
-             nextStep();
+           if (oldRole !== newRole) {
+             var member = {
+               id: d.id,
+               type: d.type,
+               role: newRole
+             };
+             context.perform(actionChangeMember(d.relation.id, member, d.index), _t('operations.change_role.annotation', {
+               n: 1
+             }));
+             context.validator().validate();
            }
          }
 
-         function closeTownHall() {
-           if (!isTownHallSelected()) return clickTownHall();
-           var selector = '.entity-editor-pane button.close svg use';
-           var href = select(selector).attr('href') || '#iD-icon-close';
-           reveal('.entity-editor-pane', helpHtml('intro.navigation.close_townhall', {
-             button: icon(href, 'inline')
+         function deleteMember(d3_event, d) {
+           // remove the hover-highlight styling
+           utilHighlightEntities([d.id], false, context);
+           context.perform(actionDeleteMember(d.relation.id, d.index), _t('operations.delete_member.annotation', {
+             n: 1
            }));
-           context.on('exit.intro', function () {
-             continueTo(searchStreet);
-           });
-           context.history().on('change.intro', function () {
-             // update the close icon in the tooltip if the user edits something.
-             var selector = '.entity-editor-pane button.close svg use';
-             var href = select(selector).attr('href') || '#iD-icon-close';
-             reveal('.entity-editor-pane', helpHtml('intro.navigation.close_townhall', {
-               button: icon(href, 'inline')
-             }), {
-               duration: 0
-             });
-           });
 
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
+           if (!context.hasEntity(d.relation.id)) {
+             // Removing the last member will also delete the relation.
+             // If this happens we need to exit the selection mode
+             context.enter(modeBrowse(context));
+           } else {
+             // Changing the mode also runs `validate`, but otherwise we need to
+             // rerun it manually
+             context.validator().validate();
            }
          }
 
-         function searchStreet() {
-           context.enter(modeBrowse(context));
-           context.history().reset('initial'); // ensure spring street exists
-
-           var msec = transitionTime(springStreet, context.map().center());
-
-           if (msec) {
-             reveal(null, null, {
-               duration: 0
+         function renderDisclosureContent(selection) {
+           var entityID = _entityIDs[0];
+           var memberships = [];
+           var entity = context.entity(entityID);
+           entity.members.slice(0, _maxMembers).forEach(function (member, index) {
+             memberships.push({
+               index: index,
+               id: member.id,
+               type: member.type,
+               role: member.role,
+               relation: entity,
+               member: context.hasEntity(member.id),
+               domId: utilUniqueDomId(entityID + '-member-' + index)
              });
-           }
-
-           context.map().centerZoomEase(springStreet, 19, msec); // ..and user can see it
+           });
+           var list = selection.selectAll('.member-list').data([0]);
+           list = list.enter().append('ul').attr('class', 'member-list').merge(list);
+           var items = list.selectAll('li').data(memberships, function (d) {
+             return osmEntity.key(d.relation) + ',' + d.index + ',' + (d.member ? osmEntity.key(d.member) : 'incomplete');
+           });
+           items.exit().each(unbind).remove();
+           var itemsEnter = items.enter().append('li').attr('class', 'member-row form-field').classed('member-incomplete', function (d) {
+             return !d.member;
+           });
+           itemsEnter.each(function (d) {
+             var item = select(this);
+             var label = item.append('label').attr('class', 'field-label').attr('for', d.domId);
 
-           timeout(function () {
-             reveal('.search-header input', helpHtml('intro.navigation.search_street', {
-               name: _t('intro.graph.name.spring-street')
-             }));
-             context.container().select('.search-header input').on('keyup.intro', checkSearchResult);
-           }, msec + 100);
-         }
+             if (d.member) {
+               // highlight the member feature in the map while hovering on the list item
+               item.on('mouseover', function () {
+                 utilHighlightEntities([d.id], true, context);
+               }).on('mouseout', function () {
+                 utilHighlightEntities([d.id], false, context);
+               });
+               var labelLink = label.append('span').attr('class', 'label-text').append('a').attr('href', '#').on('click', selectMember);
+               labelLink.append('span').attr('class', 'member-entity-type').html(function (d) {
+                 var matched = _mainPresetIndex.match(d.member, context.graph());
+                 return matched && matched.name() || utilDisplayType(d.member.id);
+               });
+               labelLink.append('span').attr('class', 'member-entity-name').html(function (d) {
+                 return utilDisplayName(d.member);
+               });
+               label.append('button').attr('title', _t('icons.remove')).attr('class', 'remove member-delete').call(svgIcon('#iD-operation-delete'));
+               label.append('button').attr('class', 'member-zoom').attr('title', _t('icons.zoom_to')).call(svgIcon('#iD-icon-framed-dot', 'monochrome')).on('click', zoomToMember);
+             } else {
+               var labelText = label.append('span').attr('class', 'label-text');
+               labelText.append('span').attr('class', 'member-entity-type').html(_t.html('inspector.' + d.type, {
+                 id: d.id
+               }));
+               labelText.append('span').attr('class', 'member-entity-name').html(_t.html('inspector.incomplete', {
+                 id: d.id
+               }));
+               label.append('button').attr('class', 'member-download').attr('title', _t('icons.download')).call(svgIcon('#iD-icon-load')).on('click', downloadMember);
+             }
+           });
+           var wrapEnter = itemsEnter.append('div').attr('class', 'form-field-input-wrap form-field-input-member');
+           wrapEnter.append('input').attr('class', 'member-role').attr('id', function (d) {
+             return d.domId;
+           }).property('type', 'text').attr('placeholder', _t('inspector.role')).call(utilNoAuto);
 
-         function checkSearchResult() {
-           var first = context.container().select('.feature-list-item:nth-child(0n+2)'); // skip "No Results" item
+           if (taginfo) {
+             wrapEnter.each(bindTypeahead);
+           } // update
 
-           var firstName = first.select('.entity-name');
-           var name = _t('intro.graph.name.spring-street');
 
-           if (!firstName.empty() && firstName.html() === name) {
-             reveal(first.node(), helpHtml('intro.navigation.choose_street', {
-               name: name
-             }), {
-               duration: 300
-             });
-             context.on('exit.intro', function () {
-               continueTo(selectedStreet);
-             });
-             context.container().select('.search-header input').on('keydown.intro', eventCancel, true).on('keyup.intro', null);
-           }
+           items = items.merge(itemsEnter).order();
+           items.select('input.member-role').property('value', function (d) {
+             return d.role;
+           }).on('blur', changeRole).on('change', changeRole);
+           items.select('button.member-delete').on('click', deleteMember);
+           var dragOrigin, targetIndex;
+           items.call(d3_drag().on('start', function (d3_event) {
+             dragOrigin = {
+               x: d3_event.x,
+               y: d3_event.y
+             };
+             targetIndex = null;
+           }).on('drag', function (d3_event) {
+             var x = d3_event.x - dragOrigin.x,
+                 y = d3_event.y - dragOrigin.y;
+             if (!select(this).classed('dragging') && // don't display drag until dragging beyond a distance threshold
+             Math.sqrt(Math.pow(x, 2) + Math.pow(y, 2)) <= 5) return;
+             var index = items.nodes().indexOf(this);
+             select(this).classed('dragging', true);
+             targetIndex = null;
+             selection.selectAll('li.member-row').style('transform', function (d2, index2) {
+               var node = select(this).node();
 
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             context.container().select('.search-header input').on('keydown.intro', null).on('keyup.intro', null);
-             nextStep();
-           }
-         }
+               if (index === index2) {
+                 return 'translate(' + x + 'px, ' + y + 'px)';
+               } else if (index2 > index && d3_event.y > node.offsetTop) {
+                 if (targetIndex === null || index2 > targetIndex) {
+                   targetIndex = index2;
+                 }
 
-         function selectedStreet() {
-           if (!context.hasEntity(springStreetEndId) || !context.hasEntity(springStreetId)) {
-             return searchStreet();
-           }
+                 return 'translateY(-100%)';
+               } else if (index2 < index && d3_event.y < node.offsetTop + node.offsetHeight) {
+                 if (targetIndex === null || index2 < targetIndex) {
+                   targetIndex = index2;
+                 }
 
-           var onClick = function onClick() {
-             continueTo(editorStreet);
-           };
+                 return 'translateY(100%)';
+               }
 
-           var entity = context.entity(springStreetEndId);
-           var box = pointBox(entity.loc, context);
-           box.height = 500;
-           reveal(box, helpHtml('intro.navigation.selected_street', {
-             name: _t('intro.graph.name.spring-street')
-           }), {
-             duration: 600,
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: onClick
-           });
-           timeout(function () {
-             context.map().on('move.intro drawn.intro', function () {
-               var entity = context.hasEntity(springStreetEndId);
-               if (!entity) return;
-               var box = pointBox(entity.loc, context);
-               box.height = 500;
-               reveal(box, helpHtml('intro.navigation.selected_street', {
-                 name: _t('intro.graph.name.spring-street')
-               }), {
-                 duration: 0,
-                 buttonText: _t.html('intro.ok'),
-                 buttonCallback: onClick
-               });
+               return null;
              });
-           }, 600); // after reveal.
+           }).on('end', function (d3_event, d) {
+             if (!select(this).classed('dragging')) return;
+             var index = items.nodes().indexOf(this);
+             select(this).classed('dragging', false);
+             selection.selectAll('li.member-row').style('transform', null);
 
-           context.on('enter.intro', function (mode) {
-             if (!context.hasEntity(springStreetId)) {
-               return continueTo(searchStreet);
+             if (targetIndex !== null) {
+               // dragged to a new position, reorder
+               context.perform(actionMoveMember(d.relation.id, index, targetIndex), _t('operations.reorder_members.annotation'));
+               context.validator().validate();
              }
+           }));
 
-             var ids = context.selectedIDs();
+           function bindTypeahead(d) {
+             var row = select(this);
+             var role = row.selectAll('input.member-role');
+             var origValue = role.property('value');
 
-             if (mode.id !== 'select' || !ids.length || ids[0] !== springStreetId) {
-               // keep Spring Street selected..
-               context.enter(modeSelect(context, [springStreetId]));
-             }
-           });
-           context.history().on('change.intro', function () {
-             if (!context.hasEntity(springStreetEndId) || !context.hasEntity(springStreetId)) {
-               timeout(function () {
-                 continueTo(searchStreet);
-               }, 300); // after any transition (e.g. if user deleted intersection)
+             function sort(value, data) {
+               var sameletter = [];
+               var other = [];
+
+               for (var i = 0; i < data.length; i++) {
+                 if (data[i].value.substring(0, value.length) === value) {
+                   sameletter.push(data[i]);
+                 } else {
+                   other.push(data[i]);
+                 }
+               }
+
+               return sameletter.concat(other);
              }
-           });
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('enter.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
-           }
-         }
+             role.call(uiCombobox(context, 'member-role').fetcher(function (role, callback) {
+               // The `geometry` param is used in the `taginfo.js` interface for
+               // filtering results, as a key into the `tag_members_fractions`
+               // object.  If we don't know the geometry because the member is
+               // not yet downloaded, it's ok to guess based on type.
+               var geometry;
 
-         function editorStreet() {
-           var selector = '.entity-editor-pane button.close svg use';
-           var href = select(selector).attr('href') || '#iD-icon-close';
-           reveal('.entity-editor-pane', helpHtml('intro.navigation.street_different_fields') + '{br}' + helpHtml('intro.navigation.editor_street', {
-             button: icon(href, 'inline'),
-             field1: onewayField.label(),
-             field2: maxspeedField.label()
-           }));
-           context.on('exit.intro', function () {
-             continueTo(play);
-           });
-           context.history().on('change.intro', function () {
-             // update the close icon in the tooltip if the user edits something.
-             var selector = '.entity-editor-pane button.close svg use';
-             var href = select(selector).attr('href') || '#iD-icon-close';
-             reveal('.entity-editor-pane', helpHtml('intro.navigation.street_different_fields') + '{br}' + helpHtml('intro.navigation.editor_street', {
-               button: icon(href, 'inline'),
-               field1: onewayField.label(),
-               field2: maxspeedField.label()
-             }), {
-               duration: 0
-             });
-           });
+               if (d.member) {
+                 geometry = context.graph().geometry(d.member.id);
+               } else if (d.type === 'relation') {
+                 geometry = 'relation';
+               } else if (d.type === 'way') {
+                 geometry = 'line';
+               } else {
+                 geometry = 'point';
+               }
 
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
+               var rtype = entity.tags.type;
+               taginfo.roles({
+                 debounce: true,
+                 rtype: rtype || '',
+                 geometry: geometry,
+                 query: role
+               }, function (err, data) {
+                 if (!err) callback(sort(role, data));
+               });
+             }).on('cancel', function () {
+               role.property('value', origValue);
+             }));
            }
-         }
 
-         function play() {
-           dispatch$1.call('done');
-           reveal('.ideditor', helpHtml('intro.navigation.play', {
-             next: _t('intro.points.title')
-           }), {
-             tooltipBox: '.intro-nav-wrap .chapter-point',
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: function buttonCallback() {
-               reveal('.ideditor');
-             }
-           });
+           function unbind() {
+             var row = select(this);
+             row.selectAll('input.member-role').call(uiCombobox.off, context);
+           }
          }
 
-         chapter.enter = function () {
-           dragMap();
+         section.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs;
+           _entityIDs = val;
+           return section;
          };
 
-         chapter.exit = function () {
-           timeouts.forEach(window.clearTimeout);
-           context.on('enter.intro exit.intro', null);
-           context.map().on('move.intro drawn.intro', null);
-           context.history().on('change.intro', null);
-           context.container().select('.inspector-wrap').on('wheel.intro', null);
-           context.container().select('.search-header input').on('keydown.intro keyup.intro', null);
-         };
+         return section;
+       }
 
-         chapter.restart = function () {
-           chapter.exit();
-           chapter.enter();
-         };
+       function actionDeleteMembers(relationId, memberIndexes) {
+         return function (graph) {
+           // Remove the members in descending order so removals won't shift what members
+           // are at the remaining indexes
+           memberIndexes.sort(function (a, b) {
+             return b - a;
+           });
 
-         return utilRebind(chapter, dispatch$1, 'on');
-       }
+           for (var i in memberIndexes) {
+             graph = actionDeleteMember(relationId, memberIndexes[i])(graph);
+           }
 
-       function uiIntroPoint(context, reveal) {
-         var dispatch$1 = dispatch('done');
-         var timeouts = [];
-         var intersection = [-85.63279, 41.94394];
-         var building = [-85.632422, 41.944045];
-         var cafePreset = _mainPresetIndex.item('amenity/cafe');
-         var _pointID = null;
-         var chapter = {
-           title: 'intro.points.title'
+           return graph;
          };
+       }
 
-         function timeout(f, t) {
-           timeouts.push(window.setTimeout(f, t));
-         }
+       function uiSectionRawMembershipEditor(context) {
+         var section = uiSection('raw-membership-editor', context).shouldDisplay(function () {
+           return _entityIDs && _entityIDs.length;
+         }).label(function () {
+           var parents = getSharedParentRelations();
+           var gt = parents.length > _maxMemberships ? '>' : '';
+           var count = gt + parents.slice(0, _maxMemberships).length;
+           return _t('inspector.title_count', {
+             title: _t.html('inspector.relations'),
+             count: count
+           });
+         }).disclosureContent(renderDisclosureContent);
+         var taginfo = services.taginfo;
+         var nearbyCombo = uiCombobox(context, 'parent-relation').minItems(1).fetcher(fetchNearbyRelations).itemsMouseEnter(function (d3_event, d) {
+           if (d.relation) utilHighlightEntities([d.relation.id], true, context);
+         }).itemsMouseLeave(function (d3_event, d) {
+           if (d.relation) utilHighlightEntities([d.relation.id], false, context);
+         });
+         var _inChange = false;
+         var _entityIDs = [];
 
-         function eventCancel(d3_event) {
-           d3_event.stopPropagation();
-           d3_event.preventDefault();
-         }
+         var _showBlank;
 
-         function addPoint() {
-           context.enter(modeBrowse(context));
-           context.history().reset('initial');
-           var msec = transitionTime(intersection, context.map().center());
+         var _maxMemberships = 1000;
 
-           if (msec) {
-             reveal(null, null, {
-               duration: 0
-             });
-           }
+         function getSharedParentRelations() {
+           var parents = [];
 
-           context.map().centerZoomEase(intersection, 19, msec);
-           timeout(function () {
-             var tooltip = reveal('button.add-point', helpHtml('intro.points.points_info') + '{br}' + helpHtml('intro.points.add_point'));
-             _pointID = null;
-             tooltip.selectAll('.popover-inner').insert('svg', 'span').attr('class', 'tooltip-illustration').append('use').attr('xlink:href', '#iD-graphic-points');
-             context.on('enter.intro', function (mode) {
-               if (mode.id !== 'add-point') return;
-               continueTo(placePoint);
-             });
-           }, msec + 100);
+           for (var i = 0; i < _entityIDs.length; i++) {
+             var entity = context.graph().hasEntity(_entityIDs[i]);
+             if (!entity) continue;
 
-           function continueTo(nextStep) {
-             context.on('enter.intro', null);
-             nextStep();
-           }
-         }
+             if (i === 0) {
+               parents = context.graph().parentRelations(entity);
+             } else {
+               parents = utilArrayIntersection(parents, context.graph().parentRelations(entity));
+             }
 
-         function placePoint() {
-           if (context.mode().id !== 'add-point') {
-             return chapter.restart();
+             if (!parents.length) break;
            }
 
-           var pointBox = pad(building, 150, context);
-           var textId = context.lastPointerType() === 'mouse' ? 'place_point' : 'place_point_touch';
-           reveal(pointBox, helpHtml('intro.points.' + textId));
-           context.map().on('move.intro drawn.intro', function () {
-             pointBox = pad(building, 150, context);
-             reveal(pointBox, helpHtml('intro.points.' + textId), {
-               duration: 0
-             });
-           });
-           context.on('enter.intro', function (mode) {
-             if (mode.id !== 'select') return chapter.restart();
-             _pointID = context.mode().selectedIDs()[0];
-             continueTo(searchPreset);
-           });
-
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
-           }
+           return parents;
          }
 
-         function searchPreset() {
-           if (context.mode().id !== 'select' || !_pointID || !context.hasEntity(_pointID)) {
-             return addPoint();
-           } // disallow scrolling
-
-
-           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
-           context.container().select('.preset-search-input').on('keydown.intro', null).on('keyup.intro', checkPresetSearch);
-           reveal('.preset-search-input', helpHtml('intro.points.search_cafe', {
-             preset: cafePreset.name()
-           }));
-           context.on('enter.intro', function (mode) {
-             if (!_pointID || !context.hasEntity(_pointID)) {
-               return continueTo(addPoint);
-             }
-
-             var ids = context.selectedIDs();
+         function getMemberships() {
+           var memberships = [];
+           var relations = getSharedParentRelations().slice(0, _maxMemberships);
+           var isMultiselect = _entityIDs.length > 1;
+           var i, relation, membership, index, member, indexedMember;
 
-             if (mode.id !== 'select' || !ids.length || ids[0] !== _pointID) {
-               // keep the user's point selected..
-               context.enter(modeSelect(context, [_pointID])); // disallow scrolling
+           for (i = 0; i < relations.length; i++) {
+             relation = relations[i];
+             membership = {
+               relation: relation,
+               members: [],
+               hash: osmEntity.key(relation)
+             };
 
-               context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
-               context.container().select('.preset-search-input').on('keydown.intro', null).on('keyup.intro', checkPresetSearch);
-               reveal('.preset-search-input', helpHtml('intro.points.search_cafe', {
-                 preset: cafePreset.name()
-               }));
-               context.history().on('change.intro', null);
-             }
-           });
+             for (index = 0; index < relation.members.length; index++) {
+               member = relation.members[index];
 
-           function checkPresetSearch() {
-             var first = context.container().select('.preset-list-item:first-child');
+               if (_entityIDs.indexOf(member.id) !== -1) {
+                 indexedMember = Object.assign({}, member, {
+                   index: index
+                 });
+                 membership.members.push(indexedMember);
+                 membership.hash += ',' + index.toString();
 
-             if (first.classed('preset-amenity-cafe')) {
-               context.container().select('.preset-search-input').on('keydown.intro', eventCancel, true).on('keyup.intro', null);
-               reveal(first.select('.preset-list-button').node(), helpHtml('intro.points.choose_cafe', {
-                 preset: cafePreset.name()
-               }), {
-                 duration: 300
-               });
-               context.history().on('change.intro', function () {
-                 continueTo(aboutFeatureEditor);
-               });
+                 if (!isMultiselect) {
+                   // For single selections, list one entry per membership per relation.
+                   // For multiselections, list one entry per relation.
+                   memberships.push(membership);
+                   membership = {
+                     relation: relation,
+                     members: [],
+                     hash: osmEntity.key(relation)
+                   };
+                 }
+               }
              }
-           }
-
-           function continueTo(nextStep) {
-             context.on('enter.intro', null);
-             context.history().on('change.intro', null);
-             context.container().select('.inspector-wrap').on('wheel.intro', null);
-             context.container().select('.preset-search-input').on('keydown.intro keyup.intro', null);
-             nextStep();
-           }
-         }
 
-         function aboutFeatureEditor() {
-           if (context.mode().id !== 'select' || !_pointID || !context.hasEntity(_pointID)) {
-             return addPoint();
+             if (membership.members.length) memberships.push(membership);
            }
 
-           timeout(function () {
-             reveal('.entity-editor-pane', helpHtml('intro.points.feature_editor'), {
-               tooltipClass: 'intro-points-describe',
-               buttonText: _t.html('intro.ok'),
-               buttonCallback: function buttonCallback() {
-                 continueTo(addName);
-               }
+           memberships.forEach(function (membership) {
+             membership.domId = utilUniqueDomId('membership-' + membership.relation.id);
+             var roles = [];
+             membership.members.forEach(function (member) {
+               if (roles.indexOf(member.role) === -1) roles.push(member.role);
              });
-           }, 400);
-           context.on('exit.intro', function () {
-             // if user leaves select mode here, just continue with the tutorial.
-             continueTo(reselectPoint);
+             membership.role = roles.length === 1 ? roles[0] : roles;
            });
-
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             nextStep();
-           }
+           return memberships;
          }
 
-         function addName() {
-           if (context.mode().id !== 'select' || !_pointID || !context.hasEntity(_pointID)) {
-             return addPoint();
-           } // reset pane, in case user happened to change it..
-
+         function selectRelation(d3_event, d) {
+           d3_event.preventDefault(); // remove the hover-highlight styling
 
-           context.container().select('.inspector-wrap .panewrap').style('right', '0%');
-           var addNameString = helpHtml('intro.points.fields_info') + '{br}' + helpHtml('intro.points.add_name');
-           timeout(function () {
-             // It's possible for the user to add a name in a previous step..
-             // If so, don't tell them to add the name in this step.
-             // Give them an OK button instead.
-             var entity = context.entity(_pointID);
+           utilHighlightEntities([d.relation.id], false, context);
+           context.enter(modeSelect(context, [d.relation.id]));
+         }
 
-             if (entity.tags.name) {
-               var tooltip = reveal('.entity-editor-pane', addNameString, {
-                 tooltipClass: 'intro-points-describe',
-                 buttonText: _t.html('intro.ok'),
-                 buttonCallback: function buttonCallback() {
-                   continueTo(addCloseEditor);
-                 }
-               });
-               tooltip.select('.instruction').style('display', 'none');
-             } else {
-               reveal('.entity-editor-pane', addNameString, {
-                 tooltipClass: 'intro-points-describe'
-               });
-             }
-           }, 400);
-           context.history().on('change.intro', function () {
-             continueTo(addCloseEditor);
-           });
-           context.on('exit.intro', function () {
-             // if user leaves select mode here, just continue with the tutorial.
-             continueTo(reselectPoint);
-           });
+         function zoomToRelation(d3_event, d) {
+           d3_event.preventDefault();
+           var entity = context.entity(d.relation.id);
+           context.map().zoomToEase(entity); // highlight the relation in case it wasn't previously on-screen
 
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
-           }
+           utilHighlightEntities([d.relation.id], true, context);
          }
 
-         function addCloseEditor() {
-           // reset pane, in case user happened to change it..
-           context.container().select('.inspector-wrap .panewrap').style('right', '0%');
-           var selector = '.entity-editor-pane button.close svg use';
-           var href = select(selector).attr('href') || '#iD-icon-close';
-           context.on('exit.intro', function () {
-             continueTo(reselectPoint);
+         function changeRole(d3_event, d) {
+           if (d === 0) return; // called on newrow (shouldn't happen)
+
+           if (_inChange) return; // avoid accidental recursive call #5731
+
+           var newRole = context.cleanRelationRole(select(this).property('value'));
+           if (!newRole.trim() && typeof d.role !== 'string') return;
+           var membersToUpdate = d.members.filter(function (member) {
+             return member.role !== newRole;
            });
-           reveal('.entity-editor-pane', helpHtml('intro.points.add_close', {
-             button: icon(href, 'inline')
-           }));
 
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             nextStep();
+           if (membersToUpdate.length) {
+             _inChange = true;
+             context.perform(function actionChangeMemberRoles(graph) {
+               membersToUpdate.forEach(function (member) {
+                 var newMember = Object.assign({}, member, {
+                   role: newRole
+                 });
+                 delete newMember.index;
+                 graph = actionChangeMember(d.relation.id, newMember, member.index)(graph);
+               });
+               return graph;
+             }, _t('operations.change_role.annotation', {
+               n: membersToUpdate.length
+             }));
+             context.validator().validate();
            }
+
+           _inChange = false;
          }
 
-         function reselectPoint() {
-           if (!_pointID) return chapter.restart();
-           var entity = context.hasEntity(_pointID);
-           if (!entity) return chapter.restart(); // make sure it's still a cafe, in case user somehow changed it..
+         function addMembership(d, role) {
+           this.blur(); // avoid keeping focus on the button
 
-           var oldPreset = _mainPresetIndex.match(entity, context.graph());
-           context.replace(actionChangePreset(_pointID, oldPreset, cafePreset));
-           context.enter(modeBrowse(context));
-           var msec = transitionTime(entity.loc, context.map().center());
+           _showBlank = false;
 
-           if (msec) {
-             reveal(null, null, {
-               duration: 0
-             });
-           }
+           function actionAddMembers(relationId, ids, role) {
+             return function (graph) {
+               for (var i in ids) {
+                 var member = {
+                   id: ids[i],
+                   type: graph.entity(ids[i]).type,
+                   role: role
+                 };
+                 graph = actionAddMember(relationId, member)(graph);
+               }
 
-           context.map().centerEase(entity.loc, msec);
-           timeout(function () {
-             var box = pointBox(entity.loc, context);
-             reveal(box, helpHtml('intro.points.reselect'), {
-               duration: 600
-             });
-             timeout(function () {
-               context.map().on('move.intro drawn.intro', function () {
-                 var entity = context.hasEntity(_pointID);
-                 if (!entity) return chapter.restart();
-                 var box = pointBox(entity.loc, context);
-                 reveal(box, helpHtml('intro.points.reselect'), {
-                   duration: 0
-                 });
-               });
-             }, 600); // after reveal..
+               return graph;
+             };
+           }
 
-             context.on('enter.intro', function (mode) {
-               if (mode.id !== 'select') return;
-               continueTo(updatePoint);
-             });
-           }, msec + 100);
+           if (d.relation) {
+             context.perform(actionAddMembers(d.relation.id, _entityIDs, role), _t('operations.add_member.annotation', {
+               n: _entityIDs.length
+             }));
+             context.validator().validate();
+           } else {
+             var relation = osmRelation();
+             context.perform(actionAddEntity(relation), actionAddMembers(relation.id, _entityIDs, role), _t('operations.add.annotation.relation')); // changing the mode also runs `validate`
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
+             context.enter(modeSelect(context, [relation.id]).newFeature(true));
            }
          }
 
-         function updatePoint() {
-           if (context.mode().id !== 'select' || !_pointID || !context.hasEntity(_pointID)) {
-             return continueTo(reselectPoint);
-           } // reset pane, in case user happened to untag the point..
+         function deleteMembership(d3_event, d) {
+           this.blur(); // avoid keeping focus on the button
 
+           if (d === 0) return; // called on newrow (shouldn't happen)
+           // remove the hover-highlight styling
 
-           context.container().select('.inspector-wrap .panewrap').style('right', '0%');
-           context.on('exit.intro', function () {
-             continueTo(reselectPoint);
-           });
-           context.history().on('change.intro', function () {
-             continueTo(updateCloseEditor);
+           utilHighlightEntities([d.relation.id], false, context);
+           var indexes = d.members.map(function (member) {
+             return member.index;
            });
-           timeout(function () {
-             reveal('.entity-editor-pane', helpHtml('intro.points.update'), {
-               tooltipClass: 'intro-points-describe'
-             });
-           }, 400);
-
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
-           }
+           context.perform(actionDeleteMembers(d.relation.id, indexes), _t('operations.delete_member.annotation', {
+             n: _entityIDs.length
+           }));
+           context.validator().validate();
          }
 
-         function updateCloseEditor() {
-           if (context.mode().id !== 'select' || !_pointID || !context.hasEntity(_pointID)) {
-             return continueTo(reselectPoint);
-           } // reset pane, in case user happened to change it..
+         function fetchNearbyRelations(q, callback) {
+           var newRelation = {
+             relation: null,
+             value: _t('inspector.new_relation'),
+             display: _t.html('inspector.new_relation')
+           };
+           var entityID = _entityIDs[0];
+           var result = [];
+           var graph = context.graph();
 
+           function baseDisplayLabel(entity) {
+             var matched = _mainPresetIndex.match(entity, graph);
+             var presetName = matched && matched.name() || _t('inspector.relation');
+             var entityName = utilDisplayName(entity) || '';
+             return presetName + ' ' + entityName;
+           }
 
-           context.container().select('.inspector-wrap .panewrap').style('right', '0%');
-           context.on('exit.intro', function () {
-             continueTo(rightClickPoint);
-           });
-           timeout(function () {
-             reveal('.entity-editor-pane', helpHtml('intro.points.update_close', {
-               button: icon('#iD-icon-close', 'inline')
-             }));
-           }, 500);
+           var explicitRelation = q && context.hasEntity(q.toLowerCase());
 
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             nextStep();
-           }
-         }
+           if (explicitRelation && explicitRelation.type === 'relation' && explicitRelation.id !== entityID) {
+             // loaded relation is specified explicitly, only show that
+             result.push({
+               relation: explicitRelation,
+               value: baseDisplayLabel(explicitRelation) + ' ' + explicitRelation.id
+             });
+           } else {
+             context.history().intersects(context.map().extent()).forEach(function (entity) {
+               if (entity.type !== 'relation' || entity.id === entityID) return;
+               var value = baseDisplayLabel(entity);
+               if (q && (value + ' ' + entity.id).toLowerCase().indexOf(q.toLowerCase()) === -1) return;
+               result.push({
+                 relation: entity,
+                 value: value
+               });
+             });
+             result.sort(function (a, b) {
+               return osmRelation.creationOrder(a.relation, b.relation);
+             }); // Dedupe identical names by appending relation id - see #2891
 
-         function rightClickPoint() {
-           if (!_pointID) return chapter.restart();
-           var entity = context.hasEntity(_pointID);
-           if (!entity) return chapter.restart();
-           context.enter(modeBrowse(context));
-           var box = pointBox(entity.loc, context);
-           var textId = context.lastPointerType() === 'mouse' ? 'rightclick' : 'edit_menu_touch';
-           reveal(box, helpHtml('intro.points.' + textId), {
-             duration: 600
-           });
-           timeout(function () {
-             context.map().on('move.intro', function () {
-               var entity = context.hasEntity(_pointID);
-               if (!entity) return chapter.restart();
-               var box = pointBox(entity.loc, context);
-               reveal(box, helpHtml('intro.points.' + textId), {
-                 duration: 0
+             var dupeGroups = Object.values(utilArrayGroupBy(result, 'value')).filter(function (v) {
+               return v.length > 1;
+             });
+             dupeGroups.forEach(function (group) {
+               group.forEach(function (obj) {
+                 obj.value += ' ' + obj.relation.id;
                });
              });
-           }, 600); // after reveal
+           }
 
-           context.on('enter.intro', function (mode) {
-             if (mode.id !== 'select') return;
-             var ids = context.selectedIDs();
-             if (ids.length !== 1 || ids[0] !== _pointID) return;
-             timeout(function () {
-               var node = selectMenuItem(context, 'delete').node();
-               if (!node) return;
-               continueTo(enterDelete);
-             }, 50); // after menu visible
+           result.forEach(function (obj) {
+             obj.title = obj.value;
            });
-
-           function continueTo(nextStep) {
-             context.on('enter.intro', null);
-             context.map().on('move.intro', null);
-             nextStep();
-           }
+           result.unshift(newRelation);
+           callback(result);
          }
 
-         function enterDelete() {
-           if (!_pointID) return chapter.restart();
-           var entity = context.hasEntity(_pointID);
-           if (!entity) return chapter.restart();
-           var node = selectMenuItem(context, 'delete').node();
+         function renderDisclosureContent(selection) {
+           var memberships = getMemberships();
+           var list = selection.selectAll('.member-list').data([0]);
+           list = list.enter().append('ul').attr('class', 'member-list').merge(list);
+           var items = list.selectAll('li.member-row-normal').data(memberships, function (d) {
+             return d.hash;
+           });
+           items.exit().each(unbind).remove(); // Enter
 
-           if (!node) {
-             return continueTo(rightClickPoint);
-           }
+           var itemsEnter = items.enter().append('li').attr('class', 'member-row member-row-normal form-field'); // highlight the relation in the map while hovering on the list item
 
-           reveal('.edit-menu', helpHtml('intro.points.delete'), {
-             padding: 50
+           itemsEnter.on('mouseover', function (d3_event, d) {
+             utilHighlightEntities([d.relation.id], true, context);
+           }).on('mouseout', function (d3_event, d) {
+             utilHighlightEntities([d.relation.id], false, context);
            });
-           timeout(function () {
-             context.map().on('move.intro', function () {
-               reveal('.edit-menu', helpHtml('intro.points.delete'), {
-                 duration: 0,
-                 padding: 50
-               });
-             });
-           }, 300); // after menu visible
-
-           context.on('exit.intro', function () {
-             if (!_pointID) return chapter.restart();
-             var entity = context.hasEntity(_pointID);
-             if (entity) return continueTo(rightClickPoint); // point still exists
+           var labelEnter = itemsEnter.append('label').attr('class', 'field-label').attr('for', function (d) {
+             return d.domId;
            });
-           context.history().on('change.intro', function (changed) {
-             if (changed.deleted().length) {
-               continueTo(undo);
-             }
+           var labelLink = labelEnter.append('span').attr('class', 'label-text').append('a').attr('href', '#').on('click', selectRelation);
+           labelLink.append('span').attr('class', 'member-entity-type').html(function (d) {
+             var matched = _mainPresetIndex.match(d.relation, context.graph());
+             return matched && matched.name() || _t('inspector.relation');
            });
+           labelLink.append('span').attr('class', 'member-entity-name').html(function (d) {
+             return utilDisplayName(d.relation);
+           });
+           labelEnter.append('button').attr('class', 'remove member-delete').call(svgIcon('#iD-operation-delete')).on('click', deleteMembership);
+           labelEnter.append('button').attr('class', 'member-zoom').attr('title', _t('icons.zoom_to')).call(svgIcon('#iD-icon-framed-dot', 'monochrome')).on('click', zoomToRelation);
+           var wrapEnter = itemsEnter.append('div').attr('class', 'form-field-input-wrap form-field-input-member');
+           wrapEnter.append('input').attr('class', 'member-role').attr('id', function (d) {
+             return d.domId;
+           }).property('type', 'text').property('value', function (d) {
+             return typeof d.role === 'string' ? d.role : '';
+           }).attr('title', function (d) {
+             return Array.isArray(d.role) ? d.role.filter(Boolean).join('\n') : d.role;
+           }).attr('placeholder', function (d) {
+             return Array.isArray(d.role) ? _t('inspector.multiple_roles') : _t('inspector.role');
+           }).classed('mixed', function (d) {
+             return Array.isArray(d.role);
+           }).call(utilNoAuto).on('blur', changeRole).on('change', changeRole);
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro', null);
-             context.history().on('change.intro', null);
-             context.on('exit.intro', null);
-             nextStep();
+           if (taginfo) {
+             wrapEnter.each(bindTypeahead);
            }
-         }
 
-         function undo() {
-           context.history().on('change.intro', function () {
-             continueTo(play);
+           var newMembership = list.selectAll('.member-row-new').data(_showBlank ? [0] : []); // Exit
+
+           newMembership.exit().remove(); // Enter
+
+           var newMembershipEnter = newMembership.enter().append('li').attr('class', 'member-row member-row-new form-field');
+           var newLabelEnter = newMembershipEnter.append('label').attr('class', 'field-label');
+           newLabelEnter.append('input').attr('placeholder', _t('inspector.choose_relation')).attr('type', 'text').attr('class', 'member-entity-input').call(utilNoAuto);
+           newLabelEnter.append('button').attr('class', 'remove member-delete').call(svgIcon('#iD-operation-delete')).on('click', function () {
+             list.selectAll('.member-row-new').remove();
            });
-           reveal('.top-toolbar button.undo-button', helpHtml('intro.points.undo'));
+           var newWrapEnter = newMembershipEnter.append('div').attr('class', 'form-field-input-wrap form-field-input-member');
+           newWrapEnter.append('input').attr('class', 'member-role').property('type', 'text').attr('placeholder', _t('inspector.role')).call(utilNoAuto); // Update
 
-           function continueTo(nextStep) {
-             context.history().on('change.intro', null);
-             nextStep();
-           }
-         }
+           newMembership = newMembership.merge(newMembershipEnter);
+           newMembership.selectAll('.member-entity-input').on('blur', cancelEntity) // if it wasn't accepted normally, cancel it
+           .call(nearbyCombo.on('accept', acceptEntity).on('cancel', cancelEntity)); // Container for the Add button
 
-         function play() {
-           dispatch$1.call('done');
-           reveal('.ideditor', helpHtml('intro.points.play', {
-             next: _t('intro.areas.title')
-           }), {
-             tooltipBox: '.intro-nav-wrap .chapter-area',
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: function buttonCallback() {
-               reveal('.ideditor');
-             }
+           var addRow = selection.selectAll('.add-row').data([0]); // enter
+
+           var addRowEnter = addRow.enter().append('div').attr('class', 'add-row');
+           var addRelationButton = addRowEnter.append('button').attr('class', 'add-relation');
+           addRelationButton.call(svgIcon('#iD-icon-plus', 'light'));
+           addRelationButton.call(uiTooltip().title(_t.html('inspector.add_to_relation')).placement(_mainLocalizer.textDirection() === 'ltr' ? 'right' : 'left'));
+           addRowEnter.append('div').attr('class', 'space-value'); // preserve space
+
+           addRowEnter.append('div').attr('class', 'space-buttons'); // preserve space
+           // update
+
+           addRow = addRow.merge(addRowEnter);
+           addRow.select('.add-relation').on('click', function () {
+             _showBlank = true;
+             section.reRender();
+             list.selectAll('.member-entity-input').node().focus();
            });
-         }
 
-         chapter.enter = function () {
-           addPoint();
-         };
+           function acceptEntity(d) {
+             if (!d) {
+               cancelEntity();
+               return;
+             } // remove hover-higlighting
 
-         chapter.exit = function () {
-           timeouts.forEach(window.clearTimeout);
-           context.on('enter.intro exit.intro', null);
-           context.map().on('move.intro drawn.intro', null);
-           context.history().on('change.intro', null);
-           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
-           context.container().select('.preset-search-input').on('keydown.intro keyup.intro', null);
-         };
 
-         chapter.restart = function () {
-           chapter.exit();
-           chapter.enter();
-         };
+             if (d.relation) utilHighlightEntities([d.relation.id], false, context);
+             var role = context.cleanRelationRole(list.selectAll('.member-row-new .member-role').property('value'));
+             addMembership(d, role);
+           }
 
-         return utilRebind(chapter, dispatch$1, 'on');
-       }
+           function cancelEntity() {
+             var input = newMembership.selectAll('.member-entity-input');
+             input.property('value', ''); // remove hover-higlighting
 
-       function uiIntroArea(context, reveal) {
-         var dispatch$1 = dispatch('done');
-         var playground = [-85.63552, 41.94159];
-         var playgroundPreset = _mainPresetIndex.item('leisure/playground');
-         var nameField = _mainPresetIndex.field('name');
-         var descriptionField = _mainPresetIndex.field('description');
-         var timeouts = [];
+             context.surface().selectAll('.highlighted').classed('highlighted', false);
+           }
 
-         var _areaID;
+           function bindTypeahead(d) {
+             var row = select(this);
+             var role = row.selectAll('input.member-role');
+             var origValue = role.property('value');
 
-         var chapter = {
-           title: 'intro.areas.title'
-         };
+             function sort(value, data) {
+               var sameletter = [];
+               var other = [];
 
-         function timeout(f, t) {
-           timeouts.push(window.setTimeout(f, t));
-         }
+               for (var i = 0; i < data.length; i++) {
+                 if (data[i].value.substring(0, value.length) === value) {
+                   sameletter.push(data[i]);
+                 } else {
+                   other.push(data[i]);
+                 }
+               }
 
-         function eventCancel(d3_event) {
-           d3_event.stopPropagation();
-           d3_event.preventDefault();
-         }
+               return sameletter.concat(other);
+             }
 
-         function revealPlayground(center, text, options) {
-           var padding = 180 * Math.pow(2, context.map().zoom() - 19.5);
-           var box = pad(center, padding, context);
-           reveal(box, text, options);
+             role.call(uiCombobox(context, 'member-role').fetcher(function (role, callback) {
+               var rtype = d.relation.tags.type;
+               taginfo.roles({
+                 debounce: true,
+                 rtype: rtype || '',
+                 geometry: context.graph().geometry(_entityIDs[0]),
+                 query: role
+               }, function (err, data) {
+                 if (!err) callback(sort(role, data));
+               });
+             }).on('cancel', function () {
+               role.property('value', origValue);
+             }));
+           }
+
+           function unbind() {
+             var row = select(this);
+             row.selectAll('input.member-role').call(uiCombobox.off, context);
+           }
          }
 
-         function addArea() {
-           context.enter(modeBrowse(context));
-           context.history().reset('initial');
-           _areaID = null;
-           var msec = transitionTime(playground, context.map().center());
+         section.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs;
+           _entityIDs = val;
+           _showBlank = false;
+           return section;
+         };
 
-           if (msec) {
-             reveal(null, null, {
-               duration: 0
-             });
+         return section;
+       }
+
+       function uiSectionSelectionList(context) {
+         var _selectedIDs = [];
+         var section = uiSection('selected-features', context).shouldDisplay(function () {
+           return _selectedIDs.length > 1;
+         }).label(function () {
+           return _t('inspector.title_count', {
+             title: _t.html('inspector.features'),
+             count: _selectedIDs.length
+           });
+         }).disclosureContent(renderDisclosureContent);
+         context.history().on('change.selectionList', function (difference) {
+           if (difference) {
+             section.reRender();
            }
+         });
 
-           context.map().centerZoomEase(playground, 19, msec);
-           timeout(function () {
-             var tooltip = reveal('button.add-area', helpHtml('intro.areas.add_playground'));
-             tooltip.selectAll('.popover-inner').insert('svg', 'span').attr('class', 'tooltip-illustration').append('use').attr('xlink:href', '#iD-graphic-areas');
-             context.on('enter.intro', function (mode) {
-               if (mode.id !== 'add-area') return;
-               continueTo(startPlayground);
-             });
-           }, msec + 100);
+         section.entityIDs = function (val) {
+           if (!arguments.length) return _selectedIDs;
+           _selectedIDs = val;
+           return section;
+         };
 
-           function continueTo(nextStep) {
-             context.on('enter.intro', null);
-             nextStep();
-           }
+         function selectEntity(d3_event, entity) {
+           context.enter(modeSelect(context, [entity.id]));
          }
 
-         function startPlayground() {
-           if (context.mode().id !== 'add-area') {
-             return chapter.restart();
-           }
+         function deselectEntity(d3_event, entity) {
+           var selectedIDs = _selectedIDs.slice();
 
-           _areaID = null;
-           context.map().zoomEase(19.5, 500);
-           timeout(function () {
-             var textId = context.lastPointerType() === 'mouse' ? 'starting_node_click' : 'starting_node_tap';
-             var startDrawString = helpHtml('intro.areas.start_playground') + helpHtml('intro.areas.' + textId);
-             revealPlayground(playground, startDrawString, {
-               duration: 250
-             });
-             timeout(function () {
-               context.map().on('move.intro drawn.intro', function () {
-                 revealPlayground(playground, startDrawString, {
-                   duration: 0
-                 });
-               });
-               context.on('enter.intro', function (mode) {
-                 if (mode.id !== 'draw-area') return chapter.restart();
-                 continueTo(continuePlayground);
-               });
-             }, 250); // after reveal
-           }, 550); // after easing
+           var index = selectedIDs.indexOf(entity.id);
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
+           if (index > -1) {
+             selectedIDs.splice(index, 1);
+             context.enter(modeSelect(context, selectedIDs));
            }
          }
 
-         function continuePlayground() {
-           if (context.mode().id !== 'draw-area') {
-             return chapter.restart();
-           }
+         function renderDisclosureContent(selection) {
+           var list = selection.selectAll('.feature-list').data([0]);
+           list = list.enter().append('ul').attr('class', 'feature-list').merge(list);
 
-           _areaID = null;
-           revealPlayground(playground, helpHtml('intro.areas.continue_playground'), {
-             duration: 250
-           });
-           timeout(function () {
-             context.map().on('move.intro drawn.intro', function () {
-               revealPlayground(playground, helpHtml('intro.areas.continue_playground'), {
-                 duration: 0
-               });
-             });
-           }, 250); // after reveal
+           var entities = _selectedIDs.map(function (id) {
+             return context.hasEntity(id);
+           }).filter(Boolean);
 
-           context.on('enter.intro', function (mode) {
-             if (mode.id === 'draw-area') {
-               var entity = context.hasEntity(context.selectedIDs()[0]);
+           var items = list.selectAll('.feature-list-item').data(entities, osmEntity.key);
+           items.exit().remove(); // Enter
 
-               if (entity && entity.nodes.length >= 6) {
-                 return continueTo(finishPlayground);
-               } else {
-                 return;
-               }
-             } else if (mode.id === 'select') {
-               _areaID = context.selectedIDs()[0];
-               return continueTo(searchPresets);
-             } else {
-               return chapter.restart();
-             }
+           var enter = items.enter().append('li').attr('class', 'feature-list-item').each(function (d) {
+             select(this).on('mouseover', function () {
+               utilHighlightEntities([d.id], true, context);
+             }).on('mouseout', function () {
+               utilHighlightEntities([d.id], false, context);
+             });
            });
+           var label = enter.append('button').attr('class', 'label').on('click', selectEntity);
+           label.append('span').attr('class', 'entity-geom-icon').call(svgIcon('', 'pre-text'));
+           label.append('span').attr('class', 'entity-type');
+           label.append('span').attr('class', 'entity-name');
+           enter.append('button').attr('class', 'close').attr('title', _t('icons.deselect')).on('click', deselectEntity).call(svgIcon('#iD-icon-close')); // Update
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
-           }
+           items = items.merge(enter);
+           items.selectAll('.entity-geom-icon use').attr('href', function () {
+             var entity = this.parentNode.parentNode.__data__;
+             return '#iD-icon-' + entity.geometry(context.graph());
+           });
+           items.selectAll('.entity-type').html(function (entity) {
+             return _mainPresetIndex.match(entity, context.graph()).name();
+           });
+           items.selectAll('.entity-name').html(function (d) {
+             // fetch latest entity
+             var entity = context.entity(d.id);
+             return utilDisplayName(entity);
+           });
          }
 
-         function finishPlayground() {
-           if (context.mode().id !== 'draw-area') {
-             return chapter.restart();
-           }
+         return section;
+       }
 
-           _areaID = null;
-           var finishString = helpHtml('intro.areas.finish_area_' + (context.lastPointerType() === 'mouse' ? 'click' : 'tap')) + helpHtml('intro.areas.finish_playground');
-           revealPlayground(playground, finishString, {
-             duration: 250
-           });
-           timeout(function () {
-             context.map().on('move.intro drawn.intro', function () {
-               revealPlayground(playground, finishString, {
-                 duration: 0
-               });
-             });
-           }, 250); // after reveal
+       function uiEntityEditor(context) {
+         var dispatch = dispatch$8('choose');
+         var _state = 'select';
+         var _coalesceChanges = false;
+         var _modified = false;
 
-           context.on('enter.intro', function (mode) {
-             if (mode.id === 'draw-area') {
-               return;
-             } else if (mode.id === 'select') {
-               _areaID = context.selectedIDs()[0];
-               return continueTo(searchPresets);
-             } else {
-               return chapter.restart();
-             }
-           });
+         var _base;
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
-           }
-         }
+         var _entityIDs;
 
-         function searchPresets() {
-           if (!_areaID || !context.hasEntity(_areaID)) {
-             return addArea();
-           }
+         var _activePresets = [];
 
-           var ids = context.selectedIDs();
+         var _newFeature;
 
-           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _areaID) {
-             context.enter(modeSelect(context, [_areaID]));
-           } // disallow scrolling
+         var _sections;
 
+         function entityEditor(selection) {
+           var combinedTags = utilCombinedTags(_entityIDs, context.graph()); // Header
 
-           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
-           timeout(function () {
-             // reset pane, in case user somehow happened to change it..
-             context.container().select('.inspector-wrap .panewrap').style('right', '-100%');
-             context.container().select('.preset-search-input').on('keydown.intro', null).on('keyup.intro', checkPresetSearch);
-             reveal('.preset-search-input', helpHtml('intro.areas.search_playground', {
-               preset: playgroundPreset.name()
-             }));
-           }, 400); // after preset list pane visible..
+           var header = selection.selectAll('.header').data([0]); // Enter
 
-           context.on('enter.intro', function (mode) {
-             if (!_areaID || !context.hasEntity(_areaID)) {
-               return continueTo(addArea);
-             }
+           var headerEnter = header.enter().append('div').attr('class', 'header fillL');
+           headerEnter.append('button').attr('class', 'preset-reset preset-choose').call(svgIcon(_mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-forward' : '#iD-icon-backward'));
+           headerEnter.append('button').attr('class', 'close').on('click', function () {
+             context.enter(modeBrowse(context));
+           }).call(svgIcon(_modified ? '#iD-icon-apply' : '#iD-icon-close'));
+           headerEnter.append('h3'); // Update
 
-             var ids = context.selectedIDs();
+           header = header.merge(headerEnter);
+           header.selectAll('h3').html(_entityIDs.length === 1 ? _t.html('inspector.edit') : _t.html('inspector.edit_features'));
+           header.selectAll('.preset-reset').on('click', function () {
+             dispatch.call('choose', this, _activePresets);
+           }); // Body
 
-             if (mode.id !== 'select' || !ids.length || ids[0] !== _areaID) {
-               // keep the user's area selected..
-               context.enter(modeSelect(context, [_areaID])); // reset pane, in case user somehow happened to change it..
+           var body = selection.selectAll('.inspector-body').data([0]); // Enter
 
-               context.container().select('.inspector-wrap .panewrap').style('right', '-100%'); // disallow scrolling
+           var bodyEnter = body.enter().append('div').attr('class', 'entity-editor inspector-body sep-top'); // Update
 
-               context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
-               context.container().select('.preset-search-input').on('keydown.intro', null).on('keyup.intro', checkPresetSearch);
-               reveal('.preset-search-input', helpHtml('intro.areas.search_playground', {
-                 preset: playgroundPreset.name()
-               }));
-               context.history().on('change.intro', null);
+           body = body.merge(bodyEnter);
+
+           if (!_sections) {
+             _sections = [uiSectionSelectionList(context), uiSectionFeatureType(context).on('choose', function (presets) {
+               dispatch.call('choose', this, presets);
+             }), uiSectionEntityIssues(context), uiSectionPresetFields(context).on('change', changeTags).on('revert', revertTags), uiSectionRawTagEditor('raw-tag-editor', context).on('change', changeTags), uiSectionRawMemberEditor(context), uiSectionRawMembershipEditor(context)];
+           }
+
+           _sections.forEach(function (section) {
+             if (section.entityIDs) {
+               section.entityIDs(_entityIDs);
              }
-           });
 
-           function checkPresetSearch() {
-             var first = context.container().select('.preset-list-item:first-child');
+             if (section.presets) {
+               section.presets(_activePresets);
+             }
 
-             if (first.classed('preset-leisure-playground')) {
-               reveal(first.select('.preset-list-button').node(), helpHtml('intro.areas.choose_playground', {
-                 preset: playgroundPreset.name()
-               }), {
-                 duration: 300
-               });
-               context.container().select('.preset-search-input').on('keydown.intro', eventCancel, true).on('keyup.intro', null);
-               context.history().on('change.intro', function () {
-                 continueTo(clickAddField);
-               });
+             if (section.tags) {
+               section.tags(combinedTags);
              }
-           }
 
-           function continueTo(nextStep) {
-             context.container().select('.inspector-wrap').on('wheel.intro', null);
-             context.on('enter.intro', null);
-             context.history().on('change.intro', null);
-             context.container().select('.preset-search-input').on('keydown.intro keyup.intro', null);
-             nextStep();
-           }
-         }
+             if (section.state) {
+               section.state(_state);
+             }
+
+             body.call(section.render);
+           });
 
-         function clickAddField() {
-           if (!_areaID || !context.hasEntity(_areaID)) {
-             return addArea();
-           }
+           context.history().on('change.entity-editor', historyChanged);
 
-           var ids = context.selectedIDs();
+           function historyChanged(difference) {
+             if (selection.selectAll('.entity-editor').empty()) return;
+             if (_state === 'hide') return;
+             var significant = !difference || difference.didChange.properties || difference.didChange.addition || difference.didChange.deletion;
+             if (!significant) return;
+             _entityIDs = _entityIDs.filter(context.hasEntity);
+             if (!_entityIDs.length) return;
+             var priorActivePreset = _activePresets.length === 1 && _activePresets[0];
+             loadActivePresets();
+             var graph = context.graph();
+             entityEditor.modified(_base !== graph);
+             entityEditor(selection);
 
-           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _areaID) {
-             return searchPresets();
+             if (priorActivePreset && _activePresets.length === 1 && priorActivePreset !== _activePresets[0]) {
+               // flash the button to indicate the preset changed
+               context.container().selectAll('.entity-editor button.preset-reset .label').style('background-color', '#fff').transition().duration(750).style('background-color', null);
+             }
            }
-
-           if (!context.container().select('.form-field-description').empty()) {
-             return continueTo(describePlayground);
-           } // disallow scrolling
+         } // Tag changes that fire on input can all get coalesced into a single
+         // history operation when the user leaves the field.  #2342
+         // Use explicit entityIDs in case the selection changes before the event is fired.
 
 
-           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
-           timeout(function () {
-             // reset pane, in case user somehow happened to change it..
-             context.container().select('.inspector-wrap .panewrap').style('right', '0%'); // It's possible for the user to add a description in a previous step..
-             // If they did this already, just continue to next step.
+         function changeTags(entityIDs, changed, onInput) {
+           var actions = [];
 
-             var entity = context.entity(_areaID);
+           for (var i in entityIDs) {
+             var entityID = entityIDs[i];
+             var entity = context.entity(entityID);
+             var tags = Object.assign({}, entity.tags); // shallow copy
 
-             if (entity.tags.description) {
-               return continueTo(play);
-             } // scroll "Add field" into view
+             for (var k in changed) {
+               if (!k) continue;
+               var v = changed[k];
 
+               if (v !== undefined || tags.hasOwnProperty(k)) {
+                 tags[k] = v;
+               }
+             }
 
-             var box = context.container().select('.more-fields').node().getBoundingClientRect();
+             if (!onInput) {
+               tags = utilCleanTags(tags);
+             }
 
-             if (box.top > 300) {
-               var pane = context.container().select('.entity-editor-pane .inspector-body');
-               var start = pane.node().scrollTop;
-               var end = start + (box.top - 300);
-               pane.transition().duration(250).tween('scroll.inspector', function () {
-                 var node = this;
-                 var i = d3_interpolateNumber(start, end);
-                 return function (t) {
-                   node.scrollTop = i(t);
-                 };
-               });
+             if (!fastDeepEqual(entity.tags, tags)) {
+               actions.push(actionChangeTags(entityID, tags));
              }
+           }
 
-             timeout(function () {
-               reveal('.more-fields .combobox-input', helpHtml('intro.areas.add_field', {
-                 name: nameField.label(),
-                 description: descriptionField.label()
-               }), {
-                 duration: 300
-               });
-               context.container().select('.more-fields .combobox-input').on('click.intro', function () {
-                 // Watch for the combobox to appear...
-                 var watcher;
-                 watcher = window.setInterval(function () {
-                   if (!context.container().select('div.combobox').empty()) {
-                     window.clearInterval(watcher);
-                     continueTo(chooseDescriptionField);
-                   }
-                 }, 300);
+           if (actions.length) {
+             var combinedAction = function combinedAction(graph) {
+               actions.forEach(function (action) {
+                 graph = action(graph);
                });
-             }, 300); // after "Add Field" visible
-           }, 400); // after editor pane visible
+               return graph;
+             };
 
-           context.on('exit.intro', function () {
-             return continueTo(searchPresets);
-           });
+             var annotation = _t('operations.change_tags.annotation');
 
-           function continueTo(nextStep) {
-             context.container().select('.inspector-wrap').on('wheel.intro', null);
-             context.container().select('.more-fields .combobox-input').on('click.intro', null);
-             context.on('exit.intro', null);
-             nextStep();
+             if (_coalesceChanges) {
+               context.overwrite(combinedAction, annotation);
+             } else {
+               context.perform(combinedAction, annotation);
+               _coalesceChanges = !!onInput;
+             }
+           } // if leaving field (blur event), rerun validation
+
+
+           if (!onInput) {
+             context.validator().validate();
            }
          }
 
-         function chooseDescriptionField() {
-           if (!_areaID || !context.hasEntity(_areaID)) {
-             return addArea();
-           }
+         function revertTags(keys) {
+           var actions = [];
 
-           var ids = context.selectedIDs();
+           for (var i in _entityIDs) {
+             var entityID = _entityIDs[i];
+             var original = context.graph().base().entities[entityID];
+             var changed = {};
 
-           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _areaID) {
-             return searchPresets();
-           }
+             for (var j in keys) {
+               var key = keys[j];
+               changed[key] = original ? original.tags[key] : undefined;
+             }
 
-           if (!context.container().select('.form-field-description').empty()) {
-             return continueTo(describePlayground);
-           } // Make sure combobox is ready..
+             var entity = context.entity(entityID);
+             var tags = Object.assign({}, entity.tags); // shallow copy
 
+             for (var k in changed) {
+               if (!k) continue;
+               var v = changed[k];
 
-           if (context.container().select('div.combobox').empty()) {
-             return continueTo(clickAddField);
-           } // Watch for the combobox to go away..
+               if (v !== undefined || tags.hasOwnProperty(k)) {
+                 tags[k] = v;
+               }
+             }
 
+             tags = utilCleanTags(tags);
 
-           var watcher;
-           watcher = window.setInterval(function () {
-             if (context.container().select('div.combobox').empty()) {
-               window.clearInterval(watcher);
-               timeout(function () {
-                 if (context.container().select('.form-field-description').empty()) {
-                   continueTo(retryChooseDescription);
-                 } else {
-                   continueTo(describePlayground);
-                 }
-               }, 300); // after description field added.
+             if (!fastDeepEqual(entity.tags, tags)) {
+               actions.push(actionChangeTags(entityID, tags));
              }
-           }, 300);
-           reveal('div.combobox', helpHtml('intro.areas.choose_field', {
-             field: descriptionField.label()
-           }), {
-             duration: 300
-           });
-           context.on('exit.intro', function () {
-             return continueTo(searchPresets);
-           });
-
-           function continueTo(nextStep) {
-             if (watcher) window.clearInterval(watcher);
-             context.on('exit.intro', null);
-             nextStep();
            }
-         }
 
-         function describePlayground() {
-           if (!_areaID || !context.hasEntity(_areaID)) {
-             return addArea();
-           }
+           if (actions.length) {
+             var combinedAction = function combinedAction(graph) {
+               actions.forEach(function (action) {
+                 graph = action(graph);
+               });
+               return graph;
+             };
 
-           var ids = context.selectedIDs();
+             var annotation = _t('operations.change_tags.annotation');
 
-           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _areaID) {
-             return searchPresets();
-           } // reset pane, in case user happened to change it..
+             if (_coalesceChanges) {
+               context.overwrite(combinedAction, annotation);
+             } else {
+               context.perform(combinedAction, annotation);
+               _coalesceChanges = false;
+             }
+           }
 
+           context.validator().validate();
+         }
 
-           context.container().select('.inspector-wrap .panewrap').style('right', '0%');
+         entityEditor.modified = function (val) {
+           if (!arguments.length) return _modified;
+           _modified = val;
+           return entityEditor;
+         };
 
-           if (context.container().select('.form-field-description').empty()) {
-             return continueTo(retryChooseDescription);
-           }
+         entityEditor.state = function (val) {
+           if (!arguments.length) return _state;
+           _state = val;
+           return entityEditor;
+         };
 
-           context.on('exit.intro', function () {
-             continueTo(play);
-           });
-           reveal('.entity-editor-pane', helpHtml('intro.areas.describe_playground', {
-             button: icon('#iD-icon-close', 'inline')
-           }), {
-             duration: 300
-           });
+         entityEditor.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs; // always reload these even if the entityIDs are unchanged, since we
+           // could be reselecting after something like dragging a node
 
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             nextStep();
-           }
-         }
+           _base = context.graph();
+           _coalesceChanges = false;
+           if (val && _entityIDs && utilArrayIdentical(_entityIDs, val)) return entityEditor; // exit early if no change
 
-         function retryChooseDescription() {
-           if (!_areaID || !context.hasEntity(_areaID)) {
-             return addArea();
-           }
+           _entityIDs = val;
+           loadActivePresets(true);
+           return entityEditor.modified(false);
+         };
 
-           var ids = context.selectedIDs();
+         entityEditor.newFeature = function (val) {
+           if (!arguments.length) return _newFeature;
+           _newFeature = val;
+           return entityEditor;
+         };
 
-           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _areaID) {
-             return searchPresets();
-           } // reset pane, in case user happened to change it..
+         function loadActivePresets(isForNewSelection) {
+           var graph = context.graph();
+           var counts = {};
 
+           for (var i in _entityIDs) {
+             var entity = graph.hasEntity(_entityIDs[i]);
+             if (!entity) return;
+             var match = _mainPresetIndex.match(entity, graph);
+             if (!counts[match.id]) counts[match.id] = 0;
+             counts[match.id] += 1;
+           }
 
-           context.container().select('.inspector-wrap .panewrap').style('right', '0%');
-           reveal('.entity-editor-pane', helpHtml('intro.areas.retry_add_field', {
-             field: descriptionField.label()
-           }), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: function buttonCallback() {
-               continueTo(clickAddField);
-             }
-           });
-           context.on('exit.intro', function () {
-             return continueTo(searchPresets);
+           var matches = Object.keys(counts).sort(function (p1, p2) {
+             return counts[p2] - counts[p1];
+           }).map(function (pID) {
+             return _mainPresetIndex.item(pID);
            });
 
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             nextStep();
+           if (!isForNewSelection) {
+             // A "weak" preset doesn't set any tags. (e.g. "Address")
+             var weakPreset = _activePresets.length === 1 && !_activePresets[0].isFallback() && Object.keys(_activePresets[0].addTags || {}).length === 0; // Don't replace a weak preset with a fallback preset (e.g. "Point")
+
+             if (weakPreset && matches.length === 1 && matches[0].isFallback()) return;
            }
-         }
 
-         function play() {
-           dispatch$1.call('done');
-           reveal('.ideditor', helpHtml('intro.areas.play', {
-             next: _t('intro.lines.title')
-           }), {
-             tooltipBox: '.intro-nav-wrap .chapter-line',
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: function buttonCallback() {
-               reveal('.ideditor');
-             }
-           });
+           entityEditor.presets(matches);
          }
 
-         chapter.enter = function () {
-           addArea();
-         };
+         entityEditor.presets = function (val) {
+           if (!arguments.length) return _activePresets; // don't reload the same preset
 
-         chapter.exit = function () {
-           timeouts.forEach(window.clearTimeout);
-           context.on('enter.intro exit.intro', null);
-           context.map().on('move.intro drawn.intro', null);
-           context.history().on('change.intro', null);
-           context.container().select('.inspector-wrap').on('wheel.intro', null);
-           context.container().select('.preset-search-input').on('keydown.intro keyup.intro', null);
-           context.container().select('.more-fields .combobox-input').on('click.intro', null);
-         };
+           if (!utilArrayIdentical(val, _activePresets)) {
+             _activePresets = val;
+           }
 
-         chapter.restart = function () {
-           chapter.exit();
-           chapter.enter();
+           return entityEditor;
          };
 
-         return utilRebind(chapter, dispatch$1, 'on');
+         return utilRebind(entityEditor, dispatch, 'on');
        }
 
-       function uiIntroLine(context, reveal) {
-         var dispatch$1 = dispatch('done');
-         var timeouts = [];
-         var _tulipRoadID = null;
-         var flowerRoadID = 'w646';
-         var tulipRoadStart = [-85.6297754121684, 41.95805253325314];
-         var tulipRoadMidpoint = [-85.62975395449628, 41.95787501510204];
-         var tulipRoadIntersection = [-85.62974496187628, 41.95742515554585];
-         var roadCategory = _mainPresetIndex.item('category-road_minor');
-         var residentialPreset = _mainPresetIndex.item('highway/residential');
-         var woodRoadID = 'w525';
-         var woodRoadEndID = 'n2862';
-         var woodRoadAddNode = [-85.62390110349587, 41.95397111462291];
-         var woodRoadDragEndpoint = [-85.623867390213, 41.95466987786487];
-         var woodRoadDragMidpoint = [-85.62386254803509, 41.95430395953872];
-         var washingtonStreetID = 'w522';
-         var twelfthAvenueID = 'w1';
-         var eleventhAvenueEndID = 'n3550';
-         var twelfthAvenueEndID = 'n5';
-         var _washingtonSegmentID = null;
-         var eleventhAvenueEnd = context.entity(eleventhAvenueEndID).loc;
-         var twelfthAvenueEnd = context.entity(twelfthAvenueEndID).loc;
-         var deleteLinesLoc = [-85.6219395542764, 41.95228033922477];
-         var twelfthAvenue = [-85.62219310052491, 41.952505413152956];
-         var chapter = {
-           title: 'intro.lines.title'
-         };
+       function uiPresetList(context) {
+         var dispatch = dispatch$8('cancel', 'choose');
 
-         function timeout(f, t) {
-           timeouts.push(window.setTimeout(f, t));
-         }
+         var _entityIDs;
 
-         function eventCancel(d3_event) {
-           d3_event.stopPropagation();
-           d3_event.preventDefault();
-         }
+         var _currLoc;
 
-         function addLine() {
-           context.enter(modeBrowse(context));
-           context.history().reset('initial');
-           var msec = transitionTime(tulipRoadStart, context.map().center());
+         var _currentPresets;
 
-           if (msec) {
-             reveal(null, null, {
-               duration: 0
-             });
-           }
+         var _autofocus = false;
 
-           context.map().centerZoomEase(tulipRoadStart, 18.5, msec);
-           timeout(function () {
-             var tooltip = reveal('button.add-line', helpHtml('intro.lines.add_line'));
-             tooltip.selectAll('.popover-inner').insert('svg', 'span').attr('class', 'tooltip-illustration').append('use').attr('xlink:href', '#iD-graphic-lines');
-             context.on('enter.intro', function (mode) {
-               if (mode.id !== 'add-line') return;
-               continueTo(startLine);
-             });
-           }, msec + 100);
+         function presetList(selection) {
+           if (!_entityIDs) return;
+           var presets = _mainPresetIndex.matchAllGeometry(entityGeometries());
+           selection.html('');
+           var messagewrap = selection.append('div').attr('class', 'header fillL');
+           var message = messagewrap.append('h3').html(_t.html('inspector.choose'));
+           messagewrap.append('button').attr('class', 'preset-choose').on('click', function () {
+             dispatch.call('cancel', this);
+           }).call(svgIcon(_mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-backward' : '#iD-icon-forward'));
 
-           function continueTo(nextStep) {
-             context.on('enter.intro', null);
-             nextStep();
+           function initialKeydown(d3_event) {
+             // hack to let delete shortcut work when search is autofocused
+             if (search.property('value').length === 0 && (d3_event.keyCode === utilKeybinding.keyCodes['⌫'] || d3_event.keyCode === utilKeybinding.keyCodes['⌦'])) {
+               d3_event.preventDefault();
+               d3_event.stopPropagation();
+               operationDelete(context, _entityIDs)(); // hack to let undo work when search is autofocused
+             } else if (search.property('value').length === 0 && (d3_event.ctrlKey || d3_event.metaKey) && d3_event.keyCode === utilKeybinding.keyCodes.z) {
+               d3_event.preventDefault();
+               d3_event.stopPropagation();
+               context.undo();
+             } else if (!d3_event.ctrlKey && !d3_event.metaKey) {
+               // don't check for delete/undo hack on future keydown events
+               select(this).on('keydown', keydown);
+               keydown.call(this, d3_event);
+             }
            }
-         }
 
-         function startLine() {
-           if (context.mode().id !== 'add-line') return chapter.restart();
-           _tulipRoadID = null;
-           var padding = 70 * Math.pow(2, context.map().zoom() - 18);
-           var box = pad(tulipRoadStart, padding, context);
-           box.height = box.height + 100;
-           var textId = context.lastPointerType() === 'mouse' ? 'start_line' : 'start_line_tap';
-           var startLineString = helpHtml('intro.lines.missing_road') + '{br}' + helpHtml('intro.lines.line_draw_info') + helpHtml('intro.lines.' + textId);
-           reveal(box, startLineString);
-           context.map().on('move.intro drawn.intro', function () {
-             padding = 70 * Math.pow(2, context.map().zoom() - 18);
-             box = pad(tulipRoadStart, padding, context);
-             box.height = box.height + 100;
-             reveal(box, startLineString, {
-               duration: 0
-             });
-           });
-           context.on('enter.intro', function (mode) {
-             if (mode.id !== 'draw-line') return chapter.restart();
-             continueTo(drawLine);
-           });
+           function keydown(d3_event) {
+             // down arrow
+             if (d3_event.keyCode === utilKeybinding.keyCodes['↓'] && // if insertion point is at the end of the string
+             search.node().selectionStart === search.property('value').length) {
+               d3_event.preventDefault();
+               d3_event.stopPropagation(); // move focus to the first item in the preset list
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
+               var buttons = list.selectAll('.preset-list-button');
+               if (!buttons.empty()) buttons.nodes()[0].focus();
+             }
            }
-         }
 
-         function drawLine() {
-           if (context.mode().id !== 'draw-line') return chapter.restart();
-           _tulipRoadID = context.mode().selectedIDs()[0];
-           context.map().centerEase(tulipRoadMidpoint, 500);
-           timeout(function () {
-             var padding = 200 * Math.pow(2, context.map().zoom() - 18.5);
-             var box = pad(tulipRoadMidpoint, padding, context);
-             box.height = box.height * 2;
-             reveal(box, helpHtml('intro.lines.intersect', {
-               name: _t('intro.graph.name.flower-street')
-             }));
-             context.map().on('move.intro drawn.intro', function () {
-               padding = 200 * Math.pow(2, context.map().zoom() - 18.5);
-               box = pad(tulipRoadMidpoint, padding, context);
-               box.height = box.height * 2;
-               reveal(box, helpHtml('intro.lines.intersect', {
-                 name: _t('intro.graph.name.flower-street')
-               }), {
-                 duration: 0
-               });
-             });
-           }, 550); // after easing..
+           function keypress(d3_event) {
+             // enter
+             var value = search.property('value');
 
-           context.history().on('change.intro', function () {
-             if (isLineConnected()) {
-               continueTo(continueLine);
+             if (d3_event.keyCode === 13 && // ↩ Return
+             value.length) {
+               list.selectAll('.preset-list-item:first-child').each(function (d) {
+                 d.choose.call(this);
+               });
              }
-           });
-           context.on('enter.intro', function (mode) {
-             if (mode.id === 'draw-line') {
-               return;
-             } else if (mode.id === 'select') {
-               continueTo(retryIntersect);
-               return;
+           }
+
+           function inputevent() {
+             var value = search.property('value');
+             list.classed('filtered', value.length);
+             var results, messageText;
+
+             if (value.length) {
+               results = presets.search(value, entityGeometries()[0], _currLoc);
+               messageText = _t('inspector.results', {
+                 n: results.collection.length,
+                 search: value
+               });
              } else {
-               return chapter.restart();
+               results = _mainPresetIndex.defaults(entityGeometries()[0], 36, !context.inIntro(), _currLoc);
+               messageText = _t('inspector.choose');
              }
-           });
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.history().on('change.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
+             list.call(drawList, results);
+             message.html(messageText);
+           }
+
+           var searchWrap = selection.append('div').attr('class', 'search-header');
+           searchWrap.call(svgIcon('#iD-icon-search', 'pre-text'));
+           var search = searchWrap.append('input').attr('class', 'preset-search-input').attr('placeholder', _t('inspector.search')).attr('type', 'search').call(utilNoAuto).on('keydown', initialKeydown).on('keypress', keypress).on('input', inputevent);
+
+           if (_autofocus) {
+             search.node().focus(); // Safari 14 doesn't always like to focus immediately,
+             // so try again on the next pass
+
+             setTimeout(function () {
+               search.node().focus();
+             }, 0);
            }
+
+           var listWrap = selection.append('div').attr('class', 'inspector-body');
+           var list = listWrap.append('div').attr('class', 'preset-list').call(drawList, _mainPresetIndex.defaults(entityGeometries()[0], 36, !context.inIntro(), _currLoc));
+           context.features().on('change.preset-list', updateForFeatureHiddenState);
          }
 
-         function isLineConnected() {
-           var entity = _tulipRoadID && context.hasEntity(_tulipRoadID);
+         function drawList(list, presets) {
+           presets = presets.matchAllGeometry(entityGeometries());
+           var collection = presets.collection.reduce(function (collection, preset) {
+             if (!preset) return collection;
 
-           if (!entity) return false;
-           var drawNodes = context.graph().childNodes(entity);
-           return drawNodes.some(function (node) {
-             return context.graph().parentWays(node).some(function (parent) {
-               return parent.id === flowerRoadID;
-             });
+             if (preset.members) {
+               if (preset.members.collection.filter(function (preset) {
+                 return preset.addable();
+               }).length > 1) {
+                 collection.push(CategoryItem(preset));
+               }
+             } else if (preset.addable()) {
+               collection.push(PresetItem(preset));
+             }
+
+             return collection;
+           }, []);
+           var items = list.selectAll('.preset-list-item').data(collection, function (d) {
+             return d.preset.id;
            });
+           items.order();
+           items.exit().remove();
+           items.enter().append('div').attr('class', function (item) {
+             return 'preset-list-item preset-' + item.preset.id.replace('/', '-');
+           }).classed('current', function (item) {
+             return _currentPresets.indexOf(item.preset) !== -1;
+           }).each(function (item) {
+             select(this).call(item);
+           }).style('opacity', 0).transition().style('opacity', 1);
+           updateForFeatureHiddenState();
          }
 
-         function retryIntersect() {
-           select(window).on('pointerdown.intro mousedown.intro', eventCancel, true);
-           var box = pad(tulipRoadIntersection, 80, context);
-           reveal(box, helpHtml('intro.lines.retry_intersect', {
-             name: _t('intro.graph.name.flower-street')
-           }));
-           timeout(chapter.restart, 3000);
-         }
+         function itemKeydown(d3_event) {
+           // the actively focused item
+           var item = select(this.closest('.preset-list-item'));
+           var parentItem = select(item.node().parentNode.closest('.preset-list-item')); // arrow down, move focus to the next, lower item
 
-         function continueLine() {
-           if (context.mode().id !== 'draw-line') return chapter.restart();
+           if (d3_event.keyCode === utilKeybinding.keyCodes['↓']) {
+             d3_event.preventDefault();
+             d3_event.stopPropagation(); // the next item in the list at the same level
 
-           var entity = _tulipRoadID && context.hasEntity(_tulipRoadID);
+             var nextItem = select(item.node().nextElementSibling); // if there is no next item in this list
 
-           if (!entity) return chapter.restart();
-           context.map().centerEase(tulipRoadIntersection, 500);
-           var continueLineText = helpHtml('intro.lines.continue_line') + '{br}' + helpHtml('intro.lines.finish_line_' + (context.lastPointerType() === 'mouse' ? 'click' : 'tap')) + helpHtml('intro.lines.finish_road');
-           reveal('.surface', continueLineText);
-           context.on('enter.intro', function (mode) {
-             if (mode.id === 'draw-line') return;else if (mode.id === 'select') return continueTo(chooseCategoryRoad);else return chapter.restart();
-           });
+             if (nextItem.empty()) {
+               // if there is a parent item
+               if (!parentItem.empty()) {
+                 // the item is the last item of a sublist,
+                 // select the next item at the parent level
+                 nextItem = select(parentItem.node().nextElementSibling);
+               } // if the focused item is expanded
 
-           function continueTo(nextStep) {
-             context.on('enter.intro', null);
-             nextStep();
-           }
-         }
+             } else if (select(this).classed('expanded')) {
+               // select the first subitem instead
+               nextItem = item.select('.subgrid .preset-list-item:first-child');
+             }
 
-         function chooseCategoryRoad() {
-           if (context.mode().id !== 'select') return chapter.restart();
-           context.on('exit.intro', function () {
-             return chapter.restart();
-           });
-           var button = context.container().select('.preset-category-road_minor .preset-list-button');
-           if (button.empty()) return chapter.restart(); // disallow scrolling
+             if (!nextItem.empty()) {
+               // focus on the next item
+               nextItem.select('.preset-list-button').node().focus();
+             } // arrow up, move focus to the previous, higher item
 
-           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
-           timeout(function () {
-             // reset pane, in case user somehow happened to change it..
-             context.container().select('.inspector-wrap .panewrap').style('right', '-100%');
-             reveal(button.node(), helpHtml('intro.lines.choose_category_road', {
-               category: roadCategory.name()
-             }));
-             button.on('click.intro', function () {
-               continueTo(choosePresetResidential);
-             });
-           }, 400); // after editor pane visible
+           } else if (d3_event.keyCode === utilKeybinding.keyCodes['↑']) {
+             d3_event.preventDefault();
+             d3_event.stopPropagation(); // the previous item in the list at the same level
 
-           function continueTo(nextStep) {
-             context.container().select('.inspector-wrap').on('wheel.intro', null);
-             context.container().select('.preset-list-button').on('click.intro', null);
-             context.on('exit.intro', null);
-             nextStep();
-           }
-         }
+             var previousItem = select(item.node().previousElementSibling); // if there is no previous item in this list
 
-         function choosePresetResidential() {
-           if (context.mode().id !== 'select') return chapter.restart();
-           context.on('exit.intro', function () {
-             return chapter.restart();
-           });
-           var subgrid = context.container().select('.preset-category-road_minor .subgrid');
-           if (subgrid.empty()) return chapter.restart();
-           subgrid.selectAll(':not(.preset-highway-residential) .preset-list-button').on('click.intro', function () {
-             continueTo(retryPresetResidential);
-           });
-           subgrid.selectAll('.preset-highway-residential .preset-list-button').on('click.intro', function () {
-             continueTo(nameRoad);
-           });
-           timeout(function () {
-             reveal(subgrid.node(), helpHtml('intro.lines.choose_preset_residential', {
-               preset: residentialPreset.name()
-             }), {
-               tooltipBox: '.preset-highway-residential .preset-list-button',
-               duration: 300
-             });
-           }, 300);
+             if (previousItem.empty()) {
+               // if there is a parent item
+               if (!parentItem.empty()) {
+                 // the item is the first subitem of a sublist select the parent item
+                 previousItem = parentItem;
+               } // if the previous item is expanded
 
-           function continueTo(nextStep) {
-             context.container().select('.preset-list-button').on('click.intro', null);
-             context.on('exit.intro', null);
-             nextStep();
-           }
-         } // selected wrong road type
+             } else if (previousItem.select('.preset-list-button').classed('expanded')) {
+               // select the last subitem of the sublist of the previous item
+               previousItem = previousItem.select('.subgrid .preset-list-item:last-child');
+             }
 
+             if (!previousItem.empty()) {
+               // focus on the previous item
+               previousItem.select('.preset-list-button').node().focus();
+             } else {
+               // the focus is at the top of the list, move focus back to the search field
+               var search = select(this.closest('.preset-list-pane')).select('.preset-search-input');
+               search.node().focus();
+             } // arrow left, move focus to the parent item if there is one
 
-         function retryPresetResidential() {
-           if (context.mode().id !== 'select') return chapter.restart();
-           context.on('exit.intro', function () {
-             return chapter.restart();
-           }); // disallow scrolling
+           } else if (d3_event.keyCode === utilKeybinding.keyCodes[_mainLocalizer.textDirection() === 'rtl' ? '→' : '←']) {
+             d3_event.preventDefault();
+             d3_event.stopPropagation(); // if there is a parent item, focus on the parent item
 
-           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
-           timeout(function () {
-             var button = context.container().select('.entity-editor-pane .preset-list-button');
-             reveal(button.node(), helpHtml('intro.lines.retry_preset_residential', {
-               preset: residentialPreset.name()
-             }));
-             button.on('click.intro', function () {
-               continueTo(chooseCategoryRoad);
-             });
-           }, 500);
+             if (!parentItem.empty()) {
+               parentItem.select('.preset-list-button').node().focus();
+             } // arrow right, choose this item
 
-           function continueTo(nextStep) {
-             context.container().select('.inspector-wrap').on('wheel.intro', null);
-             context.container().select('.preset-list-button').on('click.intro', null);
-             context.on('exit.intro', null);
-             nextStep();
+           } else if (d3_event.keyCode === utilKeybinding.keyCodes[_mainLocalizer.textDirection() === 'rtl' ? '←' : '→']) {
+             d3_event.preventDefault();
+             d3_event.stopPropagation();
+             item.datum().choose.call(select(this).node());
            }
          }
 
-         function nameRoad() {
-           context.on('exit.intro', function () {
-             continueTo(didNameRoad);
-           });
-           timeout(function () {
-             reveal('.entity-editor-pane', helpHtml('intro.lines.name_road', {
-               button: icon('#iD-icon-close', 'inline')
-             }), {
-               tooltipClass: 'intro-lines-name_road'
-             });
-           }, 500);
-
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             nextStep();
-           }
-         }
+         function CategoryItem(preset) {
+           var box,
+               sublist,
+               shown = false;
 
-         function didNameRoad() {
-           context.history().checkpoint('doneAddLine');
-           timeout(function () {
-             reveal('.surface', helpHtml('intro.lines.did_name_road'), {
-               buttonText: _t.html('intro.ok'),
-               buttonCallback: function buttonCallback() {
-                 continueTo(updateLine);
-               }
-             });
-           }, 500);
+           function item(selection) {
+             var wrap = selection.append('div').attr('class', 'preset-list-button-wrap category');
 
-           function continueTo(nextStep) {
-             nextStep();
-           }
-         }
+             function click() {
+               var isExpanded = select(this).classed('expanded');
+               var iconName = isExpanded ? _mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-backward' : '#iD-icon-forward' : '#iD-icon-down';
+               select(this).classed('expanded', !isExpanded);
+               select(this).selectAll('div.label-inner svg.icon use').attr('href', iconName);
+               item.choose();
+             }
 
-         function updateLine() {
-           context.history().reset('doneAddLine');
+             var geometries = entityGeometries();
+             var button = wrap.append('button').attr('class', 'preset-list-button').classed('expanded', false).call(uiPresetIcon().geometry(geometries.length === 1 && geometries[0]).preset(preset)).on('click', click).on('keydown', function (d3_event) {
+               // right arrow, expand the focused item
+               if (d3_event.keyCode === utilKeybinding.keyCodes[_mainLocalizer.textDirection() === 'rtl' ? '←' : '→']) {
+                 d3_event.preventDefault();
+                 d3_event.stopPropagation(); // if the item isn't expanded
 
-           if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
-             return chapter.restart();
-           }
+                 if (!select(this).classed('expanded')) {
+                   // toggle expansion (expand the item)
+                   click.call(this, d3_event);
+                 } // left arrow, collapse the focused item
 
-           var msec = transitionTime(woodRoadDragMidpoint, context.map().center());
+               } else if (d3_event.keyCode === utilKeybinding.keyCodes[_mainLocalizer.textDirection() === 'rtl' ? '→' : '←']) {
+                 d3_event.preventDefault();
+                 d3_event.stopPropagation(); // if the item is expanded
 
-           if (msec) {
-             reveal(null, null, {
-               duration: 0
+                 if (select(this).classed('expanded')) {
+                   // toggle expansion (collapse the item)
+                   click.call(this, d3_event);
+                 }
+               } else {
+                 itemKeydown.call(this, d3_event);
+               }
+             });
+             var label = button.append('div').attr('class', 'label').append('div').attr('class', 'label-inner');
+             label.append('div').attr('class', 'namepart').call(svgIcon(_mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-backward' : '#iD-icon-forward', 'inline')).append('span').html(function () {
+               return preset.nameLabel() + '&hellip;';
              });
+             box = selection.append('div').attr('class', 'subgrid').style('max-height', '0px').style('opacity', 0);
+             box.append('div').attr('class', 'arrow');
+             sublist = box.append('div').attr('class', 'preset-list fillL3');
            }
 
-           context.map().centerZoomEase(woodRoadDragMidpoint, 19, msec);
-           timeout(function () {
-             var padding = 250 * Math.pow(2, context.map().zoom() - 19);
-             var box = pad(woodRoadDragMidpoint, padding, context);
-
-             var advance = function advance() {
-               continueTo(addNode);
-             };
+           item.choose = function () {
+             if (!box || !sublist) return;
 
-             reveal(box, helpHtml('intro.lines.update_line'), {
-               buttonText: _t.html('intro.ok'),
-               buttonCallback: advance
-             });
-             context.map().on('move.intro drawn.intro', function () {
-               var padding = 250 * Math.pow(2, context.map().zoom() - 19);
-               var box = pad(woodRoadDragMidpoint, padding, context);
-               reveal(box, helpHtml('intro.lines.update_line'), {
-                 duration: 0,
-                 buttonText: _t.html('intro.ok'),
-                 buttonCallback: advance
-               });
-             });
-           }, msec + 100);
+             if (shown) {
+               shown = false;
+               box.transition().duration(200).style('opacity', '0').style('max-height', '0px').style('padding-bottom', '0px');
+             } else {
+               shown = true;
+               var members = preset.members.matchAllGeometry(entityGeometries());
+               sublist.call(drawList, members);
+               box.transition().duration(200).style('opacity', '1').style('max-height', 200 + members.collection.length * 190 + 'px').style('padding-bottom', '10px');
+             }
+           };
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             nextStep();
-           }
+           item.preset = preset;
+           return item;
          }
 
-         function addNode() {
-           context.history().reset('doneAddLine');
-
-           if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
-             return chapter.restart();
+         function PresetItem(preset) {
+           function item(selection) {
+             var wrap = selection.append('div').attr('class', 'preset-list-button-wrap');
+             var geometries = entityGeometries();
+             var button = wrap.append('button').attr('class', 'preset-list-button').call(uiPresetIcon().geometry(geometries.length === 1 && geometries[0]).preset(preset)).on('click', item.choose).on('keydown', itemKeydown);
+             var label = button.append('div').attr('class', 'label').append('div').attr('class', 'label-inner');
+             var nameparts = [preset.nameLabel(), preset.subtitleLabel()].filter(Boolean);
+             label.selectAll('.namepart').data(nameparts).enter().append('div').attr('class', 'namepart').html(function (d) {
+               return d;
+             });
+             wrap.call(item.reference.button);
+             selection.call(item.reference.body);
            }
 
-           var padding = 40 * Math.pow(2, context.map().zoom() - 19);
-           var box = pad(woodRoadAddNode, padding, context);
-           var addNodeString = helpHtml('intro.lines.add_node' + (context.lastPointerType() === 'mouse' ? '' : '_touch'));
-           reveal(box, addNodeString);
-           context.map().on('move.intro drawn.intro', function () {
-             var padding = 40 * Math.pow(2, context.map().zoom() - 19);
-             var box = pad(woodRoadAddNode, padding, context);
-             reveal(box, addNodeString, {
-               duration: 0
-             });
-           });
-           context.history().on('change.intro', function (changed) {
-             if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
-               return continueTo(updateLine);
-             }
+           item.choose = function () {
+             if (select(this).classed('disabled')) return;
 
-             if (changed.created().length === 1) {
-               timeout(function () {
-                 continueTo(startDragEndpoint);
-               }, 500);
-             }
-           });
-           context.on('enter.intro', function (mode) {
-             if (mode.id !== 'select') {
-               continueTo(updateLine);
+             if (!context.inIntro()) {
+               _mainPresetIndex.setMostRecent(preset, entityGeometries()[0]);
              }
-           });
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.history().on('change.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
-           }
+             context.perform(function (graph) {
+               for (var i in _entityIDs) {
+                 var entityID = _entityIDs[i];
+                 var oldPreset = _mainPresetIndex.match(graph.entity(entityID), graph);
+                 graph = actionChangePreset(entityID, oldPreset, preset)(graph);
+               }
+
+               return graph;
+             }, _t('operations.change_tags.annotation'));
+             context.validator().validate(); // rerun validation
+
+             dispatch.call('choose', this, preset);
+           };
+
+           item.help = function (d3_event) {
+             d3_event.stopPropagation();
+             item.reference.toggle();
+           };
+
+           item.preset = preset;
+           item.reference = uiTagReference(preset.reference());
+           return item;
          }
 
-         function startDragEndpoint() {
-           if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
-             return continueTo(updateLine);
-           }
+         function updateForFeatureHiddenState() {
+           if (!_entityIDs.every(context.hasEntity)) return;
+           var geometries = entityGeometries();
+           var button = context.container().selectAll('.preset-list .preset-list-button'); // remove existing tooltips
 
-           var padding = 100 * Math.pow(2, context.map().zoom() - 19);
-           var box = pad(woodRoadDragEndpoint, padding, context);
-           var startDragString = helpHtml('intro.lines.start_drag_endpoint' + (context.lastPointerType() === 'mouse' ? '' : '_touch')) + helpHtml('intro.lines.drag_to_intersection');
-           reveal(box, startDragString);
-           context.map().on('move.intro drawn.intro', function () {
-             if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
-               return continueTo(updateLine);
+           button.call(uiTooltip().destroyAny);
+           button.each(function (item, index) {
+             var hiddenPresetFeaturesId;
+
+             for (var i in geometries) {
+               hiddenPresetFeaturesId = context.features().isHiddenPreset(item.preset, geometries[i]);
+               if (hiddenPresetFeaturesId) break;
              }
 
-             var padding = 100 * Math.pow(2, context.map().zoom() - 19);
-             var box = pad(woodRoadDragEndpoint, padding, context);
-             reveal(box, startDragString, {
-               duration: 0
-             });
-             var entity = context.entity(woodRoadEndID);
+             var isHiddenPreset = !context.inIntro() && !!hiddenPresetFeaturesId && (_currentPresets.length !== 1 || item.preset !== _currentPresets[0]);
+             select(this).classed('disabled', isHiddenPreset);
 
-             if (geoSphericalDistance(entity.loc, woodRoadDragEndpoint) <= 4) {
-               continueTo(finishDragEndpoint);
+             if (isHiddenPreset) {
+               var isAutoHidden = context.features().autoHidden(hiddenPresetFeaturesId);
+               select(this).call(uiTooltip().title(_t.html('inspector.hidden_preset.' + (isAutoHidden ? 'zoom' : 'manual'), {
+                 features: _t.html('feature.' + hiddenPresetFeaturesId + '.description')
+               })).placement(index < 2 ? 'bottom' : 'top'));
              }
            });
-
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             nextStep();
-           }
          }
 
-         function finishDragEndpoint() {
-           if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
-             return continueTo(updateLine);
-           }
+         presetList.autofocus = function (val) {
+           if (!arguments.length) return _autofocus;
+           _autofocus = val;
+           return presetList;
+         };
 
-           var padding = 100 * Math.pow(2, context.map().zoom() - 19);
-           var box = pad(woodRoadDragEndpoint, padding, context);
-           var finishDragString = helpHtml('intro.lines.spot_looks_good') + helpHtml('intro.lines.finish_drag_endpoint' + (context.lastPointerType() === 'mouse' ? '' : '_touch'));
-           reveal(box, finishDragString);
-           context.map().on('move.intro drawn.intro', function () {
-             if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
-               return continueTo(updateLine);
-             }
+         presetList.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs;
+           _entityIDs = val;
+           _currLoc = null;
 
-             var padding = 100 * Math.pow(2, context.map().zoom() - 19);
-             var box = pad(woodRoadDragEndpoint, padding, context);
-             reveal(box, finishDragString, {
-               duration: 0
-             });
-             var entity = context.entity(woodRoadEndID);
+           if (_entityIDs && _entityIDs.length) {
+             // calculate current location
+             var extent = _entityIDs.reduce(function (extent, entityID) {
+               var entity = context.graph().entity(entityID);
+               return extent.extend(entity.extent(context.graph()));
+             }, geoExtent());
 
-             if (geoSphericalDistance(entity.loc, woodRoadDragEndpoint) > 4) {
-               continueTo(startDragEndpoint);
-             }
-           });
-           context.on('enter.intro', function () {
-             continueTo(startDragMidpoint);
-           });
+             _currLoc = extent.center(); // match presets
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
-           }
-         }
+             var presets = _entityIDs.map(function (entityID) {
+               return _mainPresetIndex.match(context.entity(entityID), context.graph());
+             });
 
-         function startDragMidpoint() {
-           if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
-             return continueTo(updateLine);
+             presetList.presets(presets);
            }
 
-           if (context.selectedIDs().indexOf(woodRoadID) === -1) {
-             context.enter(modeSelect(context, [woodRoadID]));
-           }
+           return presetList;
+         };
 
-           var padding = 80 * Math.pow(2, context.map().zoom() - 19);
-           var box = pad(woodRoadDragMidpoint, padding, context);
-           reveal(box, helpHtml('intro.lines.start_drag_midpoint'));
-           context.map().on('move.intro drawn.intro', function () {
-             if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
-               return continueTo(updateLine);
-             }
+         presetList.presets = function (val) {
+           if (!arguments.length) return _currentPresets;
+           _currentPresets = val;
+           return presetList;
+         };
 
-             var padding = 80 * Math.pow(2, context.map().zoom() - 19);
-             var box = pad(woodRoadDragMidpoint, padding, context);
-             reveal(box, helpHtml('intro.lines.start_drag_midpoint'), {
-               duration: 0
-             });
-           });
-           context.history().on('change.intro', function (changed) {
-             if (changed.created().length === 1) {
-               continueTo(continueDragMidpoint);
-             }
-           });
-           context.on('enter.intro', function (mode) {
-             if (mode.id !== 'select') {
-               // keep Wood Road selected so midpoint triangles are drawn..
-               context.enter(modeSelect(context, [woodRoadID]));
+         function entityGeometries() {
+           var counts = {};
+
+           for (var i in _entityIDs) {
+             var entityID = _entityIDs[i];
+             var entity = context.entity(entityID);
+             var geometry = entity.geometry(context.graph()); // Treat entities on addr:interpolation lines as points, not vertices (#3241)
+
+             if (geometry === 'vertex' && entity.isOnAddressLine(context.graph())) {
+               geometry = 'point';
              }
-           });
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.history().on('change.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
+             if (!counts[geometry]) counts[geometry] = 0;
+             counts[geometry] += 1;
            }
-         }
 
-         function continueDragMidpoint() {
-           if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
-             return continueTo(updateLine);
-           }
+           return Object.keys(counts).sort(function (geom1, geom2) {
+             return counts[geom2] - counts[geom1];
+           });
+         }
 
-           var padding = 100 * Math.pow(2, context.map().zoom() - 19);
-           var box = pad(woodRoadDragEndpoint, padding, context);
-           box.height += 400;
+         return utilRebind(presetList, dispatch, 'on');
+       }
 
-           var advance = function advance() {
-             context.history().checkpoint('doneUpdateLine');
-             continueTo(deleteLines);
-           };
+       function uiViewOnOSM(context) {
+         var _what; // an osmEntity or osmNote
 
-           reveal(box, helpHtml('intro.lines.continue_drag_midpoint'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: advance
-           });
-           context.map().on('move.intro drawn.intro', function () {
-             if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
-               return continueTo(updateLine);
-             }
 
-             var padding = 100 * Math.pow(2, context.map().zoom() - 19);
-             var box = pad(woodRoadDragEndpoint, padding, context);
-             box.height += 400;
-             reveal(box, helpHtml('intro.lines.continue_drag_midpoint'), {
-               duration: 0,
-               buttonText: _t.html('intro.ok'),
-               buttonCallback: advance
-             });
-           });
+         function viewOnOSM(selection) {
+           var url;
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             nextStep();
+           if (_what instanceof osmEntity) {
+             url = context.connection().entityURL(_what);
+           } else if (_what instanceof osmNote) {
+             url = context.connection().noteURL(_what);
            }
-         }
 
-         function deleteLines() {
-           context.history().reset('doneUpdateLine');
-           context.enter(modeBrowse(context));
+           var data = !_what || _what.isNew() ? [] : [_what];
+           var link = selection.selectAll('.view-on-osm').data(data, function (d) {
+             return d.id;
+           }); // exit
 
-           if (!context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
-             return chapter.restart();
-           }
+           link.exit().remove(); // enter
 
-           var msec = transitionTime(deleteLinesLoc, context.map().center());
+           var linkEnter = link.enter().append('a').attr('class', 'view-on-osm').attr('target', '_blank').attr('href', url).call(svgIcon('#iD-icon-out-link', 'inline'));
+           linkEnter.append('span').html(_t.html('inspector.view_on_osm'));
+         }
 
-           if (msec) {
-             reveal(null, null, {
-               duration: 0
-             });
-           }
+         viewOnOSM.what = function (_) {
+           if (!arguments.length) return _what;
+           _what = _;
+           return viewOnOSM;
+         };
 
-           context.map().centerZoomEase(deleteLinesLoc, 18, msec);
-           timeout(function () {
-             var padding = 200 * Math.pow(2, context.map().zoom() - 18);
-             var box = pad(deleteLinesLoc, padding, context);
-             box.top -= 200;
-             box.height += 400;
+         return viewOnOSM;
+       }
 
-             var advance = function advance() {
-               continueTo(rightClickIntersection);
-             };
+       function uiInspector(context) {
+         var presetList = uiPresetList(context);
+         var entityEditor = uiEntityEditor(context);
+         var wrap = select(null),
+             presetPane = select(null),
+             editorPane = select(null);
+         var _state = 'select';
 
-             reveal(box, helpHtml('intro.lines.delete_lines', {
-               street: _t('intro.graph.name.12th-avenue')
-             }), {
-               buttonText: _t.html('intro.ok'),
-               buttonCallback: advance
-             });
-             context.map().on('move.intro drawn.intro', function () {
-               var padding = 200 * Math.pow(2, context.map().zoom() - 18);
-               var box = pad(deleteLinesLoc, padding, context);
-               box.top -= 200;
-               box.height += 400;
-               reveal(box, helpHtml('intro.lines.delete_lines', {
-                 street: _t('intro.graph.name.12th-avenue')
-               }), {
-                 duration: 0,
-                 buttonText: _t.html('intro.ok'),
-                 buttonCallback: advance
-               });
-             });
-             context.history().on('change.intro', function () {
-               timeout(function () {
-                 continueTo(deleteLines);
-               }, 500); // after any transition (e.g. if user deleted intersection)
-             });
-           }, msec + 100);
+         var _entityIDs;
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
-           }
-         }
+         var _newFeature = false;
 
-         function rightClickIntersection() {
-           context.history().reset('doneUpdateLine');
-           context.enter(modeBrowse(context));
-           context.map().centerZoomEase(eleventhAvenueEnd, 18, 500);
-           var rightClickString = helpHtml('intro.lines.split_street', {
-             street1: _t('intro.graph.name.11th-avenue'),
-             street2: _t('intro.graph.name.washington-street')
-           }) + helpHtml('intro.lines.' + (context.lastPointerType() === 'mouse' ? 'rightclick_intersection' : 'edit_menu_intersection_touch'));
-           timeout(function () {
-             var padding = 60 * Math.pow(2, context.map().zoom() - 18);
-             var box = pad(eleventhAvenueEnd, padding, context);
-             reveal(box, rightClickString);
-             context.map().on('move.intro drawn.intro', function () {
-               var padding = 60 * Math.pow(2, context.map().zoom() - 18);
-               var box = pad(eleventhAvenueEnd, padding, context);
-               reveal(box, rightClickString, {
-                 duration: 0
-               });
-             });
-             context.on('enter.intro', function (mode) {
-               if (mode.id !== 'select') return;
-               var ids = context.selectedIDs();
-               if (ids.length !== 1 || ids[0] !== eleventhAvenueEndID) return;
-               timeout(function () {
-                 var node = selectMenuItem(context, 'split').node();
-                 if (!node) return;
-                 continueTo(splitIntersection);
-               }, 50); // after menu visible
-             });
-             context.history().on('change.intro', function () {
-               timeout(function () {
-                 continueTo(deleteLines);
-               }, 300); // after any transition (e.g. if user deleted intersection)
-             });
-           }, 600);
+         function inspector(selection) {
+           presetList.entityIDs(_entityIDs).autofocus(_newFeature).on('choose', inspector.setPreset).on('cancel', function () {
+             inspector.setPreset();
+           });
+           entityEditor.state(_state).entityIDs(_entityIDs).on('choose', inspector.showList);
+           wrap = selection.selectAll('.panewrap').data([0]);
+           var enter = wrap.enter().append('div').attr('class', 'panewrap');
+           enter.append('div').attr('class', 'preset-list-pane pane');
+           enter.append('div').attr('class', 'entity-editor-pane pane');
+           wrap = wrap.merge(enter);
+           presetPane = wrap.selectAll('.preset-list-pane');
+           editorPane = wrap.selectAll('.entity-editor-pane');
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('enter.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
-           }
-         }
+           function shouldDefaultToPresetList() {
+             // always show the inspector on hover
+             if (_state !== 'select') return false; // can only change preset on single selection
 
-         function splitIntersection() {
-           if (!context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
-             return continueTo(deleteLines);
-           }
+             if (_entityIDs.length !== 1) return false;
+             var entityID = _entityIDs[0];
+             var entity = context.hasEntity(entityID);
+             if (!entity) return false; // default to inspector if there are already tags
 
-           var node = selectMenuItem(context, 'split').node();
+             if (entity.hasNonGeometryTags()) return false; // prompt to select preset if feature is new and untagged
 
-           if (!node) {
-             return continueTo(rightClickIntersection);
-           }
+             if (_newFeature) return true; // all existing features except vertices should default to inspector
 
-           var wasChanged = false;
-           _washingtonSegmentID = null;
-           reveal('.edit-menu', helpHtml('intro.lines.split_intersection', {
-             street: _t('intro.graph.name.washington-street')
-           }), {
-             padding: 50
-           });
-           context.map().on('move.intro drawn.intro', function () {
-             var node = selectMenuItem(context, 'split').node();
+             if (entity.geometry(context.graph()) !== 'vertex') return false; // show vertex relations if any
 
-             if (!wasChanged && !node) {
-               return continueTo(rightClickIntersection);
-             }
+             if (context.graph().parentRelations(entity).length) return false; // show vertex issues if there are any
 
-             reveal('.edit-menu', helpHtml('intro.lines.split_intersection', {
-               street: _t('intro.graph.name.washington-street')
-             }), {
-               duration: 0,
-               padding: 50
-             });
-           });
-           context.history().on('change.intro', function (changed) {
-             wasChanged = true;
-             timeout(function () {
-               if (context.history().undoAnnotation() === _t('operations.split.annotation.line', {
-                 n: 1
-               })) {
-                 _washingtonSegmentID = changed.created()[0].id;
-                 continueTo(didSplit);
-               } else {
-                 _washingtonSegmentID = null;
-                 continueTo(retrySplit);
-               }
-             }, 300); // after any transition (e.g. if user deleted intersection)
-           });
+             if (context.validator().getEntityIssues(entityID).length) return false; // show turn retriction editor for junction vertices
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
+             if (entity.isHighwayIntersection(context.graph())) return false; // otherwise show preset list for uninteresting vertices
+
+             return true;
            }
-         }
 
-         function retrySplit() {
-           context.enter(modeBrowse(context));
-           context.map().centerZoomEase(eleventhAvenueEnd, 18, 500);
+           if (shouldDefaultToPresetList()) {
+             wrap.style('right', '-100%');
+             editorPane.classed('hide', true);
+             presetPane.classed('hide', false).call(presetList);
+           } else {
+             wrap.style('right', '0%');
+             presetPane.classed('hide', true);
+             editorPane.classed('hide', false).call(entityEditor);
+           }
 
-           var advance = function advance() {
-             continueTo(rightClickIntersection);
-           };
+           var footer = selection.selectAll('.footer').data([0]);
+           footer = footer.enter().append('div').attr('class', 'footer').merge(footer);
+           footer.call(uiViewOnOSM(context).what(context.hasEntity(_entityIDs.length === 1 && _entityIDs[0])));
+         }
 
-           var padding = 60 * Math.pow(2, context.map().zoom() - 18);
-           var box = pad(eleventhAvenueEnd, padding, context);
-           reveal(box, helpHtml('intro.lines.retry_split'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: advance
-           });
-           context.map().on('move.intro drawn.intro', function () {
-             var padding = 60 * Math.pow(2, context.map().zoom() - 18);
-             var box = pad(eleventhAvenueEnd, padding, context);
-             reveal(box, helpHtml('intro.lines.retry_split'), {
-               duration: 0,
-               buttonText: _t.html('intro.ok'),
-               buttonCallback: advance
-             });
+         inspector.showList = function (presets) {
+           presetPane.classed('hide', false);
+           wrap.transition().styleTween('right', function () {
+             return interpolate$1('0%', '-100%');
+           }).on('end', function () {
+             editorPane.classed('hide', true);
            });
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             nextStep();
+           if (presets) {
+             presetList.presets(presets);
            }
-         }
 
-         function didSplit() {
-           if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
-             return continueTo(rightClickIntersection);
-           }
+           presetPane.call(presetList.autofocus(true));
+         };
 
-           var ids = context.selectedIDs();
-           var string = 'intro.lines.did_split_' + (ids.length > 1 ? 'multi' : 'single');
-           var street = _t('intro.graph.name.washington-street');
-           var padding = 200 * Math.pow(2, context.map().zoom() - 18);
-           var box = pad(twelfthAvenue, padding, context);
-           box.width = box.width / 2;
-           reveal(box, helpHtml(string, {
-             street1: street,
-             street2: street
-           }), {
-             duration: 500
-           });
-           timeout(function () {
-             context.map().centerZoomEase(twelfthAvenue, 18, 500);
-             context.map().on('move.intro drawn.intro', function () {
-               var padding = 200 * Math.pow(2, context.map().zoom() - 18);
-               var box = pad(twelfthAvenue, padding, context);
-               box.width = box.width / 2;
-               reveal(box, helpHtml(string, {
-                 street1: street,
-                 street2: street
-               }), {
-                 duration: 0
-               });
+         inspector.setPreset = function (preset) {
+           // upon setting multipolygon, go to the area preset list instead of the editor
+           if (preset && preset.id === 'type/multipolygon') {
+             presetPane.call(presetList.autofocus(true));
+           } else {
+             editorPane.classed('hide', false);
+             wrap.transition().styleTween('right', function () {
+               return interpolate$1('-100%', '0%');
+             }).on('end', function () {
+               presetPane.classed('hide', true);
              });
-           }, 600); // after initial reveal and curtain cut
-
-           context.on('enter.intro', function () {
-             var ids = context.selectedIDs();
 
-             if (ids.length === 1 && ids[0] === _washingtonSegmentID) {
-               continueTo(multiSelect);
-             }
-           });
-           context.history().on('change.intro', function () {
-             if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
-               return continueTo(rightClickIntersection);
+             if (preset) {
+               entityEditor.presets([preset]);
              }
-           });
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('enter.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
+             editorPane.call(entityEditor);
            }
-         }
+         };
 
-         function multiSelect() {
-           if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
-             return continueTo(rightClickIntersection);
-           }
+         inspector.state = function (val) {
+           if (!arguments.length) return _state;
+           _state = val;
+           entityEditor.state(_state); // remove any old field help overlay that might have gotten attached to the inspector
 
-           var ids = context.selectedIDs();
-           var hasWashington = ids.indexOf(_washingtonSegmentID) !== -1;
-           var hasTwelfth = ids.indexOf(twelfthAvenueID) !== -1;
+           context.container().selectAll('.field-help-body').remove();
+           return inspector;
+         };
 
-           if (hasWashington && hasTwelfth) {
-             return continueTo(multiRightClick);
-           } else if (!hasWashington && !hasTwelfth) {
-             return continueTo(didSplit);
-           }
+         inspector.entityIDs = function (val) {
+           if (!arguments.length) return _entityIDs;
+           _entityIDs = val;
+           return inspector;
+         };
 
-           context.map().centerZoomEase(twelfthAvenue, 18, 500);
-           timeout(function () {
-             var selected, other, padding, box;
+         inspector.newFeature = function (val) {
+           if (!arguments.length) return _newFeature;
+           _newFeature = val;
+           return inspector;
+         };
 
-             if (hasWashington) {
-               selected = _t('intro.graph.name.washington-street');
-               other = _t('intro.graph.name.12th-avenue');
-               padding = 60 * Math.pow(2, context.map().zoom() - 18);
-               box = pad(twelfthAvenueEnd, padding, context);
-               box.width *= 3;
-             } else {
-               selected = _t('intro.graph.name.12th-avenue');
-               other = _t('intro.graph.name.washington-street');
-               padding = 200 * Math.pow(2, context.map().zoom() - 18);
-               box = pad(twelfthAvenue, padding, context);
-               box.width /= 2;
-             }
+         return inspector;
+       }
 
-             reveal(box, helpHtml('intro.lines.multi_select', {
-               selected: selected,
-               other1: other
-             }) + ' ' + helpHtml('intro.lines.add_to_selection_' + (context.lastPointerType() === 'mouse' ? 'click' : 'touch'), {
-               selected: selected,
-               other2: other
-             }));
-             context.map().on('move.intro drawn.intro', function () {
-               if (hasWashington) {
-                 selected = _t('intro.graph.name.washington-street');
-                 other = _t('intro.graph.name.12th-avenue');
-                 padding = 60 * Math.pow(2, context.map().zoom() - 18);
-                 box = pad(twelfthAvenueEnd, padding, context);
-                 box.width *= 3;
-               } else {
-                 selected = _t('intro.graph.name.12th-avenue');
-                 other = _t('intro.graph.name.washington-street');
-                 padding = 200 * Math.pow(2, context.map().zoom() - 18);
-                 box = pad(twelfthAvenue, padding, context);
-                 box.width /= 2;
+       function uiImproveOsmComments() {
+         var _qaItem;
+
+         function issueComments(selection) {
+           // make the div immediately so it appears above the buttons
+           var comments = selection.selectAll('.comments-container').data([0]);
+           comments = comments.enter().append('div').attr('class', 'comments-container').merge(comments); // must retrieve comments from API before they can be displayed
+
+           services.improveOSM.getComments(_qaItem).then(function (d) {
+             if (!d.comments) return; // nothing to do here
+
+             var commentEnter = comments.selectAll('.comment').data(d.comments).enter().append('div').attr('class', 'comment');
+             commentEnter.append('div').attr('class', 'comment-avatar').call(svgIcon('#iD-icon-avatar', 'comment-avatar-icon'));
+             var mainEnter = commentEnter.append('div').attr('class', 'comment-main');
+             var metadataEnter = mainEnter.append('div').attr('class', 'comment-metadata');
+             metadataEnter.append('div').attr('class', 'comment-author').each(function (d) {
+               var osm = services.osm;
+               var selection = select(this);
+
+               if (osm && d.username) {
+                 selection = selection.append('a').attr('class', 'comment-author-link').attr('href', osm.userURL(d.username)).attr('target', '_blank');
                }
 
-               reveal(box, helpHtml('intro.lines.multi_select', {
-                 selected: selected,
-                 other1: other
-               }) + ' ' + helpHtml('intro.lines.add_to_selection_' + (context.lastPointerType() === 'mouse' ? 'click' : 'touch'), {
-                 selected: selected,
-                 other2: other
-               }), {
-                 duration: 0
+               selection.html(function (d) {
+                 return d.username;
                });
              });
-             context.on('enter.intro', function () {
-               continueTo(multiSelect);
+             metadataEnter.append('div').attr('class', 'comment-date').html(function (d) {
+               return _t.html('note.status.commented', {
+                 when: localeDateString(d.timestamp)
+               });
              });
-             context.history().on('change.intro', function () {
-               if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
-                 return continueTo(rightClickIntersection);
-               }
+             mainEnter.append('div').attr('class', 'comment-text').append('p').html(function (d) {
+               return d.text;
              });
-           }, 600);
+           })["catch"](function (err) {
+             console.log(err); // eslint-disable-line no-console
+           });
+         }
+
+         function localeDateString(s) {
+           if (!s) return null;
+           var options = {
+             day: 'numeric',
+             month: 'short',
+             year: 'numeric'
+           };
+           var d = new Date(s * 1000); // timestamp is served in seconds, date takes ms
+
+           if (isNaN(d.getTime())) return null;
+           return d.toLocaleDateString(_mainLocalizer.localeCode(), options);
+         }
+
+         issueComments.issue = function (val) {
+           if (!arguments.length) return _qaItem;
+           _qaItem = val;
+           return issueComments;
+         };
+
+         return issueComments;
+       }
+
+       function uiImproveOsmDetails(context) {
+         var _qaItem;
+
+         function issueDetail(d) {
+           if (d.desc) return d.desc;
+           var issueKey = d.issueKey;
+           d.replacements = d.replacements || {};
+           d.replacements["default"] = _t.html('inspector.unknown'); // special key `default` works as a fallback string
+
+           return _t.html("QA.improveOSM.error_types.".concat(issueKey, ".description"), d.replacements);
+         }
+
+         function improveOsmDetails(selection) {
+           var details = selection.selectAll('.error-details').data(_qaItem ? [_qaItem] : [], function (d) {
+             return "".concat(d.id, "-").concat(d.status || 0);
+           });
+           details.exit().remove();
+           var detailsEnter = details.enter().append('div').attr('class', 'error-details qa-details-container'); // description
+
+           var descriptionEnter = detailsEnter.append('div').attr('class', 'qa-details-subsection');
+           descriptionEnter.append('h4').html(_t.html('QA.keepRight.detail_description'));
+           descriptionEnter.append('div').attr('class', 'qa-details-description-text').html(issueDetail); // If there are entity links in the error message..
+
+           var relatedEntities = [];
+           descriptionEnter.selectAll('.error_entity_link, .error_object_link').attr('href', '#').each(function () {
+             var link = select(this);
+             var isObjectLink = link.classed('error_object_link');
+             var entityID = isObjectLink ? utilEntityRoot(_qaItem.objectType) + _qaItem.objectId : this.textContent;
+             var entity = context.hasEntity(entityID);
+             relatedEntities.push(entityID); // Add click handler
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('enter.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
-           }
-         }
+             link.on('mouseenter', function () {
+               utilHighlightEntities([entityID], true, context);
+             }).on('mouseleave', function () {
+               utilHighlightEntities([entityID], false, context);
+             }).on('click', function (d3_event) {
+               d3_event.preventDefault();
+               utilHighlightEntities([entityID], false, context);
+               var osmlayer = context.layers().layer('osm');
 
-         function multiRightClick() {
-           if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
-             return continueTo(rightClickIntersection);
-           }
+               if (!osmlayer.enabled()) {
+                 osmlayer.enabled(true);
+               }
 
-           var padding = 200 * Math.pow(2, context.map().zoom() - 18);
-           var box = pad(twelfthAvenue, padding, context);
-           var rightClickString = helpHtml('intro.lines.multi_select_success') + helpHtml('intro.lines.multi_' + (context.lastPointerType() === 'mouse' ? 'rightclick' : 'edit_menu_touch'));
-           reveal(box, rightClickString);
-           context.map().on('move.intro drawn.intro', function () {
-             var padding = 200 * Math.pow(2, context.map().zoom() - 18);
-             var box = pad(twelfthAvenue, padding, context);
-             reveal(box, rightClickString, {
-               duration: 0
-             });
-           });
-           context.ui().editMenu().on('toggled.intro', function (open) {
-             if (!open) return;
-             timeout(function () {
-               var ids = context.selectedIDs();
+               context.map().centerZoom(_qaItem.loc, 20);
 
-               if (ids.length === 2 && ids.indexOf(twelfthAvenueID) !== -1 && ids.indexOf(_washingtonSegmentID) !== -1) {
-                 var node = selectMenuItem(context, 'delete').node();
-                 if (!node) return;
-                 continueTo(multiDelete);
-               } else if (ids.length === 1 && ids.indexOf(_washingtonSegmentID) !== -1) {
-                 return continueTo(multiSelect);
+               if (entity) {
+                 context.enter(modeSelect(context, [entityID]));
                } else {
-                 return continueTo(didSplit);
+                 context.loadEntity(entityID, function (err, result) {
+                   if (err) return;
+                   var entity = result.data.find(function (e) {
+                     return e.id === entityID;
+                   });
+                   if (entity) context.enter(modeSelect(context, [entityID]));
+                 });
                }
-             }, 300); // after edit menu visible
-           });
-           context.history().on('change.intro', function () {
-             if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
-               return continueTo(rightClickIntersection);
-             }
-           });
+             }); // Replace with friendly name if possible
+             // (The entity may not yet be loaded into the graph)
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.ui().editMenu().on('toggled.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
-           }
-         }
+             if (entity) {
+               var name = utilDisplayName(entity); // try to use common name
 
-         function multiDelete() {
-           if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
-             return continueTo(rightClickIntersection);
-           }
+               if (!name && !isObjectLink) {
+                 var preset = _mainPresetIndex.match(entity, context.graph());
+                 name = preset && !preset.isFallback() && preset.name(); // fallback to preset name
+               }
 
-           var node = selectMenuItem(context, 'delete').node();
-           if (!node) return continueTo(multiRightClick);
-           reveal('.edit-menu', helpHtml('intro.lines.multi_delete'), {
-             padding: 50
-           });
-           context.map().on('move.intro drawn.intro', function () {
-             reveal('.edit-menu', helpHtml('intro.lines.multi_delete'), {
-               duration: 0,
-               padding: 50
-             });
-           });
-           context.on('exit.intro', function () {
-             if (context.hasEntity(_washingtonSegmentID) || context.hasEntity(twelfthAvenueID)) {
-               return continueTo(multiSelect); // left select mode but roads still exist
-             }
-           });
-           context.history().on('change.intro', function () {
-             if (context.hasEntity(_washingtonSegmentID) || context.hasEntity(twelfthAvenueID)) {
-               continueTo(retryDelete); // changed something but roads still exist
-             } else {
-               continueTo(play);
+               if (name) {
+                 this.innerText = name;
+               }
              }
-           });
+           }); // Don't hide entities related to this error - #5880
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('exit.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
-           }
+           context.features().forceVisible(relatedEntities);
+           context.map().pan([0, 0]); // trigger a redraw
          }
 
-         function retryDelete() {
-           context.enter(modeBrowse(context));
-           var padding = 200 * Math.pow(2, context.map().zoom() - 18);
-           var box = pad(twelfthAvenue, padding, context);
-           reveal(box, helpHtml('intro.lines.retry_delete'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: function buttonCallback() {
-               continueTo(multiSelect);
-             }
-           });
+         improveOsmDetails.issue = function (val) {
+           if (!arguments.length) return _qaItem;
+           _qaItem = val;
+           return improveOsmDetails;
+         };
 
-           function continueTo(nextStep) {
-             nextStep();
-           }
+         return improveOsmDetails;
+       }
+
+       function uiImproveOsmHeader() {
+         var _qaItem;
+
+         function issueTitle(d) {
+           var issueKey = d.issueKey;
+           d.replacements = d.replacements || {};
+           d.replacements["default"] = _t.html('inspector.unknown'); // special key `default` works as a fallback string
+
+           return _t.html("QA.improveOSM.error_types.".concat(issueKey, ".title"), d.replacements);
          }
 
-         function play() {
-           dispatch$1.call('done');
-           reveal('.ideditor', helpHtml('intro.lines.play', {
-             next: _t('intro.buildings.title')
-           }), {
-             tooltipBox: '.intro-nav-wrap .chapter-building',
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: function buttonCallback() {
-               reveal('.ideditor');
+         function improveOsmHeader(selection) {
+           var header = selection.selectAll('.qa-header').data(_qaItem ? [_qaItem] : [], function (d) {
+             return "".concat(d.id, "-").concat(d.status || 0);
+           });
+           header.exit().remove();
+           var headerEnter = header.enter().append('div').attr('class', 'qa-header');
+           var svgEnter = headerEnter.append('div').attr('class', 'qa-header-icon').classed('new', function (d) {
+             return d.id < 0;
+           }).append('svg').attr('width', '20px').attr('height', '30px').attr('viewbox', '0 0 20 30').attr('class', function (d) {
+             return "preset-icon-28 qaItem ".concat(d.service, " itemId-").concat(d.id, " itemType-").concat(d.itemType);
+           });
+           svgEnter.append('polygon').attr('fill', 'currentColor').attr('class', 'qaItem-fill').attr('points', '16,3 4,3 1,6 1,17 4,20 7,20 10,27 13,20 16,20 19,17.033 19,6');
+           svgEnter.append('use').attr('class', 'icon-annotation').attr('width', '13px').attr('height', '13px').attr('transform', 'translate(3.5, 5)').attr('xlink:href', function (d) {
+             var picon = d.icon;
+
+             if (!picon) {
+               return '';
+             } else {
+               var isMaki = /^maki-/.test(picon);
+               return "#".concat(picon).concat(isMaki ? '-11' : '');
              }
            });
+           headerEnter.append('div').attr('class', 'qa-header-label').html(issueTitle);
          }
 
-         chapter.enter = function () {
-           addLine();
-         };
-
-         chapter.exit = function () {
-           timeouts.forEach(window.clearTimeout);
-           select(window).on('pointerdown.intro mousedown.intro', null, true);
-           context.on('enter.intro exit.intro', null);
-           context.map().on('move.intro drawn.intro', null);
-           context.history().on('change.intro', null);
-           context.container().select('.inspector-wrap').on('wheel.intro', null);
-           context.container().select('.preset-list-button').on('click.intro', null);
-         };
-
-         chapter.restart = function () {
-           chapter.exit();
-           chapter.enter();
+         improveOsmHeader.issue = function (val) {
+           if (!arguments.length) return _qaItem;
+           _qaItem = val;
+           return improveOsmHeader;
          };
 
-         return utilRebind(chapter, dispatch$1, 'on');
+         return improveOsmHeader;
        }
 
-       function uiIntroBuilding(context, reveal) {
-         var dispatch$1 = dispatch('done');
-         var house = [-85.62815, 41.95638];
-         var tank = [-85.62732, 41.95347];
-         var buildingCatetory = _mainPresetIndex.item('category-building');
-         var housePreset = _mainPresetIndex.item('building/house');
-         var tankPreset = _mainPresetIndex.item('man_made/storage_tank');
-         var timeouts = [];
-         var _houseID = null;
-         var _tankID = null;
-         var chapter = {
-           title: 'intro.buildings.title'
-         };
+       function uiImproveOsmEditor(context) {
+         var dispatch = dispatch$8('change');
+         var qaDetails = uiImproveOsmDetails(context);
+         var qaComments = uiImproveOsmComments();
+         var qaHeader = uiImproveOsmHeader();
 
-         function timeout(f, t) {
-           timeouts.push(window.setTimeout(f, t));
-         }
+         var _qaItem;
 
-         function eventCancel(d3_event) {
-           d3_event.stopPropagation();
-           d3_event.preventDefault();
+         function improveOsmEditor(selection) {
+           var headerEnter = selection.selectAll('.header').data([0]).enter().append('div').attr('class', 'header fillL');
+           headerEnter.append('button').attr('class', 'close').on('click', function () {
+             return context.enter(modeBrowse(context));
+           }).call(svgIcon('#iD-icon-close'));
+           headerEnter.append('h3').html(_t.html('QA.improveOSM.title'));
+           var body = selection.selectAll('.body').data([0]);
+           body = body.enter().append('div').attr('class', 'body').merge(body);
+           var editor = body.selectAll('.qa-editor').data([0]);
+           editor.enter().append('div').attr('class', 'modal-section qa-editor').merge(editor).call(qaHeader.issue(_qaItem)).call(qaDetails.issue(_qaItem)).call(qaComments.issue(_qaItem)).call(improveOsmSaveSection);
          }
 
-         function revealHouse(center, text, options) {
-           var padding = 160 * Math.pow(2, context.map().zoom() - 20);
-           var box = pad(center, padding, context);
-           reveal(box, text, options);
-         }
+         function improveOsmSaveSection(selection) {
+           var isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
 
-         function revealTank(center, text, options) {
-           var padding = 190 * Math.pow(2, context.map().zoom() - 19.5);
-           var box = pad(center, padding, context);
-           reveal(box, text, options);
-         }
+           var isShown = _qaItem && (isSelected || _qaItem.newComment || _qaItem.comment);
+           var saveSection = selection.selectAll('.qa-save').data(isShown ? [_qaItem] : [], function (d) {
+             return "".concat(d.id, "-").concat(d.status || 0);
+           }); // exit
 
-         function addHouse() {
-           context.enter(modeBrowse(context));
-           context.history().reset('initial');
-           _houseID = null;
-           var msec = transitionTime(house, context.map().center());
+           saveSection.exit().remove(); // enter
 
-           if (msec) {
-             reveal(null, null, {
-               duration: 0
-             });
-           }
+           var saveSectionEnter = saveSection.enter().append('div').attr('class', 'qa-save save-section cf');
+           saveSectionEnter.append('h4').attr('class', '.qa-save-header').html(_t.html('note.newComment'));
+           saveSectionEnter.append('textarea').attr('class', 'new-comment-input').attr('placeholder', _t('QA.keepRight.comment_placeholder')).attr('maxlength', 1000).property('value', function (d) {
+             return d.newComment;
+           }).call(utilNoAuto).on('input', changeInput).on('blur', changeInput); // update
 
-           context.map().centerZoomEase(house, 19, msec);
-           timeout(function () {
-             var tooltip = reveal('button.add-area', helpHtml('intro.buildings.add_building'));
-             tooltip.selectAll('.popover-inner').insert('svg', 'span').attr('class', 'tooltip-illustration').append('use').attr('xlink:href', '#iD-graphic-buildings');
-             context.on('enter.intro', function (mode) {
-               if (mode.id !== 'add-area') return;
-               continueTo(startHouse);
-             });
-           }, msec + 100);
+           saveSection = saveSectionEnter.merge(saveSection).call(qaSaveButtons);
 
-           function continueTo(nextStep) {
-             context.on('enter.intro', null);
-             nextStep();
-           }
-         }
+           function changeInput() {
+             var input = select(this);
+             var val = input.property('value').trim();
 
-         function startHouse() {
-           if (context.mode().id !== 'add-area') {
-             return continueTo(addHouse);
-           }
+             if (val === '') {
+               val = undefined;
+             } // store the unsaved comment with the issue itself
 
-           _houseID = null;
-           context.map().zoomEase(20, 500);
-           timeout(function () {
-             var startString = helpHtml('intro.buildings.start_building') + helpHtml('intro.buildings.building_corner_' + (context.lastPointerType() === 'mouse' ? 'click' : 'tap'));
-             revealHouse(house, startString);
-             context.map().on('move.intro drawn.intro', function () {
-               revealHouse(house, startString, {
-                 duration: 0
-               });
-             });
-             context.on('enter.intro', function (mode) {
-               if (mode.id !== 'draw-area') return chapter.restart();
-               continueTo(continueHouse);
+
+             _qaItem = _qaItem.update({
+               newComment: val
              });
-           }, 550); // after easing
+             var qaService = services.improveOSM;
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
+             if (qaService) {
+               qaService.replaceItem(_qaItem);
+             }
+
+             saveSection.call(qaSaveButtons);
            }
          }
 
-         function continueHouse() {
-           if (context.mode().id !== 'draw-area') {
-             return continueTo(addHouse);
-           }
+         function qaSaveButtons(selection) {
+           var isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
 
-           _houseID = null;
-           var continueString = helpHtml('intro.buildings.continue_building') + '{br}' + helpHtml('intro.areas.finish_area_' + (context.lastPointerType() === 'mouse' ? 'click' : 'tap')) + helpHtml('intro.buildings.finish_building');
-           revealHouse(house, continueString);
-           context.map().on('move.intro drawn.intro', function () {
-             revealHouse(house, continueString, {
-               duration: 0
-             });
+           var buttonSection = selection.selectAll('.buttons').data(isSelected ? [_qaItem] : [], function (d) {
+             return d.status + d.id;
+           }); // exit
+
+           buttonSection.exit().remove(); // enter
+
+           var buttonEnter = buttonSection.enter().append('div').attr('class', 'buttons');
+           buttonEnter.append('button').attr('class', 'button comment-button action').html(_t.html('QA.keepRight.save_comment'));
+           buttonEnter.append('button').attr('class', 'button close-button action');
+           buttonEnter.append('button').attr('class', 'button ignore-button action'); // update
+
+           buttonSection = buttonSection.merge(buttonEnter);
+           buttonSection.select('.comment-button').attr('disabled', function (d) {
+             return d.newComment ? null : true;
+           }).on('click.comment', function (d3_event, d) {
+             this.blur(); // avoid keeping focus on the button - #4641
+
+             var qaService = services.improveOSM;
+
+             if (qaService) {
+               qaService.postUpdate(d, function (err, item) {
+                 return dispatch.call('change', item);
+               });
+             }
            });
-           context.on('enter.intro', function (mode) {
-             if (mode.id === 'draw-area') {
-               return;
-             } else if (mode.id === 'select') {
-               var graph = context.graph();
-               var way = context.entity(context.selectedIDs()[0]);
-               var nodes = graph.childNodes(way);
-               var points = utilArrayUniq(nodes).map(function (n) {
-                 return context.projection(n.loc);
+           buttonSection.select('.close-button').html(function (d) {
+             var andComment = d.newComment ? '_comment' : '';
+             return _t.html("QA.keepRight.close".concat(andComment));
+           }).on('click.close', function (d3_event, d) {
+             this.blur(); // avoid keeping focus on the button - #4641
+
+             var qaService = services.improveOSM;
+
+             if (qaService) {
+               d.newStatus = 'SOLVED';
+               qaService.postUpdate(d, function (err, item) {
+                 return dispatch.call('change', item);
                });
+             }
+           });
+           buttonSection.select('.ignore-button').html(function (d) {
+             var andComment = d.newComment ? '_comment' : '';
+             return _t.html("QA.keepRight.ignore".concat(andComment));
+           }).on('click.ignore', function (d3_event, d) {
+             this.blur(); // avoid keeping focus on the button - #4641
 
-               if (isMostlySquare(points)) {
-                 _houseID = way.id;
-                 return continueTo(chooseCategoryBuilding);
-               } else {
-                 return continueTo(retryHouse);
-               }
-             } else {
-               return chapter.restart();
+             var qaService = services.improveOSM;
+
+             if (qaService) {
+               d.newStatus = 'INVALID';
+               qaService.postUpdate(d, function (err, item) {
+                 return dispatch.call('change', item);
+               });
              }
            });
+         } // NOTE: Don't change method name until UI v3 is merged
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
-           }
-         }
 
-         function retryHouse() {
-           var onClick = function onClick() {
-             continueTo(addHouse);
-           };
+         improveOsmEditor.error = function (val) {
+           if (!arguments.length) return _qaItem;
+           _qaItem = val;
+           return improveOsmEditor;
+         };
 
-           revealHouse(house, helpHtml('intro.buildings.retry_building'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: onClick
-           });
-           context.map().on('move.intro drawn.intro', function () {
-             revealHouse(house, helpHtml('intro.buildings.retry_building'), {
-               duration: 0,
-               buttonText: _t.html('intro.ok'),
-               buttonCallback: onClick
-             });
-           });
+         return utilRebind(improveOsmEditor, dispatch, 'on');
+       }
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             nextStep();
+       function uiKeepRightDetails(context) {
+         var _qaItem;
+
+         function issueDetail(d) {
+           var itemType = d.itemType,
+               parentIssueType = d.parentIssueType;
+           var unknown = _t.html('inspector.unknown');
+           var replacements = d.replacements || {};
+           replacements["default"] = unknown; // special key `default` works as a fallback string
+
+           var detail = _t.html("QA.keepRight.errorTypes.".concat(itemType, ".description"), replacements);
+
+           if (detail === unknown) {
+             detail = _t.html("QA.keepRight.errorTypes.".concat(parentIssueType, ".description"), replacements);
            }
+
+           return detail;
          }
 
-         function chooseCategoryBuilding() {
-           if (!_houseID || !context.hasEntity(_houseID)) {
-             return addHouse();
-           }
+         function keepRightDetails(selection) {
+           var details = selection.selectAll('.error-details').data(_qaItem ? [_qaItem] : [], function (d) {
+             return "".concat(d.id, "-").concat(d.status || 0);
+           });
+           details.exit().remove();
+           var detailsEnter = details.enter().append('div').attr('class', 'error-details qa-details-container'); // description
 
-           var ids = context.selectedIDs();
+           var descriptionEnter = detailsEnter.append('div').attr('class', 'qa-details-subsection');
+           descriptionEnter.append('h4').html(_t.html('QA.keepRight.detail_description'));
+           descriptionEnter.append('div').attr('class', 'qa-details-description-text').html(issueDetail); // If there are entity links in the error message..
 
-           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _houseID) {
-             context.enter(modeSelect(context, [_houseID]));
-           } // disallow scrolling
+           var relatedEntities = [];
+           descriptionEnter.selectAll('.error_entity_link, .error_object_link').attr('href', '#').each(function () {
+             var link = select(this);
+             var isObjectLink = link.classed('error_object_link');
+             var entityID = isObjectLink ? utilEntityRoot(_qaItem.objectType) + _qaItem.objectId : this.textContent;
+             var entity = context.hasEntity(entityID);
+             relatedEntities.push(entityID); // Add click handler
 
+             link.on('mouseenter', function () {
+               utilHighlightEntities([entityID], true, context);
+             }).on('mouseleave', function () {
+               utilHighlightEntities([entityID], false, context);
+             }).on('click', function (d3_event) {
+               d3_event.preventDefault();
+               utilHighlightEntities([entityID], false, context);
+               var osmlayer = context.layers().layer('osm');
 
-           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
-           timeout(function () {
-             // reset pane, in case user somehow happened to change it..
-             context.container().select('.inspector-wrap .panewrap').style('right', '-100%');
-             var button = context.container().select('.preset-category-building .preset-list-button');
-             reveal(button.node(), helpHtml('intro.buildings.choose_category_building', {
-               category: buildingCatetory.name()
-             }));
-             button.on('click.intro', function () {
-               button.on('click.intro', null);
-               continueTo(choosePresetHouse);
-             });
-           }, 400); // after preset list pane visible..
+               if (!osmlayer.enabled()) {
+                 osmlayer.enabled(true);
+               }
 
-           context.on('enter.intro', function (mode) {
-             if (!_houseID || !context.hasEntity(_houseID)) {
-               return continueTo(addHouse);
-             }
+               context.map().centerZoomEase(_qaItem.loc, 20);
 
-             var ids = context.selectedIDs();
+               if (entity) {
+                 context.enter(modeSelect(context, [entityID]));
+               } else {
+                 context.loadEntity(entityID, function (err, result) {
+                   if (err) return;
+                   var entity = result.data.find(function (e) {
+                     return e.id === entityID;
+                   });
+                   if (entity) context.enter(modeSelect(context, [entityID]));
+                 });
+               }
+             }); // Replace with friendly name if possible
+             // (The entity may not yet be loaded into the graph)
 
-             if (mode.id !== 'select' || !ids.length || ids[0] !== _houseID) {
-               return continueTo(chooseCategoryBuilding);
+             if (entity) {
+               var name = utilDisplayName(entity); // try to use common name
+
+               if (!name && !isObjectLink) {
+                 var preset = _mainPresetIndex.match(entity, context.graph());
+                 name = preset && !preset.isFallback() && preset.name(); // fallback to preset name
+               }
+
+               if (name) {
+                 this.innerText = name;
+               }
              }
-           });
+           }); // Don't hide entities related to this issue - #5880
 
-           function continueTo(nextStep) {
-             context.container().select('.inspector-wrap').on('wheel.intro', null);
-             context.container().select('.preset-list-button').on('click.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
-           }
+           context.features().forceVisible(relatedEntities);
+           context.map().pan([0, 0]); // trigger a redraw
          }
 
-         function choosePresetHouse() {
-           if (!_houseID || !context.hasEntity(_houseID)) {
-             return addHouse();
-           }
+         keepRightDetails.issue = function (val) {
+           if (!arguments.length) return _qaItem;
+           _qaItem = val;
+           return keepRightDetails;
+         };
 
-           var ids = context.selectedIDs();
+         return keepRightDetails;
+       }
 
-           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _houseID) {
-             context.enter(modeSelect(context, [_houseID]));
-           } // disallow scrolling
+       function uiKeepRightHeader() {
+         var _qaItem;
 
+         function issueTitle(d) {
+           var itemType = d.itemType,
+               parentIssueType = d.parentIssueType;
+           var unknown = _t.html('inspector.unknown');
+           var replacements = d.replacements || {};
+           replacements["default"] = unknown; // special key `default` works as a fallback string
 
-           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
-           timeout(function () {
-             // reset pane, in case user somehow happened to change it..
-             context.container().select('.inspector-wrap .panewrap').style('right', '-100%');
-             var button = context.container().select('.preset-building-house .preset-list-button');
-             reveal(button.node(), helpHtml('intro.buildings.choose_preset_house', {
-               preset: housePreset.name()
-             }), {
-               duration: 300
-             });
-             button.on('click.intro', function () {
-               button.on('click.intro', null);
-               continueTo(closeEditorHouse);
-             });
-           }, 400); // after preset list pane visible..
+           var title = _t.html("QA.keepRight.errorTypes.".concat(itemType, ".title"), replacements);
 
-           context.on('enter.intro', function (mode) {
-             if (!_houseID || !context.hasEntity(_houseID)) {
-               return continueTo(addHouse);
-             }
+           if (title === unknown) {
+             title = _t.html("QA.keepRight.errorTypes.".concat(parentIssueType, ".title"), replacements);
+           }
 
-             var ids = context.selectedIDs();
+           return title;
+         }
 
-             if (mode.id !== 'select' || !ids.length || ids[0] !== _houseID) {
-               return continueTo(chooseCategoryBuilding);
-             }
+         function keepRightHeader(selection) {
+           var header = selection.selectAll('.qa-header').data(_qaItem ? [_qaItem] : [], function (d) {
+             return "".concat(d.id, "-").concat(d.status || 0);
            });
-
-           function continueTo(nextStep) {
-             context.container().select('.inspector-wrap').on('wheel.intro', null);
-             context.container().select('.preset-list-button').on('click.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
-           }
+           header.exit().remove();
+           var headerEnter = header.enter().append('div').attr('class', 'qa-header');
+           var iconEnter = headerEnter.append('div').attr('class', 'qa-header-icon').classed('new', function (d) {
+             return d.id < 0;
+           });
+           iconEnter.append('div').attr('class', function (d) {
+             return "preset-icon-28 qaItem ".concat(d.service, " itemId-").concat(d.id, " itemType-").concat(d.parentIssueType);
+           }).call(svgIcon('#iD-icon-bolt', 'qaItem-fill'));
+           headerEnter.append('div').attr('class', 'qa-header-label').html(issueTitle);
          }
 
-         function closeEditorHouse() {
-           if (!_houseID || !context.hasEntity(_houseID)) {
-             return addHouse();
-           }
+         keepRightHeader.issue = function (val) {
+           if (!arguments.length) return _qaItem;
+           _qaItem = val;
+           return keepRightHeader;
+         };
 
-           var ids = context.selectedIDs();
+         return keepRightHeader;
+       }
 
-           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _houseID) {
-             context.enter(modeSelect(context, [_houseID]));
-           }
+       function uiViewOnKeepRight() {
+         var _qaItem;
 
-           context.history().checkpoint('hasHouse');
-           context.on('exit.intro', function () {
-             continueTo(rightClickHouse);
-           });
-           timeout(function () {
-             reveal('.entity-editor-pane', helpHtml('intro.buildings.close', {
-               button: icon('#iD-icon-close', 'inline')
-             }));
-           }, 500);
+         function viewOnKeepRight(selection) {
+           var url;
 
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             nextStep();
+           if (services.keepRight && _qaItem instanceof QAItem) {
+             url = services.keepRight.issueURL(_qaItem);
            }
+
+           var link = selection.selectAll('.view-on-keepRight').data(url ? [url] : []); // exit
+
+           link.exit().remove(); // enter
+
+           var linkEnter = link.enter().append('a').attr('class', 'view-on-keepRight').attr('target', '_blank').attr('rel', 'noopener') // security measure
+           .attr('href', function (d) {
+             return d;
+           }).call(svgIcon('#iD-icon-out-link', 'inline'));
+           linkEnter.append('span').html(_t.html('inspector.view_on_keepRight'));
          }
 
-         function rightClickHouse() {
-           if (!_houseID) return chapter.restart();
-           context.enter(modeBrowse(context));
-           context.history().reset('hasHouse');
-           var zoom = context.map().zoom();
+         viewOnKeepRight.what = function (val) {
+           if (!arguments.length) return _qaItem;
+           _qaItem = val;
+           return viewOnKeepRight;
+         };
 
-           if (zoom < 20) {
-             zoom = 20;
-           }
+         return viewOnKeepRight;
+       }
 
-           context.map().centerZoomEase(house, zoom, 500);
-           context.on('enter.intro', function (mode) {
-             if (mode.id !== 'select') return;
-             var ids = context.selectedIDs();
-             if (ids.length !== 1 || ids[0] !== _houseID) return;
-             timeout(function () {
-               var node = selectMenuItem(context, 'orthogonalize').node();
-               if (!node) return;
-               continueTo(clickSquare);
-             }, 50); // after menu visible
-           });
-           context.map().on('move.intro drawn.intro', function () {
-             var rightclickString = helpHtml('intro.buildings.' + (context.lastPointerType() === 'mouse' ? 'rightclick_building' : 'edit_menu_building_touch'));
-             revealHouse(house, rightclickString, {
-               duration: 0
-             });
-           });
-           context.history().on('change.intro', function () {
-             continueTo(rightClickHouse);
-           });
+       function uiKeepRightEditor(context) {
+         var dispatch = dispatch$8('change');
+         var qaDetails = uiKeepRightDetails(context);
+         var qaHeader = uiKeepRightHeader();
 
-           function continueTo(nextStep) {
-             context.on('enter.intro', null);
-             context.map().on('move.intro drawn.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
-           }
+         var _qaItem;
+
+         function keepRightEditor(selection) {
+           var headerEnter = selection.selectAll('.header').data([0]).enter().append('div').attr('class', 'header fillL');
+           headerEnter.append('button').attr('class', 'close').on('click', function () {
+             return context.enter(modeBrowse(context));
+           }).call(svgIcon('#iD-icon-close'));
+           headerEnter.append('h3').html(_t.html('QA.keepRight.title'));
+           var body = selection.selectAll('.body').data([0]);
+           body = body.enter().append('div').attr('class', 'body').merge(body);
+           var editor = body.selectAll('.qa-editor').data([0]);
+           editor.enter().append('div').attr('class', 'modal-section qa-editor').merge(editor).call(qaHeader.issue(_qaItem)).call(qaDetails.issue(_qaItem)).call(keepRightSaveSection);
+           var footer = selection.selectAll('.footer').data([0]);
+           footer.enter().append('div').attr('class', 'footer').merge(footer).call(uiViewOnKeepRight().what(_qaItem));
          }
 
-         function clickSquare() {
-           if (!_houseID) return chapter.restart();
-           var entity = context.hasEntity(_houseID);
-           if (!entity) return continueTo(rightClickHouse);
-           var node = selectMenuItem(context, 'orthogonalize').node();
+         function keepRightSaveSection(selection) {
+           var isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
 
-           if (!node) {
-             return continueTo(rightClickHouse);
-           }
+           var isShown = _qaItem && (isSelected || _qaItem.newComment || _qaItem.comment);
+           var saveSection = selection.selectAll('.qa-save').data(isShown ? [_qaItem] : [], function (d) {
+             return "".concat(d.id, "-").concat(d.status || 0);
+           }); // exit
 
-           var wasChanged = false;
-           reveal('.edit-menu', helpHtml('intro.buildings.square_building'), {
-             padding: 50
-           });
-           context.on('enter.intro', function (mode) {
-             if (mode.id === 'browse') {
-               continueTo(rightClickHouse);
-             } else if (mode.id === 'move' || mode.id === 'rotate') {
-               continueTo(retryClickSquare);
-             }
-           });
-           context.map().on('move.intro', function () {
-             var node = selectMenuItem(context, 'orthogonalize').node();
+           saveSection.exit().remove(); // enter
 
-             if (!wasChanged && !node) {
-               return continueTo(rightClickHouse);
-             }
+           var saveSectionEnter = saveSection.enter().append('div').attr('class', 'qa-save save-section cf');
+           saveSectionEnter.append('h4').attr('class', '.qa-save-header').html(_t.html('QA.keepRight.comment'));
+           saveSectionEnter.append('textarea').attr('class', 'new-comment-input').attr('placeholder', _t('QA.keepRight.comment_placeholder')).attr('maxlength', 1000).property('value', function (d) {
+             return d.newComment || d.comment;
+           }).call(utilNoAuto).on('input', changeInput).on('blur', changeInput); // update
 
-             reveal('.edit-menu', helpHtml('intro.buildings.square_building'), {
-               duration: 0,
-               padding: 50
-             });
-           });
-           context.history().on('change.intro', function () {
-             wasChanged = true;
-             context.history().on('change.intro', null); // Something changed.  Wait for transition to complete and check undo annotation.
+           saveSection = saveSectionEnter.merge(saveSection).call(qaSaveButtons);
 
-             timeout(function () {
-               if (context.history().undoAnnotation() === _t('operations.orthogonalize.annotation.feature', {
-                 n: 1
-               })) {
-                 continueTo(doneSquare);
-               } else {
-                 continueTo(retryClickSquare);
-               }
-             }, 500); // after transitioned actions
-           });
+           function changeInput() {
+             var input = select(this);
+             var val = input.property('value').trim();
 
-           function continueTo(nextStep) {
-             context.on('enter.intro', null);
-             context.map().on('move.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
-           }
-         }
+             if (val === _qaItem.comment) {
+               val = undefined;
+             } // store the unsaved comment with the issue itself
 
-         function retryClickSquare() {
-           context.enter(modeBrowse(context));
-           revealHouse(house, helpHtml('intro.buildings.retry_square'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: function buttonCallback() {
-               continueTo(rightClickHouse);
-             }
-           });
 
-           function continueTo(nextStep) {
-             nextStep();
-           }
-         }
+             _qaItem = _qaItem.update({
+               newComment: val
+             });
+             var qaService = services.keepRight;
 
-         function doneSquare() {
-           context.history().checkpoint('doneSquare');
-           revealHouse(house, helpHtml('intro.buildings.done_square'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: function buttonCallback() {
-               continueTo(addTank);
+             if (qaService) {
+               qaService.replaceItem(_qaItem); // update keepright cache
              }
-           });
 
-           function continueTo(nextStep) {
-             nextStep();
+             saveSection.call(qaSaveButtons);
            }
          }
 
-         function addTank() {
-           context.enter(modeBrowse(context));
-           context.history().reset('doneSquare');
-           _tankID = null;
-           var msec = transitionTime(tank, context.map().center());
+         function qaSaveButtons(selection) {
+           var isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
 
-           if (msec) {
-             reveal(null, null, {
-               duration: 0
-             });
-           }
+           var buttonSection = selection.selectAll('.buttons').data(isSelected ? [_qaItem] : [], function (d) {
+             return d.status + d.id;
+           }); // exit
 
-           context.map().centerZoomEase(tank, 19.5, msec);
-           timeout(function () {
-             reveal('button.add-area', helpHtml('intro.buildings.add_tank'));
-             context.on('enter.intro', function (mode) {
-               if (mode.id !== 'add-area') return;
-               continueTo(startTank);
-             });
-           }, msec + 100);
+           buttonSection.exit().remove(); // enter
 
-           function continueTo(nextStep) {
-             context.on('enter.intro', null);
-             nextStep();
-           }
-         }
+           var buttonEnter = buttonSection.enter().append('div').attr('class', 'buttons');
+           buttonEnter.append('button').attr('class', 'button comment-button action').html(_t.html('QA.keepRight.save_comment'));
+           buttonEnter.append('button').attr('class', 'button close-button action');
+           buttonEnter.append('button').attr('class', 'button ignore-button action'); // update
 
-         function startTank() {
-           if (context.mode().id !== 'add-area') {
-             return continueTo(addTank);
-           }
+           buttonSection = buttonSection.merge(buttonEnter);
+           buttonSection.select('.comment-button') // select and propagate data
+           .attr('disabled', function (d) {
+             return d.newComment ? null : true;
+           }).on('click.comment', function (d3_event, d) {
+             this.blur(); // avoid keeping focus on the button - #4641
 
-           _tankID = null;
-           timeout(function () {
-             var startString = helpHtml('intro.buildings.start_tank') + helpHtml('intro.buildings.tank_edge_' + (context.lastPointerType() === 'mouse' ? 'click' : 'tap'));
-             revealTank(tank, startString);
-             context.map().on('move.intro drawn.intro', function () {
-               revealTank(tank, startString, {
-                 duration: 0
+             var qaService = services.keepRight;
+
+             if (qaService) {
+               qaService.postUpdate(d, function (err, item) {
+                 return dispatch.call('change', item);
                });
-             });
-             context.on('enter.intro', function (mode) {
-               if (mode.id !== 'draw-area') return chapter.restart();
-               continueTo(continueTank);
-             });
-           }, 550); // after easing
+             }
+           });
+           buttonSection.select('.close-button') // select and propagate data
+           .html(function (d) {
+             var andComment = d.newComment ? '_comment' : '';
+             return _t.html("QA.keepRight.close".concat(andComment));
+           }).on('click.close', function (d3_event, d) {
+             this.blur(); // avoid keeping focus on the button - #4641
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
-           }
-         }
+             var qaService = services.keepRight;
 
-         function continueTank() {
-           if (context.mode().id !== 'draw-area') {
-             return continueTo(addTank);
-           }
+             if (qaService) {
+               d.newStatus = 'ignore_t'; // ignore temporarily (item fixed)
 
-           _tankID = null;
-           var continueString = helpHtml('intro.buildings.continue_tank') + '{br}' + helpHtml('intro.areas.finish_area_' + (context.lastPointerType() === 'mouse' ? 'click' : 'tap')) + helpHtml('intro.buildings.finish_tank');
-           revealTank(tank, continueString);
-           context.map().on('move.intro drawn.intro', function () {
-             revealTank(tank, continueString, {
-               duration: 0
-             });
-           });
-           context.on('enter.intro', function (mode) {
-             if (mode.id === 'draw-area') {
-               return;
-             } else if (mode.id === 'select') {
-               _tankID = context.selectedIDs()[0];
-               return continueTo(searchPresetTank);
-             } else {
-               return continueTo(addTank);
+               qaService.postUpdate(d, function (err, item) {
+                 return dispatch.call('change', item);
+               });
              }
            });
+           buttonSection.select('.ignore-button') // select and propagate data
+           .html(function (d) {
+             var andComment = d.newComment ? '_comment' : '';
+             return _t.html("QA.keepRight.ignore".concat(andComment));
+           }).on('click.ignore', function (d3_event, d) {
+             this.blur(); // avoid keeping focus on the button - #4641
 
-           function continueTo(nextStep) {
-             context.map().on('move.intro drawn.intro', null);
-             context.on('enter.intro', null);
-             nextStep();
-           }
-         }
-
-         function searchPresetTank() {
-           if (!_tankID || !context.hasEntity(_tankID)) {
-             return addTank();
-           }
+             var qaService = services.keepRight;
 
-           var ids = context.selectedIDs();
+             if (qaService) {
+               d.newStatus = 'ignore'; // ignore permanently (false positive)
 
-           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _tankID) {
-             context.enter(modeSelect(context, [_tankID]));
-           } // disallow scrolling
+               qaService.postUpdate(d, function (err, item) {
+                 return dispatch.call('change', item);
+               });
+             }
+           });
+         } // NOTE: Don't change method name until UI v3 is merged
 
 
-           context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
-           timeout(function () {
-             // reset pane, in case user somehow happened to change it..
-             context.container().select('.inspector-wrap .panewrap').style('right', '-100%');
-             context.container().select('.preset-search-input').on('keydown.intro', null).on('keyup.intro', checkPresetSearch);
-             reveal('.preset-search-input', helpHtml('intro.buildings.search_tank', {
-               preset: tankPreset.name()
-             }));
-           }, 400); // after preset list pane visible..
+         keepRightEditor.error = function (val) {
+           if (!arguments.length) return _qaItem;
+           _qaItem = val;
+           return keepRightEditor;
+         };
 
-           context.on('enter.intro', function (mode) {
-             if (!_tankID || !context.hasEntity(_tankID)) {
-               return continueTo(addTank);
-             }
+         return utilRebind(keepRightEditor, dispatch, 'on');
+       }
 
-             var ids = context.selectedIDs();
+       function uiOsmoseDetails(context) {
+         var _qaItem;
 
-             if (mode.id !== 'select' || !ids.length || ids[0] !== _tankID) {
-               // keep the user's area selected..
-               context.enter(modeSelect(context, [_tankID])); // reset pane, in case user somehow happened to change it..
+         function issueString(d, type) {
+           if (!d) return ''; // Issue strings are cached from Osmose API
 
-               context.container().select('.inspector-wrap .panewrap').style('right', '-100%'); // disallow scrolling
+           var s = services.osmose.getStrings(d.itemType);
+           return type in s ? s[type] : '';
+         }
 
-               context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
-               context.container().select('.preset-search-input').on('keydown.intro', null).on('keyup.intro', checkPresetSearch);
-               reveal('.preset-search-input', helpHtml('intro.buildings.search_tank', {
-                 preset: tankPreset.name()
-               }));
-               context.history().on('change.intro', null);
-             }
+         function osmoseDetails(selection) {
+           var details = selection.selectAll('.error-details').data(_qaItem ? [_qaItem] : [], function (d) {
+             return "".concat(d.id, "-").concat(d.status || 0);
            });
+           details.exit().remove();
+           var detailsEnter = details.enter().append('div').attr('class', 'error-details qa-details-container'); // Description
 
-           function checkPresetSearch() {
-             var first = context.container().select('.preset-list-item:first-child');
+           if (issueString(_qaItem, 'detail')) {
+             var div = detailsEnter.append('div').attr('class', 'qa-details-subsection');
+             div.append('h4').html(_t.html('QA.keepRight.detail_description'));
+             div.append('p').attr('class', 'qa-details-description-text').html(function (d) {
+               return issueString(d, 'detail');
+             }).selectAll('a').attr('rel', 'noopener').attr('target', '_blank');
+           } // Elements (populated later as data is requested)
 
-             if (first.classed('preset-man_made-storage_tank')) {
-               reveal(first.select('.preset-list-button').node(), helpHtml('intro.buildings.choose_tank', {
-                 preset: tankPreset.name()
-               }), {
-                 duration: 300
-               });
-               context.container().select('.preset-search-input').on('keydown.intro', eventCancel, true).on('keyup.intro', null);
-               context.history().on('change.intro', function () {
-                 continueTo(closeEditorTank);
-               });
-             }
-           }
 
-           function continueTo(nextStep) {
-             context.container().select('.inspector-wrap').on('wheel.intro', null);
-             context.on('enter.intro', null);
-             context.history().on('change.intro', null);
-             context.container().select('.preset-search-input').on('keydown.intro keyup.intro', null);
-             nextStep();
-           }
-         }
+           var detailsDiv = detailsEnter.append('div').attr('class', 'qa-details-subsection');
+           var elemsDiv = detailsEnter.append('div').attr('class', 'qa-details-subsection'); // Suggested Fix (mustn't exist for every issue type)
 
-         function closeEditorTank() {
-           if (!_tankID || !context.hasEntity(_tankID)) {
-             return addTank();
-           }
+           if (issueString(_qaItem, 'fix')) {
+             var _div = detailsEnter.append('div').attr('class', 'qa-details-subsection');
 
-           var ids = context.selectedIDs();
+             _div.append('h4').html(_t.html('QA.osmose.fix_title'));
 
-           if (context.mode().id !== 'select' || !ids.length || ids[0] !== _tankID) {
-             context.enter(modeSelect(context, [_tankID]));
-           }
+             _div.append('p').html(function (d) {
+               return issueString(d, 'fix');
+             }).selectAll('a').attr('rel', 'noopener').attr('target', '_blank');
+           } // Common Pitfalls (mustn't exist for every issue type)
 
-           context.history().checkpoint('hasTank');
-           context.on('exit.intro', function () {
-             continueTo(rightClickTank);
-           });
-           timeout(function () {
-             reveal('.entity-editor-pane', helpHtml('intro.buildings.close', {
-               button: icon('#iD-icon-close', 'inline')
-             }));
-           }, 500);
 
-           function continueTo(nextStep) {
-             context.on('exit.intro', null);
-             nextStep();
-           }
-         }
+           if (issueString(_qaItem, 'trap')) {
+             var _div2 = detailsEnter.append('div').attr('class', 'qa-details-subsection');
+
+             _div2.append('h4').html(_t.html('QA.osmose.trap_title'));
+
+             _div2.append('p').html(function (d) {
+               return issueString(d, 'trap');
+             }).selectAll('a').attr('rel', 'noopener').attr('target', '_blank');
+           } // Save current item to check if UI changed by time request resolves
+
 
-         function rightClickTank() {
-           if (!_tankID) return continueTo(addTank);
-           context.enter(modeBrowse(context));
-           context.history().reset('hasTank');
-           context.map().centerEase(tank, 500);
-           timeout(function () {
-             context.on('enter.intro', function (mode) {
-               if (mode.id !== 'select') return;
-               var ids = context.selectedIDs();
-               if (ids.length !== 1 || ids[0] !== _tankID) return;
-               timeout(function () {
-                 var node = selectMenuItem(context, 'circularize').node();
-                 if (!node) return;
-                 continueTo(clickCircle);
-               }, 50); // after menu visible
-             });
-             var rightclickString = helpHtml('intro.buildings.' + (context.lastPointerType() === 'mouse' ? 'rightclick_tank' : 'edit_menu_tank_touch'));
-             revealTank(tank, rightclickString);
-             context.map().on('move.intro drawn.intro', function () {
-               revealTank(tank, rightclickString, {
-                 duration: 0
-               });
-             });
-             context.history().on('change.intro', function () {
-               continueTo(rightClickTank);
-             });
-           }, 600);
+           var thisItem = _qaItem;
+           services.osmose.loadIssueDetail(_qaItem).then(function (d) {
+             // No details to add if there are no associated issue elements
+             if (!d.elems || d.elems.length === 0) return; // Do nothing if UI has moved on by the time this resolves
 
-           function continueTo(nextStep) {
-             context.on('enter.intro', null);
-             context.map().on('move.intro drawn.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
-           }
-         }
+             if (context.selectedErrorID() !== thisItem.id && context.container().selectAll(".qaItem.osmose.hover.itemId-".concat(thisItem.id)).empty()) return; // Things like keys and values are dynamically added to a subtitle string
 
-         function clickCircle() {
-           if (!_tankID) return chapter.restart();
-           var entity = context.hasEntity(_tankID);
-           if (!entity) return continueTo(rightClickTank);
-           var node = selectMenuItem(context, 'circularize').node();
+             if (d.detail) {
+               detailsDiv.append('h4').html(_t.html('QA.osmose.detail_title'));
+               detailsDiv.append('p').html(function (d) {
+                 return d.detail;
+               }).selectAll('a').attr('rel', 'noopener').attr('target', '_blank');
+             } // Create list of linked issue elements
 
-           if (!node) {
-             return continueTo(rightClickTank);
-           }
 
-           var wasChanged = false;
-           reveal('.edit-menu', helpHtml('intro.buildings.circle_tank'), {
-             padding: 50
-           });
-           context.on('enter.intro', function (mode) {
-             if (mode.id === 'browse') {
-               continueTo(rightClickTank);
-             } else if (mode.id === 'move' || mode.id === 'rotate') {
-               continueTo(retryClickCircle);
-             }
-           });
-           context.map().on('move.intro', function () {
-             var node = selectMenuItem(context, 'circularize').node();
+             elemsDiv.append('h4').html(_t.html('QA.osmose.elems_title'));
+             elemsDiv.append('ul').selectAll('li').data(d.elems).enter().append('li').append('a').attr('href', '#').attr('class', 'error_entity_link').html(function (d) {
+               return d;
+             }).each(function () {
+               var link = select(this);
+               var entityID = this.textContent;
+               var entity = context.hasEntity(entityID); // Add click handler
 
-             if (!wasChanged && !node) {
-               return continueTo(rightClickTank);
-             }
+               link.on('mouseenter', function () {
+                 utilHighlightEntities([entityID], true, context);
+               }).on('mouseleave', function () {
+                 utilHighlightEntities([entityID], false, context);
+               }).on('click', function (d3_event) {
+                 d3_event.preventDefault();
+                 utilHighlightEntities([entityID], false, context);
+                 var osmlayer = context.layers().layer('osm');
 
-             reveal('.edit-menu', helpHtml('intro.buildings.circle_tank'), {
-               duration: 0,
-               padding: 50
-             });
-           });
-           context.history().on('change.intro', function () {
-             wasChanged = true;
-             context.history().on('change.intro', null); // Something changed.  Wait for transition to complete and check undo annotation.
+                 if (!osmlayer.enabled()) {
+                   osmlayer.enabled(true);
+                 }
 
-             timeout(function () {
-               if (context.history().undoAnnotation() === _t('operations.circularize.annotation.feature', {
-                 n: 1
-               })) {
-                 continueTo(play);
-               } else {
-                 continueTo(retryClickCircle);
-               }
-             }, 500); // after transitioned actions
-           });
+                 context.map().centerZoom(d.loc, 20);
 
-           function continueTo(nextStep) {
-             context.on('enter.intro', null);
-             context.map().on('move.intro', null);
-             context.history().on('change.intro', null);
-             nextStep();
-           }
-         }
+                 if (entity) {
+                   context.enter(modeSelect(context, [entityID]));
+                 } else {
+                   context.loadEntity(entityID, function (err, result) {
+                     if (err) return;
+                     var entity = result.data.find(function (e) {
+                       return e.id === entityID;
+                     });
+                     if (entity) context.enter(modeSelect(context, [entityID]));
+                   });
+                 }
+               }); // Replace with friendly name if possible
+               // (The entity may not yet be loaded into the graph)
 
-         function retryClickCircle() {
-           context.enter(modeBrowse(context));
-           revealTank(tank, helpHtml('intro.buildings.retry_circle'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: function buttonCallback() {
-               continueTo(rightClickTank);
-             }
-           });
+               if (entity) {
+                 var name = utilDisplayName(entity); // try to use common name
 
-           function continueTo(nextStep) {
-             nextStep();
-           }
-         }
+                 if (!name) {
+                   var preset = _mainPresetIndex.match(entity, context.graph());
+                   name = preset && !preset.isFallback() && preset.name(); // fallback to preset name
+                 }
 
-         function play() {
-           dispatch$1.call('done');
-           reveal('.ideditor', helpHtml('intro.buildings.play', {
-             next: _t('intro.startediting.title')
-           }), {
-             tooltipBox: '.intro-nav-wrap .chapter-startEditing',
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: function buttonCallback() {
-               reveal('.ideditor');
-             }
+                 if (name) {
+                   this.innerText = name;
+                 }
+               }
+             }); // Don't hide entities related to this issue - #5880
+
+             context.features().forceVisible(d.elems);
+             context.map().pan([0, 0]); // trigger a redraw
+           })["catch"](function (err) {
+             console.log(err); // eslint-disable-line no-console
            });
          }
 
-         chapter.enter = function () {
-           addHouse();
+         osmoseDetails.issue = function (val) {
+           if (!arguments.length) return _qaItem;
+           _qaItem = val;
+           return osmoseDetails;
          };
 
-         chapter.exit = function () {
-           timeouts.forEach(window.clearTimeout);
-           context.on('enter.intro exit.intro', null);
-           context.map().on('move.intro drawn.intro', null);
-           context.history().on('change.intro', null);
-           context.container().select('.inspector-wrap').on('wheel.intro', null);
-           context.container().select('.preset-search-input').on('keydown.intro keyup.intro', null);
-           context.container().select('.more-fields .combobox-input').on('click.intro', null);
-         };
+         return osmoseDetails;
+       }
 
-         chapter.restart = function () {
-           chapter.exit();
-           chapter.enter();
-         };
+       function uiOsmoseHeader() {
+         var _qaItem;
 
-         return utilRebind(chapter, dispatch$1, 'on');
-       }
+         function issueTitle(d) {
+           var unknown = _t('inspector.unknown');
+           if (!d) return unknown; // Issue titles supplied by Osmose
 
-       function uiIntroStartEditing(context, reveal) {
-         var dispatch$1 = dispatch('done', 'startEditing');
-         var modalSelection = select(null);
-         var chapter = {
-           title: 'intro.startediting.title'
-         };
+           var s = services.osmose.getStrings(d.itemType);
+           return 'title' in s ? s.title : unknown;
+         }
 
-         function showHelp() {
-           reveal('.map-control.help-control', helpHtml('intro.startediting.help'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: function buttonCallback() {
-               shortcuts();
-             }
+         function osmoseHeader(selection) {
+           var header = selection.selectAll('.qa-header').data(_qaItem ? [_qaItem] : [], function (d) {
+             return "".concat(d.id, "-").concat(d.status || 0);
            });
-         }
+           header.exit().remove();
+           var headerEnter = header.enter().append('div').attr('class', 'qa-header');
+           var svgEnter = headerEnter.append('div').attr('class', 'qa-header-icon').classed('new', function (d) {
+             return d.id < 0;
+           }).append('svg').attr('width', '20px').attr('height', '30px').attr('viewbox', '0 0 20 30').attr('class', function (d) {
+             return "preset-icon-28 qaItem ".concat(d.service, " itemId-").concat(d.id, " itemType-").concat(d.itemType);
+           });
+           svgEnter.append('polygon').attr('fill', function (d) {
+             return services.osmose.getColor(d.item);
+           }).attr('class', 'qaItem-fill').attr('points', '16,3 4,3 1,6 1,17 4,20 7,20 10,27 13,20 16,20 19,17.033 19,6');
+           svgEnter.append('use').attr('class', 'icon-annotation').attr('width', '13px').attr('height', '13px').attr('transform', 'translate(3.5, 5)').attr('xlink:href', function (d) {
+             var picon = d.icon;
 
-         function shortcuts() {
-           reveal('.map-control.help-control', helpHtml('intro.startediting.shortcuts'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: function buttonCallback() {
-               showSave();
+             if (!picon) {
+               return '';
+             } else {
+               var isMaki = /^maki-/.test(picon);
+               return "#".concat(picon).concat(isMaki ? '-11' : '');
              }
            });
+           headerEnter.append('div').attr('class', 'qa-header-label').html(issueTitle);
          }
 
-         function showSave() {
-           context.container().selectAll('.shaded').remove(); // in case user opened keyboard shortcuts
+         osmoseHeader.issue = function (val) {
+           if (!arguments.length) return _qaItem;
+           _qaItem = val;
+           return osmoseHeader;
+         };
 
-           reveal('.top-toolbar button.save', helpHtml('intro.startediting.save'), {
-             buttonText: _t.html('intro.ok'),
-             buttonCallback: function buttonCallback() {
-               showStart();
-             }
-           });
-         }
+         return osmoseHeader;
+       }
 
-         function showStart() {
-           context.container().selectAll('.shaded').remove(); // in case user opened keyboard shortcuts
+       function uiViewOnOsmose() {
+         var _qaItem;
 
-           modalSelection = uiModal(context.container());
-           modalSelection.select('.modal').attr('class', 'modal-splash modal');
-           modalSelection.selectAll('.close').remove();
-           var startbutton = modalSelection.select('.content').attr('class', 'fillL').append('button').attr('class', 'modal-section huge-modal-button').on('click', function () {
-             modalSelection.remove();
-           });
-           startbutton.append('svg').attr('class', 'illustration').append('use').attr('xlink:href', '#iD-logo-walkthrough');
-           startbutton.append('h2').html(_t.html('intro.startediting.start'));
-           dispatch$1.call('startEditing');
-         }
+         function viewOnOsmose(selection) {
+           var url;
 
-         chapter.enter = function () {
-           showHelp();
-         };
+           if (services.osmose && _qaItem instanceof QAItem) {
+             url = services.osmose.itemURL(_qaItem);
+           }
 
-         chapter.exit = function () {
-           modalSelection.remove();
-           context.container().selectAll('.shaded').remove(); // in case user opened keyboard shortcuts
+           var link = selection.selectAll('.view-on-osmose').data(url ? [url] : []); // exit
+
+           link.exit().remove(); // enter
+
+           var linkEnter = link.enter().append('a').attr('class', 'view-on-osmose').attr('target', '_blank').attr('rel', 'noopener') // security measure
+           .attr('href', function (d) {
+             return d;
+           }).call(svgIcon('#iD-icon-out-link', 'inline'));
+           linkEnter.append('span').html(_t.html('inspector.view_on_osmose'));
+         }
+
+         viewOnOsmose.what = function (val) {
+           if (!arguments.length) return _qaItem;
+           _qaItem = val;
+           return viewOnOsmose;
          };
 
-         return utilRebind(chapter, dispatch$1, 'on');
+         return viewOnOsmose;
        }
 
-       var chapterUi = {
-         welcome: uiIntroWelcome,
-         navigation: uiIntroNavigation,
-         point: uiIntroPoint,
-         area: uiIntroArea,
-         line: uiIntroLine,
-         building: uiIntroBuilding,
-         startEditing: uiIntroStartEditing
-       };
-       var chapterFlow = ['welcome', 'navigation', 'point', 'area', 'line', 'building', 'startEditing'];
-       function uiIntro(context) {
-         var INTRO_IMAGERY = 'EsriWorldImageryClarity';
-         var _introGraph = {};
-
-         var _currChapter;
+       function uiOsmoseEditor(context) {
+         var dispatch = dispatch$8('change');
+         var qaDetails = uiOsmoseDetails(context);
+         var qaHeader = uiOsmoseHeader();
 
-         function intro(selection) {
-           _mainFileFetcher.get('intro_graph').then(function (dataIntroGraph) {
-             // create entities for intro graph and localize names
-             for (var id in dataIntroGraph) {
-               if (!_introGraph[id]) {
-                 _introGraph[id] = osmEntity(localize(dataIntroGraph[id]));
-               }
-             }
+         var _qaItem;
 
-             selection.call(startIntro);
-           })["catch"](function () {
-             /* ignore */
-           });
+         function osmoseEditor(selection) {
+           var header = selection.selectAll('.header').data([0]);
+           var headerEnter = header.enter().append('div').attr('class', 'header fillL');
+           headerEnter.append('button').attr('class', 'close').on('click', function () {
+             return context.enter(modeBrowse(context));
+           }).call(svgIcon('#iD-icon-close'));
+           headerEnter.append('h3').html(_t.html('QA.osmose.title'));
+           var body = selection.selectAll('.body').data([0]);
+           body = body.enter().append('div').attr('class', 'body').merge(body);
+           var editor = body.selectAll('.qa-editor').data([0]);
+           editor.enter().append('div').attr('class', 'modal-section qa-editor').merge(editor).call(qaHeader.issue(_qaItem)).call(qaDetails.issue(_qaItem)).call(osmoseSaveSection);
+           var footer = selection.selectAll('.footer').data([0]);
+           footer.enter().append('div').attr('class', 'footer').merge(footer).call(uiViewOnOsmose().what(_qaItem));
          }
 
-         function startIntro(selection) {
-           context.enter(modeBrowse(context)); // Save current map state
+         function osmoseSaveSection(selection) {
+           var isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
 
-           var osm = context.connection();
-           var history = context.history().toJSON();
-           var hash = window.location.hash;
-           var center = context.map().center();
-           var zoom = context.map().zoom();
-           var background = context.background().baseLayerSource();
-           var overlays = context.background().overlayLayerSources();
-           var opacity = context.container().selectAll('.main-map .layer-background').style('opacity');
-           var caches = osm && osm.caches();
-           var baseEntities = context.history().graph().base().entities; // Show sidebar and disable the sidebar resizing button
-           // (this needs to be before `context.inIntro(true)`)
+           var isShown = _qaItem && isSelected;
+           var saveSection = selection.selectAll('.qa-save').data(isShown ? [_qaItem] : [], function (d) {
+             return "".concat(d.id, "-").concat(d.status || 0);
+           }); // exit
 
-           context.ui().sidebar.expand();
-           context.container().selectAll('button.sidebar-toggle').classed('disabled', true); // Block saving
+           saveSection.exit().remove(); // enter
 
-           context.inIntro(true); // Load semi-real data used in intro
+           var saveSectionEnter = saveSection.enter().append('div').attr('class', 'qa-save save-section cf'); // update
 
-           if (osm) {
-             osm.toggle(false).reset();
-           }
+           saveSection = saveSectionEnter.merge(saveSection).call(qaSaveButtons);
+         }
 
-           context.history().reset();
-           context.history().merge(Object.values(coreGraph().load(_introGraph).entities));
-           context.history().checkpoint('initial'); // Setup imagery
+         function qaSaveButtons(selection) {
+           var isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
 
-           var imagery = context.background().findSource(INTRO_IMAGERY);
+           var buttonSection = selection.selectAll('.buttons').data(isSelected ? [_qaItem] : [], function (d) {
+             return d.status + d.id;
+           }); // exit
 
-           if (imagery) {
-             context.background().baseLayerSource(imagery);
-           } else {
-             context.background().bing();
-           }
+           buttonSection.exit().remove(); // enter
 
-           overlays.forEach(function (d) {
-             return context.background().toggleOverlayLayer(d);
-           }); // Setup data layers (only OSM)
+           var buttonEnter = buttonSection.enter().append('div').attr('class', 'buttons');
+           buttonEnter.append('button').attr('class', 'button close-button action');
+           buttonEnter.append('button').attr('class', 'button ignore-button action'); // update
 
-           var layers = context.layers();
-           layers.all().forEach(function (item) {
-             // if the layer has the function `enabled`
-             if (typeof item.layer.enabled === 'function') {
-               item.layer.enabled(item.id === 'osm');
+           buttonSection = buttonSection.merge(buttonEnter);
+           buttonSection.select('.close-button').html(_t.html('QA.keepRight.close')).on('click.close', function (d3_event, d) {
+             this.blur(); // avoid keeping focus on the button - #4641
+
+             var qaService = services.osmose;
+
+             if (qaService) {
+               d.newStatus = 'done';
+               qaService.postUpdate(d, function (err, item) {
+                 return dispatch.call('change', item);
+               });
              }
            });
-           context.container().selectAll('.main-map .layer-background').style('opacity', 1);
-           var curtain = uiCurtain(context.container().node());
-           selection.call(curtain); // Store that the user started the walkthrough..
+           buttonSection.select('.ignore-button').html(_t.html('QA.keepRight.ignore')).on('click.ignore', function (d3_event, d) {
+             this.blur(); // avoid keeping focus on the button - #4641
 
-           corePreferences('walkthrough_started', 'yes'); // Restore previous walkthrough progress..
+             var qaService = services.osmose;
 
-           var storedProgress = corePreferences('walkthrough_progress') || '';
-           var progress = storedProgress.split(';').filter(Boolean);
-           var chapters = chapterFlow.map(function (chapter, i) {
-             var s = chapterUi[chapter](context, curtain.reveal).on('done', function () {
-               buttons.filter(function (d) {
-                 return d.title === s.title;
-               }).classed('finished', true);
+             if (qaService) {
+               d.newStatus = 'false';
+               qaService.postUpdate(d, function (err, item) {
+                 return dispatch.call('change', item);
+               });
+             }
+           });
+         } // NOTE: Don't change method name until UI v3 is merged
 
-               if (i < chapterFlow.length - 1) {
-                 var next = chapterFlow[i + 1];
-                 context.container().select("button.chapter-".concat(next)).classed('next', true);
-               } // Store walkthrough progress..
 
+         osmoseEditor.error = function (val) {
+           if (!arguments.length) return _qaItem;
+           _qaItem = val;
+           return osmoseEditor;
+         };
 
-               progress.push(chapter);
-               corePreferences('walkthrough_progress', utilArrayUniq(progress).join(';'));
-             });
-             return s;
-           });
-           chapters[chapters.length - 1].on('startEditing', function () {
-             // Store walkthrough progress..
-             progress.push('startEditing');
-             corePreferences('walkthrough_progress', utilArrayUniq(progress).join(';')); // Store if walkthrough is completed..
+         return utilRebind(osmoseEditor, dispatch, 'on');
+       }
 
-             var incomplete = utilArrayDifference(chapterFlow, progress);
+       function uiNoteComments() {
+         var _note;
 
-             if (!incomplete.length) {
-               corePreferences('walkthrough_completed', 'yes');
-             }
+         function noteComments(selection) {
+           if (_note.isNew()) return; // don't draw .comments-container
 
-             curtain.remove();
-             navwrap.remove();
-             context.container().selectAll('.main-map .layer-background').style('opacity', opacity);
-             context.container().selectAll('button.sidebar-toggle').classed('disabled', false);
+           var comments = selection.selectAll('.comments-container').data([0]);
+           comments = comments.enter().append('div').attr('class', 'comments-container').merge(comments);
+           var commentEnter = comments.selectAll('.comment').data(_note.comments).enter().append('div').attr('class', 'comment');
+           commentEnter.append('div').attr('class', function (d) {
+             return 'comment-avatar user-' + d.uid;
+           }).call(svgIcon('#iD-icon-avatar', 'comment-avatar-icon'));
+           var mainEnter = commentEnter.append('div').attr('class', 'comment-main');
+           var metadataEnter = mainEnter.append('div').attr('class', 'comment-metadata');
+           metadataEnter.append('div').attr('class', 'comment-author').each(function (d) {
+             var selection = select(this);
+             var osm = services.osm;
 
-             if (osm) {
-               osm.toggle(true).reset().caches(caches);
+             if (osm && d.user) {
+               selection = selection.append('a').attr('class', 'comment-author-link').attr('href', osm.userURL(d.user)).attr('target', '_blank');
              }
 
-             context.history().reset().merge(Object.values(baseEntities));
-             context.background().baseLayerSource(background);
-             overlays.forEach(function (d) {
-               return context.background().toggleOverlayLayer(d);
+             selection.html(function (d) {
+               return d.user || _t.html('note.anonymous');
              });
-
-             if (history) {
-               context.history().fromJSON(history, false);
-             }
-
-             context.map().centerZoom(center, zoom);
-             window.location.replace(hash);
-             context.inIntro(false);
            });
-           var navwrap = selection.append('div').attr('class', 'intro-nav-wrap fillD');
-           navwrap.append('svg').attr('class', 'intro-nav-wrap-logo').append('use').attr('xlink:href', '#iD-logo-walkthrough');
-           var buttonwrap = navwrap.append('div').attr('class', 'joined').selectAll('button.chapter');
-           var buttons = buttonwrap.data(chapters).enter().append('button').attr('class', function (d, i) {
-             return "chapter chapter-".concat(chapterFlow[i]);
-           }).on('click', enterChapter);
-           buttons.append('span').html(function (d) {
-             return _t.html(d.title);
+           metadataEnter.append('div').attr('class', 'comment-date').html(function (d) {
+             return _t('note.status.' + d.action, {
+               when: localeDateString(d.date)
+             });
            });
-           buttons.append('span').attr('class', 'status').call(svgIcon(_mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-backward' : '#iD-icon-forward', 'inline'));
-           enterChapter(null, chapters[0]);
-
-           function enterChapter(d3_event, newChapter) {
-             if (_currChapter) {
-               _currChapter.exit();
-             }
+           mainEnter.append('div').attr('class', 'comment-text').html(function (d) {
+             return d.html;
+           }).selectAll('a').attr('rel', 'noopener nofollow').attr('target', '_blank');
+           comments.call(replaceAvatars);
+         }
 
-             context.enter(modeBrowse(context));
-             _currChapter = newChapter;
+         function replaceAvatars(selection) {
+           var showThirdPartyIcons = corePreferences('preferences.privacy.thirdpartyicons') || 'true';
+           var osm = services.osm;
+           if (showThirdPartyIcons !== 'true' || !osm) return;
+           var uids = {}; // gather uids in the comment thread
 
-             _currChapter.enter();
+           _note.comments.forEach(function (d) {
+             if (d.uid) uids[d.uid] = true;
+           });
 
-             buttons.classed('next', false).classed('active', function (d) {
-               return d.title === _currChapter.title;
+           Object.keys(uids).forEach(function (uid) {
+             osm.loadUser(uid, function (err, user) {
+               if (!user || !user.image_url) return;
+               selection.selectAll('.comment-avatar.user-' + uid).html('').append('img').attr('class', 'icon comment-avatar-icon').attr('src', user.image_url).attr('alt', user.display_name);
              });
-           }
+           });
          }
 
-         return intro;
-       }
+         function localeDateString(s) {
+           if (!s) return null;
+           var options = {
+             day: 'numeric',
+             month: 'short',
+             year: 'numeric'
+           };
+           s = s.replace(/-/g, '/'); // fix browser-specific Date() issues
 
-       function uiIssuesInfo(context) {
-         var warningsItem = {
-           id: 'warnings',
-           count: 0,
-           iconID: 'iD-icon-alert',
-           descriptionID: 'issues.warnings_and_errors'
-         };
-         var resolvedItem = {
-           id: 'resolved',
-           count: 0,
-           iconID: 'iD-icon-apply',
-           descriptionID: 'issues.user_resolved_issues'
+           var d = new Date(s);
+           if (isNaN(d.getTime())) return null;
+           return d.toLocaleDateString(_mainLocalizer.localeCode(), options);
+         }
+
+         noteComments.note = function (val) {
+           if (!arguments.length) return _note;
+           _note = val;
+           return noteComments;
          };
 
-         function update(selection) {
-           var shownItems = [];
-           var liveIssues = context.validator().getIssues({
-             what: corePreferences('validate-what') || 'edited',
-             where: corePreferences('validate-where') || 'all'
-           });
+         return noteComments;
+       }
 
-           if (liveIssues.length) {
-             warningsItem.count = liveIssues.length;
-             shownItems.push(warningsItem);
-           }
+       function uiNoteHeader() {
+         var _note;
 
-           if (corePreferences('validate-what') === 'all') {
-             var resolvedIssues = context.validator().getResolvedIssues();
+         function noteHeader(selection) {
+           var header = selection.selectAll('.note-header').data(_note ? [_note] : [], function (d) {
+             return d.status + d.id;
+           });
+           header.exit().remove();
+           var headerEnter = header.enter().append('div').attr('class', 'note-header');
+           var iconEnter = headerEnter.append('div').attr('class', function (d) {
+             return 'note-header-icon ' + d.status;
+           }).classed('new', function (d) {
+             return d.id < 0;
+           });
+           iconEnter.append('div').attr('class', 'preset-icon-28').call(svgIcon('#iD-icon-note', 'note-fill'));
+           iconEnter.each(function (d) {
+             var statusIcon;
 
-             if (resolvedIssues.length) {
-               resolvedItem.count = resolvedIssues.length;
-               shownItems.push(resolvedItem);
+             if (d.id < 0) {
+               statusIcon = '#iD-icon-plus';
+             } else if (d.status === 'open') {
+               statusIcon = '#iD-icon-close';
+             } else {
+               statusIcon = '#iD-icon-apply';
              }
-           }
 
-           var chips = selection.selectAll('.chip').data(shownItems, function (d) {
-             return d.id;
+             iconEnter.append('div').attr('class', 'note-icon-annotation').call(svgIcon(statusIcon, 'icon-annotation'));
            });
-           chips.exit().remove();
-           var enter = chips.enter().append('a').attr('class', function (d) {
-             return 'chip ' + d.id + '-count';
-           }).attr('href', '#').each(function (d) {
-             var chipSelection = select(this);
-             var tooltipBehavior = uiTooltip().placement('top').title(_t.html(d.descriptionID));
-             chipSelection.call(tooltipBehavior).on('click', function (d3_event) {
-               d3_event.preventDefault();
-               tooltipBehavior.hide(select(this)); // open the Issues pane
+           headerEnter.append('div').attr('class', 'note-header-label').html(function (d) {
+             if (_note.isNew()) {
+               return _t('note.new');
+             }
 
-               context.ui().togglePanes(context.container().select('.map-panes .issues-pane'));
-             });
-             chipSelection.call(svgIcon('#' + d.iconID));
-           });
-           enter.append('span').attr('class', 'count');
-           enter.merge(chips).selectAll('span.count').html(function (d) {
-             return d.count.toString();
+             return _t('note.note') + ' ' + d.id + ' ' + (d.status === 'closed' ? _t('note.closed') : '');
            });
          }
 
-         return function (selection) {
-           update(selection);
-           context.validator().on('validated.infobox', function () {
-             update(selection);
-           });
+         noteHeader.note = function (val) {
+           if (!arguments.length) return _note;
+           _note = val;
+           return noteHeader;
          };
-       }
-
-       function uiMapInMap(context) {
-         function mapInMap(selection) {
-           var backgroundLayer = rendererTileLayer(context);
-           var overlayLayers = {};
-           var projection = geoRawMercator();
-           var dataLayer = svgData(projection, context).showLabels(false);
-           var debugLayer = svgDebug(projection, context);
-           var zoom = d3_zoom().scaleExtent([geoZoomToScale(0.5), geoZoomToScale(24)]).on('start', zoomStarted).on('zoom', zoomed).on('end', zoomEnded);
-           var wrap = select(null);
-           var tiles = select(null);
-           var viewport = select(null);
-           var _isTransformed = false;
-           var _isHidden = true;
-           var _skipEvents = false;
-           var _gesture = null;
-           var _zDiff = 6; // by default, minimap renders at (main zoom - 6)
-
-           var _dMini; // dimensions of minimap
 
+         return noteHeader;
+       }
 
-           var _cMini; // center pixel of minimap
-
+       function uiNoteReport() {
+         var _note;
 
-           var _tStart; // transform at start of gesture
+         function noteReport(selection) {
+           var url;
 
+           if (services.osm && _note instanceof osmNote && !_note.isNew()) {
+             url = services.osm.noteReportURL(_note);
+           }
 
-           var _tCurr; // transform at most recent event
+           var link = selection.selectAll('.note-report').data(url ? [url] : []); // exit
 
+           link.exit().remove(); // enter
 
-           var _timeoutID;
+           var linkEnter = link.enter().append('a').attr('class', 'note-report').attr('target', '_blank').attr('href', function (d) {
+             return d;
+           }).call(svgIcon('#iD-icon-out-link', 'inline'));
+           linkEnter.append('span').html(_t.html('note.report'));
+         }
 
-           function zoomStarted() {
-             if (_skipEvents) return;
-             _tStart = _tCurr = projection.transform();
-             _gesture = null;
-           }
+         noteReport.note = function (val) {
+           if (!arguments.length) return _note;
+           _note = val;
+           return noteReport;
+         };
 
-           function zoomed(d3_event) {
-             if (_skipEvents) return;
-             var x = d3_event.transform.x;
-             var y = d3_event.transform.y;
-             var k = d3_event.transform.k;
-             var isZooming = k !== _tStart.k;
-             var isPanning = x !== _tStart.x || y !== _tStart.y;
+         return noteReport;
+       }
 
-             if (!isZooming && !isPanning) {
-               return; // no change
-             } // lock in either zooming or panning, don't allow both in minimap.
+       function uiNoteEditor(context) {
+         var dispatch = dispatch$8('change');
+         var noteComments = uiNoteComments();
+         var noteHeader = uiNoteHeader(); // var formFields = uiFormFields(context);
 
+         var _note;
 
-             if (!_gesture) {
-               _gesture = isZooming ? 'zoom' : 'pan';
-             }
+         var _newNote; // var _fieldsArr;
 
-             var tMini = projection.transform();
-             var tX, tY, scale;
 
-             if (_gesture === 'zoom') {
-               scale = k / tMini.k;
-               tX = (_cMini[0] / scale - _cMini[0]) * scale;
-               tY = (_cMini[1] / scale - _cMini[1]) * scale;
-             } else {
-               k = tMini.k;
-               scale = 1;
-               tX = x - tMini.x;
-               tY = y - tMini.y;
-             }
+         function noteEditor(selection) {
+           var header = selection.selectAll('.header').data([0]);
+           var headerEnter = header.enter().append('div').attr('class', 'header fillL');
+           headerEnter.append('button').attr('class', 'close').on('click', function () {
+             context.enter(modeBrowse(context));
+           }).call(svgIcon('#iD-icon-close'));
+           headerEnter.append('h3').html(_t.html('note.title'));
+           var body = selection.selectAll('.body').data([0]);
+           body = body.enter().append('div').attr('class', 'body').merge(body);
+           var editor = body.selectAll('.note-editor').data([0]);
+           editor.enter().append('div').attr('class', 'modal-section note-editor').merge(editor).call(noteHeader.note(_note)).call(noteComments.note(_note)).call(noteSaveSection);
+           var footer = selection.selectAll('.footer').data([0]);
+           footer.enter().append('div').attr('class', 'footer').merge(footer).call(uiViewOnOSM(context).what(_note)).call(uiNoteReport().note(_note)); // rerender the note editor on any auth change
 
-             utilSetTransform(tiles, tX, tY, scale);
-             utilSetTransform(viewport, 0, 0, scale);
-             _isTransformed = true;
-             _tCurr = identity$2.translate(x, y).scale(k);
-             var zMain = geoScaleToZoom(context.projection.scale());
-             var zMini = geoScaleToZoom(k);
-             _zDiff = zMain - zMini;
-             queueRedraw();
-           }
+           var osm = services.osm;
 
-           function zoomEnded() {
-             if (_skipEvents) return;
-             if (_gesture !== 'pan') return;
-             updateProjection();
-             _gesture = null;
-             context.map().center(projection.invert(_cMini)); // recenter main map..
+           if (osm) {
+             osm.on('change.note-save', function () {
+               selection.call(noteEditor);
+             });
            }
+         }
 
-           function updateProjection() {
-             var loc = context.map().center();
-             var tMain = context.projection.transform();
-             var zMain = geoScaleToZoom(tMain.k);
-             var zMini = Math.max(zMain - _zDiff, 0.5);
-             var kMini = geoZoomToScale(zMini);
-             projection.translate([tMain.x, tMain.y]).scale(kMini);
-             var point = projection(loc);
-             var mouse = _gesture === 'pan' ? geoVecSubtract([_tCurr.x, _tCurr.y], [_tStart.x, _tStart.y]) : [0, 0];
-             var xMini = _cMini[0] - point[0] + tMain.x + mouse[0];
-             var yMini = _cMini[1] - point[1] + tMain.y + mouse[1];
-             projection.translate([xMini, yMini]).clipExtent([[0, 0], _dMini]);
-             _tCurr = projection.transform();
+         function noteSaveSection(selection) {
+           var isSelected = _note && _note.id === context.selectedNoteID();
 
-             if (_isTransformed) {
-               utilSetTransform(tiles, 0, 0);
-               utilSetTransform(viewport, 0, 0);
-               _isTransformed = false;
-             }
+           var noteSave = selection.selectAll('.note-save').data(isSelected ? [_note] : [], function (d) {
+             return d.status + d.id;
+           }); // exit
 
-             zoom.scaleExtent([geoZoomToScale(0.5), geoZoomToScale(zMain - 3)]);
-             _skipEvents = true;
-             wrap.call(zoom.transform, _tCurr);
-             _skipEvents = false;
-           }
+           noteSave.exit().remove(); // enter
 
-           function redraw() {
-             clearTimeout(_timeoutID);
-             if (_isHidden) return;
-             updateProjection();
-             var zMini = geoScaleToZoom(projection.scale()); // setup tile container
+           var noteSaveEnter = noteSave.enter().append('div').attr('class', 'note-save save-section cf'); // // if new note, show categories to pick from
+           // if (_note.isNew()) {
+           //     var presets = presetManager;
+           //     // NOTE: this key isn't a age and therefore there is no documentation (yet)
+           //     _fieldsArr = [
+           //         uiField(context, presets.field('category'), null, { show: true, revert: false }),
+           //     ];
+           //     _fieldsArr.forEach(function(field) {
+           //         field
+           //             .on('change', changeCategory);
+           //     });
+           //     noteSaveEnter
+           //         .append('div')
+           //         .attr('class', 'note-category')
+           //         .call(formFields.fieldsArr(_fieldsArr));
+           // }
+           // function changeCategory() {
+           //     // NOTE: perhaps there is a better way to get value
+           //     var val = context.container().select('input[name=\'category\']:checked').property('__data__') || undefined;
+           //     // store the unsaved category with the note itself
+           //     _note = _note.update({ newCategory: val });
+           //     var osm = services.osm;
+           //     if (osm) {
+           //         osm.replaceNote(_note);  // update note cache
+           //     }
+           //     noteSave
+           //         .call(noteSaveButtons);
+           // }
 
-             tiles = wrap.selectAll('.map-in-map-tiles').data([0]);
-             tiles = tiles.enter().append('div').attr('class', 'map-in-map-tiles').merge(tiles); // redraw background
+           noteSaveEnter.append('h4').attr('class', '.note-save-header').html(function () {
+             return _note.isNew() ? _t('note.newDescription') : _t('note.newComment');
+           });
+           var commentTextarea = noteSaveEnter.append('textarea').attr('class', 'new-comment-input').attr('placeholder', _t('note.inputPlaceholder')).attr('maxlength', 1000).property('value', function (d) {
+             return d.newComment;
+           }).call(utilNoAuto).on('keydown.note-input', keydown).on('input.note-input', changeInput).on('blur.note-input', changeInput);
 
-             backgroundLayer.source(context.background().baseLayerSource()).projection(projection).dimensions(_dMini);
-             var background = tiles.selectAll('.map-in-map-background').data([0]);
-             background.enter().append('div').attr('class', 'map-in-map-background').merge(background).call(backgroundLayer); // redraw overlay
+           if (!commentTextarea.empty() && _newNote) {
+             // autofocus the comment field for new notes
+             commentTextarea.node().focus();
+           } // update
 
-             var overlaySources = context.background().overlayLayerSources();
-             var activeOverlayLayers = [];
 
-             for (var i = 0; i < overlaySources.length; i++) {
-               if (overlaySources[i].validZoom(zMini)) {
-                 if (!overlayLayers[i]) overlayLayers[i] = rendererTileLayer(context);
-                 activeOverlayLayers.push(overlayLayers[i].source(overlaySources[i]).projection(projection).dimensions(_dMini));
-               }
-             }
+           noteSave = noteSaveEnter.merge(noteSave).call(userDetails).call(noteSaveButtons); // fast submit if user presses cmd+enter
 
-             var overlay = tiles.selectAll('.map-in-map-overlay').data([0]);
-             overlay = overlay.enter().append('div').attr('class', 'map-in-map-overlay').merge(overlay);
-             var overlays = overlay.selectAll('div').data(activeOverlayLayers, function (d) {
-               return d.source().name();
-             });
-             overlays.exit().remove();
-             overlays = overlays.enter().append('div').merge(overlays).each(function (layer) {
-               select(this).call(layer);
-             });
-             var dataLayers = tiles.selectAll('.map-in-map-data').data([0]);
-             dataLayers.exit().remove();
-             dataLayers = dataLayers.enter().append('svg').attr('class', 'map-in-map-data').merge(dataLayers).call(dataLayer).call(debugLayer); // redraw viewport bounding box
+           function keydown(d3_event) {
+             if (!(d3_event.keyCode === 13 && // ↩ Return
+             d3_event.metaKey)) return;
+             var osm = services.osm;
+             if (!osm) return;
+             var hasAuth = osm.authenticated();
+             if (!hasAuth) return;
+             if (!_note.newComment) return;
+             d3_event.preventDefault();
+             select(this).on('keydown.note-input', null); // focus on button and submit
 
-             if (_gesture !== 'pan') {
-               var getPath = d3_geoPath(projection);
-               var bbox = {
-                 type: 'Polygon',
-                 coordinates: [context.map().extent().polygon()]
-               };
-               viewport = wrap.selectAll('.map-in-map-viewport').data([0]);
-               viewport = viewport.enter().append('svg').attr('class', 'map-in-map-viewport').merge(viewport);
-               var path = viewport.selectAll('.map-in-map-bbox').data([bbox]);
-               path.enter().append('path').attr('class', 'map-in-map-bbox').merge(path).attr('d', getPath).classed('thick', function (d) {
-                 return getPath.area(d) < 30;
-               });
-             }
+             window.setTimeout(function () {
+               if (_note.isNew()) {
+                 noteSave.selectAll('.save-button').node().focus();
+                 clickSave();
+               } else {
+                 noteSave.selectAll('.comment-button').node().focus();
+                 clickComment();
+               }
+             }, 10);
            }
 
-           function queueRedraw() {
-             clearTimeout(_timeoutID);
-             _timeoutID = setTimeout(function () {
-               redraw();
-             }, 750);
-           }
+           function changeInput() {
+             var input = select(this);
+             var val = input.property('value').trim() || undefined; // store the unsaved comment with the note itself
 
-           function toggle(d3_event) {
-             if (d3_event) d3_event.preventDefault();
-             _isHidden = !_isHidden;
-             context.container().select('.minimap-toggle-item').classed('active', !_isHidden).select('input').property('checked', !_isHidden);
+             _note = _note.update({
+               newComment: val
+             });
+             var osm = services.osm;
 
-             if (_isHidden) {
-               wrap.style('display', 'block').style('opacity', '1').transition().duration(200).style('opacity', '0').on('end', function () {
-                 selection.selectAll('.map-in-map').style('display', 'none');
-               });
-             } else {
-               wrap.style('display', 'block').style('opacity', '0').transition().duration(200).style('opacity', '1').on('end', function () {
-                 redraw();
-               });
+             if (osm) {
+               osm.replaceNote(_note); // update note cache
              }
+
+             noteSave.call(noteSaveButtons);
            }
+         }
 
-           uiMapInMap.toggle = toggle;
-           wrap = selection.selectAll('.map-in-map').data([0]);
-           wrap = wrap.enter().append('div').attr('class', 'map-in-map').style('display', _isHidden ? 'none' : 'block').call(zoom).on('dblclick.zoom', null).merge(wrap); // reflow warning: Hardcode dimensions - currently can't resize it anyway..
+         function userDetails(selection) {
+           var detailSection = selection.selectAll('.detail-section').data([0]);
+           detailSection = detailSection.enter().append('div').attr('class', 'detail-section').merge(detailSection);
+           var osm = services.osm;
+           if (!osm) return; // Add warning if user is not logged in
 
-           _dMini = [200, 150]; //utilGetDimensions(wrap);
+           var hasAuth = osm.authenticated();
+           var authWarning = detailSection.selectAll('.auth-warning').data(hasAuth ? [] : [0]);
+           authWarning.exit().transition().duration(200).style('opacity', 0).remove();
+           var authEnter = authWarning.enter().insert('div', '.tag-reference-body').attr('class', 'field-warning auth-warning').style('opacity', 0);
+           authEnter.call(svgIcon('#iD-icon-alert', 'inline'));
+           authEnter.append('span').html(_t.html('note.login'));
+           authEnter.append('a').attr('target', '_blank').call(svgIcon('#iD-icon-out-link', 'inline')).append('span').html(_t.html('login')).on('click.note-login', function (d3_event) {
+             d3_event.preventDefault();
+             osm.authenticate();
+           });
+           authEnter.transition().duration(200).style('opacity', 1);
+           var prose = detailSection.selectAll('.note-save-prose').data(hasAuth ? [0] : []);
+           prose.exit().remove();
+           prose = prose.enter().append('p').attr('class', 'note-save-prose').html(_t.html('note.upload_explanation')).merge(prose);
+           osm.userDetails(function (err, user) {
+             if (err) return;
+             var userLink = select(document.createElement('div'));
 
-           _cMini = geoVecScale(_dMini, 0.5);
-           context.map().on('drawn.map-in-map', function (drawn) {
-             if (drawn.full === true) {
-               redraw();
+             if (user.image_url) {
+               userLink.append('img').attr('src', user.image_url).attr('class', 'icon pre-text user-icon');
              }
+
+             userLink.append('a').attr('class', 'user-info').html(user.display_name).attr('href', osm.userURL(user.display_name)).attr('target', '_blank');
+             prose.html(_t.html('note.upload_explanation_with_user', {
+               user: userLink.html()
+             }));
            });
-           redraw();
-           context.keybinding().on(_t('background.minimap.key'), toggle);
          }
 
-         return mapInMap;
-       }
-
-       function uiNotice(context) {
-         return function (selection) {
-           var div = selection.append('div').attr('class', 'notice');
-           var button = div.append('button').attr('class', 'zoom-to notice fillD').on('click', function () {
-             context.map().zoomEase(context.minEditableZoom());
-           }).on('wheel', function (d3_event) {
-             // let wheel events pass through #4482
-             var e2 = new WheelEvent(d3_event.type, d3_event);
-             context.surface().node().dispatchEvent(e2);
-           });
-           button.call(svgIcon('#iD-icon-plus', 'pre-text')).append('span').attr('class', 'label').html(_t.html('zoom_in_edit'));
+         function noteSaveButtons(selection) {
+           var osm = services.osm;
+           var hasAuth = osm && osm.authenticated();
 
-           function disableTooHigh() {
-             var canEdit = context.map().zoom() >= context.minEditableZoom();
-             div.style('display', canEdit ? 'none' : 'block');
-           }
+           var isSelected = _note && _note.id === context.selectedNoteID();
 
-           context.map().on('move.notice', debounce(disableTooHigh, 500));
-           disableTooHigh();
-         };
-       }
+           var buttonSection = selection.selectAll('.buttons').data(isSelected ? [_note] : [], function (d) {
+             return d.status + d.id;
+           }); // exit
 
-       function uiPhotoviewer(context) {
-         var dispatch$1 = dispatch('resize');
+           buttonSection.exit().remove(); // enter
 
-         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
+           var buttonEnter = buttonSection.enter().append('div').attr('class', 'buttons');
 
-         function photoviewer(selection) {
-           selection.append('button').attr('class', 'thumb-hide').on('click', function () {
-             if (services.streetside) {
-               services.streetside.hideViewer(context);
-             }
+           if (_note.isNew()) {
+             buttonEnter.append('button').attr('class', 'button cancel-button secondary-action').html(_t.html('confirm.cancel'));
+             buttonEnter.append('button').attr('class', 'button save-button action').html(_t.html('note.save'));
+           } else {
+             buttonEnter.append('button').attr('class', 'button status-button action');
+             buttonEnter.append('button').attr('class', 'button comment-button action').html(_t.html('note.comment'));
+           } // update
 
-             if (services.mapillary) {
-               services.mapillary.hideViewer(context);
-             }
 
-             if (services.openstreetcam) {
-               services.openstreetcam.hideViewer(context);
-             }
-           }).append('div').call(svgIcon('#iD-icon-close'));
+           buttonSection = buttonSection.merge(buttonEnter);
+           buttonSection.select('.cancel-button') // select and propagate data
+           .on('click.cancel', clickCancel);
+           buttonSection.select('.save-button') // select and propagate data
+           .attr('disabled', isSaveDisabled).on('click.save', clickSave);
+           buttonSection.select('.status-button') // select and propagate data
+           .attr('disabled', hasAuth ? null : true).html(function (d) {
+             var action = d.status === 'open' ? 'close' : 'open';
+             var andComment = d.newComment ? '_comment' : '';
+             return _t('note.' + action + andComment);
+           }).on('click.status', clickStatus);
+           buttonSection.select('.comment-button') // select and propagate data
+           .attr('disabled', isSaveDisabled).on('click.comment', clickComment);
 
-           function preventDefault(d3_event) {
-             d3_event.preventDefault();
+           function isSaveDisabled(d) {
+             return hasAuth && d.status === 'open' && d.newComment ? null : true;
            }
+         }
 
-           selection.append('button').attr('class', 'resize-handle-xy').on('touchstart touchdown touchend', preventDefault).on(_pointerPrefix + 'down', buildResizeListener(selection, 'resize', dispatch$1, {
-             resizeOnX: true,
-             resizeOnY: true
-           }));
-           selection.append('button').attr('class', 'resize-handle-x').on('touchstart touchdown touchend', preventDefault).on(_pointerPrefix + 'down', buildResizeListener(selection, 'resize', dispatch$1, {
-             resizeOnX: true
-           }));
-           selection.append('button').attr('class', 'resize-handle-y').on('touchstart touchdown touchend', preventDefault).on(_pointerPrefix + 'down', buildResizeListener(selection, 'resize', dispatch$1, {
-             resizeOnY: true
-           }));
-
-           function buildResizeListener(target, eventName, dispatch, options) {
-             var resizeOnX = !!options.resizeOnX;
-             var resizeOnY = !!options.resizeOnY;
-             var minHeight = options.minHeight || 240;
-             var minWidth = options.minWidth || 320;
-             var pointerId;
-             var startX;
-             var startY;
-             var startWidth;
-             var startHeight;
-
-             function startResize(d3_event) {
-               if (pointerId !== (d3_event.pointerId || 'mouse')) return;
-               d3_event.preventDefault();
-               d3_event.stopPropagation();
-               var mapSize = context.map().dimensions();
+         function clickCancel(d3_event, d) {
+           this.blur(); // avoid keeping focus on the button - #4641
 
-               if (resizeOnX) {
-                 var maxWidth = mapSize[0];
-                 var newWidth = clamp(startWidth + d3_event.clientX - startX, minWidth, maxWidth);
-                 target.style('width', newWidth + 'px');
-               }
+           var osm = services.osm;
 
-               if (resizeOnY) {
-                 var maxHeight = mapSize[1] - 90; // preserve space at top/bottom of map
+           if (osm) {
+             osm.removeNote(d);
+           }
 
-                 var newHeight = clamp(startHeight + startY - d3_event.clientY, minHeight, maxHeight);
-                 target.style('height', newHeight + 'px');
-               }
+           context.enter(modeBrowse(context));
+           dispatch.call('change');
+         }
 
-               dispatch.call(eventName, target, utilGetDimensions(target, true));
-             }
+         function clickSave(d3_event, d) {
+           this.blur(); // avoid keeping focus on the button - #4641
 
-             function clamp(num, min, max) {
-               return Math.max(min, Math.min(num, max));
-             }
+           var osm = services.osm;
 
-             function stopResize(d3_event) {
-               if (pointerId !== (d3_event.pointerId || 'mouse')) return;
-               d3_event.preventDefault();
-               d3_event.stopPropagation(); // remove all the listeners we added
+           if (osm) {
+             osm.postNoteCreate(d, function (err, note) {
+               dispatch.call('change', note);
+             });
+           }
+         }
 
-               select(window).on('.' + eventName, null);
-             }
+         function clickStatus(d3_event, d) {
+           this.blur(); // avoid keeping focus on the button - #4641
 
-             return function initResize(d3_event) {
-               d3_event.preventDefault();
-               d3_event.stopPropagation();
-               pointerId = d3_event.pointerId || 'mouse';
-               startX = d3_event.clientX;
-               startY = d3_event.clientY;
-               var targetRect = target.node().getBoundingClientRect();
-               startWidth = targetRect.width;
-               startHeight = targetRect.height;
-               select(window).on(_pointerPrefix + 'move.' + eventName, startResize, false).on(_pointerPrefix + 'up.' + eventName, stopResize, false);
+           var osm = services.osm;
 
-               if (_pointerPrefix === 'pointer') {
-                 select(window).on('pointercancel.' + eventName, stopResize, false);
-               }
-             };
+           if (osm) {
+             var setStatus = d.status === 'open' ? 'closed' : 'open';
+             osm.postNoteUpdate(d, setStatus, function (err, note) {
+               dispatch.call('change', note);
+             });
            }
          }
 
-         photoviewer.onMapResize = function () {
-           var photoviewer = context.container().select('.photoviewer');
-           var content = context.container().select('.main-content');
-           var mapDimensions = utilGetDimensions(content, true); // shrink photo viewer if it is too big
-           // (-90 preserves space at top and bottom of map used by menus)
+         function clickComment(d3_event, d) {
+           this.blur(); // avoid keeping focus on the button - #4641
 
-           var photoDimensions = utilGetDimensions(photoviewer, true);
+           var osm = services.osm;
 
-           if (photoDimensions[0] > mapDimensions[0] || photoDimensions[1] > mapDimensions[1] - 90) {
-             var setPhotoDimensions = [Math.min(photoDimensions[0], mapDimensions[0]), Math.min(photoDimensions[1], mapDimensions[1] - 90)];
-             photoviewer.style('width', setPhotoDimensions[0] + 'px').style('height', setPhotoDimensions[1] + 'px');
-             dispatch$1.call('resize', photoviewer, setPhotoDimensions);
+           if (osm) {
+             osm.postNoteUpdate(d, d.status, function (err, note) {
+               dispatch.call('change', note);
+             });
            }
-         };
+         }
 
-         return utilRebind(photoviewer, dispatch$1, 'on');
-       }
+         noteEditor.note = function (val) {
+           if (!arguments.length) return _note;
+           _note = val;
+           return noteEditor;
+         };
 
-       function uiRestore(context) {
-         return function (selection) {
-           if (!context.history().hasRestorableChanges()) return;
-           var modalSelection = uiModal(selection, true);
-           modalSelection.select('.modal').attr('class', 'modal fillL');
-           var introModal = modalSelection.select('.content');
-           introModal.append('div').attr('class', 'modal-section').append('h3').html(_t.html('restore.heading'));
-           introModal.append('div').attr('class', 'modal-section').append('p').html(_t.html('restore.description'));
-           var buttonWrap = introModal.append('div').attr('class', 'modal-actions');
-           var restore = buttonWrap.append('button').attr('class', 'restore').on('click', function () {
-             context.history().restore();
-             modalSelection.remove();
-           });
-           restore.append('svg').attr('class', 'logo logo-restore').append('use').attr('xlink:href', '#iD-logo-restore');
-           restore.append('div').html(_t.html('restore.restore'));
-           var reset = buttonWrap.append('button').attr('class', 'reset').on('click', function () {
-             context.history().clearSaved();
-             modalSelection.remove();
-           });
-           reset.append('svg').attr('class', 'logo logo-reset').append('use').attr('xlink:href', '#iD-logo-reset');
-           reset.append('div').html(_t.html('restore.reset'));
-           restore.node().focus();
+         noteEditor.newNote = function (val) {
+           if (!arguments.length) return _newNote;
+           _newNote = val;
+           return noteEditor;
          };
+
+         return utilRebind(noteEditor, dispatch, 'on');
        }
 
-       function uiScale(context) {
-         var projection = context.projection,
-             isImperial = !_mainLocalizer.usesMetric(),
-             maxLength = 180,
-             tickHeight = 8;
+       function uiSidebar(context) {
+         var inspector = uiInspector(context);
+         var dataEditor = uiDataEditor(context);
+         var noteEditor = uiNoteEditor(context);
+         var improveOsmEditor = uiImproveOsmEditor(context);
+         var keepRightEditor = uiKeepRightEditor(context);
+         var osmoseEditor = uiOsmoseEditor(context);
 
-         function scaleDefs(loc1, loc2) {
-           var lat = (loc2[1] + loc1[1]) / 2,
-               conversion = isImperial ? 3.28084 : 1,
-               dist = geoLonToMeters(loc2[0] - loc1[0], lat) * conversion,
-               scale = {
-             dist: 0,
-             px: 0,
-             text: ''
-           },
-               buckets,
-               i,
-               val,
-               dLon;
+         var _current;
 
-           if (isImperial) {
-             buckets = [5280000, 528000, 52800, 5280, 500, 50, 5, 1];
-           } else {
-             buckets = [5000000, 500000, 50000, 5000, 500, 50, 5, 1];
-           } // determine a user-friendly endpoint for the scale
+         var _wasData = false;
+         var _wasNote = false;
+         var _wasQaItem = false; // use pointer events on supported platforms; fallback to mouse events
 
+         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
 
-           for (i = 0; i < buckets.length; i++) {
-             val = buckets[i];
+         function sidebar(selection) {
+           var container = context.container();
+           var minWidth = 240;
+           var sidebarWidth;
+           var containerWidth;
+           var dragOffset; // Set the initial width constraints
 
-             if (dist >= val) {
-               scale.dist = Math.floor(dist / val) * val;
-               break;
-             } else {
-               scale.dist = +dist.toFixed(2);
-             }
-           }
+           selection.style('min-width', minWidth + 'px').style('max-width', '400px').style('width', '33.3333%');
+           var resizer = selection.append('div').attr('class', 'sidebar-resizer').on(_pointerPrefix + 'down.sidebar-resizer', pointerdown);
+           var downPointerId, lastClientX, containerLocGetter;
 
-           dLon = geoMetersToLon(scale.dist / conversion, lat);
-           scale.px = Math.round(projection([loc1[0] + dLon, loc1[1]])[0]);
-           scale.text = displayLength(scale.dist / conversion, isImperial);
-           return scale;
-         }
+           function pointerdown(d3_event) {
+             if (downPointerId) return;
+             if ('button' in d3_event && d3_event.button !== 0) return;
+             downPointerId = d3_event.pointerId || 'mouse';
+             lastClientX = d3_event.clientX;
+             containerLocGetter = utilFastMouse(container.node()); // offset from edge of sidebar-resizer
 
-         function update(selection) {
-           // choose loc1, loc2 along bottom of viewport (near where the scale will be drawn)
-           var dims = context.map().dimensions(),
-               loc1 = projection.invert([0, dims[1]]),
-               loc2 = projection.invert([maxLength, dims[1]]),
-               scale = scaleDefs(loc1, loc2);
-           selection.select('.scale-path').attr('d', 'M0.5,0.5v' + tickHeight + 'h' + scale.px + 'v-' + tickHeight);
-           selection.select('.scale-text').style(_mainLocalizer.textDirection() === 'ltr' ? 'left' : 'right', scale.px + 16 + 'px').html(scale.text);
-         }
+             dragOffset = utilFastMouse(resizer.node())(d3_event)[0] - 1;
+             sidebarWidth = selection.node().getBoundingClientRect().width;
+             containerWidth = container.node().getBoundingClientRect().width;
+             var widthPct = sidebarWidth / containerWidth * 100;
+             selection.style('width', widthPct + '%') // lock in current width
+             .style('max-width', '85%'); // but allow larger widths
 
-         return function (selection) {
-           function switchUnits() {
-             isImperial = !isImperial;
-             selection.call(update);
+             resizer.classed('dragging', true);
+             select(window).on('touchmove.sidebar-resizer', function (d3_event) {
+               // disable page scrolling while resizing on touch input
+               d3_event.preventDefault();
+             }, {
+               passive: false
+             }).on(_pointerPrefix + 'move.sidebar-resizer', pointermove).on(_pointerPrefix + 'up.sidebar-resizer pointercancel.sidebar-resizer', pointerup);
            }
 
-           var scalegroup = selection.append('svg').attr('class', 'scale').on('click', switchUnits).append('g').attr('transform', 'translate(10,11)');
-           scalegroup.append('path').attr('class', 'scale-path');
-           selection.append('div').attr('class', 'scale-text');
-           selection.call(update);
-           context.map().on('move.scale', function () {
-             update(selection);
-           });
-         };
-       }
+           function pointermove(d3_event) {
+             if (downPointerId !== (d3_event.pointerId || 'mouse')) return;
+             d3_event.preventDefault();
+             var dx = d3_event.clientX - lastClientX;
+             lastClientX = d3_event.clientX;
+             var isRTL = _mainLocalizer.textDirection() === 'rtl';
+             var scaleX = isRTL ? 0 : 1;
+             var xMarginProperty = isRTL ? 'margin-right' : 'margin-left';
+             var x = containerLocGetter(d3_event)[0] - dragOffset;
+             sidebarWidth = isRTL ? containerWidth - x : x;
+             var isCollapsed = selection.classed('collapsed');
+             var shouldCollapse = sidebarWidth < minWidth;
+             selection.classed('collapsed', shouldCollapse);
 
-       function uiShortcuts(context) {
-         var detected = utilDetect();
-         var _activeTab = 0;
+             if (shouldCollapse) {
+               if (!isCollapsed) {
+                 selection.style(xMarginProperty, '-400px').style('width', '400px');
+                 context.ui().onResize([(sidebarWidth - dx) * scaleX, 0]);
+               }
+             } else {
+               var widthPct = sidebarWidth / containerWidth * 100;
+               selection.style(xMarginProperty, null).style('width', widthPct + '%');
 
-         var _modalSelection;
+               if (isCollapsed) {
+                 context.ui().onResize([-sidebarWidth * scaleX, 0]);
+               } else {
+                 context.ui().onResize([-dx * scaleX, 0]);
+               }
+             }
+           }
 
-         var _selection = select(null);
+           function pointerup(d3_event) {
+             if (downPointerId !== (d3_event.pointerId || 'mouse')) return;
+             downPointerId = null;
+             resizer.classed('dragging', false);
+             select(window).on('touchmove.sidebar-resizer', null).on(_pointerPrefix + 'move.sidebar-resizer', null).on(_pointerPrefix + 'up.sidebar-resizer pointercancel.sidebar-resizer', null);
+           }
 
-         var _dataShortcuts;
+           var featureListWrap = selection.append('div').attr('class', 'feature-list-pane').call(uiFeatureList(context));
+           var inspectorWrap = selection.append('div').attr('class', 'inspector-hidden inspector-wrap');
 
-         function shortcutsModal(_modalSelection) {
-           _modalSelection.select('.modal').classed('modal-shortcuts', true);
+           var hoverModeSelect = function hoverModeSelect(targets) {
+             context.container().selectAll('.feature-list-item button').classed('hover', false);
 
-           var content = _modalSelection.select('.content');
+             if (context.selectedIDs().length > 1 && targets && targets.length) {
+               var elements = context.container().selectAll('.feature-list-item button').filter(function (node) {
+                 return targets.indexOf(node) !== -1;
+               });
 
-           content.append('div').attr('class', 'modal-section').append('h3').html(_t.html('shortcuts.title'));
-           _mainFileFetcher.get('shortcuts').then(function (data) {
-             _dataShortcuts = data;
-             content.call(render);
-           })["catch"](function () {
-             /* ignore */
-           });
-         }
+               if (!elements.empty()) {
+                 elements.classed('hover', true);
+               }
+             }
+           };
 
-         function render(selection) {
-           if (!_dataShortcuts) return;
-           var wrapper = selection.selectAll('.wrapper').data([0]);
-           var wrapperEnter = wrapper.enter().append('div').attr('class', 'wrapper modal-section');
-           var tabsBar = wrapperEnter.append('div').attr('class', 'tabs-bar');
-           var shortcutsList = wrapperEnter.append('div').attr('class', 'shortcuts-list');
-           wrapper = wrapper.merge(wrapperEnter);
-           var tabs = tabsBar.selectAll('.tab').data(_dataShortcuts);
-           var tabsEnter = tabs.enter().append('a').attr('class', 'tab').attr('href', '#').on('click', function (d3_event, d) {
-             d3_event.preventDefault();
+           sidebar.hoverModeSelect = throttle(hoverModeSelect, 200);
 
-             var i = _dataShortcuts.indexOf(d);
+           function hover(targets) {
+             var datum = targets && targets.length && targets[0];
 
-             _activeTab = i;
-             render(selection);
-           });
-           tabsEnter.append('span').html(function (d) {
-             return _t.html(d.text);
-           }); // Update
+             if (datum && datum.__featurehash__) {
+               // hovering on data
+               _wasData = true;
+               sidebar.show(dataEditor.datum(datum));
+               selection.selectAll('.sidebar-component').classed('inspector-hover', true);
+             } else if (datum instanceof osmNote) {
+               if (context.mode().id === 'drag-note') return;
+               _wasNote = true;
+               var osm = services.osm;
 
-           wrapper.selectAll('.tab').classed('active', function (d, i) {
-             return i === _activeTab;
-           });
-           var shortcuts = shortcutsList.selectAll('.shortcut-tab').data(_dataShortcuts);
-           var shortcutsEnter = shortcuts.enter().append('div').attr('class', function (d) {
-             return 'shortcut-tab shortcut-tab-' + d.tab;
-           });
-           var columnsEnter = shortcutsEnter.selectAll('.shortcut-column').data(function (d) {
-             return d.columns;
-           }).enter().append('table').attr('class', 'shortcut-column');
-           var rowsEnter = columnsEnter.selectAll('.shortcut-row').data(function (d) {
-             return d.rows;
-           }).enter().append('tr').attr('class', 'shortcut-row');
-           var sectionRows = rowsEnter.filter(function (d) {
-             return !d.shortcuts;
-           });
-           sectionRows.append('td');
-           sectionRows.append('td').attr('class', 'shortcut-section').append('h3').html(function (d) {
-             return _t.html(d.text);
-           });
-           var shortcutRows = rowsEnter.filter(function (d) {
-             return d.shortcuts;
-           });
-           var shortcutKeys = shortcutRows.append('td').attr('class', 'shortcut-keys');
-           var modifierKeys = shortcutKeys.filter(function (d) {
-             return d.modifiers;
-           });
-           modifierKeys.selectAll('kbd.modifier').data(function (d) {
-             if (detected.os === 'win' && d.text === 'shortcuts.editing.commands.redo') {
-               return ['⌘'];
-             } else if (detected.os !== 'mac' && d.text === 'shortcuts.browsing.display_options.fullscreen') {
-               return [];
-             } else {
-               return d.modifiers;
-             }
-           }).enter().each(function () {
-             var selection = select(this);
-             selection.append('kbd').attr('class', 'modifier').html(function (d) {
-               return uiCmd.display(d);
-             });
-             selection.append('span').html('+');
-           });
-           shortcutKeys.selectAll('kbd.shortcut').data(function (d) {
-             var arr = d.shortcuts;
+               if (osm) {
+                 datum = osm.getNote(datum.id); // marker may contain stale data - get latest
+               }
 
-             if (detected.os === 'win' && d.text === 'shortcuts.editing.commands.redo') {
-               arr = ['Y'];
-             } else if (detected.os !== 'mac' && d.text === 'shortcuts.browsing.display_options.fullscreen') {
-               arr = ['F11'];
-             } // replace translations
+               sidebar.show(noteEditor.note(datum));
+               selection.selectAll('.sidebar-component').classed('inspector-hover', true);
+             } else if (datum instanceof QAItem) {
+               _wasQaItem = true;
+               var errService = services[datum.service];
 
+               if (errService) {
+                 // marker may contain stale data - get latest
+                 datum = errService.getError(datum.id);
+               } // Currently only three possible services
 
-             arr = arr.map(function (s) {
-               return uiCmd.display(s.indexOf('.') !== -1 ? _t(s) : s);
-             });
-             return utilArrayUniq(arr).map(function (s) {
-               return {
-                 shortcut: s,
-                 separator: d.separator,
-                 suffix: d.suffix
-               };
-             });
-           }).enter().each(function (d, i, nodes) {
-             var selection = select(this);
-             var click = d.shortcut.toLowerCase().match(/(.*).click/);
 
-             if (click && click[1]) {
-               // replace "left_click", "right_click" with mouse icon
-               selection.call(svgIcon('#iD-walkthrough-mouse-' + click[1], 'operation'));
-             } else if (d.shortcut.toLowerCase() === 'long-press') {
-               selection.call(svgIcon('#iD-walkthrough-longpress', 'longpress operation'));
-             } else if (d.shortcut.toLowerCase() === 'tap') {
-               selection.call(svgIcon('#iD-walkthrough-tap', 'tap operation'));
-             } else {
-               selection.append('kbd').attr('class', 'shortcut').html(function (d) {
-                 return d.shortcut;
+               var errEditor;
+
+               if (datum.service === 'keepRight') {
+                 errEditor = keepRightEditor;
+               } else if (datum.service === 'osmose') {
+                 errEditor = osmoseEditor;
+               } else {
+                 errEditor = improveOsmEditor;
+               }
+
+               context.container().selectAll('.qaItem.' + datum.service).classed('hover', function (d) {
+                 return d.id === datum.id;
                });
-             }
+               sidebar.show(errEditor.error(datum));
+               selection.selectAll('.sidebar-component').classed('inspector-hover', true);
+             } else if (!_current && datum instanceof osmEntity) {
+               featureListWrap.classed('inspector-hidden', true);
+               inspectorWrap.classed('inspector-hidden', false).classed('inspector-hover', true);
 
-             if (i < nodes.length - 1) {
-               selection.append('span').html(d.separator || "\xA0" + _t.html('shortcuts.or') + "\xA0");
-             } else if (i === nodes.length - 1 && d.suffix) {
-               selection.append('span').html(d.suffix);
+               if (!inspector.entityIDs() || !utilArrayIdentical(inspector.entityIDs(), [datum.id]) || inspector.state() !== 'hover') {
+                 inspector.state('hover').entityIDs([datum.id]).newFeature(false);
+                 inspectorWrap.call(inspector);
+               }
+             } else if (!_current) {
+               featureListWrap.classed('inspector-hidden', false);
+               inspectorWrap.classed('inspector-hidden', true);
+               inspector.state('hide');
+             } else if (_wasData || _wasNote || _wasQaItem) {
+               _wasNote = false;
+               _wasData = false;
+               _wasQaItem = false;
+               context.container().selectAll('.note').classed('hover', false);
+               context.container().selectAll('.qaItem').classed('hover', false);
+               sidebar.hide();
              }
-           });
-           shortcutKeys.filter(function (d) {
-             return d.gesture;
-           }).each(function () {
-             var selection = select(this);
-             selection.append('span').html('+');
-             selection.append('span').attr('class', 'gesture').html(function (d) {
-               return _t.html(d.gesture);
-             });
-           });
-           shortcutRows.append('td').attr('class', 'shortcut-desc').html(function (d) {
-             return d.text ? _t.html(d.text) : "\xA0";
-           }); // Update
-
-           wrapper.selectAll('.shortcut-tab').style('display', function (d, i) {
-             return i === _activeTab ? 'flex' : 'none';
-           });
-         }
+           }
 
-         return function (selection, show) {
-           _selection = selection;
+           sidebar.hover = throttle(hover, 200);
 
-           if (show) {
-             _modalSelection = uiModal(selection);
+           sidebar.intersects = function (extent) {
+             var rect = selection.node().getBoundingClientRect();
+             return extent.intersects([context.projection.invert([0, rect.height]), context.projection.invert([rect.width, 0])]);
+           };
 
-             _modalSelection.call(shortcutsModal);
-           } else {
-             context.keybinding().on([_t('shortcuts.toggle.key'), '?'], function () {
-               if (context.container().selectAll('.modal-shortcuts').size()) {
-                 // already showing
-                 if (_modalSelection) {
-                   _modalSelection.close();
+           sidebar.select = function (ids, newFeature) {
+             sidebar.hide();
 
-                   _modalSelection = null;
-                 }
-               } else {
-                 _modalSelection = uiModal(_selection);
+             if (ids && ids.length) {
+               var entity = ids.length === 1 && context.entity(ids[0]);
 
-                 _modalSelection.call(shortcutsModal);
+               if (entity && newFeature && selection.classed('collapsed')) {
+                 // uncollapse the sidebar
+                 var extent = entity.extent(context.graph());
+                 sidebar.expand(sidebar.intersects(extent));
                }
-             });
-           }
-         };
-       }
 
-       var pair_1 = pair;
+               featureListWrap.classed('inspector-hidden', true);
+               inspectorWrap.classed('inspector-hidden', false).classed('inspector-hover', false); // reload the UI even if the ids are the same since the entities
+               // themselves may have changed
 
-       function search(input, dims) {
-         if (!dims) dims = 'NSEW';
-         if (typeof input !== 'string') return null;
-         input = input.toUpperCase();
-         var regex = /^[\s\,]*([NSEW])?\s*([\-|\—|\―]?[0-9.]+)[°º˚]?\s*(?:([0-9.]+)['’′‘]\s*)?(?:([0-9.]+)(?:''|"|”|″)\s*)?([NSEW])?/;
-         var m = input.match(regex);
-         if (!m) return null; // no match
+               inspector.state('select').entityIDs(ids).newFeature(newFeature);
+               inspectorWrap.call(inspector);
+             } else {
+               inspector.state('hide');
+             }
+           };
 
-         var matched = m[0]; // extract dimension.. m[1] = leading, m[5] = trailing
+           sidebar.showPresetList = function () {
+             inspector.showList();
+           };
 
-         var dim;
+           sidebar.show = function (component, element) {
+             featureListWrap.classed('inspector-hidden', true);
+             inspectorWrap.classed('inspector-hidden', true);
+             if (_current) _current.remove();
+             _current = selection.append('div').attr('class', 'sidebar-component').call(component, element);
+           };
+
+           sidebar.hide = function () {
+             featureListWrap.classed('inspector-hidden', false);
+             inspectorWrap.classed('inspector-hidden', true);
+             if (_current) _current.remove();
+             _current = null;
+           };
 
-         if (m[1] && m[5]) {
-           // if matched both..
-           dim = m[1]; // keep leading
+           sidebar.expand = function (moveMap) {
+             if (selection.classed('collapsed')) {
+               sidebar.toggle(moveMap);
+             }
+           };
 
-           matched = matched.slice(0, -1); // remove trailing dimension from match
-         } else {
-           dim = m[1] || m[5];
-         } // if unrecognized dimension
+           sidebar.collapse = function (moveMap) {
+             if (!selection.classed('collapsed')) {
+               sidebar.toggle(moveMap);
+             }
+           };
 
+           sidebar.toggle = function (moveMap) {
+             // Don't allow sidebar to toggle when the user is in the walkthrough.
+             if (context.inIntro()) return;
+             var isCollapsed = selection.classed('collapsed');
+             var isCollapsing = !isCollapsed;
+             var isRTL = _mainLocalizer.textDirection() === 'rtl';
+             var scaleX = isRTL ? 0 : 1;
+             var xMarginProperty = isRTL ? 'margin-right' : 'margin-left';
+             sidebarWidth = selection.node().getBoundingClientRect().width; // switch from % to px
 
-         if (dim && dims.indexOf(dim) === -1) return null; // extract DMS
+             selection.style('width', sidebarWidth + 'px');
+             var startMargin, endMargin, lastMargin;
 
-         var deg = m[2] ? parseFloat(m[2]) : 0;
-         var min = m[3] ? parseFloat(m[3]) / 60 : 0;
-         var sec = m[4] ? parseFloat(m[4]) / 3600 : 0;
-         var sign = deg < 0 ? -1 : 1;
-         if (dim === 'S' || dim === 'W') sign *= -1;
-         return {
-           val: (Math.abs(deg) + min + sec) * sign,
-           dim: dim,
-           matched: matched,
-           remain: input.slice(matched.length)
-         };
-       }
+             if (isCollapsing) {
+               startMargin = lastMargin = 0;
+               endMargin = -sidebarWidth;
+             } else {
+               startMargin = lastMargin = -sidebarWidth;
+               endMargin = 0;
+             }
 
-       function pair(input, dims) {
-         input = input.trim();
-         var one = search(input, dims);
-         if (!one) return null;
-         input = one.remain.trim();
-         var two = search(input, dims);
-         if (!two || two.remain) return null;
+             if (!isCollapsing) {
+               // unhide the sidebar's content before it transitions onscreen
+               selection.classed('collapsed', isCollapsing);
+             }
 
-         if (one.dim) {
-           return swapdim(one.val, two.val, one.dim);
-         } else {
-           return [one.val, two.val];
-         }
-       }
+             selection.transition().style(xMarginProperty, endMargin + 'px').tween('panner', function () {
+               var i = d3_interpolateNumber(startMargin, endMargin);
+               return function (t) {
+                 var dx = lastMargin - Math.round(i(t));
+                 lastMargin = lastMargin - dx;
+                 context.ui().onResize(moveMap ? undefined : [dx * scaleX, 0]);
+               };
+             }).on('end', function () {
+               if (isCollapsing) {
+                 // hide the sidebar's content after it transitions offscreen
+                 selection.classed('collapsed', isCollapsing);
+               } // switch back from px to %
 
-       function swapdim(a, b, dim) {
-         if (dim === 'N' || dim === 'S') return [a, b];
-         if (dim === 'W' || dim === 'E') return [b, a];
-       }
 
-       function uiFeatureList(context) {
-         var _geocodeResults;
+               if (!isCollapsing) {
+                 var containerWidth = container.node().getBoundingClientRect().width;
+                 var widthPct = sidebarWidth / containerWidth * 100;
+                 selection.style(xMarginProperty, null).style('width', widthPct + '%');
+               }
+             });
+           }; // toggle the sidebar collapse when double-clicking the resizer
 
-         function featureList(selection) {
-           var header = selection.append('div').attr('class', 'header fillL');
-           header.append('h3').html(_t.html('inspector.feature_list'));
-           var searchWrap = selection.append('div').attr('class', 'search-header');
-           searchWrap.call(svgIcon('#iD-icon-search', 'pre-text'));
-           var search = searchWrap.append('input').attr('placeholder', _t('inspector.search')).attr('type', 'search').call(utilNoAuto).on('keypress', keypress).on('keydown', keydown).on('input', inputevent);
-           var listWrap = selection.append('div').attr('class', 'inspector-body');
-           var list = listWrap.append('div').attr('class', 'feature-list');
-           context.on('exit.feature-list', clearSearch);
-           context.map().on('drawn.feature-list', mapDrawn);
-           context.keybinding().on(uiCmd('⌘F'), focusSearch);
 
-           function focusSearch(d3_event) {
-             var mode = context.mode() && context.mode().id;
-             if (mode !== 'browse') return;
+           resizer.on('dblclick', function (d3_event) {
              d3_event.preventDefault();
-             search.node().focus();
-           }
 
-           function keydown(d3_event) {
-             if (d3_event.keyCode === 27) {
-               // escape
-               search.node().blur();
+             if (d3_event.sourceEvent) {
+               d3_event.sourceEvent.preventDefault();
              }
-           }
 
-           function keypress(d3_event) {
-             var q = search.property('value'),
-                 items = list.selectAll('.feature-list-item');
+             sidebar.toggle();
+           }); // ensure hover sidebar is closed when zooming out beyond editable zoom
 
-             if (d3_event.keyCode === 13 && // ↩ Return
-             q.length && items.size()) {
-               click(items.datum());
+           context.map().on('crossEditableZoom.sidebar', function (within) {
+             if (!within && !selection.select('.inspector-hover').empty()) {
+               hover([]);
              }
-           }
+           });
+         }
 
-           function inputevent() {
-             _geocodeResults = undefined;
-             drawList();
-           }
+         sidebar.showPresetList = function () {};
 
-           function clearSearch() {
-             search.property('value', '');
-             drawList();
-           }
+         sidebar.hover = function () {};
 
-           function mapDrawn(e) {
-             if (e.full) {
-               drawList();
-             }
-           }
+         sidebar.hover.cancel = function () {};
 
-           function features() {
-             var result = [];
-             var graph = context.graph();
-             var visibleCenter = context.map().extent().center();
-             var q = search.property('value').toLowerCase();
-             if (!q) return result;
-             var locationMatch = pair_1(q.toUpperCase()) || q.match(/^(-?\d+\.?\d*)\s+(-?\d+\.?\d*)$/);
+         sidebar.intersects = function () {};
 
-             if (locationMatch) {
-               var loc = [parseFloat(locationMatch[0]), parseFloat(locationMatch[1])];
-               result.push({
-                 id: -1,
-                 geometry: 'point',
-                 type: _t('inspector.location'),
-                 name: dmsCoordinatePair([loc[1], loc[0]]),
-                 location: loc
-               });
-             } // A location search takes priority over an ID search
+         sidebar.select = function () {};
 
+         sidebar.show = function () {};
 
-             var idMatch = !locationMatch && q.match(/(?:^|\W)(node|way|relation|[nwr])\W?0*([1-9]\d*)(?:\W|$)/i);
+         sidebar.hide = function () {};
 
-             if (idMatch) {
-               var elemType = idMatch[1].charAt(0);
-               var elemId = idMatch[2];
-               result.push({
-                 id: elemType + elemId,
-                 geometry: elemType === 'n' ? 'point' : elemType === 'w' ? 'line' : 'relation',
-                 type: elemType === 'n' ? _t('inspector.node') : elemType === 'w' ? _t('inspector.way') : _t('inspector.relation'),
-                 name: elemId
-               });
-             }
+         sidebar.expand = function () {};
 
-             var allEntities = graph.entities;
-             var localResults = [];
+         sidebar.collapse = function () {};
 
-             for (var id in allEntities) {
-               var entity = allEntities[id];
-               if (!entity) continue;
-               var name = utilDisplayName(entity) || '';
-               if (name.toLowerCase().indexOf(q) < 0) continue;
-               var matched = _mainPresetIndex.match(entity, graph);
-               var type = matched && matched.name() || utilDisplayType(entity.id);
-               var extent = entity.extent(graph);
-               var distance = extent ? geoSphericalDistance(visibleCenter, extent.center()) : 0;
-               localResults.push({
-                 id: entity.id,
-                 entity: entity,
-                 geometry: entity.geometry(graph),
-                 type: type,
-                 name: name,
-                 distance: distance
-               });
-               if (localResults.length > 100) break;
-             }
+         sidebar.toggle = function () {};
 
-             localResults = localResults.sort(function byDistance(a, b) {
-               return a.distance - b.distance;
-             });
-             result = result.concat(localResults);
+         return sidebar;
+       }
 
-             (_geocodeResults || []).forEach(function (d) {
-               if (d.osm_type && d.osm_id) {
-                 // some results may be missing these - #1890
-                 // Make a temporary osmEntity so we can preset match
-                 // and better localize the search result - #4725
-                 var id = osmEntity.id.fromOSM(d.osm_type, d.osm_id);
-                 var tags = {};
-                 tags[d["class"]] = d.type;
-                 var attrs = {
-                   id: id,
-                   type: d.osm_type,
-                   tags: tags
-                 };
+       function uiSourceSwitch(context) {
+         var keys;
 
-                 if (d.osm_type === 'way') {
-                   // for ways, add some fake closed nodes
-                   attrs.nodes = ['a', 'a']; // so that geometry area is possible
-                 }
+         function click(d3_event) {
+           d3_event.preventDefault();
+           var osm = context.connection();
+           if (!osm) return;
+           if (context.inIntro()) return;
+           if (context.history().hasChanges() && !window.confirm(_t('source_switch.lose_changes'))) return;
+           var isLive = select(this).classed('live');
+           isLive = !isLive;
+           context.enter(modeBrowse(context));
+           context.history().clearSaved(); // remove saved history
 
-                 var tempEntity = osmEntity(attrs);
-                 var tempGraph = coreGraph([tempEntity]);
-                 var matched = _mainPresetIndex.match(tempEntity, tempGraph);
-                 var type = matched && matched.name() || utilDisplayType(id);
-                 result.push({
-                   id: tempEntity.id,
-                   geometry: tempEntity.geometry(tempGraph),
-                   type: type,
-                   name: d.display_name,
-                   extent: new geoExtent([parseFloat(d.boundingbox[3]), parseFloat(d.boundingbox[0])], [parseFloat(d.boundingbox[2]), parseFloat(d.boundingbox[1])])
-                 });
-               }
-             });
+           context.flush(); // remove stored data
 
-             if (q.match(/^[0-9]+$/)) {
-               // if query is just a number, possibly an OSM ID without a prefix
-               result.push({
-                 id: 'n' + q,
-                 geometry: 'point',
-                 type: _t('inspector.node'),
-                 name: q
-               });
-               result.push({
-                 id: 'w' + q,
-                 geometry: 'line',
-                 type: _t('inspector.way'),
-                 name: q
-               });
-               result.push({
-                 id: 'r' + q,
-                 geometry: 'relation',
-                 type: _t('inspector.relation'),
-                 name: q
-               });
-             }
+           select(this).html(isLive ? _t.html('source_switch.live') : _t.html('source_switch.dev')).classed('live', isLive).classed('chip', isLive);
+           osm["switch"](isLive ? keys[0] : keys[1]); // switch connection (warning: dispatches 'change' event)
+         }
 
-             return result;
-           }
+         var sourceSwitch = function sourceSwitch(selection) {
+           selection.append('a').attr('href', '#').html(_t.html('source_switch.live')).attr('class', 'live chip').on('click', click);
+         };
 
-           function drawList() {
-             var value = search.property('value');
-             var results = features();
-             list.classed('filtered', value.length);
-             var resultsIndicator = list.selectAll('.no-results-item').data([0]).enter().append('button').property('disabled', true).attr('class', 'no-results-item').call(svgIcon('#iD-icon-alert', 'pre-text'));
-             resultsIndicator.append('span').attr('class', 'entity-name');
-             list.selectAll('.no-results-item .entity-name').html(_t.html('geocoder.no_results_worldwide'));
+         sourceSwitch.keys = function (_) {
+           if (!arguments.length) return keys;
+           keys = _;
+           return sourceSwitch;
+         };
 
-             if (services.geocoder) {
-               list.selectAll('.geocode-item').data([0]).enter().append('button').attr('class', 'geocode-item secondary-action').on('click', geocoderSearch).append('div').attr('class', 'label').append('span').attr('class', 'entity-name').html(_t.html('geocoder.search'));
-             }
+         return sourceSwitch;
+       }
 
-             list.selectAll('.no-results-item').style('display', value.length && !results.length ? 'block' : 'none');
-             list.selectAll('.geocode-item').style('display', value && _geocodeResults === undefined ? 'block' : 'none');
-             list.selectAll('.feature-list-item').data([-1]).remove();
-             var items = list.selectAll('.feature-list-item').data(results, function (d) {
-               return d.id;
-             });
-             var enter = items.enter().insert('button', '.geocode-item').attr('class', 'feature-list-item').on('mouseover', mouseover).on('mouseout', mouseout).on('click', click);
-             var label = enter.append('div').attr('class', 'label');
-             label.each(function (d) {
-               select(this).call(svgIcon('#iD-icon-' + d.geometry, 'pre-text'));
-             });
-             label.append('span').attr('class', 'entity-type').html(function (d) {
-               return d.type;
-             });
-             label.append('span').attr('class', 'entity-name').html(function (d) {
-               return d.name;
-             });
-             enter.style('opacity', 0).transition().style('opacity', 1);
-             items.order();
-             items.exit().remove();
-           }
+       function uiSpinner(context) {
+         var osm = context.connection();
+         return function (selection) {
+           var img = selection.append('img').attr('src', context.imagePath('loader-black.gif')).style('opacity', 0);
 
-           function mouseover(d3_event, d) {
-             if (d.id === -1) return;
-             utilHighlightEntities([d.id], true, context);
+           if (osm) {
+             osm.on('loading.spinner', function () {
+               img.transition().style('opacity', 1);
+             }).on('loaded.spinner', function () {
+               img.transition().style('opacity', 0);
+             });
            }
+         };
+       }
 
-           function mouseout(d3_event, d) {
-             if (d.id === -1) return;
-             utilHighlightEntities([d.id], false, context);
-           }
+       function uiSplash(context) {
+         return function (selection) {
+           // Exception - if there are restorable changes, skip this splash screen.
+           // This is because we currently only support one `uiModal` at a time
+           //  and we need to show them `uiRestore`` instead of this one.
+           if (context.history().hasRestorableChanges()) return; // If user has not seen this version of the privacy policy, show the splash again.
 
-           function click(d3_event, d) {
-             d3_event.preventDefault();
+           var updateMessage = '';
+           var sawPrivacyVersion = corePreferences('sawPrivacyVersion');
+           var showSplash = !corePreferences('sawSplash');
 
-             if (d.location) {
-               context.map().centerZoomEase([d.location[1], d.location[0]], 19);
-             } else if (d.entity) {
-               utilHighlightEntities([d.id], false, context);
-               context.enter(modeSelect(context, [d.entity.id]));
-               context.map().zoomToEase(d.entity);
-             } else {
-               // download, zoom to, and select the entity with the given ID
-               context.zoomToEntity(d.id);
-             }
+           if (sawPrivacyVersion !== context.privacyVersion) {
+             updateMessage = _t('splash.privacy_update');
+             showSplash = true;
            }
 
-           function geocoderSearch() {
-             services.geocoder.search(search.property('value'), function (err, resp) {
-               _geocodeResults = resp || [];
-               drawList();
-             });
-           }
-         }
+           if (!showSplash) return;
+           corePreferences('sawSplash', true);
+           corePreferences('sawPrivacyVersion', context.privacyVersion); // fetch intro graph data now, while user is looking at the splash screen
 
-         return featureList;
+           _mainFileFetcher.get('intro_graph');
+           var modalSelection = uiModal(selection);
+           modalSelection.select('.modal').attr('class', 'modal-splash modal');
+           var introModal = modalSelection.select('.content').append('div').attr('class', 'fillL');
+           introModal.append('div').attr('class', 'modal-section').append('h3').html(_t.html('splash.welcome'));
+           var modalSection = introModal.append('div').attr('class', 'modal-section');
+           modalSection.append('p').html(_t.html('splash.text', {
+             version: context.version,
+             website: '<a target="_blank" href="http://ideditor.blog/">ideditor.blog</a>',
+             github: '<a target="_blank" href="https://github.com/openstreetmap/iD">github.com</a>'
+           }));
+           modalSection.append('p').html(_t.html('splash.privacy', {
+             updateMessage: updateMessage,
+             privacyLink: '<a target="_blank" href="https://github.com/openstreetmap/iD/blob/release/PRIVACY.md">' + _t('splash.privacy_policy') + '</a>'
+           }));
+           var buttonWrap = introModal.append('div').attr('class', 'modal-actions');
+           var walkthrough = buttonWrap.append('button').attr('class', 'walkthrough').on('click', function () {
+             context.container().call(uiIntro(context));
+             modalSelection.close();
+           });
+           walkthrough.append('svg').attr('class', 'logo logo-walkthrough').append('use').attr('xlink:href', '#iD-logo-walkthrough');
+           walkthrough.append('div').html(_t.html('splash.walkthrough'));
+           var startEditing = buttonWrap.append('button').attr('class', 'start-editing').on('click', modalSelection.close);
+           startEditing.append('svg').attr('class', 'logo logo-features').append('use').attr('xlink:href', '#iD-logo-features');
+           startEditing.append('div').html(_t.html('splash.start'));
+           modalSelection.select('button.close').attr('class', 'hide');
+         };
        }
 
-       var getOwnPropertyDescriptor$4 = objectGetOwnPropertyDescriptor.f;
-
+       function uiStatus(context) {
+         var osm = context.connection();
+         return function (selection) {
+           if (!osm) return;
 
+           function update(err, apiStatus) {
+             selection.html('');
 
+             if (err) {
+               if (apiStatus === 'connectionSwitched') {
+                 // if the connection was just switched, we can't rely on
+                 // the status (we're getting the status of the previous api)
+                 return;
+               } else if (apiStatus === 'rateLimited') {
+                 selection.html(_t.html('osm_api_status.message.rateLimit')).append('a').attr('href', '#').attr('class', 'api-status-login').attr('target', '_blank').call(svgIcon('#iD-icon-out-link', 'inline')).append('span').html(_t.html('login')).on('click.login', function (d3_event) {
+                   d3_event.preventDefault();
+                   osm.authenticate();
+                 });
+               } else {
+                 // don't allow retrying too rapidly
+                 var throttledRetry = throttle(function () {
+                   // try loading the visible tiles
+                   context.loadTiles(context.projection); // manually reload the status too in case all visible tiles were already loaded
 
+                   osm.reloadApiStatus();
+                 }, 2000); // eslint-disable-next-line no-warning-comments
+                 // TODO: nice messages for different error types
 
 
-       var nativeStartsWith = ''.startsWith;
-       var min$9 = Math.min;
+                 selection.html(_t.html('osm_api_status.message.error') + ' ').append('a').attr('href', '#') // let the user manually retry their connection directly
+                 .html(_t.html('osm_api_status.retry')).on('click.retry', function (d3_event) {
+                   d3_event.preventDefault();
+                   throttledRetry();
+                 });
+               }
+             } else if (apiStatus === 'readonly') {
+               selection.html(_t.html('osm_api_status.message.readonly'));
+             } else if (apiStatus === 'offline') {
+               selection.html(_t.html('osm_api_status.message.offline'));
+             }
 
-       var CORRECT_IS_REGEXP_LOGIC = correctIsRegexpLogic('startsWith');
-       // https://github.com/zloirock/core-js/pull/702
-       var MDN_POLYFILL_BUG =  !CORRECT_IS_REGEXP_LOGIC && !!function () {
-         var descriptor = getOwnPropertyDescriptor$4(String.prototype, 'startsWith');
-         return descriptor && !descriptor.writable;
-       }();
+             selection.attr('class', 'api-status ' + (err ? 'error' : apiStatus));
+           }
 
-       // `String.prototype.startsWith` method
-       // https://tc39.es/ecma262/#sec-string.prototype.startswith
-       _export({ target: 'String', proto: true, forced: !MDN_POLYFILL_BUG && !CORRECT_IS_REGEXP_LOGIC }, {
-         startsWith: function startsWith(searchString /* , position = 0 */) {
-           var that = String(requireObjectCoercible(this));
-           notARegexp(searchString);
-           var index = toLength(min$9(arguments.length > 1 ? arguments[1] : undefined, that.length));
-           var search = String(searchString);
-           return nativeStartsWith
-             ? nativeStartsWith.call(that, search, index)
-             : that.slice(index, index + search.length) === search;
-         }
-       });
+           osm.on('apiStatusChange.uiStatus', update); // reload the status periodically regardless of other factors
 
-       function uiSectionEntityIssues(context) {
-         var _entityIDs = [];
-         var _issues = [];
+           window.setInterval(function () {
+             osm.reloadApiStatus();
+           }, 90000); // load the initial status in case no OSM data was loaded yet
 
-         var _activeIssueID;
+           osm.reloadApiStatus();
+         };
+       }
 
-         var section = uiSection('entity-issues', context).shouldDisplay(function () {
-           return _issues.length > 0;
-         }).label(function () {
-           return _t('inspector.title_count', {
-             title: _t.html('issues.list_title'),
-             count: _issues.length
-           });
-         }).disclosureContent(renderDisclosureContent);
-         context.validator().on('validated.entity_issues', function () {
-           // Refresh on validated events
-           reloadIssues();
-           section.reRender();
-         }).on('focusedIssue.entity_issues', function (issue) {
-           makeActiveIssue(issue.id);
+       function modeDrawArea(context, wayID, startGraph, button) {
+         var mode = {
+           button: button,
+           id: 'draw-area'
+         };
+         var behavior = behaviorDrawWay(context, wayID, mode, startGraph).on('rejectedSelfIntersection.modeDrawArea', function () {
+           context.ui().flash.iconName('#iD-icon-no').label(_t('self_intersection.error.areas'))();
          });
+         mode.wayID = wayID;
 
-         function reloadIssues() {
-           _issues = context.validator().getSharedEntityIssues(_entityIDs, {
-             includeDisabledRules: true
-           });
-         }
+         mode.enter = function () {
+           context.install(behavior);
+         };
 
-         function makeActiveIssue(issueID) {
-           _activeIssueID = issueID;
-           section.selection().selectAll('.issue-container').classed('active', function (d) {
-             return d.id === _activeIssueID;
-           });
-         }
+         mode.exit = function () {
+           context.uninstall(behavior);
+         };
 
-         function renderDisclosureContent(selection) {
-           selection.classed('grouped-items-area', true);
-           _activeIssueID = _issues.length > 0 ? _issues[0].id : null;
-           var containers = selection.selectAll('.issue-container').data(_issues, function (d) {
-             return d.id;
-           }); // Exit
+         mode.selectedIDs = function () {
+           return [wayID];
+         };
+
+         mode.activeID = function () {
+           return behavior && behavior.activeID() || [];
+         };
 
-           containers.exit().remove(); // Enter
+         return mode;
+       }
 
-           var containersEnter = containers.enter().append('div').attr('class', 'issue-container');
-           var itemsEnter = containersEnter.append('div').attr('class', function (d) {
-             return 'issue severity-' + d.severity;
-           }).on('mouseover.highlight', function (d3_event, d) {
-             // don't hover-highlight the selected entity
-             var ids = d.entityIds.filter(function (e) {
-               return _entityIDs.indexOf(e) === -1;
-             });
-             utilHighlightEntities(ids, true, context);
-           }).on('mouseout.highlight', function (d3_event, d) {
-             var ids = d.entityIds.filter(function (e) {
-               return _entityIDs.indexOf(e) === -1;
-             });
-             utilHighlightEntities(ids, false, context);
-           });
-           var labelsEnter = itemsEnter.append('div').attr('class', 'issue-label');
-           var textEnter = labelsEnter.append('button').attr('class', 'issue-text').on('click', function (d3_event, d) {
-             makeActiveIssue(d.id); // expand only the clicked item
+       function modeAddArea(context, mode) {
+         mode.id = 'add-area';
+         var behavior = behaviorAddWay(context).on('start', start).on('startFromWay', startFromWay).on('startFromNode', startFromNode);
+         var defaultTags = {
+           area: 'yes'
+         };
+         if (mode.preset) defaultTags = mode.preset.setTags(defaultTags, 'area');
 
-             var extent = d.extent(context.graph());
+         function actionClose(wayId) {
+           return function (graph) {
+             return graph.replace(graph.entity(wayId).close());
+           };
+         }
 
-             if (extent) {
-               var setZoom = Math.max(context.map().zoom(), 19);
-               context.map().unobscuredCenterZoomEase(extent.center(), setZoom);
-             }
+         function start(loc) {
+           var startGraph = context.graph();
+           var node = osmNode({
+             loc: loc
            });
-           textEnter.each(function (d) {
-             var iconName = '#iD-icon-' + (d.severity === 'warning' ? 'alert' : 'error');
-             select(this).call(svgIcon(iconName, 'issue-icon'));
+           var way = osmWay({
+             tags: defaultTags
            });
-           textEnter.append('span').attr('class', 'issue-message');
-           var infoButton = labelsEnter.append('button').attr('class', 'issue-info-button').attr('title', _t('icons.information')).call(svgIcon('#iD-icon-inspect'));
-           infoButton.on('click', function (d3_event) {
-             d3_event.stopPropagation();
-             d3_event.preventDefault();
-             this.blur(); // avoid keeping focus on the button - #4641
-
-             var container = select(this.parentNode.parentNode.parentNode);
-             var info = container.selectAll('.issue-info');
-             var isExpanded = info.classed('expanded');
+           context.perform(actionAddEntity(node), actionAddEntity(way), actionAddVertex(way.id, node.id), actionClose(way.id));
+           context.enter(modeDrawArea(context, way.id, startGraph, mode.button));
+         }
 
-             if (isExpanded) {
-               info.transition().duration(200).style('max-height', '0px').style('opacity', '0').on('end', function () {
-                 info.classed('expanded', false);
-               });
-             } else {
-               info.classed('expanded', true).transition().duration(200).style('max-height', '200px').style('opacity', '1').on('end', function () {
-                 info.style('max-height', null);
-               });
-             }
+         function startFromWay(loc, edge) {
+           var startGraph = context.graph();
+           var node = osmNode({
+             loc: loc
            });
-           itemsEnter.append('ul').attr('class', 'issue-fix-list');
-           containersEnter.append('div').attr('class', 'issue-info').style('max-height', '0').style('opacity', '0').each(function (d) {
-             if (typeof d.reference === 'function') {
-               select(this).call(d.reference);
-             } else {
-               select(this).html(_t.html('inspector.no_documentation_key'));
-             }
-           }); // Update
-
-           containers = containers.merge(containersEnter).classed('active', function (d) {
-             return d.id === _activeIssueID;
+           var way = osmWay({
+             tags: defaultTags
            });
-           containers.selectAll('.issue-message').html(function (d) {
-             return d.message(context);
-           }); // fixes
+           context.perform(actionAddEntity(node), actionAddEntity(way), actionAddVertex(way.id, node.id), actionClose(way.id), actionAddMidpoint({
+             loc: loc,
+             edge: edge
+           }, node));
+           context.enter(modeDrawArea(context, way.id, startGraph, mode.button));
+         }
 
-           var fixLists = containers.selectAll('.issue-fix-list');
-           var fixes = fixLists.selectAll('.issue-fix-item').data(function (d) {
-             return d.fixes ? d.fixes(context) : [];
-           }, function (fix) {
-             return fix.id;
+         function startFromNode(node) {
+           var startGraph = context.graph();
+           var way = osmWay({
+             tags: defaultTags
            });
-           fixes.exit().remove();
-           var fixesEnter = fixes.enter().append('li').attr('class', 'issue-fix-item');
-           var buttons = fixesEnter.append('button').on('click', function (d3_event, d) {
-             // not all fixes are actionable
-             if (select(this).attr('disabled') || !d.onClick) return; // Don't run another fix for this issue within a second of running one
-             // (Necessary for "Select a feature type" fix. Most fixes should only ever run once)
+           context.perform(actionAddEntity(way), actionAddVertex(way.id, node.id), actionClose(way.id));
+           context.enter(modeDrawArea(context, way.id, startGraph, mode.button));
+         }
 
-             if (d.issue.dateLastRanFix && new Date() - d.issue.dateLastRanFix < 1000) return;
-             d.issue.dateLastRanFix = new Date(); // remove hover-highlighting
+         mode.enter = function () {
+           context.install(behavior);
+         };
 
-             utilHighlightEntities(d.issue.entityIds.concat(d.entityIds), false, context);
-             new Promise(function (resolve, reject) {
-               d.onClick(context, resolve, reject);
+         mode.exit = function () {
+           context.uninstall(behavior);
+         };
 
-               if (d.onClick.length <= 1) {
-                 // if the fix doesn't take any completion parameters then consider it resolved
-                 resolve();
-               }
-             }).then(function () {
-               // revalidate whenever the fix has finished running successfully
-               context.validator().validate();
-             });
-           }).on('mouseover.highlight', function (d3_event, d) {
-             utilHighlightEntities(d.entityIds, true, context);
-           }).on('mouseout.highlight', function (d3_event, d) {
-             utilHighlightEntities(d.entityIds, false, context);
-           });
-           buttons.each(function (d) {
-             var iconName = d.icon || 'iD-icon-wrench';
+         return mode;
+       }
 
-             if (iconName.startsWith('maki')) {
-               iconName += '-15';
-             }
+       function modeAddLine(context, mode) {
+         mode.id = 'add-line';
+         var behavior = behaviorAddWay(context).on('start', start).on('startFromWay', startFromWay).on('startFromNode', startFromNode);
+         var defaultTags = {};
+         if (mode.preset) defaultTags = mode.preset.setTags(defaultTags, 'line');
 
-             select(this).call(svgIcon('#' + iconName, 'fix-icon'));
+         function start(loc) {
+           var startGraph = context.graph();
+           var node = osmNode({
+             loc: loc
            });
-           buttons.append('span').attr('class', 'fix-message').html(function (d) {
-             return d.title;
+           var way = osmWay({
+             tags: defaultTags
            });
-           fixesEnter.merge(fixes).selectAll('button').classed('actionable', function (d) {
-             return d.onClick;
-           }).attr('disabled', function (d) {
-             return d.onClick ? null : 'true';
-           }).attr('title', function (d) {
-             if (d.disabledReason) {
-               return d.disabledReason;
-             }
+           context.perform(actionAddEntity(node), actionAddEntity(way), actionAddVertex(way.id, node.id));
+           context.enter(modeDrawLine(context, way.id, startGraph, mode.button));
+         }
 
-             return null;
+         function startFromWay(loc, edge) {
+           var startGraph = context.graph();
+           var node = osmNode({
+             loc: loc
            });
+           var way = osmWay({
+             tags: defaultTags
+           });
+           context.perform(actionAddEntity(node), actionAddEntity(way), actionAddVertex(way.id, node.id), actionAddMidpoint({
+             loc: loc,
+             edge: edge
+           }, node));
+           context.enter(modeDrawLine(context, way.id, startGraph, mode.button));
          }
 
-         section.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs;
+         function startFromNode(node) {
+           var startGraph = context.graph();
+           var way = osmWay({
+             tags: defaultTags
+           });
+           context.perform(actionAddEntity(way), actionAddVertex(way.id, node.id));
+           context.enter(modeDrawLine(context, way.id, startGraph, mode.button));
+         }
 
-           if (!_entityIDs || !val || !utilArrayIdentical(_entityIDs, val)) {
-             _entityIDs = val;
-             _activeIssueID = null;
-             reloadIssues();
-           }
+         mode.enter = function () {
+           context.install(behavior);
+         };
 
-           return section;
+         mode.exit = function () {
+           context.uninstall(behavior);
          };
 
-         return section;
+         return mode;
        }
 
-       function uiPresetIcon() {
-         var _preset;
-
-         var _geometry;
-
-         var _sizeClass = 'medium';
-
-         function isSmall() {
-           return _sizeClass === 'small';
-         }
+       function modeAddPoint(context, mode) {
+         mode.id = 'add-point';
+         var behavior = behaviorDraw(context).on('click', add).on('clickWay', addWay).on('clickNode', addNode).on('cancel', cancel).on('finish', cancel);
+         var defaultTags = {};
+         if (mode.preset) defaultTags = mode.preset.setTags(defaultTags, 'point');
 
-         function presetIcon(selection) {
-           selection.each(render);
+         function add(loc) {
+           var node = osmNode({
+             loc: loc,
+             tags: defaultTags
+           });
+           context.perform(actionAddEntity(node), _t('operations.add.annotation.point'));
+           enterSelectMode(node);
          }
 
-         function getIcon(p, geom) {
-           if (isSmall() && p.isFallback && p.isFallback()) return 'iD-icon-' + p.id;else if (p.icon) return p.icon;else if (geom === 'line') return 'iD-other-line';else if (geom === 'vertex') return p.isFallback() ? '' : 'temaki-vertex';else if (isSmall() && geom === 'point') return '';else return 'maki-marker-stroked';
+         function addWay(loc, edge) {
+           var node = osmNode({
+             tags: defaultTags
+           });
+           context.perform(actionAddMidpoint({
+             loc: loc,
+             edge: edge
+           }, node), _t('operations.add.annotation.vertex'));
+           enterSelectMode(node);
          }
 
-         function renderPointBorder(container, drawPoint) {
-           var pointBorder = container.selectAll('.preset-icon-point-border').data(drawPoint ? [0] : []);
-           pointBorder.exit().remove();
-           var pointBorderEnter = pointBorder.enter();
-           var w = 40;
-           var h = 40;
-           pointBorderEnter.append('svg').attr('class', 'preset-icon-fill preset-icon-point-border').attr('width', w).attr('height', h).attr('viewBox', "0 0 ".concat(w, " ").concat(h)).append('path').attr('transform', 'translate(11.5, 8)').attr('d', 'M 17,8 C 17,13 11,21 8.5,23.5 C 6,21 0,13 0,8 C 0,4 4,-0.5 8.5,-0.5 C 13,-0.5 17,4 17,8 z');
-           pointBorder = pointBorderEnter.merge(pointBorder);
+         function enterSelectMode(node) {
+           context.enter(modeSelect(context, [node.id]).newFeature(true));
          }
 
-         function renderCircleFill(container, drawVertex) {
-           var vertexFill = container.selectAll('.preset-icon-fill-vertex').data(drawVertex ? [0] : []);
-           vertexFill.exit().remove();
-           var vertexFillEnter = vertexFill.enter();
-           var w = 60;
-           var h = 60;
-           var d = 40;
-           vertexFillEnter.append('svg').attr('class', 'preset-icon-fill preset-icon-fill-vertex').attr('width', w).attr('height', h).attr('viewBox', "0 0 ".concat(w, " ").concat(h)).append('circle').attr('cx', w / 2).attr('cy', h / 2).attr('r', d / 2);
-           vertexFill = vertexFillEnter.merge(vertexFill);
-         }
+         function addNode(node) {
+           if (Object.keys(defaultTags).length === 0) {
+             enterSelectMode(node);
+             return;
+           }
 
-         function renderSquareFill(container, drawArea, tagClasses) {
-           var fill = container.selectAll('.preset-icon-fill-area').data(drawArea ? [0] : []);
-           fill.exit().remove();
-           var fillEnter = fill.enter();
-           var d = isSmall() ? 40 : 60;
-           var w = d;
-           var h = d;
-           var l = d * 2 / 3;
-           var c1 = (w - l) / 2;
-           var c2 = c1 + l;
-           fillEnter = fillEnter.append('svg').attr('class', 'preset-icon-fill preset-icon-fill-area').attr('width', w).attr('height', h).attr('viewBox', "0 0 ".concat(w, " ").concat(h));
-           ['fill', 'stroke'].forEach(function (klass) {
-             fillEnter.append('path').attr('d', "M".concat(c1, " ").concat(c1, " L").concat(c1, " ").concat(c2, " L").concat(c2, " ").concat(c2, " L").concat(c2, " ").concat(c1, " Z")).attr('class', "line area ".concat(klass));
-           });
-           var rVertex = 2.5;
-           [[c1, c1], [c1, c2], [c2, c2], [c2, c1]].forEach(function (point) {
-             fillEnter.append('circle').attr('class', 'vertex').attr('cx', point[0]).attr('cy', point[1]).attr('r', rVertex);
-           });
+           var tags = Object.assign({}, node.tags); // shallow copy
 
-           if (!isSmall()) {
-             var rMidpoint = 1.25;
-             [[c1, w / 2], [c2, w / 2], [h / 2, c1], [h / 2, c2]].forEach(function (point) {
-               fillEnter.append('circle').attr('class', 'midpoint').attr('cx', point[0]).attr('cy', point[1]).attr('r', rMidpoint);
-             });
+           for (var key in defaultTags) {
+             tags[key] = defaultTags[key];
            }
 
-           fill = fillEnter.merge(fill);
-           fill.selectAll('path.stroke').attr('class', "area stroke ".concat(tagClasses));
-           fill.selectAll('path.fill').attr('class', "area fill ".concat(tagClasses));
+           context.perform(actionChangeTags(node.id, tags), _t('operations.add.annotation.point'));
+           enterSelectMode(node);
          }
 
-         function renderLine(container, drawLine, tagClasses) {
-           var line = container.selectAll('.preset-icon-line').data(drawLine ? [0] : []);
-           line.exit().remove();
-           var lineEnter = line.enter();
-           var d = isSmall() ? 40 : 60; // draw the line parametrically
-
-           var w = d;
-           var h = d;
-           var y = Math.round(d * 0.72);
-           var l = Math.round(d * 0.6);
-           var r = 2.5;
-           var x1 = (w - l) / 2;
-           var x2 = x1 + l;
-           lineEnter = lineEnter.append('svg').attr('class', 'preset-icon-line').attr('width', w).attr('height', h).attr('viewBox', "0 0 ".concat(w, " ").concat(h));
-           ['casing', 'stroke'].forEach(function (klass) {
-             lineEnter.append('path').attr('d', "M".concat(x1, " ").concat(y, " L").concat(x2, " ").concat(y)).attr('class', "line ".concat(klass));
-           });
-           [[x1 - 1, y], [x2 + 1, y]].forEach(function (point) {
-             lineEnter.append('circle').attr('class', 'vertex').attr('cx', point[0]).attr('cy', point[1]).attr('r', r);
-           });
-           line = lineEnter.merge(line);
-           line.selectAll('path.stroke').attr('class', "line stroke ".concat(tagClasses));
-           line.selectAll('path.casing').attr('class', "line casing ".concat(tagClasses));
+         function cancel() {
+           context.enter(modeBrowse(context));
          }
 
-         function renderRoute(container, drawRoute, p) {
-           var route = container.selectAll('.preset-icon-route').data(drawRoute ? [0] : []);
-           route.exit().remove();
-           var routeEnter = route.enter();
-           var d = isSmall() ? 40 : 60; // draw the route parametrically
+         mode.enter = function () {
+           context.install(behavior);
+         };
 
-           var w = d;
-           var h = d;
-           var y1 = Math.round(d * 0.80);
-           var y2 = Math.round(d * 0.68);
-           var l = Math.round(d * 0.6);
-           var r = 2;
-           var x1 = (w - l) / 2;
-           var x2 = x1 + l / 3;
-           var x3 = x2 + l / 3;
-           var x4 = x3 + l / 3;
-           routeEnter = routeEnter.append('svg').attr('class', 'preset-icon-route').attr('width', w).attr('height', h).attr('viewBox', "0 0 ".concat(w, " ").concat(h));
-           ['casing', 'stroke'].forEach(function (klass) {
-             routeEnter.append('path').attr('d', "M".concat(x1, " ").concat(y1, " L").concat(x2, " ").concat(y2)).attr('class', "segment0 line ".concat(klass));
-             routeEnter.append('path').attr('d', "M".concat(x2, " ").concat(y2, " L").concat(x3, " ").concat(y1)).attr('class', "segment1 line ".concat(klass));
-             routeEnter.append('path').attr('d', "M".concat(x3, " ").concat(y1, " L").concat(x4, " ").concat(y2)).attr('class', "segment2 line ".concat(klass));
-           });
-           [[x1, y1], [x2, y2], [x3, y1], [x4, y2]].forEach(function (point) {
-             routeEnter.append('circle').attr('class', 'vertex').attr('cx', point[0]).attr('cy', point[1]).attr('r', r);
-           });
-           route = routeEnter.merge(route);
+         mode.exit = function () {
+           context.uninstall(behavior);
+         };
 
-           if (drawRoute) {
-             var routeType = p.tags.type === 'waterway' ? 'waterway' : p.tags.route;
-             var segmentPresetIDs = routeSegments[routeType];
+         return mode;
+       }
 
-             for (var i in segmentPresetIDs) {
-               var segmentPreset = _mainPresetIndex.item(segmentPresetIDs[i]);
-               var segmentTagClasses = svgTagClasses().getClassesString(segmentPreset.tags, '');
-               route.selectAll("path.stroke.segment".concat(i)).attr('class', "segment".concat(i, " line stroke ").concat(segmentTagClasses));
-               route.selectAll("path.casing.segment".concat(i)).attr('class', "segment".concat(i, " line casing ").concat(segmentTagClasses));
-             }
-           }
-         } // Route icons are drawn with a zigzag annotation underneath:
-         //     o   o
-         //    / \ /
-         //   o   o
-         // This dataset defines the styles that are used to draw the zigzag segments.
+       function modeSelectNote(context, selectedNoteID) {
+         var mode = {
+           id: 'select-note',
+           button: 'browse'
+         };
 
+         var _keybinding = utilKeybinding('select-note');
 
-         var routeSegments = {
-           bicycle: ['highway/cycleway', 'highway/cycleway', 'highway/cycleway'],
-           bus: ['highway/unclassified', 'highway/secondary', 'highway/primary'],
-           trolleybus: ['highway/unclassified', 'highway/secondary', 'highway/primary'],
-           detour: ['highway/tertiary', 'highway/residential', 'highway/unclassified'],
-           ferry: ['route/ferry', 'route/ferry', 'route/ferry'],
-           foot: ['highway/footway', 'highway/footway', 'highway/footway'],
-           hiking: ['highway/path', 'highway/path', 'highway/path'],
-           horse: ['highway/bridleway', 'highway/bridleway', 'highway/bridleway'],
-           light_rail: ['railway/light_rail', 'railway/light_rail', 'railway/light_rail'],
-           monorail: ['railway/monorail', 'railway/monorail', 'railway/monorail'],
-           pipeline: ['man_made/pipeline', 'man_made/pipeline', 'man_made/pipeline'],
-           piste: ['piste/downhill', 'piste/hike', 'piste/nordic'],
-           power: ['power/line', 'power/line', 'power/line'],
-           road: ['highway/secondary', 'highway/primary', 'highway/trunk'],
-           subway: ['railway/subway', 'railway/subway', 'railway/subway'],
-           train: ['railway/rail', 'railway/rail', 'railway/rail'],
-           tram: ['railway/tram', 'railway/tram', 'railway/tram'],
-           waterway: ['waterway/stream', 'waterway/stream', 'waterway/stream']
-         };
+         var _noteEditor = uiNoteEditor(context).on('change', function () {
+           context.map().pan([0, 0]); // trigger a redraw
 
-         function render() {
-           var p = _preset.apply(this, arguments);
+           var note = checkSelectedID();
+           if (!note) return;
+           context.ui().sidebar.show(_noteEditor.note(note));
+         });
 
-           var geom = _geometry ? _geometry.apply(this, arguments) : null;
+         var _behaviors = [behaviorBreathe(), behaviorHover(context), behaviorSelect(context), behaviorLasso(context), modeDragNode(context).behavior, modeDragNote(context).behavior];
+         var _newFeature = false;
 
-           if (geom === 'relation' && p.tags && (p.tags.type === 'route' && p.tags.route && routeSegments[p.tags.route] || p.tags.type === 'waterway')) {
-             geom = 'route';
+         function checkSelectedID() {
+           if (!services.osm) return;
+           var note = services.osm.getNote(selectedNoteID);
+
+           if (!note) {
+             context.enter(modeBrowse(context));
            }
 
-           var showThirdPartyIcons = corePreferences('preferences.privacy.thirdpartyicons') || 'true';
-           var isFallback = isSmall() && p.isFallback && p.isFallback();
-           var imageURL = showThirdPartyIcons === 'true' && p.imageURL;
-           var picon = getIcon(p, geom);
-           var isMaki = picon && /^maki-/.test(picon);
-           var isTemaki = picon && /^temaki-/.test(picon);
-           var isFa = picon && /^fa[srb]-/.test(picon);
-           var isiDIcon = picon && !(isMaki || isTemaki || isFa);
-           var isCategory = !p.setTags;
-           var drawPoint = picon && geom === 'point' && isSmall() && !isFallback;
-           var drawVertex = picon !== null && geom === 'vertex' && (!isSmall() || !isFallback);
-           var drawLine = picon && geom === 'line' && !isFallback && !isCategory;
-           var drawArea = picon && geom === 'area' && !isFallback;
-           var drawRoute = picon && geom === 'route';
-           var isFramed = drawVertex || drawArea || drawLine || drawRoute;
-           var tags = !isCategory ? p.setTags({}, geom) : {};
+           return note;
+         } // class the note as selected, or return to browse mode if the note is gone
 
-           for (var k in tags) {
-             if (tags[k] === '*') {
-               tags[k] = 'yes';
+
+         function selectNote(d3_event, drawn) {
+           if (!checkSelectedID()) return;
+           var selection = context.surface().selectAll('.layer-notes .note-' + selectedNoteID);
+
+           if (selection.empty()) {
+             // Return to browse mode if selected DOM elements have
+             // disappeared because the user moved them out of view..
+             var source = d3_event && d3_event.type === 'zoom' && d3_event.sourceEvent;
+
+             if (drawn && source && (source.type === 'pointermove' || source.type === 'mousemove' || source.type === 'touchmove')) {
+               context.enter(modeBrowse(context));
              }
+           } else {
+             selection.classed('selected', true);
+             context.selectedNoteID(selectedNoteID);
            }
+         }
 
-           var tagClasses = svgTagClasses().getClassesString(tags, '');
-           var selection = select(this);
-           var container = selection.selectAll('.preset-icon-container').data([0]);
-           container = container.enter().append('div').attr('class', "preset-icon-container ".concat(_sizeClass)).merge(container);
-           container.classed('showing-img', !!imageURL).classed('fallback', isFallback);
-           renderPointBorder(container, drawPoint);
-           renderCircleFill(container, drawVertex);
-           renderSquareFill(container, drawArea, tagClasses);
-           renderLine(container, drawLine, tagClasses);
-           renderRoute(container, drawRoute, p);
-           var icon = container.selectAll('.preset-icon').data(picon ? [0] : []);
-           icon.exit().remove();
-           icon = icon.enter().append('div').attr('class', 'preset-icon').call(svgIcon('')).merge(icon);
-           icon.attr('class', 'preset-icon ' + (geom ? geom + '-geom' : '')).classed('framed', isFramed).classed('preset-icon-iD', isiDIcon);
-           icon.selectAll('svg').attr('class', 'icon ' + picon + ' ' + (!isiDIcon && geom !== 'line' ? '' : tagClasses));
-           icon.selectAll('use').attr('href', '#' + picon + (isMaki ? isSmall() && geom === 'point' ? '-11' : '-15' : ''));
-           var imageIcon = container.selectAll('img.image-icon').data(imageURL ? [0] : []);
-           imageIcon.exit().remove();
-           imageIcon = imageIcon.enter().append('img').attr('class', 'image-icon').on('load', function () {
-             return container.classed('showing-img', true);
-           }).on('error', function () {
-             return container.classed('showing-img', false);
-           }).merge(imageIcon);
-           imageIcon.attr('src', imageURL);
+         function esc() {
+           if (context.container().select('.combobox').size()) return;
+           context.enter(modeBrowse(context));
          }
 
-         presetIcon.preset = function (val) {
-           if (!arguments.length) return _preset;
-           _preset = utilFunctor(val);
-           return presetIcon;
+         mode.zoomToSelected = function () {
+           if (!services.osm) return;
+           var note = services.osm.getNote(selectedNoteID);
+
+           if (note) {
+             context.map().centerZoomEase(note.loc, 20);
+           }
          };
 
-         presetIcon.geometry = function (val) {
-           if (!arguments.length) return _geometry;
-           _geometry = utilFunctor(val);
-           return presetIcon;
+         mode.newFeature = function (val) {
+           if (!arguments.length) return _newFeature;
+           _newFeature = val;
+           return mode;
          };
 
-         presetIcon.sizeClass = function (val) {
-           if (!arguments.length) return _sizeClass;
-           _sizeClass = val;
-           return presetIcon;
+         mode.enter = function () {
+           var note = checkSelectedID();
+           if (!note) return;
+
+           _behaviors.forEach(context.install);
+
+           _keybinding.on(_t('inspector.zoom_to.key'), mode.zoomToSelected).on('⎋', esc, true);
+
+           select(document).call(_keybinding);
+           selectNote();
+           var sidebar = context.ui().sidebar;
+           sidebar.show(_noteEditor.note(note).newNote(_newFeature)); // expand the sidebar, avoid obscuring the note if needed
+
+           sidebar.expand(sidebar.intersects(note.extent()));
+           context.map().on('drawn.select', selectNote);
          };
 
-         return presetIcon;
-       }
+         mode.exit = function () {
+           _behaviors.forEach(context.uninstall);
 
-       function uiSectionFeatureType(context) {
-         var dispatch$1 = dispatch('choose');
-         var _entityIDs = [];
-         var _presets = [];
+           select(document).call(_keybinding.unbind);
+           context.surface().selectAll('.layer-notes .selected').classed('selected hover', false);
+           context.map().on('drawn.select', null);
+           context.ui().sidebar.hide();
+           context.selectedNoteID(null);
+         };
 
-         var _tagReference;
+         return mode;
+       }
 
-         var section = uiSection('feature-type', context).label(_t.html('inspector.feature_type')).disclosureContent(renderDisclosureContent);
+       function modeAddNote(context) {
+         var mode = {
+           id: 'add-note',
+           button: 'note',
+           description: _t.html('modes.add_note.description'),
+           key: _t('modes.add_note.key')
+         };
+         var behavior = behaviorDraw(context).on('click', add).on('cancel', cancel).on('finish', cancel);
 
-         function renderDisclosureContent(selection) {
-           selection.classed('preset-list-item', true);
-           selection.classed('mixed-types', _presets.length > 1);
-           var presetButtonWrap = selection.selectAll('.preset-list-button-wrap').data([0]).enter().append('div').attr('class', 'preset-list-button-wrap');
-           var presetButton = presetButtonWrap.append('button').attr('class', 'preset-list-button preset-reset').call(uiTooltip().title(_t.html('inspector.back_tooltip')).placement('bottom'));
-           presetButton.append('div').attr('class', 'preset-icon-container');
-           presetButton.append('div').attr('class', 'label').append('div').attr('class', 'label-inner');
-           presetButtonWrap.append('div').attr('class', 'accessory-buttons');
-           var tagReferenceBodyWrap = selection.selectAll('.tag-reference-body-wrap').data([0]);
-           tagReferenceBodyWrap = tagReferenceBodyWrap.enter().append('div').attr('class', 'tag-reference-body-wrap').merge(tagReferenceBodyWrap); // update header
+         function add(loc) {
+           var osm = services.osm;
+           if (!osm) return;
+           var note = osmNote({
+             loc: loc,
+             status: 'open',
+             comments: []
+           });
+           osm.replaceNote(note); // force a reraw (there is no history change that would otherwise do this)
 
-           if (_tagReference) {
-             selection.selectAll('.preset-list-button-wrap .accessory-buttons').style('display', _presets.length === 1 ? null : 'none').call(_tagReference.button);
-             tagReferenceBodyWrap.style('display', _presets.length === 1 ? null : 'none').call(_tagReference.body);
-           }
+           context.map().pan([0, 0]);
+           context.selectedNoteID(note.id).enter(modeSelectNote(context, note.id).newFeature(true));
+         }
 
-           selection.selectAll('.preset-reset').on('click', function () {
-             dispatch$1.call('choose', this, _presets);
-           }).on('pointerdown pointerup mousedown mouseup', function (d3_event) {
-             d3_event.preventDefault();
-             d3_event.stopPropagation();
-           });
-           var geometries = entityGeometries();
-           selection.select('.preset-list-item button').call(uiPresetIcon().geometry(_presets.length === 1 ? geometries.length === 1 && geometries[0] : null).preset(_presets.length === 1 ? _presets[0] : _mainPresetIndex.item('point')));
-           var names = _presets.length === 1 ? [_presets[0].nameLabel(), _presets[0].subtitleLabel()].filter(Boolean) : [_t('inspector.multiple_types')];
-           var label = selection.select('.label-inner');
-           var nameparts = label.selectAll('.namepart').data(names, function (d) {
-             return d;
-           });
-           nameparts.exit().remove();
-           nameparts.enter().append('div').attr('class', 'namepart').html(function (d) {
-             return d;
-           });
+         function cancel() {
+           context.enter(modeBrowse(context));
          }
 
-         section.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs;
-           _entityIDs = val;
-           return section;
+         mode.enter = function () {
+           context.install(behavior);
          };
 
-         section.presets = function (val) {
-           if (!arguments.length) return _presets; // don't reload the same preset
+         mode.exit = function () {
+           context.uninstall(behavior);
+         };
 
-           if (!utilArrayIdentical(val, _presets)) {
-             _presets = val;
+         return mode;
+       }
 
-             if (_presets.length === 1) {
-               _tagReference = uiTagReference(_presets[0].reference()).showing(false);
-             }
+       var JXON = new function () {
+         var sValueProp = 'keyValue',
+             sAttributesProp = 'keyAttributes',
+             sAttrPref = '@',
+
+         /* you can customize these values */
+         aCache = [],
+             rIsNull = /^\s*$/,
+             rIsBool = /^(?:true|false)$/i;
+
+         function parseText(sValue) {
+           if (rIsNull.test(sValue)) {
+             return null;
            }
 
-           return section;
-         };
+           if (rIsBool.test(sValue)) {
+             return sValue.toLowerCase() === 'true';
+           }
 
-         function entityGeometries() {
-           var counts = {};
+           if (isFinite(sValue)) {
+             return parseFloat(sValue);
+           }
 
-           for (var i in _entityIDs) {
-             var geometry = context.graph().geometry(_entityIDs[i]);
-             if (!counts[geometry]) counts[geometry] = 0;
-             counts[geometry] += 1;
+           if (isFinite(Date.parse(sValue))) {
+             return new Date(sValue);
            }
 
-           return Object.keys(counts).sort(function (geom1, geom2) {
-             return counts[geom2] - counts[geom1];
-           });
+           return sValue;
          }
 
-         return utilRebind(section, dispatch$1, 'on');
-       }
-
-       // It borrows some code from uiHelp
+         function EmptyTree() {}
 
-       function uiFieldHelp(context, fieldName) {
-         var fieldHelp = {};
+         EmptyTree.prototype.toString = function () {
+           return 'null';
+         };
 
-         var _inspector = select(null);
+         EmptyTree.prototype.valueOf = function () {
+           return null;
+         };
 
-         var _wrap = select(null);
+         function objectify(vValue) {
+           return vValue === null ? new EmptyTree() : vValue instanceof Object ? vValue : new vValue.constructor(vValue);
+         }
 
-         var _body = select(null);
+         function createObjTree(oParentNode, nVerb, bFreeze, bNesteAttr) {
+           var nLevelStart = aCache.length,
+               bChildren = oParentNode.hasChildNodes(),
+               bAttributes = oParentNode.hasAttributes(),
+               bHighVerb = Boolean(nVerb & 2);
+           var sProp,
+               vContent,
+               nLength = 0,
+               sCollectedTxt = '',
+               vResult = bHighVerb ? {} :
+           /* put here the default value for empty nodes: */
+           true;
 
-         var fieldHelpKeys = {
-           restrictions: [['about', ['about', 'from_via_to', 'maxdist', 'maxvia']], ['inspecting', ['about', 'from_shadow', 'allow_shadow', 'restrict_shadow', 'only_shadow', 'restricted', 'only']], ['modifying', ['about', 'indicators', 'allow_turn', 'restrict_turn', 'only_turn']], ['tips', ['simple', 'simple_example', 'indirect', 'indirect_example', 'indirect_noedit']]]
-         };
-         var fieldHelpHeadings = {};
-         var replacements = {
-           distField: _t.html('restriction.controls.distance'),
-           viaField: _t.html('restriction.controls.via'),
-           fromShadow: icon('#iD-turn-shadow', 'inline shadow from'),
-           allowShadow: icon('#iD-turn-shadow', 'inline shadow allow'),
-           restrictShadow: icon('#iD-turn-shadow', 'inline shadow restrict'),
-           onlyShadow: icon('#iD-turn-shadow', 'inline shadow only'),
-           allowTurn: icon('#iD-turn-yes', 'inline turn'),
-           restrictTurn: icon('#iD-turn-no', 'inline turn'),
-           onlyTurn: icon('#iD-turn-only', 'inline turn')
-         }; // For each section, squash all the texts into a single markdown document
+           if (bChildren) {
+             for (var oNode, nItem = 0; nItem < oParentNode.childNodes.length; nItem++) {
+               oNode = oParentNode.childNodes.item(nItem);
 
-         var docs = fieldHelpKeys[fieldName].map(function (key) {
-           var helpkey = 'help.field.' + fieldName + '.' + key[0];
-           var text = key[1].reduce(function (all, part) {
-             var subkey = helpkey + '.' + part;
-             var depth = fieldHelpHeadings[subkey]; // is this subkey a heading?
+               if (oNode.nodeType === 4) {
+                 /* nodeType is 'CDATASection' (4) */
+                 sCollectedTxt += oNode.nodeValue;
+               } else if (oNode.nodeType === 3) {
+                 /* nodeType is 'Text' (3) */
+                 sCollectedTxt += oNode.nodeValue.trim();
+               } else if (oNode.nodeType === 1 && !oNode.prefix) {
+                 /* nodeType is 'Element' (1) */
+                 aCache.push(oNode);
+               }
+             }
+           }
 
-             var hhh = depth ? Array(depth + 1).join('#') + ' ' : ''; // if so, prepend with some ##'s
+           var nLevelEnd = aCache.length,
+               vBuiltVal = parseText(sCollectedTxt);
 
-             return all + hhh + _t.html(subkey, replacements) + '\n\n';
-           }, '');
-           return {
-             key: helpkey,
-             title: _t.html(helpkey + '.title'),
-             html: marked_1(text.trim())
-           };
-         });
+           if (!bHighVerb && (bChildren || bAttributes)) {
+             vResult = nVerb === 0 ? objectify(vBuiltVal) : {};
+           }
 
-         function show() {
-           updatePosition();
+           for (var nElId = nLevelStart; nElId < nLevelEnd; nElId++) {
+             sProp = aCache[nElId].nodeName.toLowerCase();
+             vContent = createObjTree(aCache[nElId], nVerb, bFreeze, bNesteAttr);
 
-           _body.classed('hide', false).style('opacity', '0').transition().duration(200).style('opacity', '1');
-         }
+             if (vResult.hasOwnProperty(sProp)) {
+               if (vResult[sProp].constructor !== Array) {
+                 vResult[sProp] = [vResult[sProp]];
+               }
 
-         function hide() {
-           _body.classed('hide', true).transition().duration(200).style('opacity', '0').on('end', function () {
-             _body.classed('hide', true);
-           });
-         }
+               vResult[sProp].push(vContent);
+             } else {
+               vResult[sProp] = vContent;
+               nLength++;
+             }
+           }
 
-         function clickHelp(index) {
-           var d = docs[index];
-           var tkeys = fieldHelpKeys[fieldName][index][1];
+           if (bAttributes) {
+             var nAttrLen = oParentNode.attributes.length,
+                 sAPrefix = bNesteAttr ? '' : sAttrPref,
+                 oAttrParent = bNesteAttr ? {} : vResult;
 
-           _body.selectAll('.field-help-nav-item').classed('active', function (d, i) {
-             return i === index;
-           });
+             for (var oAttrib, nAttrib = 0; nAttrib < nAttrLen; nLength++, nAttrib++) {
+               oAttrib = oParentNode.attributes.item(nAttrib);
+               oAttrParent[sAPrefix + oAttrib.name.toLowerCase()] = parseText(oAttrib.value.trim());
+             }
 
-           var content = _body.selectAll('.field-help-content').html(d.html); // class the paragraphs so we can find and style them
+             if (bNesteAttr) {
+               if (bFreeze) {
+                 Object.freeze(oAttrParent);
+               }
 
+               vResult[sAttributesProp] = oAttrParent;
+               nLength -= nAttrLen - 1;
+             }
+           }
 
-           content.selectAll('p').attr('class', function (d, i) {
-             return tkeys[i];
-           }); // insert special content for certain help sections
+           if (nVerb === 3 || (nVerb === 2 || nVerb === 1 && nLength > 0) && sCollectedTxt) {
+             vResult[sValueProp] = vBuiltVal;
+           } else if (!bHighVerb && nLength === 0 && sCollectedTxt) {
+             vResult = vBuiltVal;
+           }
 
-           if (d.key === 'help.field.restrictions.inspecting') {
-             content.insert('img', 'p.from_shadow').attr('class', 'field-help-image cf').attr('src', context.imagePath('tr_inspect.gif'));
-           } else if (d.key === 'help.field.restrictions.modifying') {
-             content.insert('img', 'p.allow_turn').attr('class', 'field-help-image cf').attr('src', context.imagePath('tr_modify.gif'));
+           if (bFreeze && (bHighVerb || nLength > 0)) {
+             Object.freeze(vResult);
            }
+
+           aCache.length = nLevelStart;
+           return vResult;
          }
 
-         fieldHelp.button = function (selection) {
-           if (_body.empty()) return;
-           var button = selection.selectAll('.field-help-button').data([0]); // enter/update
+         function loadObjTree(oXMLDoc, oParentEl, oParentObj) {
+           var vValue, oChild;
 
-           button.enter().append('button').attr('class', 'field-help-button').call(svgIcon('#iD-icon-help')).merge(button).on('click', function (d3_event) {
-             d3_event.stopPropagation();
-             d3_event.preventDefault();
+           if (oParentObj instanceof String || oParentObj instanceof Number || oParentObj instanceof Boolean) {
+             oParentEl.appendChild(oXMLDoc.createTextNode(oParentObj.toString()));
+             /* verbosity level is 0 */
+           } else if (oParentObj.constructor === Date) {
+             oParentEl.appendChild(oXMLDoc.createTextNode(oParentObj.toGMTString()));
+           }
 
-             if (_body.classed('hide')) {
-               show();
-             } else {
-               hide();
+           for (var sName in oParentObj) {
+             vValue = oParentObj[sName];
+
+             if (isFinite(sName) || vValue instanceof Function) {
+               continue;
              }
-           });
-         };
+             /* verbosity level is 0 */
 
-         function updatePosition() {
-           var wrap = _wrap.node();
 
-           var inspector = _inspector.node();
+             if (sName === sValueProp) {
+               if (vValue !== null && vValue !== true) {
+                 oParentEl.appendChild(oXMLDoc.createTextNode(vValue.constructor === Date ? vValue.toGMTString() : String(vValue)));
+               }
+             } else if (sName === sAttributesProp) {
+               /* verbosity level is 3 */
+               for (var sAttrib in vValue) {
+                 oParentEl.setAttribute(sAttrib, vValue[sAttrib]);
+               }
+             } else if (sName.charAt(0) === sAttrPref) {
+               oParentEl.setAttribute(sName.slice(1), vValue);
+             } else if (vValue.constructor === Array) {
+               for (var nItem = 0; nItem < vValue.length; nItem++) {
+                 oChild = oXMLDoc.createElement(sName);
+                 loadObjTree(oXMLDoc, oChild, vValue[nItem]);
+                 oParentEl.appendChild(oChild);
+               }
+             } else {
+               oChild = oXMLDoc.createElement(sName);
 
-           var wRect = wrap.getBoundingClientRect();
-           var iRect = inspector.getBoundingClientRect();
+               if (vValue instanceof Object) {
+                 loadObjTree(oXMLDoc, oChild, vValue);
+               } else if (vValue !== null && vValue !== true) {
+                 oChild.appendChild(oXMLDoc.createTextNode(vValue.toString()));
+               }
 
-           _body.style('top', wRect.top + inspector.scrollTop - iRect.top + 'px');
+               oParentEl.appendChild(oChild);
+             }
+           }
          }
 
-         fieldHelp.body = function (selection) {
-           // This control expects the field to have a form-field-input-wrap div
-           _wrap = selection.selectAll('.form-field-input-wrap');
-           if (_wrap.empty()) return; // absolute position relative to the inspector, so it "floats" above the fields
-
-           _inspector = context.container().select('.sidebar .entity-editor-pane .inspector-body');
-           if (_inspector.empty()) return;
-           _body = _inspector.selectAll('.field-help-body').data([0]);
-
-           var enter = _body.enter().append('div').attr('class', 'field-help-body hide'); // initially hidden
+         this.build = function (oXMLParent, nVerbosity
+         /* optional */
+         , bFreeze
+         /* optional */
+         , bNesteAttributes
+         /* optional */
+         ) {
+           var _nVerb = arguments.length > 1 && typeof nVerbosity === 'number' ? nVerbosity & 3 :
+           /* put here the default verbosity level: */
+           1;
 
+           return createObjTree(oXMLParent, _nVerb, bFreeze || false, arguments.length > 3 ? bNesteAttributes : _nVerb === 3);
+         };
 
-           var titleEnter = enter.append('div').attr('class', 'field-help-title cf');
-           titleEnter.append('h2').attr('class', _mainLocalizer.textDirection() === 'rtl' ? 'fr' : 'fl').html(_t.html('help.field.' + fieldName + '.title'));
-           titleEnter.append('button').attr('class', 'fr close').on('click', function (d3_event) {
-             d3_event.stopPropagation();
-             d3_event.preventDefault();
-             hide();
-           }).call(svgIcon('#iD-icon-close'));
-           var navEnter = enter.append('div').attr('class', 'field-help-nav cf');
-           var titles = docs.map(function (d) {
-             return d.title;
-           });
-           navEnter.selectAll('.field-help-nav-item').data(titles).enter().append('div').attr('class', 'field-help-nav-item').html(function (d) {
-             return d;
-           }).on('click', function (d3_event, d) {
-             d3_event.stopPropagation();
-             d3_event.preventDefault();
-             clickHelp(titles.indexOf(d));
-           });
-           enter.append('div').attr('class', 'field-help-content');
-           _body = _body.merge(enter);
-           clickHelp(0);
+         this.unbuild = function (oObjTree) {
+           var oNewDoc = document.implementation.createDocument('', '', null);
+           loadObjTree(oNewDoc, oNewDoc, oObjTree);
+           return oNewDoc;
          };
 
-         return fieldHelp;
-       }
+         this.stringify = function (oObjTree) {
+           return new XMLSerializer().serializeToString(JXON.unbuild(oObjTree));
+         };
+       }(); // var myObject = JXON.build(doc);
+       // we got our javascript object! try: alert(JSON.stringify(myObject));
+       // var newDoc = JXON.unbuild(myObject);
+       // we got our Document instance! try: alert((new XMLSerializer()).serializeToString(newDoc));
 
-       function uiFieldCheck(field, context) {
-         var dispatch$1 = dispatch('change');
-         var options = field.strings && field.strings.options;
-         var values = [];
-         var texts = [];
+       function uiConflicts(context) {
+         var dispatch = dispatch$8('cancel', 'save');
+         var keybinding = utilKeybinding('conflicts');
 
-         var _tags;
+         var _origChanges;
 
-         var input = select(null);
-         var text = select(null);
-         var label = select(null);
-         var reverser = select(null);
+         var _conflictList;
 
-         var _impliedYes;
+         var _shownConflictIndex;
 
-         var _entityIDs = [];
+         function keybindingOn() {
+           select(document).call(keybinding.on('⎋', cancel, true));
+         }
 
-         var _value;
+         function keybindingOff() {
+           select(document).call(keybinding.unbind);
+         }
 
-         if (options) {
-           for (var k in options) {
-             values.push(k === 'undefined' ? undefined : k);
-             texts.push(field.t.html('options.' + k, {
-               'default': options[k]
-             }));
-           }
-         } else {
-           values = [undefined, 'yes'];
-           texts = [_t.html('inspector.unknown'), _t.html('inspector.check.yes')];
+         function tryAgain() {
+           keybindingOff();
+           dispatch.call('save');
+         }
 
-           if (field.type !== 'defaultCheck') {
-             values.push('no');
-             texts.push(_t.html('inspector.check.no'));
-           }
-         } // Checks tags to see whether an undefined value is "Assumed to be Yes"
+         function cancel() {
+           keybindingOff();
+           dispatch.call('cancel');
+         }
 
+         function conflicts(selection) {
+           keybindingOn();
+           var headerEnter = selection.selectAll('.header').data([0]).enter().append('div').attr('class', 'header fillL');
+           headerEnter.append('button').attr('class', 'fr').on('click', cancel).call(svgIcon('#iD-icon-close'));
+           headerEnter.append('h3').html(_t.html('save.conflict.header'));
+           var bodyEnter = selection.selectAll('.body').data([0]).enter().append('div').attr('class', 'body fillL');
+           var conflictsHelpEnter = bodyEnter.append('div').attr('class', 'conflicts-help').html(_t.html('save.conflict.help')); // Download changes link
 
-         function checkImpliedYes() {
-           _impliedYes = field.id === 'oneway_yes'; // hack: pretend `oneway` field is a `oneway_yes` field
-           // where implied oneway tag exists (e.g. `junction=roundabout`) #2220, #1841
+           var detected = utilDetect();
+           var changeset = new osmChangeset();
+           delete changeset.id; // Export without changeset_id
 
-           if (field.id === 'oneway') {
-             var entity = context.entity(_entityIDs[0]);
+           var data = JXON.stringify(changeset.osmChangeJXON(_origChanges));
+           var blob = new Blob([data], {
+             type: 'text/xml;charset=utf-8;'
+           });
+           var fileName = 'changes.osc';
+           var linkEnter = conflictsHelpEnter.selectAll('.download-changes').append('a').attr('class', 'download-changes');
 
-             for (var key in entity.tags) {
-               if (key in osmOneWayTags && entity.tags[key] in osmOneWayTags[key]) {
-                 _impliedYes = true;
-                 texts[0] = _t.html('presets.fields.oneway_yes.options.undefined');
-                 break;
-               }
-             }
+           if (detected.download) {
+             // All except IE11 and Edge
+             linkEnter // download the data as a file
+             .attr('href', window.URL.createObjectURL(blob)).attr('download', fileName);
+           } else {
+             // IE11 and Edge
+             linkEnter // open data uri in a new tab
+             .attr('target', '_blank').on('click.download', function () {
+               navigator.msSaveBlob(blob, fileName);
+             });
            }
-         }
-
-         function reverserHidden() {
-           if (!context.container().select('div.inspector-hover').empty()) return true;
-           return !(_value === 'yes' || _impliedYes && !_value);
-         }
 
-         function reverserSetText(selection) {
-           var entity = _entityIDs.length && context.hasEntity(_entityIDs[0]);
-           if (reverserHidden() || !entity) return selection;
-           var first = entity.first();
-           var last = entity.isClosed() ? entity.nodes[entity.nodes.length - 2] : entity.last();
-           var pseudoDirection = first < last;
-           var icon = pseudoDirection ? '#iD-icon-forward' : '#iD-icon-backward';
-           selection.selectAll('.reverser-span').html(_t.html('inspector.check.reverser')).call(svgIcon(icon, 'inline'));
-           return selection;
+           linkEnter.call(svgIcon('#iD-icon-load', 'inline')).append('span').html(_t.html('save.conflict.download_changes'));
+           bodyEnter.append('div').attr('class', 'conflict-container fillL3').call(showConflict, 0);
+           bodyEnter.append('div').attr('class', 'conflicts-done').attr('opacity', 0).style('display', 'none').html(_t.html('save.conflict.done'));
+           var buttonsEnter = bodyEnter.append('div').attr('class', 'buttons col12 joined conflicts-buttons');
+           buttonsEnter.append('button').attr('disabled', _conflictList.length > 1).attr('class', 'action conflicts-button col6').html(_t.html('save.title')).on('click.try_again', tryAgain);
+           buttonsEnter.append('button').attr('class', 'secondary-action conflicts-button col6').html(_t.html('confirm.cancel')).on('click.cancel', cancel);
          }
 
-         var check = function check(selection) {
-           checkImpliedYes();
-           label = selection.selectAll('.form-field-input-wrap').data([0]);
-           var enter = label.enter().append('label').attr('class', 'form-field-input-wrap form-field-input-check');
-           enter.append('input').property('indeterminate', field.type !== 'defaultCheck').attr('type', 'checkbox').attr('id', field.domId);
-           enter.append('span').html(texts[0]).attr('class', 'value');
+         function showConflict(selection, index) {
+           index = utilWrap(index, _conflictList.length);
+           _shownConflictIndex = index;
+           var parent = select(selection.node().parentNode); // enable save button if this is the last conflict being reviewed..
 
-           if (field.type === 'onewayCheck') {
-             enter.append('button').attr('class', 'reverser' + (reverserHidden() ? ' hide' : '')).append('span').attr('class', 'reverser-span');
+           if (index === _conflictList.length - 1) {
+             window.setTimeout(function () {
+               parent.select('.conflicts-button').attr('disabled', null);
+               parent.select('.conflicts-done').transition().attr('opacity', 1).style('display', 'block');
+             }, 250);
            }
 
-           label = label.merge(enter);
-           input = label.selectAll('input');
-           text = label.selectAll('span.value');
-           input.on('click', function (d3_event) {
-             d3_event.stopPropagation();
-             var t = {};
+           var conflict = selection.selectAll('.conflict').data([_conflictList[index]]);
+           conflict.exit().remove();
+           var conflictEnter = conflict.enter().append('div').attr('class', 'conflict');
+           conflictEnter.append('h4').attr('class', 'conflict-count').html(_t.html('save.conflict.count', {
+             num: index + 1,
+             total: _conflictList.length
+           }));
+           conflictEnter.append('a').attr('class', 'conflict-description').attr('href', '#').html(function (d) {
+             return d.name;
+           }).on('click', function (d3_event, d) {
+             d3_event.preventDefault();
+             zoomToEntity(d.id);
+           });
+           var details = conflictEnter.append('div').attr('class', 'conflict-detail-container');
+           details.append('ul').attr('class', 'conflict-detail-list').selectAll('li').data(function (d) {
+             return d.details || [];
+           }).enter().append('li').attr('class', 'conflict-detail-item').html(function (d) {
+             return d;
+           });
+           details.append('div').attr('class', 'conflict-choices').call(addChoices);
+           details.append('div').attr('class', 'conflict-nav-buttons joined cf').selectAll('button').data(['previous', 'next']).enter().append('button').html(function (d) {
+             return _t.html('save.conflict.' + d);
+           }).attr('class', 'conflict-nav-button action col6').attr('disabled', function (d, i) {
+             return i === 0 && index === 0 || i === 1 && index === _conflictList.length - 1 || null;
+           }).on('click', function (d3_event, d) {
+             d3_event.preventDefault();
+             var container = parent.selectAll('.conflict-container');
+             var sign = d === 'previous' ? -1 : 1;
+             container.selectAll('.conflict').remove();
+             container.call(showConflict, index + sign);
+           });
+         }
+
+         function addChoices(selection) {
+           var choices = selection.append('ul').attr('class', 'layer-list').selectAll('li').data(function (d) {
+             return d.choices || [];
+           }); // enter
 
-             if (Array.isArray(_tags[field.key])) {
-               if (values.indexOf('yes') !== -1) {
-                 t[field.key] = 'yes';
-               } else {
-                 t[field.key] = values[0];
-               }
-             } else {
-               t[field.key] = values[(values.indexOf(_value) + 1) % values.length];
-             } // Don't cycle through `alternating` or `reversible` states - #4970
-             // (They are supported as translated strings, but should not toggle with clicks)
+           var choicesEnter = choices.enter().append('li').attr('class', 'layer');
+           var labelEnter = choicesEnter.append('label');
+           labelEnter.append('input').attr('type', 'radio').attr('name', function (d) {
+             return d.id;
+           }).on('change', function (d3_event, d) {
+             var ul = this.parentNode.parentNode.parentNode;
+             ul.__data__.chosen = d.id;
+             choose(d3_event, ul, d);
+           });
+           labelEnter.append('span').html(function (d) {
+             return d.text;
+           }); // update
 
+           choicesEnter.merge(choices).each(function (d) {
+             var ul = this.parentNode;
 
-             if (t[field.key] === 'reversible' || t[field.key] === 'alternating') {
-               t[field.key] = values[0];
+             if (ul.__data__.chosen === d.id) {
+               choose(null, ul, d);
              }
+           });
+         }
 
-             dispatch$1.call('change', this, t);
+         function choose(d3_event, ul, datum) {
+           if (d3_event) d3_event.preventDefault();
+           select(ul).selectAll('li').classed('active', function (d) {
+             return d === datum;
+           }).selectAll('input').property('checked', function (d) {
+             return d === datum;
            });
+           var extent = geoExtent();
+           var entity;
+           entity = context.graph().hasEntity(datum.id);
+           if (entity) extent._extend(entity.extent(context.graph()));
+           datum.action();
+           entity = context.graph().hasEntity(datum.id);
+           if (entity) extent._extend(entity.extent(context.graph()));
+           zoomToEntity(datum.id, extent);
+         }
 
-           if (field.type === 'onewayCheck') {
-             reverser = label.selectAll('.reverser');
-             reverser.call(reverserSetText).on('click', function (d3_event) {
-               d3_event.preventDefault();
-               d3_event.stopPropagation();
-               context.perform(function (graph) {
-                 for (var i in _entityIDs) {
-                   graph = actionReverse(_entityIDs[i])(graph);
-                 }
+         function zoomToEntity(id, extent) {
+           context.surface().selectAll('.hover').classed('hover', false);
+           var entity = context.graph().hasEntity(id);
 
-                 return graph;
-               }, _t('operations.reverse.annotation.line', {
-                 n: 1
-               })); // must manually revalidate since no 'change' event was called
+           if (entity) {
+             if (extent) {
+               context.map().trimmedExtent(extent);
+             } else {
+               context.map().zoomToEase(entity);
+             }
 
-               context.validator().validate();
-               select(this).call(reverserSetText);
-             });
+             context.surface().selectAll(utilEntityOrMemberSelector([entity.id], context.graph())).classed('hover', true);
            }
-         };
+         } // The conflict list should be an array of objects like:
+         // {
+         //     id: id,
+         //     name: entityName(local),
+         //     details: merge.conflicts(),
+         //     chosen: 1,
+         //     choices: [
+         //         choice(id, keepMine, forceLocal),
+         //         choice(id, keepTheirs, forceRemote)
+         //     ]
+         // }
 
-         check.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs;
-           _entityIDs = val;
-           return check;
-         };
 
-         check.tags = function (tags) {
-           _tags = tags;
+         conflicts.conflictList = function (_) {
+           if (!arguments.length) return _conflictList;
+           _conflictList = _;
+           return conflicts;
+         };
 
-           function isChecked(val) {
-             return val !== 'no' && val !== '' && val !== undefined && val !== null;
-           }
+         conflicts.origChanges = function (_) {
+           if (!arguments.length) return _origChanges;
+           _origChanges = _;
+           return conflicts;
+         };
 
-           function textFor(val) {
-             if (val === '') val = undefined;
-             var index = values.indexOf(val);
-             return index !== -1 ? texts[index] : '"' + val + '"';
+         conflicts.shownEntityIds = function () {
+           if (_conflictList && typeof _shownConflictIndex === 'number') {
+             return [_conflictList[_shownConflictIndex].id];
            }
 
-           checkImpliedYes();
-           var isMixed = Array.isArray(tags[field.key]);
-           _value = !isMixed && tags[field.key] && tags[field.key].toLowerCase();
-
-           if (field.type === 'onewayCheck' && (_value === '1' || _value === '-1')) {
-             _value = 'yes';
-           }
+           return [];
+         };
 
-           input.property('indeterminate', isMixed || field.type !== 'defaultCheck' && !_value).property('checked', isChecked(_value));
-           text.html(isMixed ? _t.html('inspector.multiple_values') : textFor(_value)).classed('mixed', isMixed);
-           label.classed('set', !!_value);
+         return utilRebind(conflicts, dispatch, 'on');
+       }
 
-           if (field.type === 'onewayCheck') {
-             reverser.classed('hide', reverserHidden()).call(reverserSetText);
-           }
-         };
+       function uiConfirm(selection) {
+         var modalSelection = uiModal(selection);
+         modalSelection.select('.modal').classed('modal-alert', true);
+         var section = modalSelection.select('.content');
+         section.append('div').attr('class', 'modal-section header');
+         section.append('div').attr('class', 'modal-section message-text');
+         var buttons = section.append('div').attr('class', 'modal-section buttons cf');
 
-         check.focus = function () {
-           input.node().focus();
+         modalSelection.okButton = function () {
+           buttons.append('button').attr('class', 'button ok-button action').on('click.confirm', function () {
+             modalSelection.remove();
+           }).html(_t.html('confirm.okay')).node().focus();
+           return modalSelection;
          };
 
-         return utilRebind(check, dispatch$1, 'on');
+         return modalSelection;
        }
 
-       function uiFieldCombo(field, context) {
-         var dispatch$1 = dispatch('change');
+       function uiChangesetEditor(context) {
+         var dispatch = dispatch$8('change');
+         var formFields = uiFormFields(context);
+         var commentCombo = uiCombobox(context, 'comment').caseSensitive(true);
 
-         var _isMulti = field.type === 'multiCombo' || field.type === 'manyCombo';
+         var _fieldsArr;
 
-         var _isNetwork = field.type === 'networkCombo';
+         var _tags;
 
-         var _isSemi = field.type === 'semiCombo';
+         var _changesetID;
 
-         var _optstrings = field.strings && field.strings.options;
+         function changesetEditor(selection) {
+           render(selection);
+         }
 
-         var _optarray = field.options;
+         function render(selection) {
+           var initial = false;
 
-         var _snake_case = field.snake_case || field.snake_case === undefined;
+           if (!_fieldsArr) {
+             initial = true;
+             var presets = _mainPresetIndex;
+             _fieldsArr = [uiField(context, presets.field('comment'), null, {
+               show: true,
+               revert: false
+             }), uiField(context, presets.field('source'), null, {
+               show: false,
+               revert: false
+             }), uiField(context, presets.field('hashtags'), null, {
+               show: false,
+               revert: false
+             })];
 
-         var _combobox = uiCombobox(context, 'combo-' + field.safeid).caseSensitive(field.caseSensitive).minItems(_isMulti || _isSemi ? 1 : 2);
+             _fieldsArr.forEach(function (field) {
+               field.on('change', function (t, onInput) {
+                 dispatch.call('change', field, undefined, t, onInput);
+               });
+             });
+           }
 
-         var _container = select(null);
+           _fieldsArr.forEach(function (field) {
+             field.tags(_tags);
+           });
 
-         var _inputWrap = select(null);
+           selection.call(formFields.fieldsArr(_fieldsArr));
 
-         var _input = select(null);
+           if (initial) {
+             var commentField = selection.select('.form-field-comment textarea');
+             var commentNode = commentField.node();
 
-         var _comboData = [];
-         var _multiData = [];
-         var _entityIDs = [];
+             if (commentNode) {
+               commentNode.focus();
+               commentNode.select();
+             } // trigger a 'blur' event so that comment field can be cleaned
+             // and checked for hashtags, even if retrieved from localstorage
 
-         var _tags;
 
-         var _countryCode;
+             utilTriggerEvent(commentField, 'blur');
+             var osm = context.connection();
 
-         var _staticPlaceholder; // initialize deprecated tags array
+             if (osm) {
+               osm.userChangesets(function (err, changesets) {
+                 if (err) return;
+                 var comments = changesets.map(function (changeset) {
+                   var comment = changeset.tags.comment;
+                   return comment ? {
+                     title: comment,
+                     value: comment
+                   } : null;
+                 }).filter(Boolean);
+                 commentField.call(commentCombo.data(utilArrayUniqBy(comments, 'title')));
+               });
+             }
+           } // Add warning if comment mentions Google
 
 
-         var _dataDeprecated = [];
-         _mainFileFetcher.get('deprecated').then(function (d) {
-           _dataDeprecated = d;
-         })["catch"](function () {
-           /* ignore */
-         }); // ensure multiCombo field.key ends with a ':'
+           var hasGoogle = _tags.comment.match(/google/i);
 
-         if (_isMulti && field.key && /[^:]$/.test(field.key)) {
-           field.key += ':';
+           var commentWarning = selection.select('.form-field-comment').selectAll('.comment-warning').data(hasGoogle ? [0] : []);
+           commentWarning.exit().transition().duration(200).style('opacity', 0).remove();
+           var commentEnter = commentWarning.enter().insert('div', '.tag-reference-body').attr('class', 'field-warning comment-warning').style('opacity', 0);
+           commentEnter.append('a').attr('target', '_blank').call(svgIcon('#iD-icon-alert', 'inline')).attr('href', _t('commit.google_warning_link')).append('span').html(_t.html('commit.google_warning'));
+           commentEnter.transition().duration(200).style('opacity', 1);
          }
 
-         function snake(s) {
-           return s.replace(/\s+/g, '_');
-         }
+         changesetEditor.tags = function (_) {
+           if (!arguments.length) return _tags;
+           _tags = _; // Don't reset _fieldsArr here.
 
-         function unsnake(s) {
-           return s.replace(/_+/g, ' ');
-         }
+           return changesetEditor;
+         };
 
-         function clean(s) {
-           return s.split(';').map(function (s) {
-             return s.trim();
-           }).join(';');
-         } // returns the tag value for a display value
-         // (for multiCombo, dval should be the key suffix, not the entire key)
+         changesetEditor.changesetID = function (_) {
+           if (!arguments.length) return _changesetID;
+           if (_changesetID === _) return changesetEditor;
+           _changesetID = _;
+           _fieldsArr = null;
+           return changesetEditor;
+         };
 
+         return utilRebind(changesetEditor, dispatch, 'on');
+       }
 
-         function tagValue(dval) {
-           dval = clean(dval || '');
+       function uiSectionChanges(context) {
+         var detected = utilDetect();
+         var _discardTags = {};
+         _mainFileFetcher.get('discarded').then(function (d) {
+           _discardTags = d;
+         })["catch"](function () {
+           /* ignore */
+         });
+         var section = uiSection('changes-list', context).label(function () {
+           var history = context.history();
+           var summary = history.difference().summary();
+           return _t('inspector.title_count', {
+             title: _t.html('commit.changes'),
+             count: summary.length
+           });
+         }).disclosureContent(renderDisclosureContent);
 
-           if (_optstrings) {
-             var found = _comboData.find(function (o) {
-               return o.key && clean(o.value) === dval;
-             });
+         function renderDisclosureContent(selection) {
+           var history = context.history();
+           var summary = history.difference().summary();
+           var container = selection.selectAll('.commit-section').data([0]);
+           var containerEnter = container.enter().append('div').attr('class', 'commit-section');
+           containerEnter.append('ul').attr('class', 'changeset-list');
+           container = containerEnter.merge(container);
+           var items = container.select('ul').selectAll('li').data(summary);
+           var itemsEnter = items.enter().append('li').attr('class', 'change-item');
+           var buttons = itemsEnter.append('button').on('mouseover', mouseover).on('mouseout', mouseout).on('click', click);
+           buttons.each(function (d) {
+             select(this).call(svgIcon('#iD-icon-' + d.entity.geometry(d.graph), 'pre-text ' + d.changeType));
+           });
+           buttons.append('span').attr('class', 'change-type').html(function (d) {
+             return _t.html('commit.' + d.changeType) + ' ';
+           });
+           buttons.append('strong').attr('class', 'entity-type').html(function (d) {
+             var matched = _mainPresetIndex.match(d.entity, d.graph);
+             return matched && matched.name() || utilDisplayType(d.entity.id);
+           });
+           buttons.append('span').attr('class', 'entity-name').html(function (d) {
+             var name = utilDisplayName(d.entity) || '',
+                 string = '';
 
-             if (found) {
-               return found.key;
+             if (name !== '') {
+               string += ':';
              }
-           }
-
-           if (field.type === 'typeCombo' && !dval) {
-             return 'yes';
-           }
 
-           return (_snake_case ? snake(dval) : dval) || undefined;
-         } // returns the display value for a tag value
-         // (for multiCombo, tval should be the key suffix, not the entire key)
+             return string += ' ' + name;
+           });
+           items = itemsEnter.merge(items); // Download changeset link
 
+           var changeset = new osmChangeset().update({
+             id: undefined
+           });
+           var changes = history.changes(actionDiscardTags(history.difference(), _discardTags));
+           delete changeset.id; // Export without chnageset_id
 
-         function displayValue(tval) {
-           tval = tval || '';
+           var data = JXON.stringify(changeset.osmChangeJXON(changes));
+           var blob = new Blob([data], {
+             type: 'text/xml;charset=utf-8;'
+           });
+           var fileName = 'changes.osc';
+           var linkEnter = container.selectAll('.download-changes').data([0]).enter().append('a').attr('class', 'download-changes');
 
-           if (_optstrings) {
-             var found = _comboData.find(function (o) {
-               return o.key === tval && o.value;
+           if (detected.download) {
+             // All except IE11 and Edge
+             linkEnter // download the data as a file
+             .attr('href', window.URL.createObjectURL(blob)).attr('download', fileName);
+           } else {
+             // IE11 and Edge
+             linkEnter // open data uri in a new tab
+             .attr('target', '_blank').on('click.download', function () {
+               navigator.msSaveBlob(blob, fileName);
              });
+           }
 
-             if (found) {
-               return found.value;
+           linkEnter.call(svgIcon('#iD-icon-load', 'inline')).append('span').html(_t.html('commit.download_changes'));
+
+           function mouseover(d) {
+             if (d.entity) {
+               context.surface().selectAll(utilEntityOrMemberSelector([d.entity.id], context.graph())).classed('hover', true);
              }
            }
 
-           if (field.type === 'typeCombo' && tval.toLowerCase() === 'yes') {
-             return '';
+           function mouseout() {
+             context.surface().selectAll('.hover').classed('hover', false);
            }
 
-           return _snake_case ? unsnake(tval) : tval;
-         } // Compute the difference between arrays of objects by `value` property
-         //
-         // objectDifference([{value:1}, {value:2}, {value:3}], [{value:2}])
-         // > [{value:1}, {value:3}]
-         //
+           function click(d3_event, change) {
+             if (change.changeType !== 'deleted') {
+               var entity = change.entity;
+               context.map().zoomToEase(entity);
+               context.surface().selectAll(utilEntityOrMemberSelector([entity.id], context.graph())).classed('hover', true);
+             }
+           }
+         }
 
+         return section;
+       }
 
-         function objectDifference(a, b) {
-           return a.filter(function (d1) {
-             return !b.some(function (d2) {
-               return !d2.isMixed && d1.value === d2.value;
-             });
+       function uiCommitWarnings(context) {
+         function commitWarnings(selection) {
+           var issuesBySeverity = context.validator().getIssuesBySeverity({
+             what: 'edited',
+             where: 'all',
+             includeDisabledRules: true
            });
-         }
 
-         function initCombo(selection, attachTo) {
-           if (_optstrings) {
-             selection.attr('readonly', 'readonly');
-             selection.call(_combobox, attachTo);
-             setStaticValues(setPlaceholder);
-           } else if (_optarray) {
-             selection.call(_combobox, attachTo);
-             setStaticValues(setPlaceholder);
-           } else if (services.taginfo) {
-             selection.call(_combobox.fetcher(setTaginfoValues), attachTo);
-             setTaginfoValues('', setPlaceholder);
+           for (var severity in issuesBySeverity) {
+             var issues = issuesBySeverity[severity];
+             var section = severity + '-section';
+             var issueItem = severity + '-item';
+             var container = selection.selectAll('.' + section).data(issues.length ? [0] : []);
+             container.exit().remove();
+             var containerEnter = container.enter().append('div').attr('class', 'modal-section ' + section + ' fillL2');
+             containerEnter.append('h3').html(severity === 'warning' ? _t.html('commit.warnings') : _t.html('commit.errors'));
+             containerEnter.append('ul').attr('class', 'changeset-list');
+             container = containerEnter.merge(container);
+             var items = container.select('ul').selectAll('li').data(issues, function (d) {
+               return d.id;
+             });
+             items.exit().remove();
+             var itemsEnter = items.enter().append('li').attr('class', issueItem);
+             var buttons = itemsEnter.append('button').on('mouseover', function (d3_event, d) {
+               if (d.entityIds) {
+                 context.surface().selectAll(utilEntityOrMemberSelector(d.entityIds, context.graph())).classed('hover', true);
+               }
+             }).on('mouseout', function () {
+               context.surface().selectAll('.hover').classed('hover', false);
+             }).on('click', function (d3_event, d) {
+               context.validator().focusIssue(d);
+             });
+             buttons.call(svgIcon('#iD-icon-alert', 'pre-text'));
+             buttons.append('strong').attr('class', 'issue-message');
+             buttons.filter(function (d) {
+               return d.tooltip;
+             }).call(uiTooltip().title(function (d) {
+               return d.tooltip;
+             }).placement('top'));
+             items = itemsEnter.merge(items);
+             items.selectAll('.issue-message').html(function (d) {
+               return d.message(context);
+             });
            }
          }
 
-         function setStaticValues(callback) {
-           if (!(_optstrings || _optarray)) return;
+         return commitWarnings;
+       }
 
-           if (_optstrings) {
-             _comboData = Object.keys(_optstrings).map(function (k) {
-               var v = field.t('options.' + k, {
-                 'default': _optstrings[k]
-               });
-               return {
-                 key: k,
-                 value: v,
-                 title: v,
-                 display: field.t.html('options.' + k, {
-                   'default': _optstrings[k]
-                 })
-               };
-             });
-           } else if (_optarray) {
-             _comboData = _optarray.map(function (k) {
-               var v = _snake_case ? unsnake(k) : k;
-               return {
-                 key: k,
-                 value: v,
-                 title: v
-               };
-             });
-           }
+       var readOnlyTags = [/^changesets_count$/, /^created_by$/, /^ideditor:/, /^imagery_used$/, /^host$/, /^locale$/, /^warnings:/, /^resolved:/, /^closed:note$/, /^closed:keepright$/, /^closed:improveosm:/, /^closed:osmose:/]; // treat most punctuation (except -, _, +, &) as hashtag delimiters - #4398
+       // from https://stackoverflow.com/a/25575009
 
-           _combobox.data(objectDifference(_comboData, _multiData));
+       var hashtagRegex = /(#[^\u2000-\u206F\u2E00-\u2E7F\s\\'!"#$%()*,.\/:;<=>?@\[\]^`{|}~]+)/g;
+       function uiCommit(context) {
+         var dispatch = dispatch$8('cancel');
 
-           if (callback) callback(_comboData);
+         var _userDetails;
+
+         var _selection;
+
+         var changesetEditor = uiChangesetEditor(context).on('change', changeTags);
+         var rawTagEditor = uiSectionRawTagEditor('changeset-tag-editor', context).on('change', changeTags).readOnlyTags(readOnlyTags);
+         var commitChanges = uiSectionChanges(context);
+         var commitWarnings = uiCommitWarnings(context);
+
+         function commit(selection) {
+           _selection = selection; // Initialize changeset if one does not exist yet.
+
+           if (!context.changeset) initChangeset();
+           loadDerivedChangesetTags();
+           selection.call(render);
          }
 
-         function setTaginfoValues(q, callback) {
-           var fn = _isMulti ? 'multikeys' : 'values';
-           var query = (_isMulti ? field.key : '') + q;
-           var hasCountryPrefix = _isNetwork && _countryCode && _countryCode.indexOf(q.toLowerCase()) === 0;
+         function initChangeset() {
+           // expire stored comment, hashtags, source after cutoff datetime - #3947 #4899
+           var commentDate = +corePreferences('commentDate') || 0;
+           var currDate = Date.now();
+           var cutoff = 2 * 86400 * 1000; // 2 days
 
-           if (hasCountryPrefix) {
-             query = _countryCode + ':';
+           if (commentDate > currDate || currDate - commentDate > cutoff) {
+             corePreferences('comment', null);
+             corePreferences('hashtags', null);
+             corePreferences('source', null);
+           } // load in explicitly-set values, if any
+
+
+           if (context.defaultChangesetComment()) {
+             corePreferences('comment', context.defaultChangesetComment());
+             corePreferences('commentDate', Date.now());
            }
 
-           var params = {
-             debounce: q !== '',
-             key: field.key,
-             query: query
-           };
+           if (context.defaultChangesetSource()) {
+             corePreferences('source', context.defaultChangesetSource());
+             corePreferences('commentDate', Date.now());
+           }
 
-           if (_entityIDs.length) {
-             params.geometry = context.graph().geometry(_entityIDs[0]);
+           if (context.defaultChangesetHashtags()) {
+             corePreferences('hashtags', context.defaultChangesetHashtags());
+             corePreferences('commentDate', Date.now());
            }
 
-           services.taginfo[fn](params, function (err, data) {
-             if (err) return;
-             data = data.filter(function (d) {
-               if (field.type === 'typeCombo' && d.value === 'yes') {
-                 // don't show the fallback value
-                 return false;
-               } // don't show values with very low usage
+           var detected = utilDetect();
+           var tags = {
+             comment: corePreferences('comment') || '',
+             created_by: context.cleanTagValue('iD ' + context.version),
+             host: context.cleanTagValue(detected.host),
+             locale: context.cleanTagValue(_mainLocalizer.localeCode())
+           }; // call findHashtags initially - this will remove stored
+           // hashtags if any hashtags are found in the comment - #4304
+
+           findHashtags(tags, true);
+           var hashtags = corePreferences('hashtags');
 
+           if (hashtags) {
+             tags.hashtags = hashtags;
+           }
 
-               return !d.count || d.count > 10;
-             });
-             var deprecatedValues = osmEntity.deprecatedTagValuesByKey(_dataDeprecated)[field.key];
+           var source = corePreferences('source');
 
-             if (deprecatedValues) {
-               // don't suggest deprecated tag values
-               data = data.filter(function (d) {
-                 return deprecatedValues.indexOf(d.value) === -1;
-               });
-             }
+           if (source) {
+             tags.source = source;
+           }
 
-             if (hasCountryPrefix) {
-               data = data.filter(function (d) {
-                 return d.value.toLowerCase().indexOf(_countryCode + ':') === 0;
-               });
-             } // hide the caret if there are no suggestions
+           var photoOverlaysUsed = context.history().photoOverlaysUsed();
 
+           if (photoOverlaysUsed.length) {
+             var sources = (tags.source || '').split(';'); // include this tag for any photo layer
 
-             _container.classed('empty-combobox', data.length === 0);
+             if (sources.indexOf('streetlevel imagery') === -1) {
+               sources.push('streetlevel imagery');
+             } // add the photo overlays used during editing as sources
 
-             _comboData = data.map(function (d) {
-               var k = d.value;
-               if (_isMulti) k = k.replace(field.key, '');
-               var v = _snake_case ? unsnake(k) : k;
-               return {
-                 key: k,
-                 value: v,
-                 title: _isMulti ? v : d.title
-               };
-             });
-             _comboData = objectDifference(_comboData, _multiData);
-             if (callback) callback(_comboData);
-           });
-         }
 
-         function setPlaceholder(values) {
-           if (_isMulti || _isSemi) {
-             _staticPlaceholder = field.placeholder() || _t('inspector.add');
-           } else {
-             var vals = values.map(function (d) {
-               return d.value;
-             }).filter(function (s) {
-               return s.length < 20;
-             });
-             var placeholders = vals.length > 1 ? vals : values.map(function (d) {
-               return d.key;
+             photoOverlaysUsed.forEach(function (photoOverlay) {
+               if (sources.indexOf(photoOverlay) === -1) {
+                 sources.push(photoOverlay);
+               }
              });
-             _staticPlaceholder = field.placeholder() || placeholders.slice(0, 3).join(', ');
+             tags.source = context.cleanTagValue(sources.join(';'));
            }
 
-           if (!/(…|\.\.\.)$/.test(_staticPlaceholder)) {
-             _staticPlaceholder += '…';
-           }
+           context.changeset = new osmChangeset({
+             tags: tags
+           });
+         } // Calculates read-only metadata tags based on the user's editing session and applies
+         // them to the changeset.
 
-           var ph;
 
-           if (!_isMulti && !_isSemi && _tags && Array.isArray(_tags[field.key])) {
-             ph = _t('inspector.multiple_values');
-           } else {
-             ph = _staticPlaceholder;
-           }
+         function loadDerivedChangesetTags() {
+           var osm = context.connection();
+           if (!osm) return;
+           var tags = Object.assign({}, context.changeset.tags); // shallow copy
+           // assign tags for imagery used
 
-           _container.selectAll('input').attr('placeholder', ph);
-         }
+           var imageryUsed = context.cleanTagValue(context.history().imageryUsed().join(';'));
+           tags.imagery_used = imageryUsed || 'None'; // assign tags for closed issues and notes
 
-         function change() {
-           var t = {};
-           var val;
+           var osmClosed = osm.getClosedIDs();
+           var itemType;
 
-           if (_isMulti || _isSemi) {
-             val = tagValue(utilGetSetValue(_input).replace(/,/g, ';')) || '';
+           if (osmClosed.length) {
+             tags['closed:note'] = context.cleanTagValue(osmClosed.join(';'));
+           }
 
-             _container.classed('active', false);
+           if (services.keepRight) {
+             var krClosed = services.keepRight.getClosedIDs();
 
-             utilGetSetValue(_input, '');
-             var vals = val.split(';').filter(Boolean);
-             if (!vals.length) return;
+             if (krClosed.length) {
+               tags['closed:keepright'] = context.cleanTagValue(krClosed.join(';'));
+             }
+           }
 
-             if (_isMulti) {
-               utilArrayUniq(vals).forEach(function (v) {
-                 var key = (field.key || '') + v;
+           if (services.improveOSM) {
+             var iOsmClosed = services.improveOSM.getClosedCounts();
 
-                 if (_tags) {
-                   // don't set a multicombo value to 'yes' if it already has a non-'no' value
-                   // e.g. `language:de=main`
-                   var old = _tags[key];
-                   if (typeof old === 'string' && old.toLowerCase() !== 'no') return;
-                 }
+             for (itemType in iOsmClosed) {
+               tags['closed:improveosm:' + itemType] = context.cleanTagValue(iOsmClosed[itemType].toString());
+             }
+           }
 
-                 key = context.cleanTagKey(key);
-                 field.keys.push(key);
-                 t[key] = 'yes';
-               });
-             } else if (_isSemi) {
-               var arr = _multiData.map(function (d) {
-                 return d.key;
-               });
+           if (services.osmose) {
+             var osmoseClosed = services.osmose.getClosedCounts();
 
-               arr = arr.concat(vals);
-               t[field.key] = context.cleanTagValue(utilArrayUniq(arr).filter(Boolean).join(';'));
+             for (itemType in osmoseClosed) {
+               tags['closed:osmose:' + itemType] = context.cleanTagValue(osmoseClosed[itemType].toString());
              }
+           } // remove existing issue counts
 
-             window.setTimeout(function () {
-               _input.node().focus();
-             }, 10);
-           } else {
-             var rawValue = utilGetSetValue(_input); // don't override multiple values with blank string
 
-             if (!rawValue && Array.isArray(_tags[field.key])) return;
-             val = context.cleanTagValue(tagValue(rawValue));
-             t[field.key] = val || undefined;
+           for (var key in tags) {
+             if (key.match(/(^warnings:)|(^resolved:)/)) {
+               delete tags[key];
+             }
            }
 
-           dispatch$1.call('change', this, t);
-         }
+           function addIssueCounts(issues, prefix) {
+             var issuesByType = utilArrayGroupBy(issues, 'type');
 
-         function removeMultikey(d3_event, d) {
-           d3_event.preventDefault();
-           d3_event.stopPropagation();
-           var t = {};
+             for (var issueType in issuesByType) {
+               var issuesOfType = issuesByType[issueType];
 
-           if (_isMulti) {
-             t[d.key] = undefined;
-           } else if (_isSemi) {
-             var arr = _multiData.map(function (md) {
-               return md.key === d.key ? null : md.key;
-             }).filter(Boolean);
+               if (issuesOfType[0].subtype) {
+                 var issuesBySubtype = utilArrayGroupBy(issuesOfType, 'subtype');
 
-             arr = utilArrayUniq(arr);
-             t[field.key] = arr.length ? arr.join(';') : undefined;
-           }
+                 for (var issueSubtype in issuesBySubtype) {
+                   var issuesOfSubtype = issuesBySubtype[issueSubtype];
+                   tags[prefix + ':' + issueType + ':' + issueSubtype] = context.cleanTagValue(issuesOfSubtype.length.toString());
+                 }
+               } else {
+                 tags[prefix + ':' + issueType] = context.cleanTagValue(issuesOfType.length.toString());
+               }
+             }
+           } // add counts of warnings generated by the user's edits
 
-           dispatch$1.call('change', this, t);
-         }
 
-         function combo(selection) {
-           _container = selection.selectAll('.form-field-input-wrap').data([0]);
-           var type = _isMulti || _isSemi ? 'multicombo' : 'combo';
-           _container = _container.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + type).merge(_container);
+           var warnings = context.validator().getIssuesBySeverity({
+             what: 'edited',
+             where: 'all',
+             includeIgnored: true,
+             includeDisabledRules: true
+           }).warning;
+           addIssueCounts(warnings, 'warnings'); // add counts of issues resolved by the user's edits
 
-           if (_isMulti || _isSemi) {
-             _container = _container.selectAll('.chiplist').data([0]);
-             var listClass = 'chiplist'; // Use a separate line for each value in the Destinations field
-             // to mimic highway exit signs
+           var resolvedIssues = context.validator().getResolvedIssues();
+           addIssueCounts(resolvedIssues, 'resolved');
+           context.changeset = context.changeset.update({
+             tags: tags
+           });
+         }
 
-             if (field.key === 'destination') {
-               listClass += ' full-line-chips';
-             }
+         function render(selection) {
+           var osm = context.connection();
+           if (!osm) return;
+           var header = selection.selectAll('.header').data([0]);
+           var headerTitle = header.enter().append('div').attr('class', 'header fillL');
+           headerTitle.append('div').append('h3').html(_t.html('commit.title'));
+           headerTitle.append('button').attr('class', 'close').on('click', function () {
+             dispatch.call('cancel', this);
+           }).call(svgIcon('#iD-icon-close'));
+           var body = selection.selectAll('.body').data([0]);
+           body = body.enter().append('div').attr('class', 'body').merge(body); // Changeset Section
 
-             _container = _container.enter().append('ul').attr('class', listClass).on('click', function () {
-               window.setTimeout(function () {
-                 _input.node().focus();
-               }, 10);
-             }).merge(_container);
-             _inputWrap = _container.selectAll('.input-wrap').data([0]);
-             _inputWrap = _inputWrap.enter().append('li').attr('class', 'input-wrap').merge(_inputWrap);
-             _input = _inputWrap.selectAll('input').data([0]);
-           } else {
-             _input = _container.selectAll('input').data([0]);
-           }
+           var changesetSection = body.selectAll('.changeset-editor').data([0]);
+           changesetSection = changesetSection.enter().append('div').attr('class', 'modal-section changeset-editor').merge(changesetSection);
+           changesetSection.call(changesetEditor.changesetID(context.changeset.id).tags(context.changeset.tags)); // Warnings
 
-           _input = _input.enter().append('input').attr('type', 'text').attr('id', field.domId).call(utilNoAuto).call(initCombo, selection).merge(_input);
+           body.call(commitWarnings); // Upload Explanation
 
-           if (_isNetwork) {
-             var extent = combinedEntityExtent();
-             var countryCode = extent && iso1A2Code(extent.center());
-             _countryCode = countryCode && countryCode.toLowerCase();
+           var saveSection = body.selectAll('.save-section').data([0]);
+           saveSection = saveSection.enter().append('div').attr('class', 'modal-section save-section fillL').merge(saveSection);
+           var prose = saveSection.selectAll('.commit-info').data([0]);
+
+           if (prose.enter().size()) {
+             // first time, make sure to update user details in prose
+             _userDetails = null;
            }
 
-           _input.on('change', change).on('blur', change);
+           prose = prose.enter().append('p').attr('class', 'commit-info').html(_t.html('commit.upload_explanation')).merge(prose); // always check if this has changed, but only update prose.html()
+           // if needed, because it can trigger a style recalculation
 
-           _input.on('keydown.field', function (d3_event) {
-             switch (d3_event.keyCode) {
-               case 13:
-                 // ↩ Return
-                 _input.node().blur(); // blurring also enters the value
+           osm.userDetails(function (err, user) {
+             if (err) return;
+             if (_userDetails === user) return; // no change
 
+             _userDetails = user;
+             var userLink = select(document.createElement('div'));
 
-                 d3_event.stopPropagation();
-                 break;
+             if (user.image_url) {
+               userLink.append('img').attr('src', user.image_url).attr('class', 'icon pre-text user-icon');
              }
-           });
 
-           if (_isMulti || _isSemi) {
-             _combobox.on('accept', function () {
-               _input.node().blur();
+             userLink.append('a').attr('class', 'user-info').html(user.display_name).attr('href', osm.userURL(user.display_name)).attr('target', '_blank');
+             prose.html(_t.html('commit.upload_explanation_with_user', {
+               user: userLink.html()
+             }));
+           }); // Request Review
 
-               _input.node().focus();
-             });
+           var requestReview = saveSection.selectAll('.request-review').data([0]); // Enter
 
-             _input.on('focus', function () {
-               _container.classed('active', true);
-             });
+           var requestReviewEnter = requestReview.enter().append('div').attr('class', 'request-review');
+           var requestReviewDomId = utilUniqueDomId('commit-input-request-review');
+           var labelEnter = requestReviewEnter.append('label').attr('for', requestReviewDomId);
+
+           if (!labelEnter.empty()) {
+             labelEnter.call(uiTooltip().title(_t.html('commit.request_review_info')).placement('top'));
            }
-         }
 
-         combo.tags = function (tags) {
-           _tags = tags;
+           labelEnter.append('input').attr('type', 'checkbox').attr('id', requestReviewDomId);
+           labelEnter.append('span').html(_t.html('commit.request_review')); // Update
 
-           if (_isMulti || _isSemi) {
-             _multiData = [];
-             var maxLength;
+           requestReview = requestReview.merge(requestReviewEnter);
+           var requestReviewInput = requestReview.selectAll('input').property('checked', isReviewRequested(context.changeset.tags)).on('change', toggleRequestReview); // Buttons
 
-             if (_isMulti) {
-               // Build _multiData array containing keys already set..
-               for (var k in tags) {
-                 if (field.key && k.indexOf(field.key) !== 0) continue;
-                 if (!field.key && field.keys.indexOf(k) === -1) continue;
-                 var v = tags[k];
-                 if (!v || typeof v === 'string' && v.toLowerCase() === 'no') continue;
-                 var suffix = field.key ? k.substr(field.key.length) : k;
+           var buttonSection = saveSection.selectAll('.buttons').data([0]); // enter
 
-                 _multiData.push({
-                   key: k,
-                   value: displayValue(suffix),
-                   isMixed: Array.isArray(v)
-                 });
-               }
+           var buttonEnter = buttonSection.enter().append('div').attr('class', 'buttons fillL');
+           buttonEnter.append('button').attr('class', 'secondary-action button cancel-button').append('span').attr('class', 'label').html(_t.html('commit.cancel'));
+           var uploadButton = buttonEnter.append('button').attr('class', 'action button save-button');
+           uploadButton.append('span').attr('class', 'label').html(_t.html('commit.save'));
+           var uploadBlockerTooltipText = getUploadBlockerMessage(); // update
 
-               if (field.key) {
-                 // Set keys for form-field modified (needed for undo and reset buttons)..
-                 field.keys = _multiData.map(function (d) {
-                   return d.key;
-                 }); // limit the input length so it fits after prepending the key prefix
+           buttonSection = buttonSection.merge(buttonEnter);
+           buttonSection.selectAll('.cancel-button').on('click.cancel', function () {
+             dispatch.call('cancel', this);
+           });
+           buttonSection.selectAll('.save-button').classed('disabled', uploadBlockerTooltipText !== null).on('click.save', function () {
+             if (!select(this).classed('disabled')) {
+               this.blur(); // avoid keeping focus on the button - #4641
 
-                 maxLength = context.maxCharsForTagKey() - utilUnicodeCharsCount(field.key);
-               } else {
-                 maxLength = context.maxCharsForTagKey();
+               for (var key in context.changeset.tags) {
+                 // remove any empty keys before upload
+                 if (!key) delete context.changeset.tags[key];
                }
-             } else if (_isSemi) {
-               var allValues = [];
-               var commonValues;
 
-               if (Array.isArray(tags[field.key])) {
-                 tags[field.key].forEach(function (tagVal) {
-                   var thisVals = utilArrayUniq((tagVal || '').split(';')).filter(Boolean);
-                   allValues = allValues.concat(thisVals);
+               context.uploader().save(context.changeset);
+             }
+           }); // remove any existing tooltip
 
-                   if (!commonValues) {
-                     commonValues = thisVals;
-                   } else {
-                     commonValues = commonValues.filter(function (value) {
-                       return thisVals.includes(value);
-                     });
-                   }
-                 });
-                 allValues = utilArrayUniq(allValues).filter(Boolean);
-               } else {
-                 allValues = utilArrayUniq((tags[field.key] || '').split(';')).filter(Boolean);
-                 commonValues = allValues;
-               }
+           uiTooltip().destroyAny(buttonSection.selectAll('.save-button'));
 
-               _multiData = allValues.map(function (v) {
-                 return {
-                   key: v,
-                   value: displayValue(v),
-                   isMixed: !commonValues.includes(v)
-                 };
-               });
-               var currLength = utilUnicodeCharsCount(commonValues.join(';')); // limit the input length to the remaining available characters
+           if (uploadBlockerTooltipText) {
+             buttonSection.selectAll('.save-button').call(uiTooltip().title(uploadBlockerTooltipText).placement('top'));
+           } // Raw Tag Editor
 
-               maxLength = context.maxCharsForTagValue() - currLength;
 
-               if (currLength > 0) {
-                 // account for the separator if a new value will be appended to existing
-                 maxLength -= 1;
-               }
-             } // a negative maxlength doesn't make sense
+           var tagSection = body.selectAll('.tag-section.raw-tag-editor').data([0]);
+           tagSection = tagSection.enter().append('div').attr('class', 'modal-section tag-section raw-tag-editor').merge(tagSection);
+           tagSection.call(rawTagEditor.tags(Object.assign({}, context.changeset.tags)) // shallow copy
+           .render);
+           var changesSection = body.selectAll('.commit-changes-section').data([0]);
+           changesSection = changesSection.enter().append('div').attr('class', 'modal-section commit-changes-section').merge(changesSection); // Change summary
 
+           changesSection.call(commitChanges.render);
 
-             maxLength = Math.max(0, maxLength);
-             var allowDragAndDrop = _isSemi // only semiCombo values are ordered
-             && !Array.isArray(tags[field.key]); // Exclude existing multikeys from combo options..
+           function toggleRequestReview() {
+             var rr = requestReviewInput.property('checked');
+             updateChangeset({
+               review_requested: rr ? 'yes' : undefined
+             });
+             tagSection.call(rawTagEditor.tags(Object.assign({}, context.changeset.tags)) // shallow copy
+             .render);
+           }
+         }
 
-             var available = objectDifference(_comboData, _multiData);
+         function getUploadBlockerMessage() {
+           var errors = context.validator().getIssuesBySeverity({
+             what: 'edited',
+             where: 'all'
+           }).error;
 
-             _combobox.data(available); // Hide 'Add' button if this field uses fixed set of
-             // translateable _optstrings and they're all currently used,
-             // or if the field is already at its character limit
+           if (errors.length) {
+             return _t('commit.outstanding_errors_message', {
+               count: errors.length
+             });
+           } else {
+             var hasChangesetComment = context.changeset && context.changeset.tags.comment && context.changeset.tags.comment.trim().length;
 
+             if (!hasChangesetComment) {
+               return _t('commit.comment_needed_message');
+             }
+           }
 
-             var hideAdd = _optstrings && !available.length || maxLength <= 0;
+           return null;
+         }
 
-             _container.selectAll('.chiplist .input-wrap').style('display', hideAdd ? 'none' : null); // Render chips
+         function changeTags(_, changed, onInput) {
+           if (changed.hasOwnProperty('comment')) {
+             if (changed.comment === undefined) {
+               changed.comment = '';
+             }
 
+             if (!onInput) {
+               corePreferences('comment', changed.comment);
+               corePreferences('commentDate', Date.now());
+             }
+           }
 
-             var chips = _container.selectAll('.chip').data(_multiData);
+           if (changed.hasOwnProperty('source')) {
+             if (changed.source === undefined) {
+               corePreferences('source', null);
+             } else if (!onInput) {
+               corePreferences('source', changed.source);
+               corePreferences('commentDate', Date.now());
+             }
+           } // no need to update `prefs` for `hashtags` here since it's done in `updateChangeset`
 
-             chips.exit().remove();
-             var enter = chips.enter().insert('li', '.input-wrap').attr('class', 'chip');
-             enter.append('span');
-             enter.append('a');
-             chips = chips.merge(enter).order().classed('draggable', allowDragAndDrop).classed('mixed', function (d) {
-               return d.isMixed;
-             }).attr('title', function (d) {
-               return d.isMixed ? _t('inspector.unshared_value_tooltip') : null;
-             });
 
-             if (allowDragAndDrop) {
-               registerDragAndDrop(chips);
-             }
+           updateChangeset(changed, onInput);
 
-             chips.select('span').html(function (d) {
-               return d.value;
-             });
-             chips.select('a').attr('href', '#').on('click', removeMultikey).attr('class', 'remove').html('×');
-           } else {
-             var isMixed = Array.isArray(tags[field.key]);
-             var mixedValues = isMixed && tags[field.key].map(function (val) {
-               return displayValue(val);
-             }).filter(Boolean);
-             utilGetSetValue(_input, !isMixed ? displayValue(tags[field.key]) : '').attr('title', isMixed ? mixedValues.join('\n') : undefined).attr('placeholder', isMixed ? _t('inspector.multiple_values') : _staticPlaceholder || '').classed('mixed', isMixed);
+           if (_selection) {
+             _selection.call(render);
            }
-         };
+         }
 
-         function registerDragAndDrop(selection) {
-           // allow drag and drop re-ordering of chips
-           var dragOrigin, targetIndex;
-           selection.call(d3_drag().on('start', function (d3_event) {
-             dragOrigin = {
-               x: d3_event.x,
-               y: d3_event.y
-             };
-             targetIndex = null;
-           }).on('drag', function (d3_event) {
-             var x = d3_event.x - dragOrigin.x,
-                 y = d3_event.y - dragOrigin.y;
-             if (!select(this).classed('dragging') && // don't display drag until dragging beyond a distance threshold
-             Math.sqrt(Math.pow(x, 2) + Math.pow(y, 2)) <= 5) return;
-             var index = selection.nodes().indexOf(this);
-             select(this).classed('dragging', true);
-             targetIndex = null;
-             var targetIndexOffsetTop = null;
-             var draggedTagWidth = select(this).node().offsetWidth;
+         function findHashtags(tags, commentOnly) {
+           var detectedHashtags = commentHashtags();
 
-             if (field.key === 'destination') {
-               // meaning tags are full width
-               _container.selectAll('.chip').style('transform', function (d2, index2) {
-                 var node = select(this).node();
+           if (detectedHashtags.length) {
+             // always remove stored hashtags if there are hashtags in the comment - #4304
+             corePreferences('hashtags', null);
+           }
 
-                 if (index === index2) {
-                   return 'translate(' + x + 'px, ' + y + 'px)'; // move the dragged tag up the order
-                 } else if (index2 > index && d3_event.y > node.offsetTop) {
-                   if (targetIndex === null || index2 > targetIndex) {
-                     targetIndex = index2;
-                   }
+           if (!detectedHashtags.length || !commentOnly) {
+             detectedHashtags = detectedHashtags.concat(hashtagHashtags());
+           }
 
-                   return 'translateY(-100%)'; // move the dragged tag down the order
-                 } else if (index2 < index && d3_event.y < node.offsetTop + node.offsetHeight) {
-                   if (targetIndex === null || index2 < targetIndex) {
-                     targetIndex = index2;
-                   }
+           var allLowerCase = new Set();
+           return detectedHashtags.filter(function (hashtag) {
+             // Compare tags as lowercase strings, but keep original case tags
+             var lowerCase = hashtag.toLowerCase();
 
-                   return 'translateY(100%)';
-                 }
+             if (!allLowerCase.has(lowerCase)) {
+               allLowerCase.add(lowerCase);
+               return true;
+             }
 
-                 return null;
-               });
-             } else {
-               _container.selectAll('.chip').each(function (d2, index2) {
-                 var node = select(this).node(); // check the cursor is in the bounding box
+             return false;
+           }); // Extract hashtags from `comment`
+
+           function commentHashtags() {
+             var matches = (tags.comment || '').replace(/http\S*/g, '') // drop anything that looks like a URL - #4289
+             .match(hashtagRegex);
+             return matches || [];
+           } // Extract and clean hashtags from `hashtags`
+
+
+           function hashtagHashtags() {
+             var matches = (tags.hashtags || '').split(/[,;\s]+/).map(function (s) {
+               if (s[0] !== '#') {
+                 s = '#' + s;
+               } // prepend '#'
+
+
+               var matched = s.match(hashtagRegex);
+               return matched && matched[0];
+             }).filter(Boolean); // exclude falsy
 
-                 if (index !== index2 && d3_event.x < node.offsetLeft + node.offsetWidth + 5 && d3_event.x > node.offsetLeft && d3_event.y < node.offsetTop + node.offsetHeight && d3_event.y > node.offsetTop) {
-                   targetIndex = index2;
-                   targetIndexOffsetTop = node.offsetTop;
-                 }
-               }).style('transform', function (d2, index2) {
-                 var node = select(this).node();
+             return matches || [];
+           }
+         }
 
-                 if (index === index2) {
-                   return 'translate(' + x + 'px, ' + y + 'px)';
-                 } // only translate tags in the same row
+         function isReviewRequested(tags) {
+           var rr = tags.review_requested;
+           if (rr === undefined) return false;
+           rr = rr.trim().toLowerCase();
+           return !(rr === '' || rr === 'no');
+         }
 
+         function updateChangeset(changed, onInput) {
+           var tags = Object.assign({}, context.changeset.tags); // shallow copy
 
-                 if (node.offsetTop === targetIndexOffsetTop) {
-                   if (index2 < index && index2 >= targetIndex) {
-                     return 'translateX(' + draggedTagWidth + 'px)';
-                   } else if (index2 > index && index2 <= targetIndex) {
-                     return 'translateX(-' + draggedTagWidth + 'px)';
-                   }
-                 }
+           Object.keys(changed).forEach(function (k) {
+             var v = changed[k];
+             k = context.cleanTagKey(k);
+             if (readOnlyTags.indexOf(k) !== -1) return;
 
-                 return null;
-               });
+             if (v === undefined) {
+               delete tags[k];
+             } else if (onInput) {
+               tags[k] = v;
+             } else {
+               tags[k] = context.cleanTagValue(v);
              }
-           }).on('end', function () {
-             if (!select(this).classed('dragging')) {
-               return;
+           });
+
+           if (!onInput) {
+             // when changing the comment, override hashtags with any found in comment.
+             var commentOnly = changed.hasOwnProperty('comment') && changed.comment !== '';
+             var arr = findHashtags(tags, commentOnly);
+
+             if (arr.length) {
+               tags.hashtags = context.cleanTagValue(arr.join(';'));
+               corePreferences('hashtags', tags.hashtags);
+             } else {
+               delete tags.hashtags;
+               corePreferences('hashtags', null);
              }
+           } // always update userdetails, just in case user reauthenticates as someone else
 
-             var index = selection.nodes().indexOf(this);
-             select(this).classed('dragging', false);
 
-             _container.selectAll('.chip').style('transform', null);
+           if (_userDetails && _userDetails.changesets_count !== undefined) {
+             var changesetsCount = parseInt(_userDetails.changesets_count, 10) + 1; // #4283
 
-             if (typeof targetIndex === 'number') {
-               var element = _multiData[index];
+             tags.changesets_count = String(changesetsCount); // first 100 edits - new user
 
-               _multiData.splice(index, 1);
+             if (changesetsCount <= 100) {
+               var s;
+               s = corePreferences('walkthrough_completed');
 
-               _multiData.splice(targetIndex, 0, element);
+               if (s) {
+                 tags['ideditor:walkthrough_completed'] = s;
+               }
 
-               var t = {};
+               s = corePreferences('walkthrough_progress');
 
-               if (_multiData.length) {
-                 t[field.key] = _multiData.map(function (element) {
-                   return element.key;
-                 }).join(';');
-               } else {
-                 t[field.key] = undefined;
+               if (s) {
+                 tags['ideditor:walkthrough_progress'] = s;
                }
 
-               dispatch$1.call('change', this, t);
+               s = corePreferences('walkthrough_started');
+
+               if (s) {
+                 tags['ideditor:walkthrough_started'] = s;
+               }
              }
+           } else {
+             delete tags.changesets_count;
+           }
 
-             dragOrigin = undefined;
-             targetIndex = undefined;
-           }));
+           if (!fastDeepEqual(context.changeset.tags, tags)) {
+             context.changeset = context.changeset.update({
+               tags: tags
+             });
+           }
          }
 
-         combo.focus = function () {
-           _input.node().focus();
+         commit.reset = function () {
+           context.changeset = null;
          };
 
-         combo.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs;
-           _entityIDs = val;
-           return combo;
-         };
+         return utilRebind(commit, dispatch, 'on');
+       }
 
-         function combinedEntityExtent() {
-           return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
-         }
+       // for punction see https://stackoverflow.com/a/21224179
 
-         return utilRebind(combo, dispatch$1, 'on');
+       function simplify(str) {
+         if (typeof str !== 'string') return '';
+         return diacritics.remove(str.replace(/&/g, 'and').replace(/İ/ig, 'i').replace(/[\s\-=_!"#%'*{},.\/:;?\(\)\[\]@\\$\^*+<>«»~`’\u00a1\u00a7\u00b6\u00b7\u00bf\u037e\u0387\u055a-\u055f\u0589\u05c0\u05c3\u05c6\u05f3\u05f4\u0609\u060a\u060c\u060d\u061b\u061e\u061f\u066a-\u066d\u06d4\u0700-\u070d\u07f7-\u07f9\u0830-\u083e\u085e\u0964\u0965\u0970\u0af0\u0df4\u0e4f\u0e5a\u0e5b\u0f04-\u0f12\u0f14\u0f85\u0fd0-\u0fd4\u0fd9\u0fda\u104a-\u104f\u10fb\u1360-\u1368\u166d\u166e\u16eb-\u16ed\u1735\u1736\u17d4-\u17d6\u17d8-\u17da\u1800-\u1805\u1807-\u180a\u1944\u1945\u1a1e\u1a1f\u1aa0-\u1aa6\u1aa8-\u1aad\u1b5a-\u1b60\u1bfc-\u1bff\u1c3b-\u1c3f\u1c7e\u1c7f\u1cc0-\u1cc7\u1cd3\u200b-\u200f\u2016\u2017\u2020-\u2027\u2030-\u2038\u203b-\u203e\u2041-\u2043\u2047-\u2051\u2053\u2055-\u205e\u2cf9-\u2cfc\u2cfe\u2cff\u2d70\u2e00\u2e01\u2e06-\u2e08\u2e0b\u2e0e-\u2e16\u2e18\u2e19\u2e1b\u2e1e\u2e1f\u2e2a-\u2e2e\u2e30-\u2e39\u3001-\u3003\u303d\u30fb\ua4fe\ua4ff\ua60d-\ua60f\ua673\ua67e\ua6f2-\ua6f7\ua874-\ua877\ua8ce\ua8cf\ua8f8-\ua8fa\ua92e\ua92f\ua95f\ua9c1-\ua9cd\ua9de\ua9df\uaa5c-\uaa5f\uaade\uaadf\uaaf0\uaaf1\uabeb\ufe10-\ufe16\ufe19\ufe30\ufe45\ufe46\ufe49-\ufe4c\ufe50-\ufe52\ufe54-\ufe57\ufe5f-\ufe61\ufe68\ufe6a\ufe6b\ufeff\uff01-\uff03\uff05-\uff07\uff0a\uff0c\uff0e\uff0f\uff1a\uff1b\uff1f\uff20\uff3c\uff61\uff64\uff65]+/g, '').toLowerCase());
        }
 
-       function uiFieldText(field, context) {
-         var dispatch$1 = dispatch('change');
-         var input = select(null);
-         var outlinkButton = select(null);
-         var _entityIDs = [];
+       // `resolveStrings`
+       // Resolves the text strings for a given community index item
+       //
+       // Arguments
+       //   `item`:  Object containing the community index item
+       //   `defaults`: Object containing the community index default strings
+       //   `localizerFn?`: optional function we will call to do the localization.
+       //      This function should be like the iD `t()` function that
+       //      accepts a `stringID` and returns a localized string
+       //
+       // Returns
+       //   An Object containing all the resolved strings:
+       //   {
+       //     name:                     'talk-ru Mailing List',
+       //     url:                      'https://lists.openstreetmap.org/listinfo/talk-ru',
+       //     signupUrl:                'https://example.url/signup',
+       //     description:              'A one line description',
+       //     extendedDescription:      'Extended description',
+       //     nameHTML:                 '<a href="the url">the name</a>',
+       //     urlHTML:                  '<a href="the url">the url</a>',
+       //     signupUrlHTML:            '<a href="the signupUrl">the signupUrl</a>',
+       //     descriptionHTML:          the description, with urls and signupUrls linkified,
+       //     extendedDescriptionHTML:  the extendedDescription with urls and signupUrls linkified
+       //   }
+       //
 
-         var _tags;
+       function resolveStrings(item, defaults, localizerFn) {
+         var itemStrings = Object.assign({}, item.strings); // shallow clone
 
-         var _phoneFormats = {};
+         var defaultStrings = Object.assign({}, defaults[item.type]); // shallow clone
 
-         if (field.type === 'tel') {
-           _mainFileFetcher.get('phone_formats').then(function (d) {
-             _phoneFormats = d;
-             updatePhonePlaceholder();
-           })["catch"](function () {
-             /* ignore */
+         var anyToken = new RegExp(/(\{\w+\})/, 'gi'); // Pre-localize the item and default strings
+
+         if (localizerFn) {
+           if (itemStrings.community) {
+             var communityID = simplify(itemStrings.community);
+             itemStrings.community = localizerFn("_communities.".concat(communityID));
+           }
+
+           ['name', 'description', 'extendedDescription'].forEach(function (prop) {
+             if (defaultStrings[prop]) defaultStrings[prop] = localizerFn("_defaults.".concat(item.type, ".").concat(prop));
+             if (itemStrings[prop]) itemStrings[prop] = localizerFn("".concat(item.id, ".").concat(prop));
            });
          }
 
-         function i(selection) {
-           var entity = _entityIDs.length && context.hasEntity(_entityIDs[0]);
-           var preset = entity && _mainPresetIndex.match(entity, context.graph());
-           var isLocked = preset && preset.suggestion && field.id === 'brand';
-           field.locked(isLocked);
-           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
-           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
-           input = wrap.selectAll('input').data([0]);
-           input = input.enter().append('input').attr('type', field.type === 'identifier' ? 'text' : field.type).attr('id', field.domId).classed(field.type, true).call(utilNoAuto).merge(input);
-           input.classed('disabled', !!isLocked).attr('readonly', isLocked || null).on('input', change(true)).on('blur', change()).on('change', change());
+         var replacements = {
+           account: item.account,
+           community: itemStrings.community,
+           signupUrl: itemStrings.signupUrl,
+           url: itemStrings.url
+         }; // Resolve URLs first (which may refer to {account})
 
-           if (field.type === 'tel') {
-             updatePhonePlaceholder();
-           } else if (field.type === 'number') {
-             var rtl = _mainLocalizer.textDirection() === 'rtl';
-             input.attr('type', 'text');
-             var inc = field.increment;
-             var buttons = wrap.selectAll('.increment, .decrement').data(rtl ? [inc, -inc] : [-inc, inc]);
-             buttons.enter().append('button').attr('class', function (d) {
-               var which = d > 0 ? 'increment' : 'decrement';
-               return 'form-field-button ' + which;
-             }).merge(buttons).on('click', function (d3_event, d) {
-               d3_event.preventDefault();
-               var raw_vals = input.node().value || '0';
-               var vals = raw_vals.split(';');
-               vals = vals.map(function (v) {
-                 var num = parseFloat(v.trim(), 10);
-                 return isFinite(num) ? clamped(num + d) : v.trim();
-               });
-               input.node().value = vals.join(';');
-               change()();
-             });
-           } else if (field.type === 'identifier' && field.urlFormat && field.pattern) {
-             input.attr('type', 'text');
-             outlinkButton = wrap.selectAll('.foreign-id-permalink').data([0]);
-             outlinkButton.enter().append('button').call(svgIcon('#iD-icon-out-link')).attr('class', 'form-field-button foreign-id-permalink').attr('title', function () {
-               var domainResults = /^https?:\/\/(.{1,}?)\//.exec(field.urlFormat);
+         if (!replacements.signupUrl) {
+           replacements.signupUrl = resolve(itemStrings.signupUrl || defaultStrings.signupUrl);
+         }
 
-               if (domainResults.length >= 2 && domainResults[1]) {
-                 var domain = domainResults[1];
-                 return _t('icons.view_on', {
-                   domain: domain
-                 });
-               }
+         if (!replacements.url) {
+           replacements.url = resolve(itemStrings.url || defaultStrings.url);
+         }
 
-               return '';
-             }).on('click', function (d3_event) {
-               d3_event.preventDefault();
-               var value = validIdentifierValueForLink();
+         var resolved = {
+           name: resolve(itemStrings.name || defaultStrings.name),
+           url: resolve(itemStrings.url || defaultStrings.url),
+           signupUrl: resolve(itemStrings.signupUrl || defaultStrings.signupUrl),
+           description: resolve(itemStrings.description || defaultStrings.description),
+           extendedDescription: resolve(itemStrings.extendedDescription || defaultStrings.extendedDescription)
+         }; // Generate linkified strings
 
-               if (value) {
-                 var url = field.urlFormat.replace(/{value}/, encodeURIComponent(value));
-                 window.open(url, '_blank');
-               }
-             }).merge(outlinkButton);
-           }
-         }
+         resolved.nameHTML = linkify(resolved.url, resolved.name);
+         resolved.urlHTML = linkify(resolved.url);
+         resolved.signupUrlHTML = linkify(resolved.signupUrl);
+         resolved.descriptionHTML = resolve(itemStrings.description || defaultStrings.description, true);
+         resolved.extendedDescriptionHTML = resolve(itemStrings.extendedDescription || defaultStrings.extendedDescription, true);
+         return resolved;
 
-         function updatePhonePlaceholder() {
-           if (input.empty() || !Object.keys(_phoneFormats).length) return;
-           var extent = combinedEntityExtent();
-           var countryCode = extent && iso1A2Code(extent.center());
+         function resolve(s, addLinks) {
+           if (!s) return undefined;
+           var result = s;
 
-           var format = countryCode && _phoneFormats[countryCode.toLowerCase()];
+           for (var key in replacements) {
+             var token = "{".concat(key, "}");
+             var regex = new RegExp(token, 'g');
 
-           if (format) input.attr('placeholder', format);
-         }
+             if (regex.test(result)) {
+               var replacement = replacements[key];
 
-         function validIdentifierValueForLink() {
-           if (field.type === 'identifier' && field.pattern) {
-             var value = utilGetSetValue(input).trim().split(';')[0];
-             return value && value.match(new RegExp(field.pattern));
-           }
+               if (!replacement) {
+                 throw new Error("Cannot resolve token: ".concat(token));
+               } else {
+                 if (addLinks && (key === 'signupUrl' || key === 'url')) {
+                   replacement = linkify(replacement);
+                 }
 
-           return null;
-         } // clamp number to min/max
+                 result = result.replace(regex, replacement);
+               }
+             }
+           } // There shouldn't be any leftover tokens in a resolved string
 
 
-         function clamped(num) {
-           if (field.minValue !== undefined) {
-             num = Math.max(num, field.minValue);
-           }
+           var leftovers = result.match(anyToken);
 
-           if (field.maxValue !== undefined) {
-             num = Math.min(num, field.maxValue);
+           if (leftovers) {
+             throw new Error("Cannot resolve tokens: ".concat(leftovers));
+           } // Linkify subreddits like `/r/openstreetmap`
+           // https://github.com/osmlab/osm-community-index/issues/82
+           // https://github.com/openstreetmap/iD/issues/4997
+
+
+           if (addLinks && item.type === 'reddit') {
+             result = result.replace(/(\/r\/\w+\/*)/i, function (match) {
+               return linkify(resolved.url, match);
+             });
            }
 
-           return num;
+           return result;
          }
 
-         function change(onInput) {
-           return function () {
-             var t = {};
-             var val = utilGetSetValue(input);
-             if (!onInput) val = context.cleanTagValue(val); // don't override multiple values with blank string
+         function linkify(url, text) {
+           if (!url) return undefined;
+           text = text || url;
+           return "<a target=\"_blank\" href=\"".concat(url, "\">").concat(text, "</a>");
+         }
+       }
 
-             if (!val && Array.isArray(_tags[field.key])) return;
+       var _oci = null;
+       function uiSuccess(context) {
+         var MAXEVENTS = 2;
+         var dispatch = dispatch$8('cancel');
 
-             if (!onInput) {
-               if (field.type === 'number' && val) {
-                 var vals = val.split(';');
-                 vals = vals.map(function (v) {
-                   var num = parseFloat(v.trim(), 10);
-                   return isFinite(num) ? clamped(num) : v.trim();
-                 });
-                 val = vals.join(';');
-               }
+         var _changeset;
 
-               utilGetSetValue(input, val);
+         var _location;
+
+         ensureOSMCommunityIndex(); // start fetching the data
+
+         function ensureOSMCommunityIndex() {
+           var data = _mainFileFetcher;
+           return Promise.all([data.get('oci_features'), data.get('oci_resources'), data.get('oci_defaults')]).then(function (vals) {
+             if (_oci) return _oci; // Merge Custom Features
+
+             if (vals[0] && Array.isArray(vals[0].features)) {
+               _mainLocations.mergeCustomGeoJSON(vals[0]);
              }
 
-             t[field.key] = val || undefined;
-             dispatch$1.call('change', this, t, onInput);
-           };
-         }
+             var ociResources = Object.values(vals[1].resources);
 
-         i.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs;
-           _entityIDs = val;
-           return i;
-         };
+             if (ociResources.length) {
+               // Resolve all locationSet features.
+               return _mainLocations.mergeLocationSets(ociResources).then(function () {
+                 _oci = {
+                   resources: ociResources,
+                   defaults: vals[2].defaults
+                 };
+                 return _oci;
+               });
+             } else {
+               _oci = {
+                 resources: [],
+                 // no resources?
+                 defaults: vals[2].defaults
+               };
+               return _oci;
+             }
+           });
+         } // string-to-date parsing in JavaScript is weird
 
-         i.tags = function (tags) {
-           _tags = tags;
-           var isMixed = Array.isArray(tags[field.key]);
-           utilGetSetValue(input, !isMixed && tags[field.key] ? tags[field.key] : '').attr('title', isMixed ? tags[field.key].filter(Boolean).join('\n') : undefined).attr('placeholder', isMixed ? _t('inspector.multiple_values') : field.placeholder() || _t('inspector.unknown')).classed('mixed', isMixed);
 
-           if (outlinkButton && !outlinkButton.empty()) {
-             var disabled = !validIdentifierValueForLink();
-             outlinkButton.classed('disabled', disabled);
-           }
-         };
+         function parseEventDate(when) {
+           if (!when) return;
+           var raw = when.trim();
+           if (!raw) return;
 
-         i.focus = function () {
-           var node = input.node();
-           if (node) node.focus();
-         };
+           if (!/Z$/.test(raw)) {
+             // if no trailing 'Z', add one
+             raw += 'Z'; // this forces date to be parsed as a UTC date
+           }
 
-         function combinedEntityExtent() {
-           return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+           var parsed = new Date(raw);
+           return new Date(parsed.toUTCString().substr(0, 25)); // convert to local timezone
          }
 
-         return utilRebind(i, dispatch$1, 'on');
-       }
+         function success(selection) {
+           var header = selection.append('div').attr('class', 'header fillL');
+           header.append('h3').html(_t.html('success.just_edited'));
+           header.append('button').attr('class', 'close').on('click', function () {
+             return dispatch.call('cancel');
+           }).call(svgIcon('#iD-icon-close'));
+           var body = selection.append('div').attr('class', 'body save-success fillL');
+           var summary = body.append('div').attr('class', 'save-summary');
+           summary.append('h3').html(_t.html('success.thank_you' + (_location ? '_location' : ''), {
+             where: _location
+           }));
+           summary.append('p').html(_t.html('success.help_html')).append('a').attr('class', 'link-out').attr('target', '_blank').attr('href', _t('success.help_link_url')).call(svgIcon('#iD-icon-out-link', 'inline')).append('span').html(_t.html('success.help_link_text'));
+           var osm = context.connection();
+           if (!osm) return;
+           var changesetURL = osm.changesetURL(_changeset.id);
+           var table = summary.append('table').attr('class', 'summary-table');
+           var row = table.append('tr').attr('class', 'summary-row');
+           row.append('td').attr('class', 'cell-icon summary-icon').append('a').attr('target', '_blank').attr('href', changesetURL).append('svg').attr('class', 'logo-small').append('use').attr('xlink:href', '#iD-logo-osm');
+           var summaryDetail = row.append('td').attr('class', 'cell-detail summary-detail');
+           summaryDetail.append('a').attr('class', 'cell-detail summary-view-on-osm').attr('target', '_blank').attr('href', changesetURL).html(_t.html('success.view_on_osm'));
+           summaryDetail.append('div').html(_t.html('success.changeset_id', {
+             changeset_id: "<a href=\"".concat(changesetURL, "\" target=\"_blank\">").concat(_changeset.id, "</a>")
+           })); // Get OSM community index features intersecting the map..
 
-       function uiFieldAccess(field, context) {
-         var dispatch$1 = dispatch('change');
-         var items = select(null);
+           ensureOSMCommunityIndex().then(function (oci) {
+             var loc = context.map().center();
+             var validLocations = _mainLocations.locationsAt(loc); // Gather the communities
 
-         var _tags;
+             var communities = [];
+             oci.resources.forEach(function (resource) {
+               var area = validLocations[resource.locationSetID];
+               if (!area) return; // Resolve strings
 
-         function access(selection) {
-           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
-           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
-           var list = wrap.selectAll('ul').data([0]);
-           list = list.enter().append('ul').attr('class', 'rows').merge(list);
-           items = list.selectAll('li').data(field.keys); // Enter
+               var localizer = function localizer(stringID) {
+                 return _t.html("community.".concat(stringID));
+               };
 
-           var enter = items.enter().append('li').attr('class', function (d) {
-             return 'labeled-input preset-access-' + d;
-           });
-           enter.append('span').attr('class', 'label preset-label-access').attr('for', function (d) {
-             return 'preset-input-access-' + d;
-           }).html(function (d) {
-             return field.t.html('types.' + d);
-           });
-           enter.append('div').attr('class', 'preset-input-access-wrap').append('input').attr('type', 'text').attr('class', function (d) {
-             return 'preset-input-access preset-input-access-' + d;
-           }).call(utilNoAuto).each(function (d) {
-             select(this).call(uiCombobox(context, 'access-' + d).data(access.options(d)));
-           }); // Update
+               resource.resolved = resolveStrings(resource, oci.defaults, localizer);
+               communities.push({
+                 area: area,
+                 order: resource.order || 0,
+                 resource: resource
+               });
+             }); // sort communities by feature area ascending, community order descending
 
-           items = items.merge(enter);
-           wrap.selectAll('.preset-input-access').on('change', change).on('blur', change);
+             communities.sort(function (a, b) {
+               return a.area - b.area || b.order - a.order;
+             });
+             body.call(showCommunityLinks, communities.map(function (c) {
+               return c.resource;
+             }));
+           });
          }
 
-         function change(d3_event, d) {
-           var tag = {};
-           var value = context.cleanTagValue(utilGetSetValue(select(this))); // don't override multiple values with blank string
-
-           if (!value && typeof _tags[d] !== 'string') return;
-           tag[d] = value || undefined;
-           dispatch$1.call('change', this, tag);
+         function showCommunityLinks(selection, resources) {
+           var communityLinks = selection.append('div').attr('class', 'save-communityLinks');
+           communityLinks.append('h3').html(_t.html('success.like_osm'));
+           var table = communityLinks.append('table').attr('class', 'community-table');
+           var row = table.selectAll('.community-row').data(resources);
+           var rowEnter = row.enter().append('tr').attr('class', 'community-row');
+           rowEnter.append('td').attr('class', 'cell-icon community-icon').append('a').attr('target', '_blank').attr('href', function (d) {
+             return d.resolved.url;
+           }).append('svg').attr('class', 'logo-small').append('use').attr('xlink:href', function (d) {
+             return "#community-".concat(d.type);
+           });
+           var communityDetail = rowEnter.append('td').attr('class', 'cell-detail community-detail');
+           communityDetail.each(showCommunityDetails);
+           communityLinks.append('div').attr('class', 'community-missing').html(_t.html('success.missing')).append('a').attr('class', 'link-out').attr('target', '_blank').call(svgIcon('#iD-icon-out-link', 'inline')).attr('href', 'https://github.com/osmlab/osm-community-index/issues').append('span').html(_t.html('success.tell_us'));
          }
 
-         access.options = function (type) {
-           var options = ['no', 'permissive', 'private', 'permit', 'destination'];
-
-           if (type !== 'access') {
-             options.unshift('yes');
-             options.push('designated');
+         function showCommunityDetails(d) {
+           var selection = select(this);
+           var communityID = d.id;
+           selection.append('div').attr('class', 'community-name').html(d.resolved.nameHTML);
+           selection.append('div').attr('class', 'community-description').html(d.resolved.descriptionHTML); // Create an expanding section if any of these are present..
 
-             if (type === 'bicycle') {
-               options.push('dismount');
-             }
+           if (d.resolved.extendedDescriptionHTML || d.languageCodes && d.languageCodes.length) {
+             selection.append('div').call(uiDisclosure(context, "community-more-".concat(d.id), false).expanded(false).updatePreference(false).label(_t.html('success.more')).content(showMore));
            }
 
-           return options.map(function (option) {
-             return {
-               title: field.t('options.' + option + '.description'),
-               value: option
-             };
-           });
-         };
-
-         var placeholdersByHighway = {
-           footway: {
-             foot: 'designated',
-             motor_vehicle: 'no'
-           },
-           steps: {
-             foot: 'yes',
-             motor_vehicle: 'no',
-             bicycle: 'no',
-             horse: 'no'
-           },
-           pedestrian: {
-             foot: 'yes',
-             motor_vehicle: 'no'
-           },
-           cycleway: {
-             motor_vehicle: 'no',
-             bicycle: 'designated'
-           },
-           bridleway: {
-             motor_vehicle: 'no',
-             horse: 'designated'
-           },
-           path: {
-             foot: 'yes',
-             motor_vehicle: 'no',
-             bicycle: 'yes',
-             horse: 'yes'
-           },
-           motorway: {
-             foot: 'no',
-             motor_vehicle: 'yes',
-             bicycle: 'no',
-             horse: 'no'
-           },
-           trunk: {
-             motor_vehicle: 'yes'
-           },
-           primary: {
-             foot: 'yes',
-             motor_vehicle: 'yes',
-             bicycle: 'yes',
-             horse: 'yes'
-           },
-           secondary: {
-             foot: 'yes',
-             motor_vehicle: 'yes',
-             bicycle: 'yes',
-             horse: 'yes'
-           },
-           tertiary: {
-             foot: 'yes',
-             motor_vehicle: 'yes',
-             bicycle: 'yes',
-             horse: 'yes'
-           },
-           residential: {
-             foot: 'yes',
-             motor_vehicle: 'yes',
-             bicycle: 'yes',
-             horse: 'yes'
-           },
-           unclassified: {
-             foot: 'yes',
-             motor_vehicle: 'yes',
-             bicycle: 'yes',
-             horse: 'yes'
-           },
-           service: {
-             foot: 'yes',
-             motor_vehicle: 'yes',
-             bicycle: 'yes',
-             horse: 'yes'
-           },
-           motorway_link: {
-             foot: 'no',
-             motor_vehicle: 'yes',
-             bicycle: 'no',
-             horse: 'no'
-           },
-           trunk_link: {
-             motor_vehicle: 'yes'
-           },
-           primary_link: {
-             foot: 'yes',
-             motor_vehicle: 'yes',
-             bicycle: 'yes',
-             horse: 'yes'
-           },
-           secondary_link: {
-             foot: 'yes',
-             motor_vehicle: 'yes',
-             bicycle: 'yes',
-             horse: 'yes'
-           },
-           tertiary_link: {
-             foot: 'yes',
-             motor_vehicle: 'yes',
-             bicycle: 'yes',
-             horse: 'yes'
-           }
-         };
+           var nextEvents = (d.events || []).map(function (event) {
+             event.date = parseEventDate(event.when);
+             return event;
+           }).filter(function (event) {
+             // date is valid and future (or today)
+             var t = event.date.getTime();
+             var now = new Date().setHours(0, 0, 0, 0);
+             return !isNaN(t) && t >= now;
+           }).sort(function (a, b) {
+             // sort by date ascending
+             return a.date < b.date ? -1 : a.date > b.date ? 1 : 0;
+           }).slice(0, MAXEVENTS); // limit number of events shown
 
-         access.tags = function (tags) {
-           _tags = tags;
-           utilGetSetValue(items.selectAll('.preset-input-access'), function (d) {
-             return typeof tags[d] === 'string' ? tags[d] : '';
-           }).classed('mixed', function (d) {
-             return tags[d] && Array.isArray(tags[d]);
-           }).attr('title', function (d) {
-             return tags[d] && Array.isArray(tags[d]) && tags[d].filter(Boolean).join('\n');
-           }).attr('placeholder', function (d) {
-             if (tags[d] && Array.isArray(tags[d])) {
-               return _t('inspector.multiple_values');
-             }
+           if (nextEvents.length) {
+             selection.append('div').call(uiDisclosure(context, "community-events-".concat(d.id), false).expanded(false).updatePreference(false).label(_t.html('success.events')).content(showNextEvents)).select('.hide-toggle').append('span').attr('class', 'badge-text').html(nextEvents.length);
+           }
 
-             if (d === 'access') {
-               return 'yes';
+           function showMore(selection) {
+             var more = selection.selectAll('.community-more').data([0]);
+             var moreEnter = more.enter().append('div').attr('class', 'community-more');
+
+             if (d.resolved.extendedDescriptionHTML) {
+               moreEnter.append('div').attr('class', 'community-extended-description').html(d.resolved.extendedDescriptionHTML);
              }
 
-             if (tags.access && typeof tags.access === 'string') {
-               return tags.access;
+             if (d.languageCodes && d.languageCodes.length) {
+               var languageList = d.languageCodes.map(function (code) {
+                 return _mainLocalizer.languageName(code);
+               }).join(', ');
+               moreEnter.append('div').attr('class', 'community-languages').html(_t.html('success.languages', {
+                 languages: languageList
+               }));
              }
+           }
 
-             if (tags.highway) {
-               if (typeof tags.highway === 'string') {
-                 if (placeholdersByHighway[tags.highway] && placeholdersByHighway[tags.highway][d]) {
-                   return placeholdersByHighway[tags.highway][d];
-                 }
-               } else {
-                 var impliedAccesses = tags.highway.filter(Boolean).map(function (highwayVal) {
-                   return placeholdersByHighway[highwayVal] && placeholdersByHighway[highwayVal][d];
-                 }).filter(Boolean);
+           function showNextEvents(selection) {
+             var events = selection.append('div').attr('class', 'community-events');
+             var item = events.selectAll('.community-event').data(nextEvents);
+             var itemEnter = item.enter().append('div').attr('class', 'community-event');
+             itemEnter.append('div').attr('class', 'community-event-name').append('a').attr('target', '_blank').attr('href', function (d) {
+               return d.url;
+             }).html(function (d) {
+               var name = d.name;
 
-                 if (impliedAccesses.length === tags.highway.length && new Set(impliedAccesses).size === 1) {
-                   // if all the highway values have the same implied access for this type then use that
-                   return impliedAccesses[0];
-                 }
+               if (d.i18n && d.id) {
+                 name = _t("community.".concat(communityID, ".events.").concat(d.id, ".name"), {
+                   "default": name
+                 });
                }
-             }
 
-             return field.placeholder();
-           });
+               return name;
+             });
+             itemEnter.append('div').attr('class', 'community-event-when').html(function (d) {
+               var options = {
+                 weekday: 'short',
+                 day: 'numeric',
+                 month: 'short',
+                 year: 'numeric'
+               };
+
+               if (d.date.getHours() || d.date.getMinutes()) {
+                 // include time if it has one
+                 options.hour = 'numeric';
+                 options.minute = 'numeric';
+               }
+
+               return d.date.toLocaleString(_mainLocalizer.localeCode(), options);
+             });
+             itemEnter.append('div').attr('class', 'community-event-where').html(function (d) {
+               var where = d.where;
+
+               if (d.i18n && d.id) {
+                 where = _t("community.".concat(communityID, ".events.").concat(d.id, ".where"), {
+                   "default": where
+                 });
+               }
+
+               return where;
+             });
+             itemEnter.append('div').attr('class', 'community-event-description').html(function (d) {
+               var description = d.description;
+
+               if (d.i18n && d.id) {
+                 description = _t("community.".concat(communityID, ".events.").concat(d.id, ".description"), {
+                   "default": description
+                 });
+               }
+
+               return description;
+             });
+           }
+         }
+
+         success.changeset = function (val) {
+           if (!arguments.length) return _changeset;
+           _changeset = val;
+           return success;
          };
 
-         access.focus = function () {
-           items.selectAll('.preset-input-access').node().focus();
+         success.location = function (val) {
+           if (!arguments.length) return _location;
+           _location = val;
+           return success;
          };
 
-         return utilRebind(access, dispatch$1, 'on');
+         return utilRebind(success, dispatch, 'on');
        }
 
-       function uiFieldAddress(field, context) {
-         var dispatch$1 = dispatch('change');
-
-         var _selection = select(null);
-
-         var _wrap = select(null);
+       function modeSave(context) {
+         var mode = {
+           id: 'save'
+         };
+         var keybinding = utilKeybinding('modeSave');
+         var commit = uiCommit(context).on('cancel', cancel);
 
-         var addrField = _mainPresetIndex.field('address'); // needed for placeholder strings
+         var _conflictsUi; // uiConflicts
 
-         var _entityIDs = [];
 
-         var _tags;
+         var _location;
 
-         var _countryCode;
+         var _success;
 
-         var _addressFormats = [{
-           format: [['housenumber', 'street'], ['city', 'postcode']]
-         }];
-         _mainFileFetcher.get('address_formats').then(function (d) {
-           _addressFormats = d;
+         var uploader = context.uploader().on('saveStarted.modeSave', function () {
+           keybindingOff();
+         }) // fire off some async work that we want to be ready later
+         .on('willAttemptUpload.modeSave', prepareForSuccess).on('progressChanged.modeSave', showProgress).on('resultNoChanges.modeSave', function () {
+           cancel();
+         }).on('resultErrors.modeSave', showErrors).on('resultConflicts.modeSave', showConflicts).on('resultSuccess.modeSave', showSuccess);
 
-           if (!_selection.empty()) {
-             _selection.call(address);
-           }
-         })["catch"](function () {
-           /* ignore */
-         });
+         function cancel() {
+           context.enter(modeBrowse(context));
+         }
 
-         function getNearStreets() {
-           var extent = combinedEntityExtent();
-           var l = extent.center();
-           var box = geoExtent(l).padByMeters(200);
-           var streets = context.history().intersects(box).filter(isAddressable).map(function (d) {
-             var loc = context.projection([(extent[0][0] + extent[1][0]) / 2, (extent[0][1] + extent[1][1]) / 2]);
-             var choice = geoChooseEdge(context.graph().childNodes(d), loc, context.projection);
-             return {
-               title: d.tags.name,
-               value: d.tags.name,
-               dist: choice.distance
-             };
-           }).sort(function (a, b) {
-             return a.dist - b.dist;
-           });
-           return utilArrayUniqBy(streets, 'value');
+         function showProgress(num, total) {
+           var modal = context.container().select('.loading-modal .modal-section');
+           var progress = modal.selectAll('.progress').data([0]); // enter/update
 
-           function isAddressable(d) {
-             return d.tags.highway && d.tags.name && d.type === 'way';
-           }
+           progress.enter().append('div').attr('class', 'progress').merge(progress).text(_t('save.conflict_progress', {
+             num: num,
+             total: total
+           }));
          }
 
-         function getNearCities() {
-           var extent = combinedEntityExtent();
-           var l = extent.center();
-           var box = geoExtent(l).padByMeters(200);
-           var cities = context.history().intersects(box).filter(isAddressable).map(function (d) {
-             return {
-               title: d.tags['addr:city'] || d.tags.name,
-               value: d.tags['addr:city'] || d.tags.name,
-               dist: geoSphericalDistance(d.extent(context.graph()).center(), l)
-             };
-           }).sort(function (a, b) {
-             return a.dist - b.dist;
+         function showConflicts(changeset, conflicts, origChanges) {
+           var selection = context.container().select('.sidebar').append('div').attr('class', 'sidebar-component');
+           context.container().selectAll('.main-content').classed('active', true).classed('inactive', false);
+           _conflictsUi = uiConflicts(context).conflictList(conflicts).origChanges(origChanges).on('cancel', function () {
+             context.container().selectAll('.main-content').classed('active', false).classed('inactive', true);
+             selection.remove();
+             keybindingOn();
+             uploader.cancelConflictResolution();
+           }).on('save', function () {
+             context.container().selectAll('.main-content').classed('active', false).classed('inactive', true);
+             selection.remove();
+             uploader.processResolvedConflicts(changeset);
            });
-           return utilArrayUniqBy(cities, 'value');
-
-           function isAddressable(d) {
-             if (d.tags.name) {
-               if (d.tags.admin_level === '8' && d.tags.boundary === 'administrative') return true;
-               if (d.tags.border_type === 'city') return true;
-               if (d.tags.place === 'city' || d.tags.place === 'town' || d.tags.place === 'village') return true;
-             }
+           selection.call(_conflictsUi);
+         }
 
-             if (d.tags['addr:city']) return true;
-             return false;
-           }
+         function showErrors(errors) {
+           keybindingOn();
+           var selection = uiConfirm(context.container());
+           selection.select('.modal-section.header').append('h3').text(_t('save.error'));
+           addErrors(selection, errors);
+           selection.okButton();
          }
 
-         function getNearValues(key) {
-           var extent = combinedEntityExtent();
-           var l = extent.center();
-           var box = geoExtent(l).padByMeters(200);
-           var results = context.history().intersects(box).filter(function hasTag(d) {
-             return _entityIDs.indexOf(d.id) === -1 && d.tags[key];
-           }).map(function (d) {
-             return {
-               title: d.tags[key],
-               value: d.tags[key],
-               dist: geoSphericalDistance(d.extent(context.graph()).center(), l)
-             };
-           }).sort(function (a, b) {
-             return a.dist - b.dist;
+         function addErrors(selection, data) {
+           var message = selection.select('.modal-section.message-text');
+           var items = message.selectAll('.error-container').data(data);
+           var enter = items.enter().append('div').attr('class', 'error-container');
+           enter.append('a').attr('class', 'error-description').attr('href', '#').classed('hide-toggle', true).text(function (d) {
+             return d.msg || _t('save.unknown_error_details');
+           }).on('click', function (d3_event) {
+             d3_event.preventDefault();
+             var error = select(this);
+             var detail = select(this.nextElementSibling);
+             var exp = error.classed('expanded');
+             detail.style('display', exp ? 'none' : 'block');
+             error.classed('expanded', !exp);
            });
-           return utilArrayUniqBy(results, 'value');
+           var details = enter.append('div').attr('class', 'error-detail-container').style('display', 'none');
+           details.append('ul').attr('class', 'error-detail-list').selectAll('li').data(function (d) {
+             return d.details || [];
+           }).enter().append('li').attr('class', 'error-detail-item').text(function (d) {
+             return d;
+           });
+           items.exit().remove();
          }
 
-         function updateForCountryCode() {
-           if (!_countryCode) return;
-           var addressFormat;
+         function showSuccess(changeset) {
+           commit.reset();
 
-           for (var i = 0; i < _addressFormats.length; i++) {
-             var format = _addressFormats[i];
+           var ui = _success.changeset(changeset).location(_location).on('cancel', function () {
+             context.ui().sidebar.hide();
+           });
 
-             if (!format.countryCodes) {
-               addressFormat = format; // choose the default format, keep going
-             } else if (format.countryCodes.indexOf(_countryCode) !== -1) {
-               addressFormat = format; // choose the country format, stop here
+           context.enter(modeBrowse(context).sidebar(ui));
+         }
 
-               break;
-             }
-           }
+         function keybindingOn() {
+           select(document).call(keybinding.on('⎋', cancel, true));
+         }
 
-           var dropdowns = addressFormat.dropdowns || ['city', 'county', 'country', 'district', 'hamlet', 'neighbourhood', 'place', 'postcode', 'province', 'quarter', 'state', 'street', 'subdistrict', 'suburb'];
-           var widths = addressFormat.widths || {
-             housenumber: 1 / 3,
-             street: 2 / 3,
-             city: 2 / 3,
-             state: 1 / 4,
-             postcode: 1 / 3
-           };
+         function keybindingOff() {
+           select(document).call(keybinding.unbind);
+         } // Reverse geocode current map location so we can display a message on
+         // the success screen like "Thank you for editing around place, region."
 
-           function row(r) {
-             // Normalize widths.
-             var total = r.reduce(function (sum, key) {
-               return sum + (widths[key] || 0.5);
-             }, 0);
-             return r.map(function (key) {
-               return {
-                 id: key,
-                 width: (widths[key] || 0.5) / total
-               };
-             });
-           }
 
-           var rows = _wrap.selectAll('.addr-row').data(addressFormat.format, function (d) {
-             return d.toString();
+         function prepareForSuccess() {
+           _success = uiSuccess(context);
+           _location = null;
+           if (!services.geocoder) return;
+           services.geocoder.reverse(context.map().center(), function (err, result) {
+             if (err || !result || !result.address) return;
+             var addr = result.address;
+             var place = addr && (addr.town || addr.city || addr.county) || '';
+             var region = addr && (addr.state || addr.country) || '';
+             var separator = place && region ? _t('success.thank_you_where.separator') : '';
+             _location = _t('success.thank_you_where.format', {
+               place: place,
+               separator: separator,
+               region: region
+             });
            });
+         }
 
-           rows.exit().remove();
-           rows.enter().append('div').attr('class', 'addr-row').selectAll('input').data(row).enter().append('input').property('type', 'text').call(updatePlaceholder).attr('class', function (d) {
-             return 'addr-' + d.id;
-           }).call(utilNoAuto).each(addDropdown).style('width', function (d) {
-             return d.width * 100 + '%';
-           });
+         mode.selectedIDs = function () {
+           return _conflictsUi ? _conflictsUi.shownEntityIds() : [];
+         };
 
-           function addDropdown(d) {
-             if (dropdowns.indexOf(d.id) === -1) return; // not a dropdown
+         mode.enter = function () {
+           // Show sidebar
+           context.ui().sidebar.expand();
 
-             var nearValues = d.id === 'street' ? getNearStreets : d.id === 'city' ? getNearCities : getNearValues;
-             select(this).call(uiCombobox(context, 'address-' + d.id).minItems(1).caseSensitive(true).fetcher(function (value, callback) {
-               callback(nearValues('addr:' + d.id));
-             }));
+           function done() {
+             context.ui().sidebar.show(commit);
            }
 
-           _wrap.selectAll('input').on('blur', change()).on('change', change());
-
-           _wrap.selectAll('input:not(.combobox-input)').on('input', change(true));
-
-           if (_tags) updateTags(_tags);
-         }
+           keybindingOn();
+           context.container().selectAll('.main-content').classed('active', false).classed('inactive', true);
+           var osm = context.connection();
 
-         function address(selection) {
-           _selection = selection;
-           _wrap = selection.selectAll('.form-field-input-wrap').data([0]);
-           _wrap = _wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(_wrap);
-           var extent = combinedEntityExtent();
+           if (!osm) {
+             cancel();
+             return;
+           }
 
-           if (extent) {
-             var countryCode;
+           if (osm.authenticated()) {
+             done();
+           } else {
+             osm.authenticate(function (err) {
+               if (err) {
+                 cancel();
+               } else {
+                 done();
+               }
+             });
+           }
+         };
 
-             if (context.inIntro()) {
-               // localize the address format for the walkthrough
-               countryCode = _t('intro.graph.countrycode');
-             } else {
-               var center = extent.center();
-               countryCode = iso1A2Code(center);
-             }
+         mode.exit = function () {
+           keybindingOff();
+           context.container().selectAll('.main-content').classed('active', true).classed('inactive', false);
+           context.ui().sidebar.hide();
+         };
 
-             if (countryCode) {
-               _countryCode = countryCode.toLowerCase();
-               updateForCountryCode();
-             }
-           }
-         }
+         return mode;
+       }
 
-         function change(onInput) {
-           return function () {
-             var tags = {};
+       function modeSelectError(context, selectedErrorID, selectedErrorService) {
+         var mode = {
+           id: 'select-error',
+           button: 'browse'
+         };
+         var keybinding = utilKeybinding('select-error');
+         var errorService = services[selectedErrorService];
+         var errorEditor;
 
-             _wrap.selectAll('input').each(function (subfield) {
-               var key = field.key + ':' + subfield.id;
-               var value = this.value;
-               if (!onInput) value = context.cleanTagValue(value); // don't override multiple values with blank string
+         switch (selectedErrorService) {
+           case 'improveOSM':
+             errorEditor = uiImproveOsmEditor(context).on('change', function () {
+               context.map().pan([0, 0]); // trigger a redraw
 
-               if (Array.isArray(_tags[key]) && !value) return;
-               tags[key] = value || undefined;
+               var error = checkSelectedID();
+               if (!error) return;
+               context.ui().sidebar.show(errorEditor.error(error));
              });
+             break;
 
-             dispatch$1.call('change', this, tags, onInput);
-           };
-         }
-
-         function updatePlaceholder(inputSelection) {
-           return inputSelection.attr('placeholder', function (subfield) {
-             if (_tags && Array.isArray(_tags[field.key + ':' + subfield.id])) {
-               return _t('inspector.multiple_values');
-             }
+           case 'keepRight':
+             errorEditor = uiKeepRightEditor(context).on('change', function () {
+               context.map().pan([0, 0]); // trigger a redraw
 
-             if (_countryCode) {
-               var localkey = subfield.id + '!' + _countryCode;
-               var tkey = addrField.strings.placeholders[localkey] ? localkey : subfield.id;
-               return addrField.t('placeholders.' + tkey);
-             }
-           });
-         }
+               var error = checkSelectedID();
+               if (!error) return;
+               context.ui().sidebar.show(errorEditor.error(error));
+             });
+             break;
 
-         function updateTags(tags) {
-           utilGetSetValue(_wrap.selectAll('input'), function (subfield) {
-             var val = tags[field.key + ':' + subfield.id];
-             return typeof val === 'string' ? val : '';
-           }).attr('title', function (subfield) {
-             var val = tags[field.key + ':' + subfield.id];
-             return val && Array.isArray(val) && val.filter(Boolean).join('\n');
-           }).classed('mixed', function (subfield) {
-             return Array.isArray(tags[field.key + ':' + subfield.id]);
-           }).call(updatePlaceholder);
-         }
+           case 'osmose':
+             errorEditor = uiOsmoseEditor(context).on('change', function () {
+               context.map().pan([0, 0]); // trigger a redraw
 
-         function combinedEntityExtent() {
-           return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+               var error = checkSelectedID();
+               if (!error) return;
+               context.ui().sidebar.show(errorEditor.error(error));
+             });
+             break;
          }
 
-         address.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs;
-           _entityIDs = val;
-           return address;
-         };
-
-         address.tags = function (tags) {
-           _tags = tags;
-           updateTags(tags);
-         };
-
-         address.focus = function () {
-           var node = _wrap.selectAll('input').node();
+         var behaviors = [behaviorBreathe(), behaviorHover(context), behaviorSelect(context), behaviorLasso(context), modeDragNode(context).behavior, modeDragNote(context).behavior];
 
-           if (node) node.focus();
-         };
+         function checkSelectedID() {
+           if (!errorService) return;
+           var error = errorService.getError(selectedErrorID);
 
-         return utilRebind(address, dispatch$1, 'on');
-       }
+           if (!error) {
+             context.enter(modeBrowse(context));
+           }
 
-       function uiFieldCycleway(field, context) {
-         var dispatch$1 = dispatch('change');
-         var items = select(null);
-         var wrap = select(null);
+           return error;
+         }
 
-         var _tags;
+         mode.zoomToSelected = function () {
+           if (!errorService) return;
+           var error = errorService.getError(selectedErrorID);
 
-         function cycleway(selection) {
-           function stripcolon(s) {
-             return s.replace(':', '');
+           if (error) {
+             context.map().centerZoomEase(error.loc, 20);
            }
+         };
 
-           wrap = selection.selectAll('.form-field-input-wrap').data([0]);
-           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
-           var div = wrap.selectAll('ul').data([0]);
-           div = div.enter().append('ul').attr('class', 'rows').merge(div);
-           var keys = ['cycleway:left', 'cycleway:right'];
-           items = div.selectAll('li').data(keys);
-           var enter = items.enter().append('li').attr('class', function (d) {
-             return 'labeled-input preset-cycleway-' + stripcolon(d);
-           });
-           enter.append('span').attr('class', 'label preset-label-cycleway').attr('for', function (d) {
-             return 'preset-input-cycleway-' + stripcolon(d);
-           }).html(function (d) {
-             return field.t.html('types.' + d);
-           });
-           enter.append('div').attr('class', 'preset-input-cycleway-wrap').append('input').attr('type', 'text').attr('class', function (d) {
-             return 'preset-input-cycleway preset-input-' + stripcolon(d);
-           }).call(utilNoAuto).each(function (d) {
-             select(this).call(uiCombobox(context, 'cycleway-' + stripcolon(d)).data(cycleway.options(d)));
-           });
-           items = items.merge(enter); // Update
-
-           wrap.selectAll('.preset-input-cycleway').on('change', change).on('blur', change);
-         }
+         mode.enter = function () {
+           var error = checkSelectedID();
+           if (!error) return;
+           behaviors.forEach(context.install);
+           keybinding.on(_t('inspector.zoom_to.key'), mode.zoomToSelected).on('⎋', esc, true);
+           select(document).call(keybinding);
+           selectError();
+           var sidebar = context.ui().sidebar;
+           sidebar.show(errorEditor.error(error));
+           context.map().on('drawn.select-error', selectError); // class the error as selected, or return to browse mode if the error is gone
 
-         function change(d3_event, key) {
-           var newValue = context.cleanTagValue(utilGetSetValue(select(this))); // don't override multiple values with blank string
+           function selectError(d3_event, drawn) {
+             if (!checkSelectedID()) return;
+             var selection = context.surface().selectAll('.itemId-' + selectedErrorID + '.' + selectedErrorService);
 
-           if (!newValue && (Array.isArray(_tags.cycleway) || Array.isArray(_tags[key]))) return;
+             if (selection.empty()) {
+               // Return to browse mode if selected DOM elements have
+               // disappeared because the user moved them out of view..
+               var source = d3_event && d3_event.type === 'zoom' && d3_event.sourceEvent;
 
-           if (newValue === 'none' || newValue === '') {
-             newValue = undefined;
+               if (drawn && source && (source.type === 'pointermove' || source.type === 'mousemove' || source.type === 'touchmove')) {
+                 context.enter(modeBrowse(context));
+               }
+             } else {
+               selection.classed('selected', true);
+               context.selectedErrorID(selectedErrorID);
+             }
            }
 
-           var otherKey = key === 'cycleway:left' ? 'cycleway:right' : 'cycleway:left';
-           var otherValue = typeof _tags.cycleway === 'string' ? _tags.cycleway : _tags[otherKey];
-
-           if (otherValue && Array.isArray(otherValue)) {
-             // we must always have an explicit value for comparison
-             otherValue = otherValue[0];
+           function esc() {
+             if (context.container().select('.combobox').size()) return;
+             context.enter(modeBrowse(context));
            }
+         };
 
-           if (otherValue === 'none' || otherValue === '') {
-             otherValue = undefined;
-           }
+         mode.exit = function () {
+           behaviors.forEach(context.uninstall);
+           select(document).call(keybinding.unbind);
+           context.surface().selectAll('.qaItem.selected').classed('selected hover', false);
+           context.map().on('drawn.select-error', null);
+           context.ui().sidebar.hide();
+           context.selectedErrorID(null);
+           context.features().forceVisible([]);
+         };
 
-           var tag = {}; // If the left and right tags match, use the cycleway tag to tag both
-           // sides the same way
+         return mode;
+       }
 
-           if (newValue === otherValue) {
-             tag = {
-               cycleway: newValue,
-               'cycleway:left': undefined,
-               'cycleway:right': undefined
-             };
-           } else {
-             // Always set both left and right as changing one can affect the other
-             tag = {
-               cycleway: undefined
-             };
-             tag[key] = newValue;
-             tag[otherKey] = otherValue;
-           }
+       function uiToolOldDrawModes(context) {
+         var tool = {
+           id: 'old_modes',
+           label: _t.html('toolbar.add_feature')
+         };
+         var modes = [modeAddPoint(context, {
+           title: _t.html('modes.add_point.title'),
+           button: 'point',
+           description: _t.html('modes.add_point.description'),
+           preset: _mainPresetIndex.item('point'),
+           key: '1'
+         }), modeAddLine(context, {
+           title: _t.html('modes.add_line.title'),
+           button: 'line',
+           description: _t.html('modes.add_line.description'),
+           preset: _mainPresetIndex.item('line'),
+           key: '2'
+         }), modeAddArea(context, {
+           title: _t.html('modes.add_area.title'),
+           button: 'area',
+           description: _t.html('modes.add_area.description'),
+           preset: _mainPresetIndex.item('area'),
+           key: '3'
+         })];
 
-           dispatch$1.call('change', this, tag);
+         function enabled() {
+           return osmEditable();
          }
 
-         cycleway.options = function () {
-           return Object.keys(field.strings.options).map(function (option) {
-             return {
-               title: field.t('options.' + option + '.description'),
-               value: option
-             };
+         function osmEditable() {
+           return context.editable();
+         }
+
+         modes.forEach(function (mode) {
+           context.keybinding().on(mode.key, function () {
+             if (!enabled()) return;
+
+             if (mode.id === context.mode().id) {
+               context.enter(modeBrowse(context));
+             } else {
+               context.enter(mode);
+             }
            });
-         };
+         });
 
-         cycleway.tags = function (tags) {
-           _tags = tags; // If cycleway is set, use that instead of individual values
+         tool.render = function (selection) {
+           var wrap = selection.append('div').attr('class', 'joined').style('display', 'flex');
 
-           var commonValue = typeof tags.cycleway === 'string' && tags.cycleway;
-           utilGetSetValue(items.selectAll('.preset-input-cycleway'), function (d) {
-             if (commonValue) return commonValue;
-             return !tags.cycleway && typeof tags[d] === 'string' ? tags[d] : '';
-           }).attr('title', function (d) {
-             if (Array.isArray(tags.cycleway) || Array.isArray(tags[d])) {
-               var vals = [];
+           var debouncedUpdate = debounce(update, 500, {
+             leading: true,
+             trailing: true
+           });
 
-               if (Array.isArray(tags.cycleway)) {
-                 vals = vals.concat(tags.cycleway);
-               }
+           context.map().on('move.modes', debouncedUpdate).on('drawn.modes', debouncedUpdate);
+           context.on('enter.modes', update);
+           update();
 
-               if (Array.isArray(tags[d])) {
-                 vals = vals.concat(tags[d]);
+           function update() {
+             var buttons = wrap.selectAll('button.add-button').data(modes, function (d) {
+               return d.id;
+             }); // exit
+
+             buttons.exit().remove(); // enter
+
+             var buttonsEnter = buttons.enter().append('button').attr('class', function (d) {
+               return d.id + ' add-button bar-button';
+             }).on('click.mode-buttons', function (d3_event, d) {
+               if (!enabled()) return; // When drawing, ignore accidental clicks on mode buttons - #4042
+
+               var currMode = context.mode().id;
+               if (/^draw/.test(currMode)) return;
+
+               if (d.id === currMode) {
+                 context.enter(modeBrowse(context));
+               } else {
+                 context.enter(d);
                }
+             }).call(uiTooltip().placement('bottom').title(function (d) {
+               return d.description;
+             }).keys(function (d) {
+               return [d.key];
+             }).scrollContainer(context.container().select('.top-toolbar')));
+             buttonsEnter.each(function (d) {
+               select(this).call(svgIcon('#iD-icon-' + d.button));
+             });
+             buttonsEnter.append('span').attr('class', 'label').html(function (mode) {
+               return mode.title;
+             }); // if we are adding/removing the buttons, check if toolbar has overflowed
 
-               return vals.filter(Boolean).join('\n');
-             }
+             if (buttons.enter().size() || buttons.exit().size()) {
+               context.ui().checkOverflow('.top-toolbar', true);
+             } // update
 
-             return null;
-           }).attr('placeholder', function (d) {
-             if (Array.isArray(tags.cycleway) || Array.isArray(tags[d])) {
-               return _t('inspector.multiple_values');
-             }
 
-             return field.placeholder();
-           }).classed('mixed', function (d) {
-             return Array.isArray(tags.cycleway) || Array.isArray(tags[d]);
-           });
+             buttons = buttons.merge(buttonsEnter).classed('disabled', function (d) {
+               return !enabled();
+             }).classed('active', function (d) {
+               return context.mode() && context.mode().button === d.button;
+             });
+           }
          };
 
-         cycleway.focus = function () {
-           var node = wrap.selectAll('input').node();
-           if (node) node.focus();
+         return tool;
+       }
+
+       function uiToolNotes(context) {
+         var tool = {
+           id: 'notes',
+           label: _t.html('modes.add_note.label')
          };
+         var mode = modeAddNote(context);
 
-         return utilRebind(cycleway, dispatch$1, 'on');
-       }
+         function enabled() {
+           return notesEnabled() && notesEditable();
+         }
 
-       function uiFieldLanes(field, context) {
-         var dispatch$1 = dispatch('change');
-         var LANE_WIDTH = 40;
-         var LANE_HEIGHT = 200;
-         var _entityIDs = [];
+         function notesEnabled() {
+           var noteLayer = context.layers().layer('notes');
+           return noteLayer && noteLayer.enabled();
+         }
 
-         function lanes(selection) {
-           var lanesData = context.entity(_entityIDs[0]).lanes();
+         function notesEditable() {
+           var mode = context.mode();
+           return context.map().notesEditable() && mode && mode.id !== 'save';
+         }
 
-           if (!context.container().select('.inspector-wrap.inspector-hidden').empty() || !selection.node().parentNode) {
-             selection.call(lanes.off);
-             return;
+         context.keybinding().on(mode.key, function () {
+           if (!enabled()) return;
+
+           if (mode.id === context.mode().id) {
+             context.enter(modeBrowse(context));
+           } else {
+             context.enter(mode);
            }
+         });
 
-           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
-           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
-           var surface = wrap.selectAll('.surface').data([0]);
-           var d = utilGetDimensions(wrap);
-           var freeSpace = d[0] - lanesData.lanes.length * LANE_WIDTH * 1.5 + LANE_WIDTH * 0.5;
-           surface = surface.enter().append('svg').attr('width', d[0]).attr('height', 300).attr('class', 'surface').merge(surface);
-           var lanesSelection = surface.selectAll('.lanes').data([0]);
-           lanesSelection = lanesSelection.enter().append('g').attr('class', 'lanes').merge(lanesSelection);
-           lanesSelection.attr('transform', function () {
-             return 'translate(' + freeSpace / 2 + ', 0)';
-           });
-           var lane = lanesSelection.selectAll('.lane').data(lanesData.lanes);
-           lane.exit().remove();
-           var enter = lane.enter().append('g').attr('class', 'lane');
-           enter.append('g').append('rect').attr('y', 50).attr('width', LANE_WIDTH).attr('height', LANE_HEIGHT);
-           enter.append('g').attr('class', 'forward').append('text').attr('y', 40).attr('x', 14).html('▲');
-           enter.append('g').attr('class', 'bothways').append('text').attr('y', 40).attr('x', 14).html('▲▼');
-           enter.append('g').attr('class', 'backward').append('text').attr('y', 40).attr('x', 14).html('▼');
-           lane = lane.merge(enter);
-           lane.attr('transform', function (d) {
-             return 'translate(' + LANE_WIDTH * d.index * 1.5 + ', 0)';
-           });
-           lane.select('.forward').style('visibility', function (d) {
-             return d.direction === 'forward' ? 'visible' : 'hidden';
-           });
-           lane.select('.bothways').style('visibility', function (d) {
-             return d.direction === 'bothways' ? 'visible' : 'hidden';
-           });
-           lane.select('.backward').style('visibility', function (d) {
-             return d.direction === 'backward' ? 'visible' : 'hidden';
+         tool.render = function (selection) {
+           var debouncedUpdate = debounce(update, 500, {
+             leading: true,
+             trailing: true
            });
-         }
 
-         lanes.entityIDs = function (val) {
-           _entityIDs = val;
-         };
+           context.map().on('move.notes', debouncedUpdate).on('drawn.notes', debouncedUpdate);
+           context.on('enter.notes', update);
+           update();
 
-         lanes.tags = function () {};
+           function update() {
+             var showNotes = notesEnabled();
+             var data = showNotes ? [mode] : [];
+             var buttons = selection.selectAll('button.add-button').data(data, function (d) {
+               return d.id;
+             }); // exit
 
-         lanes.focus = function () {};
+             buttons.exit().remove(); // enter
 
-         lanes.off = function () {};
+             var buttonsEnter = buttons.enter().append('button').attr('class', function (d) {
+               return d.id + ' add-button bar-button';
+             }).on('click.notes', function (d3_event, d) {
+               if (!enabled()) return; // When drawing, ignore accidental clicks on mode buttons - #4042
 
-         return utilRebind(lanes, dispatch$1, 'on');
-       }
-       uiFieldLanes.supportsMultiselection = false;
+               var currMode = context.mode().id;
+               if (/^draw/.test(currMode)) return;
 
-       var _languagesArray = [];
-       function uiFieldLocalized(field, context) {
-         var dispatch$1 = dispatch('change', 'input');
-         var wikipedia = services.wikipedia;
-         var input = select(null);
-         var localizedInputs = select(null);
+               if (d.id === currMode) {
+                 context.enter(modeBrowse(context));
+               } else {
+                 context.enter(d);
+               }
+             }).call(uiTooltip().placement('bottom').title(function (d) {
+               return d.description;
+             }).keys(function (d) {
+               return [d.key];
+             }).scrollContainer(context.container().select('.top-toolbar')));
+             buttonsEnter.each(function (d) {
+               select(this).call(svgIcon(d.icon || '#iD-icon-' + d.button));
+             }); // if we are adding/removing the buttons, check if toolbar has overflowed
 
-         var _countryCode;
+             if (buttons.enter().size() || buttons.exit().size()) {
+               context.ui().checkOverflow('.top-toolbar', true);
+             } // update
 
-         var _tags; // A concern here in switching to async data means that _languagesArray will not
-         // be available the first time through, so things like the fetchers and
-         // the language() function will not work immediately.
 
+             buttons = buttons.merge(buttonsEnter).classed('disabled', function (d) {
+               return !enabled();
+             }).classed('active', function (d) {
+               return context.mode() && context.mode().button === d.button;
+             });
+           }
+         };
 
-         _mainFileFetcher.get('languages').then(loadLanguagesArray)["catch"](function () {
-           /* ignore */
-         });
-         var _territoryLanguages = {};
-         _mainFileFetcher.get('territory_languages').then(function (d) {
-           _territoryLanguages = d;
-         })["catch"](function () {
-           /* ignore */
-         });
-         var allSuggestions = _mainPresetIndex.collection.filter(function (p) {
-           return p.suggestion === true;
-         }); // reuse these combos
+         tool.uninstall = function () {
+           context.on('enter.editor.notes', null).on('exit.editor.notes', null).on('enter.notes', null);
+           context.map().on('move.notes', null).on('drawn.notes', null);
+         };
 
-         var langCombo = uiCombobox(context, 'localized-lang').fetcher(fetchLanguages).minItems(0);
-         var brandCombo = uiCombobox(context, 'localized-brand').canAutocomplete(false).minItems(1);
+         return tool;
+       }
 
-         var _selection = select(null);
+       function uiToolSave(context) {
+         var tool = {
+           id: 'save',
+           label: _t.html('save.title')
+         };
+         var button = null;
+         var tooltipBehavior = null;
+         var history = context.history();
+         var key = uiCmd('⌘S');
+         var _numChanges = 0;
 
-         var _multilingual = [];
+         function isSaving() {
+           var mode = context.mode();
+           return mode && mode.id === 'save';
+         }
 
-         var _buttonTip = uiTooltip().title(_t.html('translate.translate')).placement('left');
+         function isDisabled() {
+           return _numChanges === 0 || isSaving();
+         }
 
-         var _wikiTitles;
+         function save(d3_event) {
+           d3_event.preventDefault();
 
-         var _entityIDs = [];
+           if (!context.inIntro() && !isSaving() && history.hasChanges()) {
+             context.enter(modeSave(context));
+           }
+         }
 
-         function loadLanguagesArray(dataLanguages) {
-           if (_languagesArray.length !== 0) return; // some conversion is needed to ensure correct OSM tags are used
+         function bgColor() {
+           var step;
 
-           var replacements = {
-             sr: 'sr-Cyrl',
-             // in OSM, `sr` implies Cyrillic
-             'sr-Cyrl': false // `sr-Cyrl` isn't used in OSM
+           if (_numChanges === 0) {
+             return null;
+           } else if (_numChanges <= 50) {
+             step = _numChanges / 50;
+             return d3_interpolateRgb('#fff', '#ff8')(step); // white -> yellow
+           } else {
+             step = Math.min((_numChanges - 50) / 50, 1.0);
+             return d3_interpolateRgb('#ff8', '#f88')(step); // yellow -> red
+           }
+         }
 
-           };
+         function updateCount() {
+           var val = history.difference().summary().length;
+           if (val === _numChanges) return;
+           _numChanges = val;
 
-           for (var code in dataLanguages) {
-             if (replacements[code] === false) continue;
-             var metaCode = code;
-             if (replacements[code]) metaCode = replacements[code];
+           if (tooltipBehavior) {
+             tooltipBehavior.title(_t.html(_numChanges > 0 ? 'save.help' : 'save.no_changes')).keys([key]);
+           }
 
-             _languagesArray.push({
-               localName: _mainLocalizer.languageName(metaCode, {
-                 localOnly: true
-               }),
-               nativeName: dataLanguages[metaCode].nativeName,
-               code: code,
-               label: _mainLocalizer.languageName(metaCode)
-             });
+           if (button) {
+             button.classed('disabled', isDisabled()).style('background', bgColor());
+             button.select('span.count').html(_numChanges);
            }
          }
 
-         function calcLocked() {
-           // only lock the Name field
-           var isLocked = field.id === 'name' && _entityIDs.length && // lock the field if any feature needs it
-           _entityIDs.some(function (entityID) {
-             var entity = context.graph().hasEntity(entityID);
-             if (!entity) return false;
+         tool.render = function (selection) {
+           tooltipBehavior = uiTooltip().placement('bottom').title(_t.html('save.no_changes')).keys([key]).scrollContainer(context.container().select('.top-toolbar'));
+           var lastPointerUpType;
+           button = selection.append('button').attr('class', 'save disabled bar-button').on('pointerup', function (d3_event) {
+             lastPointerUpType = d3_event.pointerType;
+           }).on('click', function (d3_event) {
+             save(d3_event);
 
-             var original = context.graph().base().entities[_entityIDs[0]];
+             if (_numChanges === 0 && (lastPointerUpType === 'touch' || lastPointerUpType === 'pen')) {
+               // there are no tooltips for touch interactions so flash feedback instead
+               context.ui().flash.duration(2000).iconName('#iD-icon-save').iconClass('disabled').label(_t.html('save.no_changes'))();
+             }
 
-             var hasOriginalName = original && entity.tags.name && entity.tags.name === original.tags.name; // if the name was already edited manually then allow further editing
+             lastPointerUpType = null;
+           }).call(tooltipBehavior);
+           button.call(svgIcon('#iD-icon-save'));
+           button.append('span').attr('class', 'count').attr('aria-hidden', 'true').html('0');
+           updateCount();
+           context.keybinding().on(key, save, true);
+           context.history().on('change.save', updateCount);
+           context.on('enter.save', function () {
+             if (button) {
+               button.classed('disabled', isDisabled());
 
-             if (!hasOriginalName) return false; // features linked to Wikidata are likely important and should be protected
+               if (isSaving()) {
+                 button.call(tooltipBehavior.hide);
+               }
+             }
+           });
+         };
 
-             if (entity.tags.wikidata) return true; // assume the name has already been confirmed if its source has been researched
+         tool.uninstall = function () {
+           context.keybinding().off(key, true);
+           context.history().on('change.save', null);
+           context.on('enter.save', null);
+           button = null;
+           tooltipBehavior = null;
+         };
 
-             if (entity.tags['name:etymology:wikidata']) return true;
-             var preset = _mainPresetIndex.match(entity, context.graph());
-             var isSuggestion = preset && preset.suggestion;
-             var showsBrand = preset && preset.originalFields.filter(function (d) {
-               return d.id === 'brand';
-             }).length; // protect standardized brand names
+         return tool;
+       }
 
-             return isSuggestion && !showsBrand;
-           });
+       function uiToolSidebarToggle(context) {
+         var tool = {
+           id: 'sidebar_toggle',
+           label: _t.html('toolbar.inspect')
+         };
 
-           field.locked(isLocked);
-         } // update _multilingual, maintaining the existing order
+         tool.render = function (selection) {
+           selection.append('button').attr('class', 'bar-button').on('click', function () {
+             context.ui().sidebar.toggle();
+           }).call(uiTooltip().placement('bottom').title(_t.html('sidebar.tooltip')).keys([_t('sidebar.key')]).scrollContainer(context.container().select('.top-toolbar'))).call(svgIcon('#iD-icon-sidebar-' + (_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left')));
+         };
 
+         return tool;
+       }
 
-         function calcMultilingual(tags) {
-           var existingLangsOrdered = _multilingual.map(function (item) {
-             return item.lang;
-           });
+       function uiToolUndoRedo(context) {
+         var tool = {
+           id: 'undo_redo',
+           label: _t.html('toolbar.undo_redo')
+         };
+         var commands = [{
+           id: 'undo',
+           cmd: uiCmd('⌘Z'),
+           action: function action() {
+             context.undo();
+           },
+           annotation: function annotation() {
+             return context.history().undoAnnotation();
+           },
+           icon: 'iD-icon-' + (_mainLocalizer.textDirection() === 'rtl' ? 'redo' : 'undo')
+         }, {
+           id: 'redo',
+           cmd: uiCmd('⌘⇧Z'),
+           action: function action() {
+             context.redo();
+           },
+           annotation: function annotation() {
+             return context.history().redoAnnotation();
+           },
+           icon: 'iD-icon-' + (_mainLocalizer.textDirection() === 'rtl' ? 'undo' : 'redo')
+         }];
 
-           var existingLangs = new Set(existingLangsOrdered.filter(Boolean));
+         function editable() {
+           return context.mode() && context.mode().id !== 'save' && context.map().editableDataEnabled(true
+           /* ignore min zoom */
+           );
+         }
 
-           for (var k in tags) {
-             var m = k.match(/^(.*):(.+)$/);
+         tool.render = function (selection) {
+           var tooltipBehavior = uiTooltip().placement('bottom').title(function (d) {
+             return d.annotation() ? _t.html(d.id + '.tooltip', {
+               action: d.annotation()
+             }) : _t.html(d.id + '.nothing');
+           }).keys(function (d) {
+             return [d.cmd];
+           }).scrollContainer(context.container().select('.top-toolbar'));
+           var lastPointerUpType;
+           var buttons = selection.selectAll('button').data(commands).enter().append('button').attr('class', function (d) {
+             return 'disabled ' + d.id + '-button bar-button';
+           }).on('pointerup', function (d3_event) {
+             // `pointerup` is always called before `click`
+             lastPointerUpType = d3_event.pointerType;
+           }).on('click', function (d3_event, d) {
+             d3_event.preventDefault();
+             var annotation = d.annotation();
 
-             if (m && m[1] === field.key && m[2]) {
-               var item = {
-                 lang: m[2],
-                 value: tags[k]
-               };
+             if (editable() && annotation) {
+               d.action();
+             }
 
-               if (existingLangs.has(item.lang)) {
-                 // update the value
-                 _multilingual[existingLangsOrdered.indexOf(item.lang)].value = item.value;
-                 existingLangs["delete"](item.lang);
-               } else {
-                 _multilingual.push(item);
-               }
+             if (editable() && (lastPointerUpType === 'touch' || lastPointerUpType === 'pen')) {
+               // there are no tooltips for touch interactions so flash feedback instead
+               var text = annotation ? _t(d.id + '.tooltip', {
+                 action: annotation
+               }) : _t(d.id + '.nothing');
+               context.ui().flash.duration(2000).iconName('#' + d.icon).iconClass(annotation ? '' : 'disabled').label(text)();
              }
-           }
 
-           _multilingual = _multilingual.filter(function (item) {
-             return !item.lang || !existingLangs.has(item.lang);
+             lastPointerUpType = null;
+           }).call(tooltipBehavior);
+           buttons.each(function (d) {
+             select(this).call(svgIcon('#' + d.icon));
+           });
+           context.keybinding().on(commands[0].cmd, function (d3_event) {
+             d3_event.preventDefault();
+             if (editable()) commands[0].action();
+           }).on(commands[1].cmd, function (d3_event) {
+             d3_event.preventDefault();
+             if (editable()) commands[1].action();
            });
-         }
-
-         function localized(selection) {
-           _selection = selection;
-           calcLocked();
-           var isLocked = field.locked();
-           var singularEntity = _entityIDs.length === 1 && context.hasEntity(_entityIDs[0]);
-           var preset = singularEntity && _mainPresetIndex.match(singularEntity, context.graph());
-           var wrap = selection.selectAll('.form-field-input-wrap').data([0]); // enter/update
-
-           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
-           input = wrap.selectAll('.localized-main').data([0]); // enter/update
 
-           input = input.enter().append('input').attr('type', 'text').attr('id', field.domId).attr('class', 'localized-main').call(utilNoAuto).merge(input);
+           var debouncedUpdate = debounce(update, 500, {
+             leading: true,
+             trailing: true
+           });
 
-           if (preset && field.id === 'name') {
-             var pTag = preset.id.split('/', 2);
-             var pKey = pTag[0];
-             var pValue = pTag[1];
+           context.map().on('move.undo_redo', debouncedUpdate).on('drawn.undo_redo', debouncedUpdate);
+           context.history().on('change.undo_redo', function (difference) {
+             if (difference) update();
+           });
+           context.on('enter.undo_redo', update);
 
-             if (!preset.suggestion) {
-               // Not a suggestion preset - Add a suggestions dropdown if it makes sense to.
-               // This code attempts to determine if the matched preset is the
-               // kind of preset that even can benefit from name suggestions..
-               // - true = shops, cafes, hotels, etc. (also generic and fallback presets)
-               // - false = churches, parks, hospitals, etc. (things not in the index)
-               var isFallback = preset.isFallback();
-               var goodSuggestions = allSuggestions.filter(function (s) {
-                 if (isFallback) return true;
-                 var sTag = s.id.split('/', 2);
-                 var sKey = sTag[0];
-                 var sValue = sTag[1];
-                 return pKey === sKey && (!pValue || pValue === sValue);
-               }); // Show the suggestions.. If the user picks one, change the tags..
+           function update() {
+             buttons.classed('disabled', function (d) {
+               return !editable() || !d.annotation();
+             }).each(function () {
+               var selection = select(this);
 
-               if (allSuggestions.length && goodSuggestions.length) {
-                 input.on('blur.localized', checkBrandOnBlur).call(brandCombo.fetcher(fetchBrandNames(preset, allSuggestions)).on('accept', acceptBrand).on('cancel', cancelBrand));
+               if (!selection.select('.tooltip.in').empty()) {
+                 selection.call(tooltipBehavior.updateContent);
                }
-             }
-           }
-
-           input.classed('disabled', !!isLocked).attr('readonly', isLocked || null).on('input', change(true)).on('blur', change()).on('change', change());
-           var translateButton = wrap.selectAll('.localized-add').data([0]);
-           translateButton = translateButton.enter().append('button').attr('class', 'localized-add form-field-button').call(svgIcon('#iD-icon-plus')).merge(translateButton);
-           translateButton.classed('disabled', !!isLocked).call(isLocked ? _buttonTip.destroy : _buttonTip).on('click', addNew);
-
-           if (_tags && !_multilingual.length) {
-             calcMultilingual(_tags);
+             });
            }
+         };
 
-           localizedInputs = selection.selectAll('.localized-multilingual').data([0]);
-           localizedInputs = localizedInputs.enter().append('div').attr('class', 'localized-multilingual').merge(localizedInputs);
-           localizedInputs.call(renderMultilingual);
-           localizedInputs.selectAll('button, input').classed('disabled', !!isLocked).attr('readonly', isLocked || null); // We are not guaranteed to get an `accept` or `cancel` when blurring the field.
-           // (This can happen if the user actives the combo, arrows down, and then clicks off to blur)
-           // So compare the current field value against the suggestions one last time.
+         tool.uninstall = function () {
+           context.keybinding().off(commands[0].cmd).off(commands[1].cmd);
+           context.map().on('move.undo_redo', null).on('drawn.undo_redo', null);
+           context.history().on('change.undo_redo', null);
+           context.on('enter.undo_redo', null);
+         };
 
-           function checkBrandOnBlur() {
-             var latest = _entityIDs.length === 1 && context.hasEntity(_entityIDs[0]);
-             if (!latest) return; // deleting the entity blurred the field?
+         return tool;
+       }
 
-             var preset = _mainPresetIndex.match(latest, context.graph());
-             if (preset && preset.suggestion) return; // already accepted
+       function uiTopToolbar(context) {
+         var sidebarToggle = uiToolSidebarToggle(context),
+             modes = uiToolOldDrawModes(context),
+             notes = uiToolNotes(context),
+             undoRedo = uiToolUndoRedo(context),
+             save = uiToolSave(context);
 
-             var name = utilGetSetValue(input).trim();
-             var matched = allSuggestions.filter(function (s) {
-               return name === s.name();
-             });
+         function notesEnabled() {
+           var noteLayer = context.layers().layer('notes');
+           return noteLayer && noteLayer.enabled();
+         }
 
-             if (matched.length === 1) {
-               acceptBrand({
-                 suggestion: matched[0]
-               });
-             } else {
-               cancelBrand();
+         function topToolbar(bar) {
+           bar.on('wheel.topToolbar', function (d3_event) {
+             if (!d3_event.deltaX) {
+               // translate vertical scrolling into horizontal scrolling in case
+               // the user doesn't have an input device that can scroll horizontally
+               bar.node().scrollLeft += d3_event.deltaY;
              }
-           }
+           });
 
-           function acceptBrand(d) {
-             var entity = _entityIDs.length === 1 && context.hasEntity(_entityIDs[0]);
+           var debouncedUpdate = debounce(update, 500, {
+             leading: true,
+             trailing: true
+           });
 
-             if (!d || !entity) {
-               cancelBrand();
-               return;
-             }
+           context.layers().on('change.topToolbar', debouncedUpdate);
+           update();
 
-             var tags = entity.tags;
-             var geometry = entity.geometry(context.graph());
-             var removed = preset.unsetTags(tags, geometry);
+           function update() {
+             var tools = [sidebarToggle, 'spacer', modes];
+             tools.push('spacer');
 
-             for (var k in tags) {
-               tags[k] = removed[k]; // set removed tags to `undefined`
+             if (notesEnabled()) {
+               tools = tools.concat([notes, 'spacer']);
              }
 
-             tags = d.suggestion.setTags(tags, geometry);
-             utilGetSetValue(input, tags.name);
-             dispatch$1.call('change', this, tags);
-           } // user hit escape
-
-
-           function cancelBrand() {
-             var name = utilGetSetValue(input);
-             dispatch$1.call('change', this, {
-               name: name
+             tools = tools.concat([undoRedo, save]);
+             var toolbarItems = bar.selectAll('.toolbar-item').data(tools, function (d) {
+               return d.id || d;
              });
-           }
-
-           function fetchBrandNames(preset, suggestions) {
-             var pTag = preset.id.split('/', 2);
-             var pKey = pTag[0];
-             var pValue = pTag[1];
-             return function (value, callback) {
-               var results = [];
-
-               if (value && value.length > 2) {
-                 for (var i = 0; i < suggestions.length; i++) {
-                   var s = suggestions[i]; // don't suggest brands from incompatible countries
-
-                   if (_countryCode && s.countryCodes && s.countryCodes.indexOf(_countryCode) === -1) continue;
-                   var sTag = s.id.split('/', 2);
-                   var sKey = sTag[0];
-                   var sValue = sTag[1];
-                   var subtitle = s.subtitle();
-                   var name = s.name();
-                   if (subtitle) name += ' – ' + subtitle;
-                   var dist = utilEditDistance(value, name.substring(0, value.length));
-                   var matchesPreset = pKey === sKey && (!pValue || pValue === sValue);
-
-                   if (dist < 1 || matchesPreset && dist < 3) {
-                     var obj = {
-                       value: s.name(),
-                       title: name,
-                       display: s.nameLabel() + (subtitle ? ' – ' + s.subtitleLabel() : ''),
-                       suggestion: s,
-                       dist: dist + (matchesPreset ? 0 : 1) // penalize if not matched preset
-
-                     };
-                     results.push(obj);
-                   }
-                 }
-
-                 results.sort(function (a, b) {
-                   return a.dist - b.dist;
-                 });
+             toolbarItems.exit().each(function (d) {
+               if (d.uninstall) {
+                 d.uninstall();
                }
-
-               results = results.slice(0, 10);
-               callback(results);
-             };
-           }
-
-           function addNew(d3_event) {
-             d3_event.preventDefault();
-             if (field.locked()) return;
-             var defaultLang = _mainLocalizer.languageCode().toLowerCase();
-
-             var langExists = _multilingual.find(function (datum) {
-               return datum.lang === defaultLang;
+             }).remove();
+             var itemsEnter = toolbarItems.enter().append('div').attr('class', function (d) {
+               var classes = 'toolbar-item ' + (d.id || d).replace('_', '-');
+               if (d.klass) classes += ' ' + d.klass;
+               return classes;
+             });
+             var actionableItems = itemsEnter.filter(function (d) {
+               return d !== 'spacer';
+             });
+             actionableItems.append('div').attr('class', 'item-content').each(function (d) {
+               select(this).call(d.render, bar);
+             });
+             actionableItems.append('div').attr('class', 'item-label').html(function (d) {
+               return d.label;
              });
-
-             var isLangEn = defaultLang.indexOf('en') > -1;
-
-             if (isLangEn || langExists) {
-               defaultLang = '';
-               langExists = _multilingual.find(function (datum) {
-                 return datum.lang === defaultLang;
-               });
-             }
-
-             if (!langExists) {
-               // prepend the value so it appears at the top
-               _multilingual.unshift({
-                 lang: defaultLang,
-                 value: ''
-               });
-
-               localizedInputs.call(renderMultilingual);
-             }
            }
+         }
 
-           function change(onInput) {
-             return function (d3_event) {
-               if (field.locked()) {
-                 d3_event.preventDefault();
-                 return;
-               }
+         return topToolbar;
+       }
 
-               var val = utilGetSetValue(select(this));
-               if (!onInput) val = context.cleanTagValue(val); // don't override multiple values with blank string
+       var sawVersion = null;
+       var isNewVersion = false;
+       var isNewUser = false;
+       function uiVersion(context) {
+         var currVersion = context.version;
+         var matchedVersion = currVersion.match(/\d+\.\d+\.\d+.*/);
 
-               if (!val && Array.isArray(_tags[field.key])) return;
-               var t = {};
-               t[field.key] = val || undefined;
-               dispatch$1.call('change', this, t, onInput);
-             };
+         if (sawVersion === null && matchedVersion !== null) {
+           if (corePreferences('sawVersion')) {
+             isNewUser = false;
+             isNewVersion = corePreferences('sawVersion') !== currVersion && currVersion.indexOf('-') === -1;
+           } else {
+             isNewUser = true;
+             isNewVersion = true;
            }
-         }
 
-         function key(lang) {
-           return field.key + ':' + lang;
+           corePreferences('sawVersion', currVersion);
+           sawVersion = currVersion;
          }
 
-         function changeLang(d3_event, d) {
-           var tags = {}; // make sure unrecognized suffixes are lowercase - #7156
-
-           var lang = utilGetSetValue(select(this)).toLowerCase();
-
-           var language = _languagesArray.find(function (d) {
-             return d.label.toLowerCase() === lang || d.localName && d.localName.toLowerCase() === lang || d.nativeName && d.nativeName.toLowerCase() === lang;
-           });
-
-           if (language) lang = language.code;
+         return function (selection) {
+           selection.append('a').attr('target', '_blank').attr('href', 'https://github.com/openstreetmap/iD').html(currVersion); // only show new version indicator to users that have used iD before
 
-           if (d.lang && d.lang !== lang) {
-             tags[key(d.lang)] = undefined;
+           if (isNewVersion && !isNewUser) {
+             selection.append('a').attr('class', 'badge').attr('target', '_blank').attr('href', 'https://github.com/openstreetmap/iD/blob/release/CHANGELOG.md#whats-new').call(svgIcon('#maki-gift-11')).call(uiTooltip().title(_t.html('version.whats_new', {
+               version: currVersion
+             })).placement('top').scrollContainer(context.container().select('.main-footer-wrap')));
            }
+         };
+       }
 
-           var newKey = lang && context.cleanTagKey(key(lang));
-           var value = utilGetSetValue(select(this.parentNode).selectAll('.localized-value'));
-
-           if (newKey && value) {
-             tags[newKey] = value;
-           } else if (newKey && _wikiTitles && _wikiTitles[d.lang]) {
-             tags[newKey] = _wikiTitles[d.lang];
-           }
+       function uiZoom(context) {
+         var zooms = [{
+           id: 'zoom-in',
+           icon: 'iD-icon-plus',
+           title: _t.html('zoom.in'),
+           action: zoomIn,
+           disabled: function disabled() {
+             return !context.map().canZoomIn();
+           },
+           disabledTitle: _t.html('zoom.disabled.in'),
+           key: '+'
+         }, {
+           id: 'zoom-out',
+           icon: 'iD-icon-minus',
+           title: _t.html('zoom.out'),
+           action: zoomOut,
+           disabled: function disabled() {
+             return !context.map().canZoomOut();
+           },
+           disabledTitle: _t.html('zoom.disabled.out'),
+           key: '-'
+         }];
 
-           d.lang = lang;
-           dispatch$1.call('change', this, tags);
+         function zoomIn(d3_event) {
+           if (d3_event.shiftKey) return;
+           d3_event.preventDefault();
+           context.map().zoomIn();
          }
 
-         function changeValue(d3_event, d) {
-           if (!d.lang) return;
-           var value = context.cleanTagValue(utilGetSetValue(select(this))) || undefined; // don't override multiple values with blank string
-
-           if (!value && Array.isArray(d.value)) return;
-           var t = {};
-           t[key(d.lang)] = value;
-           d.value = value;
-           dispatch$1.call('change', this, t);
+         function zoomOut(d3_event) {
+           if (d3_event.shiftKey) return;
+           d3_event.preventDefault();
+           context.map().zoomOut();
          }
 
-         function fetchLanguages(value, cb) {
-           var v = value.toLowerCase(); // show the user's language first
-
-           var langCodes = [_mainLocalizer.localeCode(), _mainLocalizer.languageCode()];
-
-           if (_countryCode && _territoryLanguages[_countryCode]) {
-             langCodes = langCodes.concat(_territoryLanguages[_countryCode]);
-           }
-
-           var langItems = [];
-           langCodes.forEach(function (code) {
-             var langItem = _languagesArray.find(function (item) {
-               return item.code === code;
-             });
-
-             if (langItem) langItems.push(langItem);
-           });
-           langItems = utilArrayUniq(langItems.concat(_languagesArray));
-           cb(langItems.filter(function (d) {
-             return d.label.toLowerCase().indexOf(v) >= 0 || d.localName && d.localName.toLowerCase().indexOf(v) >= 0 || d.nativeName && d.nativeName.toLowerCase().indexOf(v) >= 0 || d.code.toLowerCase().indexOf(v) >= 0;
-           }).map(function (d) {
-             return {
-               value: d.label
-             };
-           }));
+         function zoomInFurther(d3_event) {
+           if (d3_event.shiftKey) return;
+           d3_event.preventDefault();
+           context.map().zoomInFurther();
          }
 
-         function renderMultilingual(selection) {
-           var entries = selection.selectAll('div.entry').data(_multilingual, function (d) {
-             return d.lang;
-           });
-           entries.exit().style('top', '0').style('max-height', '240px').transition().duration(200).style('opacity', '0').style('max-height', '0px').remove();
-           var entriesEnter = entries.enter().append('div').attr('class', 'entry').each(function (_, index) {
-             var wrap = select(this);
-             var domId = utilUniqueDomId(index);
-             var label = wrap.append('label').attr('class', 'field-label').attr('for', domId);
-             var text = label.append('span').attr('class', 'label-text');
-             text.append('span').attr('class', 'label-textvalue').html(_t.html('translate.localized_translation_label'));
-             text.append('span').attr('class', 'label-textannotation');
-             label.append('button').attr('class', 'remove-icon-multilingual').on('click', function (d3_event, d) {
-               if (field.locked()) return;
-               d3_event.preventDefault();
+         function zoomOutFurther(d3_event) {
+           if (d3_event.shiftKey) return;
+           d3_event.preventDefault();
+           context.map().zoomOutFurther();
+         }
 
-               if (!d.lang || !d.value) {
-                 _multilingual.splice(index, 1);
+         return function (selection) {
+           var tooltipBehavior = uiTooltip().placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left').title(function (d) {
+             if (d.disabled()) {
+               return d.disabledTitle;
+             }
 
-                 renderMultilingual(selection);
-               } else {
-                 // remove from entity tags
-                 var t = {};
-                 t[key(d.lang)] = undefined;
-                 dispatch$1.call('change', this, t);
-               }
-             }).call(svgIcon('#iD-operation-delete'));
-             wrap.append('input').attr('class', 'localized-lang').attr('id', domId).attr('type', 'text').attr('placeholder', _t('translate.localized_translation_language')).on('blur', changeLang).on('change', changeLang).call(langCombo);
-             wrap.append('input').attr('type', 'text').attr('class', 'localized-value').on('blur', changeValue).on('change', changeValue);
-           });
-           entriesEnter.style('margin-top', '0px').style('max-height', '0px').style('opacity', '0').transition().duration(200).style('margin-top', '10px').style('max-height', '240px').style('opacity', '1').on('end', function () {
-             select(this).style('max-height', '').style('overflow', 'visible');
-           });
-           entries = entries.merge(entriesEnter);
-           entries.order();
-           entries.classed('present', function (d) {
-             return d.lang && d.value;
+             return d.title;
+           }).keys(function (d) {
+             return [d.key];
            });
-           utilGetSetValue(entries.select('.localized-lang'), function (d) {
-             var langItem = _languagesArray.find(function (item) {
-               return item.code === d.lang;
-             });
+           var lastPointerUpType;
+           var buttons = selection.selectAll('button').data(zooms).enter().append('button').attr('class', function (d) {
+             return d.id;
+           }).on('pointerup.editor', function (d3_event) {
+             lastPointerUpType = d3_event.pointerType;
+           }).on('click.editor', function (d3_event, d) {
+             if (!d.disabled()) {
+               d.action(d3_event);
+             } else if (lastPointerUpType === 'touch' || lastPointerUpType === 'pen') {
+               context.ui().flash.duration(2000).iconName('#' + d.icon).iconClass('disabled').label(d.disabledTitle)();
+             }
 
-             if (langItem) return langItem.label;
-             return d.lang;
+             lastPointerUpType = null;
+           }).call(tooltipBehavior);
+           buttons.each(function (d) {
+             select(this).call(svgIcon('#' + d.icon, 'light'));
            });
-           utilGetSetValue(entries.select('.localized-value'), function (d) {
-             return typeof d.value === 'string' ? d.value : '';
-           }).attr('title', function (d) {
-             return Array.isArray(d.value) ? d.value.filter(Boolean).join('\n') : null;
-           }).attr('placeholder', function (d) {
-             return Array.isArray(d.value) ? _t('inspector.multiple_values') : _t('translate.localized_translation_name');
-           }).classed('mixed', function (d) {
-             return Array.isArray(d.value);
+           utilKeybinding.plusKeys.forEach(function (key) {
+             context.keybinding().on([key], zoomIn);
+             context.keybinding().on([uiCmd('⌥' + key)], zoomInFurther);
+           });
+           utilKeybinding.minusKeys.forEach(function (key) {
+             context.keybinding().on([key], zoomOut);
+             context.keybinding().on([uiCmd('⌥' + key)], zoomOutFurther);
            });
-         }
-
-         localized.tags = function (tags) {
-           _tags = tags; // Fetch translations from wikipedia
 
-           if (typeof tags.wikipedia === 'string' && !_wikiTitles) {
-             _wikiTitles = {};
-             var wm = tags.wikipedia.match(/([^:]+):(.+)/);
+           function updateButtonStates() {
+             buttons.classed('disabled', function (d) {
+               return d.disabled();
+             }).each(function () {
+               var selection = select(this);
 
-             if (wm && wm[0] && wm[1]) {
-               wikipedia.translations(wm[1], wm[2], function (err, d) {
-                 if (err || !d) return;
-                 _wikiTitles = d;
-               });
-             }
+               if (!selection.select('.tooltip.in').empty()) {
+                 selection.call(tooltipBehavior.updateContent);
+               }
+             });
            }
 
-           var isMixed = Array.isArray(tags[field.key]);
-           utilGetSetValue(input, typeof tags[field.key] === 'string' ? tags[field.key] : '').attr('title', isMixed ? tags[field.key].filter(Boolean).join('\n') : undefined).attr('placeholder', isMixed ? _t('inspector.multiple_values') : field.placeholder()).classed('mixed', isMixed);
-           calcMultilingual(tags);
-
-           _selection.call(localized);
-         };
-
-         localized.focus = function () {
-           input.node().focus();
-         };
-
-         localized.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs;
-           _entityIDs = val;
-           _multilingual = [];
-           loadCountryCode();
-           return localized;
+           updateButtonStates();
+           context.map().on('move.uiZoom', updateButtonStates);
          };
-
-         function loadCountryCode() {
-           var extent = combinedEntityExtent();
-           var countryCode = extent && iso1A2Code(extent.center());
-           _countryCode = countryCode && countryCode.toLowerCase();
-         }
-
-         function combinedEntityExtent() {
-           return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
-         }
-
-         return utilRebind(localized, dispatch$1, 'on');
        }
 
-       function uiFieldMaxspeed(field, context) {
-         var dispatch$1 = dispatch('change');
-         var unitInput = select(null);
-         var input = select(null);
-         var _entityIDs = [];
-
-         var _tags;
-
-         var _isImperial;
-
-         var speedCombo = uiCombobox(context, 'maxspeed');
-         var unitCombo = uiCombobox(context, 'maxspeed-unit').data(['km/h', 'mph'].map(comboValues));
-         var metricValues = [20, 30, 40, 50, 60, 70, 80, 90, 100, 110, 120];
-         var imperialValues = [5, 10, 15, 20, 25, 30, 35, 40, 45, 50, 55, 60, 65, 70, 75, 80];
-
-         function maxspeed(selection) {
-           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
-           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
-           input = wrap.selectAll('input.maxspeed-number').data([0]);
-           input = input.enter().append('input').attr('type', 'text').attr('class', 'maxspeed-number').attr('id', field.domId).call(utilNoAuto).call(speedCombo).merge(input);
-           input.on('change', change).on('blur', change);
-           var loc = combinedEntityExtent().center();
-           _isImperial = roadSpeedUnit(loc) === 'mph';
-           unitInput = wrap.selectAll('input.maxspeed-unit').data([0]);
-           unitInput = unitInput.enter().append('input').attr('type', 'text').attr('class', 'maxspeed-unit').call(unitCombo).merge(unitInput);
-           unitInput.on('blur', changeUnits).on('change', changeUnits);
-
-           function changeUnits() {
-             _isImperial = utilGetSetValue(unitInput) === 'mph';
-             utilGetSetValue(unitInput, _isImperial ? 'mph' : 'km/h');
-             setUnitSuggestions();
-             change();
-           }
+       function uiZoomToSelection(context) {
+         function isDisabled() {
+           var mode = context.mode();
+           return !mode || !mode.zoomToSelected;
          }
 
-         function setUnitSuggestions() {
-           speedCombo.data((_isImperial ? imperialValues : metricValues).map(comboValues));
-           utilGetSetValue(unitInput, _isImperial ? 'mph' : 'km/h');
-         }
+         var _lastPointerUpType;
 
-         function comboValues(d) {
-           return {
-             value: d.toString(),
-             title: d.toString()
-           };
+         function pointerup(d3_event) {
+           _lastPointerUpType = d3_event.pointerType;
          }
 
-         function change() {
-           var tag = {};
-           var value = utilGetSetValue(input).trim(); // don't override multiple values with blank string
-
-           if (!value && Array.isArray(_tags[field.key])) return;
+         function click(d3_event) {
+           d3_event.preventDefault();
 
-           if (!value) {
-             tag[field.key] = undefined;
-           } else if (isNaN(value) || !_isImperial) {
-             tag[field.key] = context.cleanTagValue(value);
+           if (isDisabled()) {
+             if (_lastPointerUpType === 'touch' || _lastPointerUpType === 'pen') {
+               context.ui().flash.duration(2000).iconName('#iD-icon-framed-dot').iconClass('disabled').label(_t.html('inspector.zoom_to.no_selection'))();
+             }
            } else {
-             tag[field.key] = context.cleanTagValue(value + ' mph');
+             var mode = context.mode();
+
+             if (mode && mode.zoomToSelected) {
+               mode.zoomToSelected();
+             }
            }
 
-           dispatch$1.call('change', this, tag);
+           _lastPointerUpType = null;
          }
 
-         maxspeed.tags = function (tags) {
-           _tags = tags;
-           var value = tags[field.key];
-           var isMixed = Array.isArray(value);
+         return function (selection) {
+           var tooltipBehavior = uiTooltip().placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left').title(function () {
+             if (isDisabled()) {
+               return _t.html('inspector.zoom_to.no_selection');
+             }
 
-           if (!isMixed) {
-             if (value && value.indexOf('mph') >= 0) {
-               value = parseInt(value, 10).toString();
-               _isImperial = true;
-             } else if (value) {
-               _isImperial = false;
+             return _t.html('inspector.zoom_to.title');
+           }).keys([_t('inspector.zoom_to.key')]);
+           var button = selection.append('button').on('pointerup', pointerup).on('click', click).call(svgIcon('#iD-icon-framed-dot', 'light')).call(tooltipBehavior);
+
+           function setEnabledState() {
+             button.classed('disabled', isDisabled());
+
+             if (!button.select('.tooltip.in').empty()) {
+               button.call(tooltipBehavior.updateContent);
              }
            }
 
-           setUnitSuggestions();
-           utilGetSetValue(input, typeof value === 'string' ? value : '').attr('title', isMixed ? value.filter(Boolean).join('\n') : null).attr('placeholder', isMixed ? _t('inspector.multiple_values') : field.placeholder()).classed('mixed', isMixed);
+           context.on('enter.uiZoomToSelection', setEnabledState);
+           setEnabledState();
          };
+       }
 
-         maxspeed.focus = function () {
-           input.node().focus();
-         };
+       function uiPane(id, context) {
+         var _key;
 
-         maxspeed.entityIDs = function (val) {
-           _entityIDs = val;
-         };
+         var _label = '';
+         var _description = '';
+         var _iconName = '';
 
-         function combinedEntityExtent() {
-           return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
-         }
+         var _sections; // array of uiSection objects
 
-         return utilRebind(maxspeed, dispatch$1, 'on');
-       }
 
-       function uiFieldRadio(field, context) {
-         var dispatch$1 = dispatch('change');
-         var placeholder = select(null);
-         var wrap = select(null);
-         var labels = select(null);
-         var radios = select(null);
-         var radioData = (field.options || field.strings && field.strings.options && Object.keys(field.strings.options) || field.keys).slice(); // shallow copy
+         var _paneSelection = select(null);
 
-         var typeField;
-         var layerField;
-         var _oldType = {};
-         var _entityIDs = [];
+         var _paneTooltip;
 
-         function selectedKey() {
-           var node = wrap.selectAll('.form-field-input-radio label.active input');
-           return !node.empty() && node.datum();
-         }
+         var pane = {
+           id: id
+         };
 
-         function radio(selection) {
-           selection.classed('preset-radio', true);
-           wrap = selection.selectAll('.form-field-input-wrap').data([0]);
-           var enter = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-radio');
-           enter.append('span').attr('class', 'placeholder');
-           wrap = wrap.merge(enter);
-           placeholder = wrap.selectAll('.placeholder');
-           labels = wrap.selectAll('label').data(radioData);
-           enter = labels.enter().append('label');
-           enter.append('input').attr('type', 'radio').attr('name', field.id).attr('value', function (d) {
-             return field.t('options.' + d, {
-               'default': d
-             });
-           }).attr('checked', false);
-           enter.append('span').html(function (d) {
-             return field.t.html('options.' + d, {
-               'default': d
-             });
-           });
-           labels = labels.merge(enter);
-           radios = labels.selectAll('input').on('change', changeRadio);
-         }
+         pane.label = function (val) {
+           if (!arguments.length) return _label;
+           _label = val;
+           return pane;
+         };
 
-         function structureExtras(selection, tags) {
-           var selected = selectedKey() || tags.layer !== undefined;
-           var type = _mainPresetIndex.field(selected);
-           var layer = _mainPresetIndex.field('layer');
-           var showLayer = selected === 'bridge' || selected === 'tunnel' || tags.layer !== undefined;
-           var extrasWrap = selection.selectAll('.structure-extras-wrap').data(selected ? [0] : []);
-           extrasWrap.exit().remove();
-           extrasWrap = extrasWrap.enter().append('div').attr('class', 'structure-extras-wrap').merge(extrasWrap);
-           var list = extrasWrap.selectAll('ul').data([0]);
-           list = list.enter().append('ul').attr('class', 'rows').merge(list); // Type
+         pane.key = function (val) {
+           if (!arguments.length) return _key;
+           _key = val;
+           return pane;
+         };
 
-           if (type) {
-             if (!typeField || typeField.id !== selected) {
-               typeField = uiField(context, type, _entityIDs, {
-                 wrap: false
-               }).on('change', changeType);
-             }
+         pane.description = function (val) {
+           if (!arguments.length) return _description;
+           _description = val;
+           return pane;
+         };
 
-             typeField.tags(tags);
-           } else {
-             typeField = null;
-           }
+         pane.iconName = function (val) {
+           if (!arguments.length) return _iconName;
+           _iconName = val;
+           return pane;
+         };
 
-           var typeItem = list.selectAll('.structure-type-item').data(typeField ? [typeField] : [], function (d) {
-             return d.id;
-           }); // Exit
+         pane.sections = function (val) {
+           if (!arguments.length) return _sections;
+           _sections = val;
+           return pane;
+         };
 
-           typeItem.exit().remove(); // Enter
+         pane.selection = function () {
+           return _paneSelection;
+         };
 
-           var typeEnter = typeItem.enter().insert('li', ':first-child').attr('class', 'labeled-input structure-type-item');
-           typeEnter.append('span').attr('class', 'label structure-label-type').attr('for', 'preset-input-' + selected).html(_t.html('inspector.radio.structure.type'));
-           typeEnter.append('div').attr('class', 'structure-input-type-wrap'); // Update
+         function hidePane() {
+           context.ui().togglePanes();
+         }
 
-           typeItem = typeItem.merge(typeEnter);
+         pane.togglePane = function (d3_event) {
+           if (d3_event) d3_event.preventDefault();
 
-           if (typeField) {
-             typeItem.selectAll('.structure-input-type-wrap').call(typeField.render);
-           } // Layer
+           _paneTooltip.hide();
+
+           context.ui().togglePanes(!_paneSelection.classed('shown') ? _paneSelection : undefined);
+         };
 
+         pane.renderToggleButton = function (selection) {
+           if (!_paneTooltip) {
+             _paneTooltip = uiTooltip().placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left').title(_description).keys([_key]);
+           }
 
-           if (layer && showLayer) {
-             if (!layerField) {
-               layerField = uiField(context, layer, _entityIDs, {
-                 wrap: false
-               }).on('change', changeLayer);
-             }
+           selection.append('button').on('click', pane.togglePane).call(svgIcon('#' + _iconName, 'light')).call(_paneTooltip);
+         };
 
-             layerField.tags(tags);
-             field.keys = utilArrayUnion(field.keys, ['layer']);
-           } else {
-             layerField = null;
-             field.keys = field.keys.filter(function (k) {
-               return k !== 'layer';
+         pane.renderContent = function (selection) {
+           // override to fully customize content
+           if (_sections) {
+             _sections.forEach(function (section) {
+               selection.call(section.render);
              });
            }
+         };
 
-           var layerItem = list.selectAll('.structure-layer-item').data(layerField ? [layerField] : []); // Exit
+         pane.renderPane = function (selection) {
+           _paneSelection = selection.append('div').attr('class', 'fillL map-pane hide ' + id + '-pane').attr('pane', id);
 
-           layerItem.exit().remove(); // Enter
+           var heading = _paneSelection.append('div').attr('class', 'pane-heading');
 
-           var layerEnter = layerItem.enter().append('li').attr('class', 'labeled-input structure-layer-item');
-           layerEnter.append('span').attr('class', 'label structure-label-layer').attr('for', 'preset-input-layer').html(_t.html('inspector.radio.structure.layer'));
-           layerEnter.append('div').attr('class', 'structure-input-layer-wrap'); // Update
+           heading.append('h2').html(_label);
+           heading.append('button').on('click', hidePane).call(svgIcon('#iD-icon-close'));
 
-           layerItem = layerItem.merge(layerEnter);
+           _paneSelection.append('div').attr('class', 'pane-content').call(pane.renderContent);
 
-           if (layerField) {
-             layerItem.selectAll('.structure-input-layer-wrap').call(layerField.render);
+           if (_key) {
+             context.keybinding().on(_key, pane.togglePane);
            }
-         }
+         };
 
-         function changeType(t, onInput) {
-           var key = selectedKey();
-           if (!key) return;
-           var val = t[key];
+         return pane;
+       }
 
-           if (val !== 'no') {
-             _oldType[key] = val;
-           }
+       function uiSectionBackgroundDisplayOptions(context) {
+         var section = uiSection('background-display-options', context).label(_t.html('background.display_options')).disclosureContent(renderDisclosureContent);
 
-           if (field.type === 'structureRadio') {
-             // remove layer if it should not be set
-             if (val === 'no' || key !== 'bridge' && key !== 'tunnel' || key === 'tunnel' && val === 'building_passage') {
-               t.layer = undefined;
-             } // add layer if it should be set
+         var _detected = utilDetect();
 
+         var _storedOpacity = corePreferences('background-opacity');
 
-             if (t.layer === undefined) {
-               if (key === 'bridge' && val !== 'no') {
-                 t.layer = '1';
-               }
+         var _minVal = 0;
 
-               if (key === 'tunnel' && val !== 'no' && val !== 'building_passage') {
-                 t.layer = '-1';
-               }
-             }
-           }
+         var _maxVal = _detected.cssfilters ? 3 : 1;
 
-           dispatch$1.call('change', this, t, onInput);
-         }
+         var _sliders = _detected.cssfilters ? ['brightness', 'contrast', 'saturation', 'sharpness'] : ['brightness'];
 
-         function changeLayer(t, onInput) {
-           if (t.layer === '0') {
-             t.layer = undefined;
-           }
+         var _options = {
+           brightness: _storedOpacity !== null ? +_storedOpacity : 1,
+           contrast: 1,
+           saturation: 1,
+           sharpness: 1
+         };
 
-           dispatch$1.call('change', this, t, onInput);
+         function clamp(x, min, max) {
+           return Math.max(min, Math.min(x, max));
          }
 
-         function changeRadio() {
-           var t = {};
-           var activeKey;
-
-           if (field.key) {
-             t[field.key] = undefined;
-           }
-
-           radios.each(function (d) {
-             var active = select(this).property('checked');
-             if (active) activeKey = d;
-
-             if (field.key) {
-               if (active) t[field.key] = d;
-             } else {
-               var val = _oldType[activeKey] || 'yes';
-               t[d] = active ? val : undefined;
-             }
-           });
+         function updateValue(d, val) {
+           val = clamp(val, _minVal, _maxVal);
+           _options[d] = val;
+           context.background()[d](val);
 
-           if (field.type === 'structureRadio') {
-             if (activeKey === 'bridge') {
-               t.layer = '1';
-             } else if (activeKey === 'tunnel' && t.tunnel !== 'building_passage') {
-               t.layer = '-1';
-             } else {
-               t.layer = undefined;
-             }
+           if (d === 'brightness') {
+             corePreferences('background-opacity', val);
            }
 
-           dispatch$1.call('change', this, t);
+           section.reRender();
          }
 
-         radio.tags = function (tags) {
-           radios.property('checked', function (d) {
-             if (field.key) {
-               return tags[field.key] === d;
-             }
+         function renderDisclosureContent(selection) {
+           var container = selection.selectAll('.display-options-container').data([0]);
+           var containerEnter = container.enter().append('div').attr('class', 'display-options-container controls-list'); // add slider controls
 
-             return !!(typeof tags[d] === 'string' && tags[d].toLowerCase() !== 'no');
+           var slidersEnter = containerEnter.selectAll('.display-control').data(_sliders).enter().append('div').attr('class', function (d) {
+             return 'display-control display-control-' + d;
+           });
+           slidersEnter.append('h5').html(function (d) {
+             return _t.html('background.' + d);
+           }).append('span').attr('class', function (d) {
+             return 'display-option-value display-option-value-' + d;
            });
+           var sildersControlEnter = slidersEnter.append('div').attr('class', 'control-wrap');
+           sildersControlEnter.append('input').attr('class', function (d) {
+             return 'display-option-input display-option-input-' + d;
+           }).attr('type', 'range').attr('min', _minVal).attr('max', _maxVal).attr('step', '0.05').on('input', function (d3_event, d) {
+             var val = select(this).property('value');
 
-           function isMixed(d) {
-             if (field.key) {
-               return Array.isArray(tags[field.key]) && tags[field.key].includes(d);
+             if (!val && d3_event && d3_event.target) {
+               val = d3_event.target.value;
              }
 
-             return Array.isArray(tags[d]);
-           }
+             updateValue(d, val);
+           });
+           sildersControlEnter.append('button').attr('title', _t('background.reset')).attr('class', function (d) {
+             return 'display-option-reset display-option-reset-' + d;
+           }).on('click', function (d3_event, d) {
+             if (d3_event.button !== 0) return;
+             updateValue(d, 1);
+           }).call(svgIcon('#iD-icon-' + (_mainLocalizer.textDirection() === 'rtl' ? 'redo' : 'undo'))); // reset all button
 
-           labels.classed('active', function (d) {
-             if (field.key) {
-               return Array.isArray(tags[field.key]) && tags[field.key].includes(d) || tags[field.key] === d;
+           containerEnter.append('a').attr('class', 'display-option-resetlink').attr('href', '#').html(_t.html('background.reset_all')).on('click', function (d3_event) {
+             d3_event.preventDefault();
+
+             for (var i = 0; i < _sliders.length; i++) {
+               updateValue(_sliders[i], 1);
              }
+           }); // update
 
-             return Array.isArray(tags[d]) || !!(tags[d] && tags[d].toLowerCase() !== 'no');
-           }).classed('mixed', isMixed).attr('title', function (d) {
-             return isMixed(d) ? _t('inspector.unshared_value_tooltip') : null;
+           container = containerEnter.merge(container);
+           container.selectAll('.display-option-input').property('value', function (d) {
+             return _options[d];
            });
-           var selection = radios.filter(function () {
-             return this.checked;
+           container.selectAll('.display-option-value').html(function (d) {
+             return Math.floor(_options[d] * 100) + '%';
            });
+           container.selectAll('.display-option-reset').classed('disabled', function (d) {
+             return _options[d] === 1;
+           }); // first time only, set brightness if needed
 
-           if (selection.empty()) {
-             placeholder.html(_t.html('inspector.none'));
-           } else {
-             placeholder.html(selection.attr('value'));
-             _oldType[selection.datum()] = tags[selection.datum()];
-           }
-
-           if (field.type === 'structureRadio') {
-             // For waterways without a tunnel tag, set 'culvert' as
-             // the _oldType to default to if the user picks 'tunnel'
-             if (!!tags.waterway && !_oldType.tunnel) {
-               _oldType.tunnel = 'culvert';
-             }
-
-             wrap.call(structureExtras, tags);
+           if (containerEnter.size() && _options.brightness !== 1) {
+             context.background().brightness(_options.brightness);
            }
-         };
-
-         radio.focus = function () {
-           radios.node().focus();
-         };
+         }
 
-         radio.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs;
-           _entityIDs = val;
-           _oldType = {};
-           return radio;
-         };
+         return section;
+       }
 
-         radio.isAllowed = function () {
-           return _entityIDs.length === 1;
-         };
+       function uiSettingsCustomBackground() {
+         var dispatch = dispatch$8('change');
 
-         return utilRebind(radio, dispatch$1, 'on');
-       }
+         function render(selection) {
+           // keep separate copies of original and current settings
+           var _origSettings = {
+             template: corePreferences('background-custom-template')
+           };
+           var _currSettings = {
+             template: corePreferences('background-custom-template')
+           };
+           var example = 'https://{switch:a,b,c}.tile.openstreetmap.org/{zoom}/{x}/{y}.png';
+           var modal = uiConfirm(selection).okButton();
+           modal.classed('settings-modal settings-custom-background', true);
+           modal.select('.modal-section.header').append('h3').html(_t.html('settings.custom_background.header'));
+           var textSection = modal.select('.modal-section.message-text');
+           var instructions = "".concat(_t.html('settings.custom_background.instructions.info'), "\n") + '\n' + "#### ".concat(_t.html('settings.custom_background.instructions.wms.tokens_label'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.wms.tokens.proj'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.wms.tokens.wkid'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.wms.tokens.dimensions'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.wms.tokens.bbox'), "\n") + '\n' + "#### ".concat(_t.html('settings.custom_background.instructions.tms.tokens_label'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.tms.tokens.xyz'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.tms.tokens.flipped_y'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.tms.tokens.switch'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.tms.tokens.quadtile'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.tms.tokens.scale_factor'), "\n") + '\n' + "#### ".concat(_t.html('settings.custom_background.instructions.example'), "\n") + "`".concat(example, "`");
+           textSection.append('div').attr('class', 'instructions-template').html(marked_1(instructions));
+           textSection.append('textarea').attr('class', 'field-template').attr('placeholder', _t('settings.custom_background.template.placeholder')).call(utilNoAuto).property('value', _currSettings.template); // insert a cancel button
 
-       function uiFieldRestrictions(field, context) {
-         var dispatch$1 = dispatch('change');
-         var breathe = behaviorBreathe();
-         corePreferences('turn-restriction-via-way', null); // remove old key
+           var buttonSection = modal.select('.modal-section.buttons');
+           buttonSection.insert('button', '.ok-button').attr('class', 'button cancel-button secondary-action').html(_t.html('confirm.cancel'));
+           buttonSection.select('.cancel-button').on('click.cancel', clickCancel);
+           buttonSection.select('.ok-button').attr('disabled', isSaveDisabled).on('click.save', clickSave);
 
-         var storedViaWay = corePreferences('turn-restriction-via-way0'); // use new key #6922
+           function isSaveDisabled() {
+             return null;
+           } // restore the original template
 
-         var storedDistance = corePreferences('turn-restriction-distance');
 
-         var _maxViaWay = storedViaWay !== null ? +storedViaWay : 0;
+           function clickCancel() {
+             textSection.select('.field-template').property('value', _origSettings.template);
+             corePreferences('background-custom-template', _origSettings.template);
+             this.blur();
+             modal.close();
+           } // accept the current template
 
-         var _maxDistance = storedDistance ? +storedDistance : 30;
 
-         var _initialized = false;
+           function clickSave() {
+             _currSettings.template = textSection.select('.field-template').property('value');
+             corePreferences('background-custom-template', _currSettings.template);
+             this.blur();
+             modal.close();
+             dispatch.call('change', this, _currSettings);
+           }
+         }
 
-         var _parent = select(null); // the entire field
+         return utilRebind(render, dispatch, 'on');
+       }
 
+       function uiSectionBackgroundList(context) {
+         var _backgroundList = select(null);
 
-         var _container = select(null); // just the map
+         var _customSource = context.background().findSource('custom');
 
+         var _settingsCustomBackground = uiSettingsCustomBackground().on('change', customChanged);
 
-         var _oldTurns;
+         var section = uiSection('background-list', context).label(_t.html('background.backgrounds')).disclosureContent(renderDisclosureContent);
 
-         var _graph;
+         function previousBackgroundID() {
+           return corePreferences('background-last-used-toggle');
+         }
 
-         var _vertexID;
+         function renderDisclosureContent(selection) {
+           // the background list
+           var container = selection.selectAll('.layer-background-list').data([0]);
+           _backgroundList = container.enter().append('ul').attr('class', 'layer-list layer-background-list').attr('dir', 'auto').merge(container); // add minimap toggle below list
 
-         var _intersection;
+           var bgExtrasListEnter = selection.selectAll('.bg-extras-list').data([0]).enter().append('ul').attr('class', 'layer-list bg-extras-list');
+           var minimapLabelEnter = bgExtrasListEnter.append('li').attr('class', 'minimap-toggle-item').append('label').call(uiTooltip().title(_t.html('background.minimap.tooltip')).keys([_t('background.minimap.key')]).placement('top'));
+           minimapLabelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event) {
+             d3_event.preventDefault();
+             uiMapInMap.toggle();
+           });
+           minimapLabelEnter.append('span').html(_t.html('background.minimap.description'));
+           var panelLabelEnter = bgExtrasListEnter.append('li').attr('class', 'background-panel-toggle-item').append('label').call(uiTooltip().title(_t.html('background.panel.tooltip')).keys([uiCmd('⌘⇧' + _t('info_panels.background.key'))]).placement('top'));
+           panelLabelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event) {
+             d3_event.preventDefault();
+             context.ui().info.toggle('background');
+           });
+           panelLabelEnter.append('span').html(_t.html('background.panel.description'));
+           var locPanelLabelEnter = bgExtrasListEnter.append('li').attr('class', 'location-panel-toggle-item').append('label').call(uiTooltip().title(_t.html('background.location_panel.tooltip')).keys([uiCmd('⌘⇧' + _t('info_panels.location.key'))]).placement('top'));
+           locPanelLabelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event) {
+             d3_event.preventDefault();
+             context.ui().info.toggle('location');
+           });
+           locPanelLabelEnter.append('span').html(_t.html('background.location_panel.description')); // "Info / Report a Problem" link
 
-         var _fromWayID;
+           selection.selectAll('.imagery-faq').data([0]).enter().append('div').attr('class', 'imagery-faq').append('a').attr('target', '_blank').call(svgIcon('#iD-icon-out-link', 'inline')).attr('href', 'https://github.com/openstreetmap/iD/blob/develop/FAQ.md#how-can-i-report-an-issue-with-background-imagery').append('span').html(_t.html('background.imagery_problem_faq'));
 
-         var _lastXPos;
+           _backgroundList.call(drawListItems, 'radio', function (d3_event, d) {
+             chooseBackground(d);
+           }, function (d) {
+             return !d.isHidden() && !d.overlay;
+           });
+         }
 
-         function restrictions(selection) {
-           _parent = selection; // try to reuse the intersection, but always rebuild it if the graph has changed
+         function setTooltips(selection) {
+           selection.each(function (d, i, nodes) {
+             var item = select(this).select('label');
+             var span = item.select('span');
+             var placement = i < nodes.length / 2 ? 'bottom' : 'top';
+             var description = d.description();
+             var isOverflowing = span.property('clientWidth') !== span.property('scrollWidth');
+             item.call(uiTooltip().destroyAny);
 
-           if (_vertexID && (context.graph() !== _graph || !_intersection)) {
-             _graph = context.graph();
-             _intersection = osmIntersection(_graph, _vertexID, _maxDistance);
-           } // It's possible for there to be no actual intersection here.
-           // for example, a vertex of two `highway=path`
-           // In this case, hide the field.
+             if (d.id === previousBackgroundID()) {
+               item.call(uiTooltip().placement(placement).title('<div>' + _t.html('background.switch') + '</div>').keys([uiCmd('⌘' + _t('background.key'))]));
+             } else if (description || isOverflowing) {
+               item.call(uiTooltip().placement(placement).title(description || d.label()));
+             }
+           });
+         }
 
+         function drawListItems(layerList, type, change, filter) {
+           var sources = context.background().sources(context.map().extent(), context.map().zoom(), true).filter(filter).sort(function (a, b) {
+             return a.best() && !b.best() ? -1 : b.best() && !a.best() ? 1 : d3_descending(a.area(), b.area()) || d3_ascending(a.name(), b.name()) || 0;
+           });
+           var layerLinks = layerList.selectAll('li') // We have to be a bit inefficient about reordering the list since
+           // arrow key navigation of radio values likes to work in the order
+           // they were added, not the display document order.
+           .data(sources, function (d, i) {
+             return d.id + '---' + i;
+           });
+           layerLinks.exit().remove();
+           var enter = layerLinks.enter().append('li').classed('layer-custom', function (d) {
+             return d.id === 'custom';
+           }).classed('best', function (d) {
+             return d.best();
+           });
+           var label = enter.append('label');
+           label.append('input').attr('type', type).attr('name', 'background-layer').attr('value', function (d) {
+             return d.id;
+           }).on('change', change);
+           label.append('span').html(function (d) {
+             return d.label();
+           });
+           enter.filter(function (d) {
+             return d.id === 'custom';
+           }).append('button').attr('class', 'layer-browse').call(uiTooltip().title(_t.html('settings.custom_background.tooltip')).placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left')).on('click', function (d3_event) {
+             d3_event.preventDefault();
+             editCustom();
+           }).call(svgIcon('#iD-icon-more'));
+           enter.filter(function (d) {
+             return d.best();
+           }).append('div').attr('class', 'best').call(uiTooltip().title(_t.html('background.best_imagery')).placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left')).append('span').html('&#9733;');
+           layerList.call(updateLayerSelections);
+         }
 
-           var isOK = _intersection && _intersection.vertices.length && // has vertices
-           _intersection.vertices // has the vertex that the user selected
-           .filter(function (vertex) {
-             return vertex.id === _vertexID;
-           }).length && _intersection.ways.length > 2 && // has more than 2 ways
-           _intersection.ways // has more than 1 TO way
-           .filter(function (way) {
-             return way.__to;
-           }).length > 1; // Also hide in the case where
+         function updateLayerSelections(selection) {
+           function active(d) {
+             return context.background().showsLayer(d);
+           }
 
-           select(selection.node().parentNode).classed('hide', !isOK); // if form field is hidden or has detached from dom, clean up.
+           selection.selectAll('li').classed('active', active).classed('switch', function (d) {
+             return d.id === previousBackgroundID();
+           }).call(setTooltips).selectAll('input').property('checked', active);
+         }
 
-           if (!isOK || !context.container().select('.inspector-wrap.inspector-hidden').empty() || !selection.node().parentNode || !selection.node().parentNode.parentNode) {
-             selection.call(restrictions.off);
-             return;
+         function chooseBackground(d) {
+           if (d.id === 'custom' && !d.template()) {
+             return editCustom();
            }
 
-           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
-           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
-           var container = wrap.selectAll('.restriction-container').data([0]); // enter
+           var previousBackground = context.background().baseLayerSource();
+           corePreferences('background-last-used-toggle', previousBackground.id);
+           corePreferences('background-last-used', d.id);
+           context.background().baseLayerSource(d);
+         }
 
-           var containerEnter = container.enter().append('div').attr('class', 'restriction-container');
-           containerEnter.append('div').attr('class', 'restriction-help'); // update
+         function customChanged(d) {
+           if (d && d.template) {
+             _customSource.template(d.template);
 
-           _container = containerEnter.merge(container).call(renderViewer);
-           var controls = wrap.selectAll('.restriction-controls').data([0]); // enter/update
+             chooseBackground(_customSource);
+           } else {
+             _customSource.template('');
 
-           controls.enter().append('div').attr('class', 'restriction-controls-container').append('div').attr('class', 'restriction-controls').merge(controls).call(renderControls);
+             chooseBackground(context.background().findSource('none'));
+           }
          }
 
-         function renderControls(selection) {
-           var distControl = selection.selectAll('.restriction-distance').data([0]);
-           distControl.exit().remove();
-           var distControlEnter = distControl.enter().append('div').attr('class', 'restriction-control restriction-distance');
-           distControlEnter.append('span').attr('class', 'restriction-control-label restriction-distance-label').html(_t.html('restriction.controls.distance') + ':');
-           distControlEnter.append('input').attr('class', 'restriction-distance-input').attr('type', 'range').attr('min', '20').attr('max', '50').attr('step', '5');
-           distControlEnter.append('span').attr('class', 'restriction-distance-text'); // update
-
-           selection.selectAll('.restriction-distance-input').property('value', _maxDistance).on('input', function () {
-             var val = select(this).property('value');
-             _maxDistance = +val;
-             _intersection = null;
+         function editCustom() {
+           context.container().call(_settingsCustomBackground);
+         }
 
-             _container.selectAll('.layer-osm .layer-turns *').remove();
+         context.background().on('change.background_list', function () {
+           _backgroundList.call(updateLayerSelections);
+         });
+         context.map().on('move.background_list', debounce(function () {
+           // layers in-view may have changed due to map move
+           window.requestIdleCallback(section.reRender);
+         }, 1000));
+         return section;
+       }
 
-             corePreferences('turn-restriction-distance', _maxDistance);
+       function uiSectionBackgroundOffset(context) {
+         var section = uiSection('background-offset', context).label(_t.html('background.fix_misalignment')).disclosureContent(renderDisclosureContent).expandedByDefault(false);
 
-             _parent.call(restrictions);
-           });
-           selection.selectAll('.restriction-distance-text').html(displayMaxDistance(_maxDistance));
-           var viaControl = selection.selectAll('.restriction-via-way').data([0]);
-           viaControl.exit().remove();
-           var viaControlEnter = viaControl.enter().append('div').attr('class', 'restriction-control restriction-via-way');
-           viaControlEnter.append('span').attr('class', 'restriction-control-label restriction-via-way-label').html(_t.html('restriction.controls.via') + ':');
-           viaControlEnter.append('input').attr('class', 'restriction-via-way-input').attr('type', 'range').attr('min', '0').attr('max', '2').attr('step', '1');
-           viaControlEnter.append('span').attr('class', 'restriction-via-way-text'); // update
+         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
 
-           selection.selectAll('.restriction-via-way-input').property('value', _maxViaWay).on('input', function () {
-             var val = select(this).property('value');
-             _maxViaWay = +val;
+         var _directions = [['top', [0, -0.5]], ['left', [-0.5, 0]], ['right', [0.5, 0]], ['bottom', [0, 0.5]]];
 
-             _container.selectAll('.layer-osm .layer-turns *').remove();
+         function updateValue() {
+           var meters = geoOffsetToMeters(context.background().offset());
+           var x = +meters[0].toFixed(2);
+           var y = +meters[1].toFixed(2);
+           context.container().selectAll('.nudge-inner-rect').select('input').classed('error', false).property('value', x + ', ' + y);
+           context.container().selectAll('.nudge-reset').classed('disabled', function () {
+             return x === 0 && y === 0;
+           });
+         }
 
-             corePreferences('turn-restriction-via-way0', _maxViaWay);
+         function resetOffset() {
+           context.background().offset([0, 0]);
+           updateValue();
+         }
 
-             _parent.call(restrictions);
-           });
-           selection.selectAll('.restriction-via-way-text').html(displayMaxVia(_maxViaWay));
+         function nudge(d) {
+           context.background().nudge(d, context.map().zoom());
+           updateValue();
          }
 
-         function renderViewer(selection) {
-           if (!_intersection) return;
-           var vgraph = _intersection.graph;
-           var filter = utilFunctor(true);
-           var projection = geoRawMercator(); // Reflow warning: `utilGetDimensions` calls `getBoundingClientRect`
-           // Instead of asking the restriction-container for its dimensions,
-           //  we can ask the .sidebar, which can have its dimensions cached.
-           // width: calc as sidebar - padding
-           // height: hardcoded (from `80_app.css`)
-           // var d = utilGetDimensions(selection);
+         function inputOffset() {
+           var input = select(this);
+           var d = input.node().value;
+           if (d === '') return resetOffset();
+           d = d.replace(/;/g, ',').split(',').map(function (n) {
+             // if n is NaN, it will always get mapped to false.
+             return !isNaN(n) && n;
+           });
 
-           var sdims = utilGetDimensions(context.container().select('.sidebar'));
-           var d = [sdims[0] - 50, 370];
-           var c = geoVecScale(d, 0.5);
-           var z = 22;
-           projection.scale(geoZoomToScale(z)); // Calculate extent of all key vertices
+           if (d.length !== 2 || !d[0] || !d[1]) {
+             input.classed('error', true);
+             return;
+           }
 
-           var extent = geoExtent();
+           context.background().offset(geoMetersToOffset(d));
+           updateValue();
+         }
 
-           for (var i = 0; i < _intersection.vertices.length; i++) {
-             extent._extend(_intersection.vertices[i].extent());
-           } // If this is a large intersection, adjust zoom to fit extent
+         function dragOffset(d3_event) {
+           if (d3_event.button !== 0) return;
+           var origin = [d3_event.clientX, d3_event.clientY];
+           var pointerId = d3_event.pointerId || 'mouse';
+           context.container().append('div').attr('class', 'nudge-surface');
+           select(window).on(_pointerPrefix + 'move.drag-bg-offset', pointermove).on(_pointerPrefix + 'up.drag-bg-offset', pointerup);
 
+           if (_pointerPrefix === 'pointer') {
+             select(window).on('pointercancel.drag-bg-offset', pointerup);
+           }
 
-           if (_intersection.vertices.length > 1) {
-             var padding = 180; // in z22 pixels
+           function pointermove(d3_event) {
+             if (pointerId !== (d3_event.pointerId || 'mouse')) return;
+             var latest = [d3_event.clientX, d3_event.clientY];
+             var d = [-(origin[0] - latest[0]) / 4, -(origin[1] - latest[1]) / 4];
+             origin = latest;
+             nudge(d);
+           }
 
-             var tl = projection([extent[0][0], extent[1][1]]);
-             var br = projection([extent[1][0], extent[0][1]]);
-             var hFactor = (br[0] - tl[0]) / (d[0] - padding);
-             var vFactor = (br[1] - tl[1]) / (d[1] - padding);
-             var hZoomDiff = Math.log(Math.abs(hFactor)) / Math.LN2;
-             var vZoomDiff = Math.log(Math.abs(vFactor)) / Math.LN2;
-             z = z - Math.max(hZoomDiff, vZoomDiff);
-             projection.scale(geoZoomToScale(z));
+           function pointerup(d3_event) {
+             if (pointerId !== (d3_event.pointerId || 'mouse')) return;
+             if (d3_event.button !== 0) return;
+             context.container().selectAll('.nudge-surface').remove();
+             select(window).on('.drag-bg-offset', null);
            }
+         }
 
-           var padTop = 35; // reserve top space for hint text
+         function renderDisclosureContent(selection) {
+           var container = selection.selectAll('.nudge-container').data([0]);
+           var containerEnter = container.enter().append('div').attr('class', 'nudge-container');
+           containerEnter.append('div').attr('class', 'nudge-instructions').html(_t.html('background.offset'));
+           var nudgeWrapEnter = containerEnter.append('div').attr('class', 'nudge-controls-wrap');
+           var nudgeEnter = nudgeWrapEnter.append('div').attr('class', 'nudge-outer-rect').on(_pointerPrefix + 'down', dragOffset);
+           nudgeEnter.append('div').attr('class', 'nudge-inner-rect').append('input').attr('type', 'text').on('change', inputOffset);
+           nudgeWrapEnter.append('div').selectAll('button').data(_directions).enter().append('button').attr('class', function (d) {
+             return d[0] + ' nudge';
+           }).on('click', function (d3_event, d) {
+             nudge(d[1]);
+           });
+           nudgeWrapEnter.append('button').attr('title', _t('background.reset')).attr('class', 'nudge-reset disabled').on('click', function (d3_event) {
+             d3_event.preventDefault();
+             resetOffset();
+           }).call(svgIcon('#iD-icon-' + (_mainLocalizer.textDirection() === 'rtl' ? 'redo' : 'undo')));
+           updateValue();
+         }
 
-           var extentCenter = projection(extent.center());
-           extentCenter[1] = extentCenter[1] - padTop;
-           projection.translate(geoVecSubtract(c, extentCenter)).clipExtent([[0, 0], d]);
-           var drawLayers = svgLayers(projection, context).only(['osm', 'touch']).dimensions(d);
-           var drawVertices = svgVertices(projection, context);
-           var drawLines = svgLines(projection, context);
-           var drawTurns = svgTurns(projection, context);
-           var firstTime = selection.selectAll('.surface').empty();
-           selection.call(drawLayers);
-           var surface = selection.selectAll('.surface').classed('tr', true);
+         context.background().on('change.backgroundOffset-update', updateValue);
+         return section;
+       }
 
-           if (firstTime) {
-             _initialized = true;
-             surface.call(breathe);
-           } // This can happen if we've lowered the detail while a FROM way
-           // is selected, and that way is no longer part of the intersection.
+       function uiSectionOverlayList(context) {
+         var section = uiSection('overlay-list', context).label(_t.html('background.overlays')).disclosureContent(renderDisclosureContent);
 
+         var _overlayList = select(null);
 
-           if (_fromWayID && !vgraph.hasEntity(_fromWayID)) {
-             _fromWayID = null;
-             _oldTurns = null;
-           }
+         function setTooltips(selection) {
+           selection.each(function (d, i, nodes) {
+             var item = select(this).select('label');
+             var span = item.select('span');
+             var placement = i < nodes.length / 2 ? 'bottom' : 'top';
+             var description = d.description();
+             var isOverflowing = span.property('clientWidth') !== span.property('scrollWidth');
+             item.call(uiTooltip().destroyAny);
 
-           surface.call(utilSetDimensions, d).call(drawVertices, vgraph, _intersection.vertices, filter, extent, z).call(drawLines, vgraph, _intersection.ways, filter).call(drawTurns, vgraph, _intersection.turns(_fromWayID, _maxViaWay));
-           surface.on('click.restrictions', click).on('mouseover.restrictions', mouseover);
-           surface.selectAll('.selected').classed('selected', false);
-           surface.selectAll('.related').classed('related', false);
-           var way;
+             if (description || isOverflowing) {
+               item.call(uiTooltip().placement(placement).title(description || d.name()));
+             }
+           });
+         }
 
-           if (_fromWayID) {
-             way = vgraph.entity(_fromWayID);
-             surface.selectAll('.' + _fromWayID).classed('selected', true).classed('related', true);
+         function updateLayerSelections(selection) {
+           function active(d) {
+             return context.background().showsLayer(d);
            }
 
-           document.addEventListener('resizeWindow', function () {
-             utilSetDimensions(_container, null);
-             redraw(1);
-           }, false);
-           updateHints(null);
-
-           function click(d3_event) {
-             surface.call(breathe.off).call(breathe);
-             var datum = d3_event.target.__data__;
-             var entity = datum && datum.properties && datum.properties.entity;
+           selection.selectAll('li').classed('active', active).call(setTooltips).selectAll('input').property('checked', active);
+         }
 
-             if (entity) {
-               datum = entity;
-             }
+         function chooseOverlay(d3_event, d) {
+           d3_event.preventDefault();
+           context.background().toggleOverlayLayer(d);
 
-             if (datum instanceof osmWay && (datum.__from || datum.__via)) {
-               _fromWayID = datum.id;
-               _oldTurns = null;
-               redraw();
-             } else if (datum instanceof osmTurn) {
-               var actions, extraActions, turns, i;
-               var restrictionType = osmInferRestriction(vgraph, datum, projection);
+           _overlayList.call(updateLayerSelections);
 
-               if (datum.restrictionID && !datum.direct) {
-                 return;
-               } else if (datum.restrictionID && !datum.only) {
-                 // NO -> ONLY
-                 var seen = {};
-                 var datumOnly = JSON.parse(JSON.stringify(datum)); // deep clone the datum
+           document.activeElement.blur();
+         }
 
-                 datumOnly.only = true; // but change this property
+         function drawListItems(layerList, type, change, filter) {
+           var sources = context.background().sources(context.map().extent(), context.map().zoom(), true).filter(filter);
+           var layerLinks = layerList.selectAll('li').data(sources, function (d) {
+             return d.name();
+           });
+           layerLinks.exit().remove();
+           var enter = layerLinks.enter().append('li');
+           var label = enter.append('label');
+           label.append('input').attr('type', type).attr('name', 'layers').on('change', change);
+           label.append('span').html(function (d) {
+             return d.label();
+           });
+           layerList.selectAll('li').sort(sortSources);
+           layerList.call(updateLayerSelections);
 
-                 restrictionType = restrictionType.replace(/^no/, 'only'); // Adding an ONLY restriction should destroy all other direct restrictions from the FROM towards the VIA.
-                 // We will remember them in _oldTurns, and restore them if the user clicks again.
+           function sortSources(a, b) {
+             return a.best() && !b.best() ? -1 : b.best() && !a.best() ? 1 : d3_descending(a.area(), b.area()) || d3_ascending(a.name(), b.name()) || 0;
+           }
+         }
 
-                 turns = _intersection.turns(_fromWayID, 2);
-                 extraActions = [];
-                 _oldTurns = [];
+         function renderDisclosureContent(selection) {
+           var container = selection.selectAll('.layer-overlay-list').data([0]);
+           _overlayList = container.enter().append('ul').attr('class', 'layer-list layer-overlay-list').attr('dir', 'auto').merge(container);
 
-                 for (i = 0; i < turns.length; i++) {
-                   var turn = turns[i];
-                   if (seen[turn.restrictionID]) continue; // avoid deleting the turn twice (#4968, #4928)
+           _overlayList.call(drawListItems, 'checkbox', chooseOverlay, function (d) {
+             return !d.isHidden() && d.overlay;
+           });
+         }
 
-                   if (turn.direct && turn.path[1] === datum.path[1]) {
-                     seen[turns[i].restrictionID] = true;
-                     turn.restrictionType = osmInferRestriction(vgraph, turn, projection);
+         context.map().on('move.overlay_list', debounce(function () {
+           // layers in-view may have changed due to map move
+           window.requestIdleCallback(section.reRender);
+         }, 1000));
+         return section;
+       }
 
-                     _oldTurns.push(turn);
+       function uiPaneBackground(context) {
+         var backgroundPane = uiPane('background', context).key(_t('background.key')).label(_t.html('background.title')).description(_t.html('background.description')).iconName('iD-icon-layers').sections([uiSectionBackgroundList(context), uiSectionOverlayList(context), uiSectionBackgroundDisplayOptions(context), uiSectionBackgroundOffset(context)]);
+         return backgroundPane;
+       }
 
-                     extraActions.push(actionUnrestrictTurn(turn));
-                   }
-                 }
+       function uiPaneHelp(context) {
+         var docKeys = [['help', ['welcome', 'open_data_h', 'open_data', 'before_start_h', 'before_start', 'open_source_h', 'open_source', 'open_source_help']], ['overview', ['navigation_h', 'navigation_drag', 'navigation_zoom', 'features_h', 'features', 'nodes_ways']], ['editing', ['select_h', 'select_left_click', 'select_right_click', 'select_space', 'multiselect_h', 'multiselect', 'multiselect_shift_click', 'multiselect_lasso', 'undo_redo_h', 'undo_redo', 'save_h', 'save', 'save_validation', 'upload_h', 'upload', 'backups_h', 'backups', 'keyboard_h', 'keyboard']], ['feature_editor', ['intro', 'definitions', 'type_h', 'type', 'type_picker', 'fields_h', 'fields_all_fields', 'fields_example', 'fields_add_field', 'tags_h', 'tags_all_tags', 'tags_resources']], ['points', ['intro', 'add_point_h', 'add_point', 'add_point_finish', 'move_point_h', 'move_point', 'delete_point_h', 'delete_point', 'delete_point_command']], ['lines', ['intro', 'add_line_h', 'add_line', 'add_line_draw', 'add_line_continue', 'add_line_finish', 'modify_line_h', 'modify_line_dragnode', 'modify_line_addnode', 'connect_line_h', 'connect_line', 'connect_line_display', 'connect_line_drag', 'connect_line_tag', 'disconnect_line_h', 'disconnect_line_command', 'move_line_h', 'move_line_command', 'move_line_connected', 'delete_line_h', 'delete_line', 'delete_line_command']], ['areas', ['intro', 'point_or_area_h', 'point_or_area', 'add_area_h', 'add_area_command', 'add_area_draw', 'add_area_continue', 'add_area_finish', 'square_area_h', 'square_area_command', 'modify_area_h', 'modify_area_dragnode', 'modify_area_addnode', 'delete_area_h', 'delete_area', 'delete_area_command']], ['relations', ['intro', 'edit_relation_h', 'edit_relation', 'edit_relation_add', 'edit_relation_delete', 'maintain_relation_h', 'maintain_relation', 'relation_types_h', 'multipolygon_h', 'multipolygon', 'multipolygon_create', 'multipolygon_merge', 'turn_restriction_h', 'turn_restriction', 'turn_restriction_field', 'turn_restriction_editing', 'route_h', 'route', 'route_add', 'boundary_h', 'boundary', 'boundary_add']], ['operations', ['intro', 'intro_2', 'straighten', 'orthogonalize', 'circularize', 'move', 'rotate', 'reflect', 'continue', 'reverse', 'disconnect', 'split', 'extract', 'merge', 'delete', 'downgrade', 'copy_paste']], ['notes', ['intro', 'add_note_h', 'add_note', 'place_note', 'move_note', 'update_note_h', 'update_note', 'save_note_h', 'save_note']], ['imagery', ['intro', 'sources_h', 'choosing', 'sources', 'offsets_h', 'offset', 'offset_change']], ['streetlevel', ['intro', 'using_h', 'using', 'photos', 'viewer']], ['gps', ['intro', 'survey', 'using_h', 'using', 'tracing', 'upload']], ['qa', ['intro', 'tools_h', 'tools', 'issues_h', 'issues']]];
+         var headings = {
+           'help.help.open_data_h': 3,
+           'help.help.before_start_h': 3,
+           'help.help.open_source_h': 3,
+           'help.overview.navigation_h': 3,
+           'help.overview.features_h': 3,
+           'help.editing.select_h': 3,
+           'help.editing.multiselect_h': 3,
+           'help.editing.undo_redo_h': 3,
+           'help.editing.save_h': 3,
+           'help.editing.upload_h': 3,
+           'help.editing.backups_h': 3,
+           'help.editing.keyboard_h': 3,
+           'help.feature_editor.type_h': 3,
+           'help.feature_editor.fields_h': 3,
+           'help.feature_editor.tags_h': 3,
+           'help.points.add_point_h': 3,
+           'help.points.move_point_h': 3,
+           'help.points.delete_point_h': 3,
+           'help.lines.add_line_h': 3,
+           'help.lines.modify_line_h': 3,
+           'help.lines.connect_line_h': 3,
+           'help.lines.disconnect_line_h': 3,
+           'help.lines.move_line_h': 3,
+           'help.lines.delete_line_h': 3,
+           'help.areas.point_or_area_h': 3,
+           'help.areas.add_area_h': 3,
+           'help.areas.square_area_h': 3,
+           'help.areas.modify_area_h': 3,
+           'help.areas.delete_area_h': 3,
+           'help.relations.edit_relation_h': 3,
+           'help.relations.maintain_relation_h': 3,
+           'help.relations.relation_types_h': 2,
+           'help.relations.multipolygon_h': 3,
+           'help.relations.turn_restriction_h': 3,
+           'help.relations.route_h': 3,
+           'help.relations.boundary_h': 3,
+           'help.notes.add_note_h': 3,
+           'help.notes.update_note_h': 3,
+           'help.notes.save_note_h': 3,
+           'help.imagery.sources_h': 3,
+           'help.imagery.offsets_h': 3,
+           'help.streetlevel.using_h': 3,
+           'help.gps.using_h': 3,
+           'help.qa.tools_h': 3,
+           'help.qa.issues_h': 3
+         }; // For each section, squash all the texts into a single markdown document
 
-                 actions = _intersection.actions.concat(extraActions, [actionRestrictTurn(datumOnly, restrictionType), _t('operations.restriction.annotation.create')]);
-               } else if (datum.restrictionID) {
-                 // ONLY -> Allowed
-                 // Restore whatever restrictions we might have destroyed by cycling thru the ONLY state.
-                 // This relies on the assumption that the intersection was already split up when we
-                 // performed the previous action (NO -> ONLY), so the IDs in _oldTurns shouldn't have changed.
-                 turns = _oldTurns || [];
-                 extraActions = [];
+         var docs = docKeys.map(function (key) {
+           var helpkey = 'help.' + key[0];
+           var helpPaneReplacements = {
+             version: context.version
+           };
+           var text = key[1].reduce(function (all, part) {
+             var subkey = helpkey + '.' + part;
+             var depth = headings[subkey]; // is this subkey a heading?
 
-                 for (i = 0; i < turns.length; i++) {
-                   if (turns[i].key !== datum.key) {
-                     extraActions.push(actionRestrictTurn(turns[i], turns[i].restrictionType));
-                   }
-                 }
+             var hhh = depth ? Array(depth + 1).join('#') + ' ' : ''; // if so, prepend with some ##'s
 
-                 _oldTurns = null;
-                 actions = _intersection.actions.concat(extraActions, [actionUnrestrictTurn(datum), _t('operations.restriction.annotation.delete')]);
-               } else {
-                 // Allowed -> NO
-                 actions = _intersection.actions.concat([actionRestrictTurn(datum, restrictionType), _t('operations.restriction.annotation.create')]);
-               }
+             return all + hhh + helpHtml(subkey, helpPaneReplacements) + '\n\n';
+           }, '');
+           return {
+             title: _t.html(helpkey + '.title'),
+             content: marked_1(text.trim()) // use keyboard key styling for shortcuts
+             .replace(/<code>/g, '<kbd>').replace(/<\/code>/g, '<\/kbd>')
+           };
+         });
+         var helpPane = uiPane('help', context).key(_t('help.key')).label(_t.html('help.title')).description(_t.html('help.title')).iconName('iD-icon-help');
 
-               context.perform.apply(context, actions); // At this point the datum will be changed, but will have same key..
-               // Refresh it and update the help..
+         helpPane.renderContent = function (content) {
+           function clickHelp(d, i) {
+             var rtl = _mainLocalizer.textDirection() === 'rtl';
+             content.property('scrollTop', 0);
+             helpPane.selection().select('.pane-heading h2').html(d.title);
+             body.html(d.content);
+             body.selectAll('a').attr('target', '_blank');
+             menuItems.classed('selected', function (m) {
+               return m.title === d.title;
+             });
+             nav.html('');
 
-               var s = surface.selectAll('.' + datum.key);
-               datum = s.empty() ? null : s.datum();
-               updateHints(datum);
+             if (rtl) {
+               nav.call(drawNext).call(drawPrevious);
              } else {
-               _fromWayID = null;
-               _oldTurns = null;
-               redraw();
+               nav.call(drawPrevious).call(drawNext);
              }
-           }
-
-           function mouseover(d3_event) {
-             var datum = d3_event.target.__data__;
-             updateHints(datum);
-           }
-
-           _lastXPos = _lastXPos || sdims[0];
-
-           function redraw(minChange) {
-             var xPos = -1;
 
-             if (minChange) {
-               xPos = utilGetDimensions(context.container().select('.sidebar'))[0];
+             function drawNext(selection) {
+               if (i < docs.length - 1) {
+                 var nextLink = selection.append('a').attr('href', '#').attr('class', 'next').on('click', function (d3_event) {
+                   d3_event.preventDefault();
+                   clickHelp(docs[i + 1], i + 1);
+                 });
+                 nextLink.append('span').html(docs[i + 1].title).call(svgIcon(rtl ? '#iD-icon-backward' : '#iD-icon-forward', 'inline'));
+               }
              }
 
-             if (!minChange || minChange && Math.abs(xPos - _lastXPos) >= minChange) {
-               if (context.hasEntity(_vertexID)) {
-                 _lastXPos = xPos;
-
-                 _container.call(renderViewer);
+             function drawPrevious(selection) {
+               if (i > 0) {
+                 var prevLink = selection.append('a').attr('href', '#').attr('class', 'previous').on('click', function (d3_event) {
+                   d3_event.preventDefault();
+                   clickHelp(docs[i - 1], i - 1);
+                 });
+                 prevLink.call(svgIcon(rtl ? '#iD-icon-forward' : '#iD-icon-backward', 'inline')).append('span').html(docs[i - 1].title);
                }
              }
            }
 
-           function highlightPathsFrom(wayID) {
-             surface.selectAll('.related').classed('related', false).classed('allow', false).classed('restrict', false).classed('only', false);
-             surface.selectAll('.' + wayID).classed('related', true);
+           function clickWalkthrough(d3_event) {
+             d3_event.preventDefault();
+             if (context.inIntro()) return;
+             context.container().call(uiIntro(context));
+             context.ui().togglePanes();
+           }
 
-             if (wayID) {
-               var turns = _intersection.turns(wayID, _maxViaWay);
+           function clickShortcuts(d3_event) {
+             d3_event.preventDefault();
+             context.container().call(context.ui().shortcuts, true);
+           }
 
-               for (var i = 0; i < turns.length; i++) {
-                 var turn = turns[i];
-                 var ids = [turn.to.way];
-                 var klass = turn.no ? 'restrict' : turn.only ? 'only' : 'allow';
+           var toc = content.append('ul').attr('class', 'toc');
+           var menuItems = toc.selectAll('li').data(docs).enter().append('li').append('a').attr('href', '#').html(function (d) {
+             return d.title;
+           }).on('click', function (d3_event, d) {
+             d3_event.preventDefault();
+             clickHelp(d, docs.indexOf(d));
+           });
+           var shortcuts = toc.append('li').attr('class', 'shortcuts').call(uiTooltip().title(_t.html('shortcuts.tooltip')).keys(['?']).placement('top')).append('a').attr('href', '#').on('click', clickShortcuts);
+           shortcuts.append('div').html(_t.html('shortcuts.title'));
+           var walkthrough = toc.append('li').attr('class', 'walkthrough').append('a').attr('href', '#').on('click', clickWalkthrough);
+           walkthrough.append('svg').attr('class', 'logo logo-walkthrough').append('use').attr('xlink:href', '#iD-logo-walkthrough');
+           walkthrough.append('div').html(_t.html('splash.walkthrough'));
+           var helpContent = content.append('div').attr('class', 'left-content');
+           var body = helpContent.append('div').attr('class', 'body');
+           var nav = helpContent.append('div').attr('class', 'nav');
+           clickHelp(docs[0], 0);
+         };
 
-                 if (turn.only || turns.length === 1) {
-                   if (turn.via.ways) {
-                     ids = ids.concat(turn.via.ways);
-                   }
-                 } else if (turn.to.way === wayID) {
-                   continue;
-                 }
+         return helpPane;
+       }
+
+       function uiSectionValidationIssues(id, severity, context) {
+         var _issues = [];
+         var section = uiSection(id, context).label(function () {
+           if (!_issues) return '';
+           var issueCountText = _issues.length > 1000 ? '1000+' : String(_issues.length);
+           return _t('inspector.title_count', {
+             title: _t.html('issues.' + severity + 's.list_title'),
+             count: issueCountText
+           });
+         }).disclosureContent(renderDisclosureContent).shouldDisplay(function () {
+           return _issues && _issues.length;
+         });
+
+         function getOptions() {
+           return {
+             what: corePreferences('validate-what') || 'edited',
+             where: corePreferences('validate-where') || 'all'
+           };
+         } // get and cache the issues to display, unordered
 
-                 surface.selectAll(utilEntitySelector(ids)).classed('related', true).classed('allow', klass === 'allow').classed('restrict', klass === 'restrict').classed('only', klass === 'only');
-               }
-             }
-           }
 
-           function updateHints(datum) {
-             var help = _container.selectAll('.restriction-help').html('');
+         function reloadIssues() {
+           _issues = context.validator().getIssuesBySeverity(getOptions())[severity];
+         }
 
-             var placeholders = {};
-             ['from', 'via', 'to'].forEach(function (k) {
-               placeholders[k] = '<span class="qualifier">' + _t('restriction.help.' + k) + '</span>';
+         function renderDisclosureContent(selection) {
+           var center = context.map().center();
+           var graph = context.graph(); // sort issues by distance away from the center of the map
+
+           var issues = _issues.map(function withDistance(issue) {
+             var extent = issue.extent(graph);
+             var dist = extent ? geoSphericalDistance(center, extent.center()) : 0;
+             return Object.assign(issue, {
+               dist: dist
              });
-             var entity = datum && datum.properties && datum.properties.entity;
+           }).sort(function byDistance(a, b) {
+             return a.dist - b.dist;
+           }); // cut off at 1000
 
-             if (entity) {
-               datum = entity;
-             }
 
-             if (_fromWayID) {
-               way = vgraph.entity(_fromWayID);
-               surface.selectAll('.' + _fromWayID).classed('selected', true).classed('related', true);
-             } // Hovering a way
+           issues = issues.slice(0, 1000); //renderIgnoredIssuesReset(_warningsSelection);
 
+           selection.call(drawIssuesList, issues);
+         }
 
-             if (datum instanceof osmWay && datum.__from) {
-               way = datum;
-               highlightPathsFrom(_fromWayID ? null : way.id);
-               surface.selectAll('.' + way.id).classed('related', true);
-               var clickSelect = !_fromWayID || _fromWayID !== way.id;
-               help.append('div') // "Click to select FROM {fromName}." / "FROM {fromName}"
-               .html(_t.html('restriction.help.' + (clickSelect ? 'select_from_name' : 'from_name'), {
-                 from: placeholders.from,
-                 fromName: displayName(way.id, vgraph)
-               })); // Hovering a turn arrow
-             } else if (datum instanceof osmTurn) {
-               var restrictionType = osmInferRestriction(vgraph, datum, projection);
-               var turnType = restrictionType.replace(/^(only|no)\_/, '');
-               var indirect = datum.direct === false ? _t.html('restriction.help.indirect') : '';
-               var klass, turnText, nextText;
+         function drawIssuesList(selection, issues) {
+           var list = selection.selectAll('.issues-list').data([0]);
+           list = list.enter().append('ul').attr('class', 'layer-list issues-list ' + severity + 's-list').merge(list);
+           var items = list.selectAll('li').data(issues, function (d) {
+             return d.id;
+           }); // Exit
 
-               if (datum.no) {
-                 klass = 'restrict';
-                 turnText = _t.html('restriction.help.turn.no_' + turnType, {
-                   indirect: indirect
-                 });
-                 nextText = _t.html('restriction.help.turn.only_' + turnType, {
-                   indirect: ''
-                 });
-               } else if (datum.only) {
-                 klass = 'only';
-                 turnText = _t.html('restriction.help.turn.only_' + turnType, {
-                   indirect: indirect
-                 });
-                 nextText = _t.html('restriction.help.turn.allowed_' + turnType, {
-                   indirect: ''
-                 });
-               } else {
-                 klass = 'allow';
-                 turnText = _t.html('restriction.help.turn.allowed_' + turnType, {
-                   indirect: indirect
-                 });
-                 nextText = _t.html('restriction.help.turn.no_' + turnType, {
-                   indirect: ''
-                 });
-               }
+           items.exit().remove(); // Enter
 
-               help.append('div') // "NO Right Turn (indirect)"
-               .attr('class', 'qualifier ' + klass).html(turnText);
-               help.append('div') // "FROM {fromName} TO {toName}"
-               .html(_t.html('restriction.help.from_name_to_name', {
-                 from: placeholders.from,
-                 fromName: displayName(datum.from.way, vgraph),
-                 to: placeholders.to,
-                 toName: displayName(datum.to.way, vgraph)
-               }));
+           var itemsEnter = items.enter().append('li').attr('class', function (d) {
+             return 'issue severity-' + d.severity;
+           });
+           var labelsEnter = itemsEnter.append('button').attr('class', 'issue-label').on('click', function (d3_event, d) {
+             context.validator().focusIssue(d);
+           }).on('mouseover', function (d3_event, d) {
+             utilHighlightEntities(d.entityIds, true, context);
+           }).on('mouseout', function (d3_event, d) {
+             utilHighlightEntities(d.entityIds, false, context);
+           });
+           var textEnter = labelsEnter.append('span').attr('class', 'issue-text');
+           textEnter.append('span').attr('class', 'issue-icon').each(function (d) {
+             var iconName = '#iD-icon-' + (d.severity === 'warning' ? 'alert' : 'error');
+             select(this).call(svgIcon(iconName));
+           });
+           textEnter.append('span').attr('class', 'issue-message');
+           /*
+           labelsEnter
+               .append('span')
+               .attr('class', 'issue-autofix')
+               .each(function(d) {
+                   if (!d.autoFix) return;
+                    d3_select(this)
+                       .append('button')
+                       .attr('title', t('issues.fix_one.title'))
+                       .datum(d.autoFix)  // set button datum to the autofix
+                       .attr('class', 'autofix action')
+                       .on('click', function(d3_event, d) {
+                           d3_event.preventDefault();
+                           d3_event.stopPropagation();
+                            var issuesEntityIDs = d.issue.entityIds;
+                           utilHighlightEntities(issuesEntityIDs.concat(d.entityIds), false, context);
+                            context.perform.apply(context, d.autoArgs);
+                           context.validator().validate();
+                       })
+                       .call(svgIcon('#iD-icon-wrench'));
+               });
+           */
+           // Update
 
-               if (datum.via.ways && datum.via.ways.length) {
-                 var names = [];
+           items = items.merge(itemsEnter).order();
+           items.selectAll('.issue-message').html(function (d) {
+             return d.message(context);
+           });
+           /*
+           // autofix
+           var canAutoFix = issues.filter(function(issue) { return issue.autoFix; });
+            var autoFixAll = selection.selectAll('.autofix-all')
+               .data(canAutoFix.length ? [0] : []);
+            // exit
+           autoFixAll.exit()
+               .remove();
+            // enter
+           var autoFixAllEnter = autoFixAll.enter()
+               .insert('div', '.issues-list')
+               .attr('class', 'autofix-all');
+            var linkEnter = autoFixAllEnter
+               .append('a')
+               .attr('class', 'autofix-all-link')
+               .attr('href', '#');
+            linkEnter
+               .append('span')
+               .attr('class', 'autofix-all-link-text')
+               .html(t.html('issues.fix_all.title'));
+            linkEnter
+               .append('span')
+               .attr('class', 'autofix-all-link-icon')
+               .call(svgIcon('#iD-icon-wrench'));
+            if (severity === 'warning') {
+               renderIgnoredIssuesReset(selection);
+           }
+            // update
+           autoFixAll = autoFixAll
+               .merge(autoFixAllEnter);
+            autoFixAll.selectAll('.autofix-all-link')
+               .on('click', function() {
+                   context.pauseChangeDispatch();
+                   context.perform(actionNoop());
+                   canAutoFix.forEach(function(issue) {
+                       var args = issue.autoFix.autoArgs.slice();  // copy
+                       if (typeof args[args.length - 1] !== 'function') {
+                           args.pop();
+                       }
+                       args.push(t('issues.fix_all.annotation'));
+                       context.replace.apply(context, args);
+                   });
+                   context.resumeChangeDispatch();
+                   context.validator().validate();
+               });
+           */
+         }
 
-                 for (var i = 0; i < datum.via.ways.length; i++) {
-                   var prev = names[names.length - 1];
-                   var curr = displayName(datum.via.ways[i], vgraph);
-                   if (!prev || curr !== prev) // collapse identical names
-                     names.push(curr);
-                 }
+         context.validator().on('validated.uiSectionValidationIssues' + id, function () {
+           window.requestIdleCallback(function () {
+             reloadIssues();
+             section.reRender();
+           });
+         });
+         context.map().on('move.uiSectionValidationIssues' + id, debounce(function () {
+           window.requestIdleCallback(function () {
+             if (getOptions().where === 'visible') {
+               // must refetch issues if they are viewport-dependent
+               reloadIssues();
+             } // always reload list to re-sort-by-distance
 
-                 help.append('div') // "VIA {viaNames}"
-                 .html(_t.html('restriction.help.via_names', {
-                   via: placeholders.via,
-                   viaNames: names.join(', ')
-                 }));
-               }
 
-               if (!indirect) {
-                 help.append('div') // Click for "No Right Turn"
-                 .html(_t.html('restriction.help.toggle', {
-                   turn: nextText.trim()
-                 }));
-               }
+             section.reRender();
+           });
+         }, 1000));
+         return section;
+       }
 
-               highlightPathsFrom(null);
-               var alongIDs = datum.path.slice();
-               surface.selectAll(utilEntitySelector(alongIDs)).classed('related', true).classed('allow', klass === 'allow').classed('restrict', klass === 'restrict').classed('only', klass === 'only'); // Hovering empty surface
-             } else {
-               highlightPathsFrom(null);
+       function uiSectionValidationOptions(context) {
+         var section = uiSection('issues-options', context).content(renderContent);
 
-               if (_fromWayID) {
-                 help.append('div') // "FROM {fromName}"
-                 .html(_t.html('restriction.help.from_name', {
-                   from: placeholders.from,
-                   fromName: displayName(_fromWayID, vgraph)
-                 }));
-               } else {
-                 help.append('div') // "Click to select a FROM segment."
-                 .html(_t.html('restriction.help.select_from', {
-                   from: placeholders.from
-                 }));
-               }
-             }
-           }
+         function renderContent(selection) {
+           var container = selection.selectAll('.issues-options-container').data([0]);
+           container = container.enter().append('div').attr('class', 'issues-options-container').merge(container);
+           var data = [{
+             key: 'what',
+             values: ['edited', 'all']
+           }, {
+             key: 'where',
+             values: ['visible', 'all']
+           }];
+           var options = container.selectAll('.issues-option').data(data, function (d) {
+             return d.key;
+           });
+           var optionsEnter = options.enter().append('div').attr('class', function (d) {
+             return 'issues-option issues-option-' + d.key;
+           });
+           optionsEnter.append('div').attr('class', 'issues-option-title').html(function (d) {
+             return _t.html('issues.options.' + d.key + '.title');
+           });
+           var valuesEnter = optionsEnter.selectAll('label').data(function (d) {
+             return d.values.map(function (val) {
+               return {
+                 value: val,
+                 key: d.key
+               };
+             });
+           }).enter().append('label');
+           valuesEnter.append('input').attr('type', 'radio').attr('name', function (d) {
+             return 'issues-option-' + d.key;
+           }).attr('value', function (d) {
+             return d.value;
+           }).property('checked', function (d) {
+             return getOptions()[d.key] === d.value;
+           }).on('change', function (d3_event, d) {
+             updateOptionValue(d3_event, d.key, d.value);
+           });
+           valuesEnter.append('span').html(function (d) {
+             return _t.html('issues.options.' + d.key + '.' + d.value);
+           });
          }
 
-         function displayMaxDistance(maxDist) {
-           var isImperial = !_mainLocalizer.usesMetric();
-           var opts;
+         function getOptions() {
+           return {
+             what: corePreferences('validate-what') || 'edited',
+             // 'all', 'edited'
+             where: corePreferences('validate-where') || 'all' // 'all', 'visible'
 
-           if (isImperial) {
-             var distToFeet = {
-               // imprecise conversion for prettier display
-               20: 70,
-               25: 85,
-               30: 100,
-               35: 115,
-               40: 130,
-               45: 145,
-               50: 160
-             }[maxDist];
-             opts = {
-               distance: _t('units.feet', {
-                 quantity: distToFeet
-               })
-             };
-           } else {
-             opts = {
-               distance: _t('units.meters', {
-                 quantity: maxDist
-               })
-             };
+           };
+         }
+
+         function updateOptionValue(d3_event, d, val) {
+           if (!val && d3_event && d3_event.target) {
+             val = d3_event.target.value;
            }
 
-           return _t.html('restriction.controls.distance_up_to', opts);
+           corePreferences('validate-' + d, val);
+           context.validator().validate();
          }
 
-         function displayMaxVia(maxVia) {
-           return maxVia === 0 ? _t.html('restriction.controls.via_node_only') : maxVia === 1 ? _t.html('restriction.controls.via_up_to_one') : _t.html('restriction.controls.via_up_to_two');
-         }
+         return section;
+       }
 
-         function displayName(entityID, graph) {
-           var entity = graph.entity(entityID);
-           var name = utilDisplayName(entity) || '';
-           var matched = _mainPresetIndex.match(entity, graph);
-           var type = matched && matched.name() || utilDisplayType(entity.id);
-           return name || type;
-         }
+       function uiSectionValidationRules(context) {
+         var MINSQUARE = 0;
+         var MAXSQUARE = 20;
+         var DEFAULTSQUARE = 5; // see also unsquare_way.js
 
-         restrictions.entityIDs = function (val) {
-           _intersection = null;
-           _fromWayID = null;
-           _oldTurns = null;
-           _vertexID = val[0];
-         };
+         var section = uiSection('issues-rules', context).disclosureContent(renderDisclosureContent).label(_t.html('issues.rules.title'));
 
-         restrictions.tags = function () {};
+         var _ruleKeys = context.validator().getRuleKeys().filter(function (key) {
+           return key !== 'maprules';
+         }).sort(function (key1, key2) {
+           // alphabetize by localized title
+           return _t('issues.' + key1 + '.title') < _t('issues.' + key2 + '.title') ? -1 : 1;
+         });
 
-         restrictions.focus = function () {};
+         function renderDisclosureContent(selection) {
+           var container = selection.selectAll('.issues-rulelist-container').data([0]);
+           var containerEnter = container.enter().append('div').attr('class', 'issues-rulelist-container');
+           containerEnter.append('ul').attr('class', 'layer-list issue-rules-list');
+           var ruleLinks = containerEnter.append('div').attr('class', 'issue-rules-links section-footer');
+           ruleLinks.append('a').attr('class', 'issue-rules-link').attr('href', '#').html(_t.html('issues.disable_all')).on('click', function (d3_event) {
+             d3_event.preventDefault();
+             context.validator().disableRules(_ruleKeys);
+           });
+           ruleLinks.append('a').attr('class', 'issue-rules-link').attr('href', '#').html(_t.html('issues.enable_all')).on('click', function (d3_event) {
+             d3_event.preventDefault();
+             context.validator().disableRules([]);
+           }); // Update
 
-         restrictions.off = function (selection) {
-           if (!_initialized) return;
-           selection.selectAll('.surface').call(breathe.off).on('click.restrictions', null).on('mouseover.restrictions', null);
-           select(window).on('resize.restrictions', null);
-         };
+           container = container.merge(containerEnter);
+           container.selectAll('.issue-rules-list').call(drawListItems, _ruleKeys, 'checkbox', 'rule', toggleRule, isRuleEnabled);
+         }
 
-         return utilRebind(restrictions, dispatch$1, 'on');
-       }
-       uiFieldRestrictions.supportsMultiselection = false;
+         function drawListItems(selection, data, type, name, change, active) {
+           var items = selection.selectAll('li').data(data); // Exit
 
-       function uiFieldTextarea(field, context) {
-         var dispatch$1 = dispatch('change');
-         var input = select(null);
+           items.exit().remove(); // Enter
 
-         var _tags;
+           var enter = items.enter().append('li');
 
-         function textarea(selection) {
-           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
-           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
-           input = wrap.selectAll('textarea').data([0]);
-           input = input.enter().append('textarea').attr('id', field.domId).call(utilNoAuto).on('input', change(true)).on('blur', change()).on('change', change()).merge(input);
-         }
+           if (name === 'rule') {
+             enter.call(uiTooltip().title(function (d) {
+               return _t.html('issues.' + d + '.tip');
+             }).placement('top'));
+           }
 
-         function change(onInput) {
-           return function () {
-             var val = utilGetSetValue(input);
-             if (!onInput) val = context.cleanTagValue(val); // don't override multiple values with blank string
+           var label = enter.append('label');
+           label.append('input').attr('type', type).attr('name', name).on('change', change);
+           label.append('span').html(function (d) {
+             var params = {};
 
-             if (!val && Array.isArray(_tags[field.key])) return;
-             var t = {};
-             t[field.key] = val || undefined;
-             dispatch$1.call('change', this, t, onInput);
-           };
-         }
+             if (d === 'unsquare_way') {
+               params.val = '<span class="square-degrees"></span>';
+             }
 
-         textarea.tags = function (tags) {
-           _tags = tags;
-           var isMixed = Array.isArray(tags[field.key]);
-           utilGetSetValue(input, !isMixed && tags[field.key] ? tags[field.key] : '').attr('title', isMixed ? tags[field.key].filter(Boolean).join('\n') : undefined).attr('placeholder', isMixed ? _t('inspector.multiple_values') : field.placeholder() || _t('inspector.unknown')).classed('mixed', isMixed);
-         };
+             return _t.html('issues.' + d + '.title', params);
+           }); // Update
 
-         textarea.focus = function () {
-           input.node().focus();
-         };
+           items = items.merge(enter);
+           items.classed('active', active).selectAll('input').property('checked', active).property('indeterminate', false); // user-configurable square threshold
 
-         return utilRebind(textarea, dispatch$1, 'on');
-       }
+           var degStr = corePreferences('validate-square-degrees');
 
-       var getOwnPropertyDescriptor$5 = objectGetOwnPropertyDescriptor.f;
+           if (degStr === null) {
+             degStr = DEFAULTSQUARE.toString();
+           }
 
+           var span = items.selectAll('.square-degrees');
+           var input = span.selectAll('.square-degrees-input').data([0]); // enter / update
 
+           input.enter().append('input').attr('type', 'number').attr('min', MINSQUARE.toString()).attr('max', MAXSQUARE.toString()).attr('step', '0.5').attr('class', 'square-degrees-input').call(utilNoAuto).on('click', function (d3_event) {
+             d3_event.preventDefault();
+             d3_event.stopPropagation();
+             this.select();
+           }).on('keyup', function (d3_event) {
+             if (d3_event.keyCode === 13) {
+               // ↩ Return
+               this.blur();
+               this.select();
+             }
+           }).on('blur', changeSquare).merge(input).property('value', degStr);
+         }
 
+         function changeSquare() {
+           var input = select(this);
+           var degStr = utilGetSetValue(input).trim();
+           var degNum = parseFloat(degStr, 10);
 
+           if (!isFinite(degNum)) {
+             degNum = DEFAULTSQUARE;
+           } else if (degNum > MAXSQUARE) {
+             degNum = MAXSQUARE;
+           } else if (degNum < MINSQUARE) {
+             degNum = MINSQUARE;
+           }
 
+           degNum = Math.round(degNum * 10) / 10; // round to 1 decimal
 
-       var nativeEndsWith = ''.endsWith;
-       var min$a = Math.min;
+           degStr = degNum.toString();
+           input.property('value', degStr);
+           corePreferences('validate-square-degrees', degStr);
+           context.validator().revalidateUnsquare();
+         }
 
-       var CORRECT_IS_REGEXP_LOGIC$1 = correctIsRegexpLogic('endsWith');
-       // https://github.com/zloirock/core-js/pull/702
-       var MDN_POLYFILL_BUG$1 =  !CORRECT_IS_REGEXP_LOGIC$1 && !!function () {
-         var descriptor = getOwnPropertyDescriptor$5(String.prototype, 'endsWith');
-         return descriptor && !descriptor.writable;
-       }();
+         function isRuleEnabled(d) {
+           return context.validator().isRuleEnabled(d);
+         }
 
-       // `String.prototype.endsWith` method
-       // https://tc39.es/ecma262/#sec-string.prototype.endswith
-       _export({ target: 'String', proto: true, forced: !MDN_POLYFILL_BUG$1 && !CORRECT_IS_REGEXP_LOGIC$1 }, {
-         endsWith: function endsWith(searchString /* , endPosition = @length */) {
-           var that = String(requireObjectCoercible(this));
-           notARegexp(searchString);
-           var endPosition = arguments.length > 1 ? arguments[1] : undefined;
-           var len = toLength(that.length);
-           var end = endPosition === undefined ? len : min$a(toLength(endPosition), len);
-           var search = String(searchString);
-           return nativeEndsWith
-             ? nativeEndsWith.call(that, search, end)
-             : that.slice(end - search.length, end) === search;
+         function toggleRule(d3_event, d) {
+           context.validator().toggleRule(d);
          }
-       });
 
-       function uiFieldWikidata(field, context) {
-         var wikidata = services.wikidata;
-         var dispatch$1 = dispatch('change');
+         context.validator().on('validated.uiSectionValidationRules', function () {
+           window.requestIdleCallback(section.reRender);
+         });
+         return section;
+       }
 
-         var _selection = select(null);
+       function uiSectionValidationStatus(context) {
+         var section = uiSection('issues-status', context).content(renderContent).shouldDisplay(function () {
+           var issues = context.validator().getIssues(getOptions());
+           return issues.length === 0;
+         });
 
-         var _searchInput = select(null);
+         function getOptions() {
+           return {
+             what: corePreferences('validate-what') || 'edited',
+             where: corePreferences('validate-where') || 'all'
+           };
+         }
 
-         var _qid = null;
-         var _wikidataEntity = null;
-         var _wikiURL = '';
-         var _entityIDs = [];
+         function renderContent(selection) {
+           var box = selection.selectAll('.box').data([0]);
+           var boxEnter = box.enter().append('div').attr('class', 'box');
+           boxEnter.append('div').call(svgIcon('#iD-icon-apply', 'pre-text'));
+           var noIssuesMessage = boxEnter.append('span');
+           noIssuesMessage.append('strong').attr('class', 'message');
+           noIssuesMessage.append('br');
+           noIssuesMessage.append('span').attr('class', 'details');
+           renderIgnoredIssuesReset(selection);
+           setNoIssuesText(selection);
+         }
 
-         var _wikipediaKey = field.keys && field.keys.find(function (key) {
-           return key.includes('wikipedia');
-         }),
-             _hintKey = field.key === 'wikidata' ? 'name' : field.key.split(':')[0];
+         function renderIgnoredIssuesReset(selection) {
+           var ignoredIssues = context.validator().getIssues({
+             what: 'all',
+             where: 'all',
+             includeDisabledRules: true,
+             includeIgnored: 'only'
+           });
+           var resetIgnored = selection.selectAll('.reset-ignored').data(ignoredIssues.length ? [0] : []); // exit
 
-         var combobox = uiCombobox(context, 'combo-' + field.safeid).caseSensitive(true).minItems(1);
+           resetIgnored.exit().remove(); // enter
 
-         function wiki(selection) {
-           _selection = selection;
-           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
-           wrap = wrap.enter().append('div').attr('class', 'form-field-input-wrap form-field-input-' + field.type).merge(wrap);
-           var list = wrap.selectAll('ul').data([0]);
-           list = list.enter().append('ul').attr('class', 'rows').merge(list);
-           var searchRow = list.selectAll('li.wikidata-search').data([0]);
-           var searchRowEnter = searchRow.enter().append('li').attr('class', 'wikidata-search');
-           searchRowEnter.append('input').attr('type', 'text').attr('id', field.domId).style('flex', '1').call(utilNoAuto).on('focus', function () {
-             var node = select(this).node();
-             node.setSelectionRange(0, node.value.length);
-           }).on('blur', function () {
-             setLabelForEntity();
-           }).call(combobox.fetcher(fetchWikidataItems));
-           combobox.on('accept', function (d) {
-             if (d) {
-               _qid = d.id;
-               change();
-             }
-           }).on('cancel', function () {
-             setLabelForEntity();
-           });
-           searchRowEnter.append('button').attr('class', 'form-field-button wiki-link').attr('title', _t('icons.view_on', {
-             domain: 'wikidata.org'
-           })).call(svgIcon('#iD-icon-out-link')).on('click', function (d3_event) {
-             d3_event.preventDefault();
-             if (_wikiURL) window.open(_wikiURL, '_blank');
-           });
-           searchRow = searchRow.merge(searchRowEnter);
-           _searchInput = searchRow.select('input');
-           var wikidataProperties = ['description', 'identifier'];
-           var items = list.selectAll('li.labeled-input').data(wikidataProperties); // Enter
+           var resetIgnoredEnter = resetIgnored.enter().append('div').attr('class', 'reset-ignored section-footer');
+           resetIgnoredEnter.append('a').attr('href', '#'); // update
 
-           var enter = items.enter().append('li').attr('class', function (d) {
-             return 'labeled-input preset-wikidata-' + d;
-           });
-           enter.append('span').attr('class', 'label').html(function (d) {
-             return _t.html('wikidata.' + d);
-           });
-           enter.append('input').attr('type', 'text').call(utilNoAuto).classed('disabled', 'true').attr('readonly', 'true');
-           enter.append('button').attr('class', 'form-field-button').attr('title', _t('icons.copy')).call(svgIcon('#iD-operation-copy')).on('click', function (d3_event) {
+           resetIgnored = resetIgnored.merge(resetIgnoredEnter);
+           resetIgnored.select('a').html(_t('inspector.title_count', {
+             title: _t.html('issues.reset_ignored'),
+             count: ignoredIssues.length
+           }));
+           resetIgnored.on('click', function (d3_event) {
              d3_event.preventDefault();
-             select(this.parentNode).select('input').node().select();
-             document.execCommand('copy');
+             context.validator().resetIgnoredIssues();
            });
          }
 
-         function fetchWikidataItems(q, callback) {
-           if (!q && _hintKey) {
-             // other tags may be good search terms
-             for (var i in _entityIDs) {
-               var entity = context.hasEntity(_entityIDs[i]);
+         function setNoIssuesText(selection) {
+           var opts = getOptions();
 
-               if (entity.tags[_hintKey]) {
-                 q = entity.tags[_hintKey];
-                 break;
+           function checkForHiddenIssues(cases) {
+             for (var type in cases) {
+               var hiddenOpts = cases[type];
+               var hiddenIssues = context.validator().getIssues(hiddenOpts);
+
+               if (hiddenIssues.length) {
+                 selection.select('.box .details').html(_t.html('issues.no_issues.hidden_issues.' + type, {
+                   count: hiddenIssues.length.toString()
+                 }));
+                 return;
                }
              }
+
+             selection.select('.box .details').html(_t.html('issues.no_issues.hidden_issues.none'));
            }
 
-           wikidata.itemsForSearchQuery(q, function (err, data) {
-             if (err) return;
+           var messageType;
 
-             for (var i in data) {
-               data[i].value = data[i].label + ' (' + data[i].id + ')';
-               data[i].title = data[i].description;
-             }
+           if (opts.what === 'edited' && opts.where === 'visible') {
+             messageType = 'edits_in_view';
+             checkForHiddenIssues({
+               elsewhere: {
+                 what: 'edited',
+                 where: 'all'
+               },
+               everything_else: {
+                 what: 'all',
+                 where: 'visible'
+               },
+               disabled_rules: {
+                 what: 'edited',
+                 where: 'visible',
+                 includeDisabledRules: 'only'
+               },
+               everything_else_elsewhere: {
+                 what: 'all',
+                 where: 'all'
+               },
+               disabled_rules_elsewhere: {
+                 what: 'edited',
+                 where: 'all',
+                 includeDisabledRules: 'only'
+               },
+               ignored_issues: {
+                 what: 'edited',
+                 where: 'visible',
+                 includeIgnored: 'only'
+               },
+               ignored_issues_elsewhere: {
+                 what: 'edited',
+                 where: 'all',
+                 includeIgnored: 'only'
+               }
+             });
+           } else if (opts.what === 'edited' && opts.where === 'all') {
+             messageType = 'edits';
+             checkForHiddenIssues({
+               everything_else: {
+                 what: 'all',
+                 where: 'all'
+               },
+               disabled_rules: {
+                 what: 'edited',
+                 where: 'all',
+                 includeDisabledRules: 'only'
+               },
+               ignored_issues: {
+                 what: 'edited',
+                 where: 'all',
+                 includeIgnored: 'only'
+               }
+             });
+           } else if (opts.what === 'all' && opts.where === 'visible') {
+             messageType = 'everything_in_view';
+             checkForHiddenIssues({
+               elsewhere: {
+                 what: 'all',
+                 where: 'all'
+               },
+               disabled_rules: {
+                 what: 'all',
+                 where: 'visible',
+                 includeDisabledRules: 'only'
+               },
+               disabled_rules_elsewhere: {
+                 what: 'all',
+                 where: 'all',
+                 includeDisabledRules: 'only'
+               },
+               ignored_issues: {
+                 what: 'all',
+                 where: 'visible',
+                 includeIgnored: 'only'
+               },
+               ignored_issues_elsewhere: {
+                 what: 'all',
+                 where: 'all',
+                 includeIgnored: 'only'
+               }
+             });
+           } else if (opts.what === 'all' && opts.where === 'all') {
+             messageType = 'everything';
+             checkForHiddenIssues({
+               disabled_rules: {
+                 what: 'all',
+                 where: 'all',
+                 includeDisabledRules: 'only'
+               },
+               ignored_issues: {
+                 what: 'all',
+                 where: 'all',
+                 includeIgnored: 'only'
+               }
+             });
+           }
 
-             if (callback) callback(data);
-           });
+           if (opts.what === 'edited' && context.history().difference().summary().length === 0) {
+             messageType = 'no_edits';
+           }
+
+           selection.select('.box .message').html(_t.html('issues.no_issues.message.' + messageType));
          }
 
-         function change() {
-           var syncTags = {};
-           syncTags[field.key] = _qid;
-           dispatch$1.call('change', this, syncTags); // attempt asynchronous update of wikidata tag..
+         context.validator().on('validated.uiSectionValidationStatus', function () {
+           window.requestIdleCallback(section.reRender);
+         });
+         context.map().on('move.uiSectionValidationStatus', debounce(function () {
+           window.requestIdleCallback(section.reRender);
+         }, 1000));
+         return section;
+       }
 
-           var initGraph = context.graph();
-           var initEntityIDs = _entityIDs;
-           wikidata.entityByQID(_qid, function (err, entity) {
-             if (err) return; // If graph has changed, we can't apply this update.
+       function uiPaneIssues(context) {
+         var issuesPane = uiPane('issues', context).key(_t('issues.key')).label(_t.html('issues.title')).description(_t.html('issues.title')).iconName('iD-icon-alert').sections([uiSectionValidationOptions(context), uiSectionValidationStatus(context), uiSectionValidationIssues('issues-errors', 'error', context), uiSectionValidationIssues('issues-warnings', 'warning', context), uiSectionValidationRules(context)]);
+         return issuesPane;
+       }
 
-             if (context.graph() !== initGraph) return;
-             if (!entity.sitelinks) return;
-             var langs = wikidata.languagesToQuery(); // use the label and description languages as fallbacks
+       function uiSettingsCustomData(context) {
+         var dispatch = dispatch$8('change');
 
-             ['labels', 'descriptions'].forEach(function (key) {
-               if (!entity[key]) return;
-               var valueLangs = Object.keys(entity[key]);
-               if (valueLangs.length === 0) return;
-               var valueLang = valueLangs[0];
+         function render(selection) {
+           var dataLayer = context.layers().layer('data'); // keep separate copies of original and current settings
 
-               if (langs.indexOf(valueLang) === -1) {
-                 langs.push(valueLang);
-               }
-             });
-             var newWikipediaValue;
+           var _origSettings = {
+             fileList: dataLayer && dataLayer.fileList() || null,
+             url: corePreferences('settings-custom-data-url')
+           };
+           var _currSettings = {
+             fileList: dataLayer && dataLayer.fileList() || null,
+             url: corePreferences('settings-custom-data-url')
+           }; // var example = 'https://{switch:a,b,c}.tile.openstreetmap.org/{zoom}/{x}/{y}.png';
 
-             if (_wikipediaKey) {
-               var foundPreferred;
+           var modal = uiConfirm(selection).okButton();
+           modal.classed('settings-modal settings-custom-data', true);
+           modal.select('.modal-section.header').append('h3').html(_t.html('settings.custom_data.header'));
+           var textSection = modal.select('.modal-section.message-text');
+           textSection.append('pre').attr('class', 'instructions-file').html(_t.html('settings.custom_data.file.instructions'));
+           textSection.append('input').attr('class', 'field-file').attr('type', 'file').property('files', _currSettings.fileList) // works for all except IE11
+           .on('change', function (d3_event) {
+             var files = d3_event.target.files;
 
-               for (var i in langs) {
-                 var lang = langs[i];
-                 var siteID = lang.replace('-', '_') + 'wiki';
+             if (files && files.length) {
+               _currSettings.url = '';
+               textSection.select('.field-url').property('value', '');
+               _currSettings.fileList = files;
+             } else {
+               _currSettings.fileList = null;
+             }
+           });
+           textSection.append('h4').html(_t.html('settings.custom_data.or'));
+           textSection.append('pre').attr('class', 'instructions-url').html(_t.html('settings.custom_data.url.instructions'));
+           textSection.append('textarea').attr('class', 'field-url').attr('placeholder', _t('settings.custom_data.url.placeholder')).call(utilNoAuto).property('value', _currSettings.url); // insert a cancel button
 
-                 if (entity.sitelinks[siteID]) {
-                   foundPreferred = true;
-                   newWikipediaValue = lang + ':' + entity.sitelinks[siteID].title; // use the first match
+           var buttonSection = modal.select('.modal-section.buttons');
+           buttonSection.insert('button', '.ok-button').attr('class', 'button cancel-button secondary-action').html(_t.html('confirm.cancel'));
+           buttonSection.select('.cancel-button').on('click.cancel', clickCancel);
+           buttonSection.select('.ok-button').attr('disabled', isSaveDisabled).on('click.save', clickSave);
 
-                   break;
-                 }
-               }
+           function isSaveDisabled() {
+             return null;
+           } // restore the original url
 
-               if (!foundPreferred) {
-                 // No wikipedia sites available in the user's language or the fallback languages,
-                 // default to any wikipedia sitelink
-                 var wikiSiteKeys = Object.keys(entity.sitelinks).filter(function (site) {
-                   return site.endsWith('wiki');
-                 });
 
-                 if (wikiSiteKeys.length === 0) {
-                   // if no wikipedia pages are linked to this wikidata entity, delete that tag
-                   newWikipediaValue = null;
-                 } else {
-                   var wikiLang = wikiSiteKeys[0].slice(0, -4).replace('_', '-');
-                   var wikiTitle = entity.sitelinks[wikiSiteKeys[0]].title;
-                   newWikipediaValue = wikiLang + ':' + wikiTitle;
-                 }
-               }
-             }
+           function clickCancel() {
+             textSection.select('.field-url').property('value', _origSettings.url);
+             corePreferences('settings-custom-data-url', _origSettings.url);
+             this.blur();
+             modal.close();
+           } // accept the current url
 
-             if (newWikipediaValue) {
-               newWikipediaValue = context.cleanTagValue(newWikipediaValue);
-             }
 
-             if (typeof newWikipediaValue === 'undefined') return;
-             var actions = initEntityIDs.map(function (entityID) {
-               var entity = context.hasEntity(entityID);
-               if (!entity) return null;
-               var currTags = Object.assign({}, entity.tags); // shallow copy
+           function clickSave() {
+             _currSettings.url = textSection.select('.field-url').property('value').trim(); // one or the other but not both
 
-               if (newWikipediaValue === null) {
-                 if (!currTags[_wikipediaKey]) return null;
-                 delete currTags[_wikipediaKey];
-               } else {
-                 currTags[_wikipediaKey] = newWikipediaValue;
-               }
+             if (_currSettings.url) {
+               _currSettings.fileList = null;
+             }
 
-               return actionChangeTags(entityID, currTags);
-             }).filter(Boolean);
-             if (!actions.length) return; // Coalesce the update of wikidata tag into the previous tag change
+             if (_currSettings.fileList) {
+               _currSettings.url = '';
+             }
 
-             context.overwrite(function actionUpdateWikipediaTags(graph) {
-               actions.forEach(function (action) {
-                 graph = action(graph);
-               });
-               return graph;
-             }, context.history().undoAnnotation()); // do not dispatch.call('change') here, because entity_editor
-             // changeTags() is not intended to be called asynchronously
-           });
+             corePreferences('settings-custom-data-url', _currSettings.url);
+             this.blur();
+             modal.close();
+             dispatch.call('change', this, _currSettings);
+           }
          }
 
-         function setLabelForEntity() {
-           var label = '';
-
-           if (_wikidataEntity) {
-             label = entityPropertyForDisplay(_wikidataEntity, 'labels');
+         return utilRebind(render, dispatch, 'on');
+       }
 
-             if (label.length === 0) {
-               label = _wikidataEntity.id.toString();
-             }
-           }
+       function uiSectionDataLayers(context) {
+         var settingsCustomData = uiSettingsCustomData(context).on('change', customChanged);
+         var layers = context.layers();
+         var section = uiSection('data-layers', context).label(_t.html('map_data.data_layers')).disclosureContent(renderDisclosureContent);
 
-           utilGetSetValue(_searchInput, label);
+         function renderDisclosureContent(selection) {
+           var container = selection.selectAll('.data-layer-container').data([0]);
+           container.enter().append('div').attr('class', 'data-layer-container').merge(container).call(drawOsmItems).call(drawQAItems).call(drawCustomDataItems).call(drawVectorItems) // Beta - Detroit mapping challenge
+           .call(drawPanelItems);
          }
 
-         wiki.tags = function (tags) {
-           var isMixed = Array.isArray(tags[field.key]);
+         function showsLayer(which) {
+           var layer = layers.layer(which);
 
-           _searchInput.attr('title', isMixed ? tags[field.key].filter(Boolean).join('\n') : null).attr('placeholder', isMixed ? _t('inspector.multiple_values') : '').classed('mixed', isMixed);
+           if (layer) {
+             return layer.enabled();
+           }
 
-           _qid = typeof tags[field.key] === 'string' && tags[field.key] || '';
+           return false;
+         }
 
-           if (!/^Q[0-9]*$/.test(_qid)) {
-             // not a proper QID
-             unrecognized();
-             return;
-           } // QID value in correct format
+         function setLayer(which, enabled) {
+           // Don't allow layer changes while drawing - #6584
+           var mode = context.mode();
+           if (mode && /^draw/.test(mode.id)) return;
+           var layer = layers.layer(which);
 
+           if (layer) {
+             layer.enabled(enabled);
 
-           _wikiURL = 'https://wikidata.org/wiki/' + _qid;
-           wikidata.entityByQID(_qid, function (err, entity) {
-             if (err) {
-               unrecognized();
-               return;
+             if (!enabled && (which === 'osm' || which === 'notes')) {
+               context.enter(modeBrowse(context));
              }
+           }
+         }
 
-             _wikidataEntity = entity;
-             setLabelForEntity();
-             var description = entityPropertyForDisplay(entity, 'descriptions');
-
-             _selection.select('button.wiki-link').classed('disabled', false);
-
-             _selection.select('.preset-wikidata-description').style('display', function () {
-               return description.length > 0 ? 'flex' : 'none';
-             }).select('input').attr('value', description);
+         function toggleLayer(which) {
+           setLayer(which, !showsLayer(which));
+         }
 
-             _selection.select('.preset-wikidata-identifier').style('display', function () {
-               return entity.id ? 'flex' : 'none';
-             }).select('input').attr('value', entity.id);
-           }); // not a proper QID
+         function drawOsmItems(selection) {
+           var osmKeys = ['osm', 'notes'];
+           var osmLayers = layers.all().filter(function (obj) {
+             return osmKeys.indexOf(obj.id) !== -1;
+           });
+           var ul = selection.selectAll('.layer-list-osm').data([0]);
+           ul = ul.enter().append('ul').attr('class', 'layer-list layer-list-osm').merge(ul);
+           var li = ul.selectAll('.list-item').data(osmLayers);
+           li.exit().remove();
+           var liEnter = li.enter().append('li').attr('class', function (d) {
+             return 'list-item list-item-' + d.id;
+           });
+           var labelEnter = liEnter.append('label').each(function (d) {
+             if (d.id === 'osm') {
+               select(this).call(uiTooltip().title(_t.html('map_data.layers.' + d.id + '.tooltip')).keys([uiCmd('⌥' + _t('area_fill.wireframe.key'))]).placement('bottom'));
+             } else {
+               select(this).call(uiTooltip().title(_t.html('map_data.layers.' + d.id + '.tooltip')).placement('bottom'));
+             }
+           });
+           labelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event, d) {
+             toggleLayer(d.id);
+           });
+           labelEnter.append('span').html(function (d) {
+             return _t.html('map_data.layers.' + d.id + '.title');
+           }); // Update
 
-           function unrecognized() {
-             _wikidataEntity = null;
-             setLabelForEntity();
+           li.merge(liEnter).classed('active', function (d) {
+             return d.layer.enabled();
+           }).selectAll('input').property('checked', function (d) {
+             return d.layer.enabled();
+           });
+         }
 
-             _selection.select('.preset-wikidata-description').style('display', 'none');
+         function drawQAItems(selection) {
+           var qaKeys = ['keepRight', 'improveOSM', 'osmose'];
+           var qaLayers = layers.all().filter(function (obj) {
+             return qaKeys.indexOf(obj.id) !== -1;
+           });
+           var ul = selection.selectAll('.layer-list-qa').data([0]);
+           ul = ul.enter().append('ul').attr('class', 'layer-list layer-list-qa').merge(ul);
+           var li = ul.selectAll('.list-item').data(qaLayers);
+           li.exit().remove();
+           var liEnter = li.enter().append('li').attr('class', function (d) {
+             return 'list-item list-item-' + d.id;
+           });
+           var labelEnter = liEnter.append('label').each(function (d) {
+             select(this).call(uiTooltip().title(_t.html('map_data.layers.' + d.id + '.tooltip')).placement('bottom'));
+           });
+           labelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event, d) {
+             toggleLayer(d.id);
+           });
+           labelEnter.append('span').html(function (d) {
+             return _t.html('map_data.layers.' + d.id + '.title');
+           }); // Update
 
-             _selection.select('.preset-wikidata-identifier').style('display', 'none');
+           li.merge(liEnter).classed('active', function (d) {
+             return d.layer.enabled();
+           }).selectAll('input').property('checked', function (d) {
+             return d.layer.enabled();
+           });
+         } // Beta feature - sample vector layers to support Detroit Mapping Challenge
+         // https://github.com/osmus/detroit-mapping-challenge
 
-             _selection.select('button.wiki-link').classed('disabled', true);
 
-             if (_qid && _qid !== '') {
-               _wikiURL = 'https://wikidata.org/wiki/Special:Search?search=' + _qid;
-             } else {
-               _wikiURL = '';
-             }
-           }
-         };
+         function drawVectorItems(selection) {
+           var dataLayer = layers.layer('data');
+           var vtData = [{
+             name: 'Detroit Neighborhoods/Parks',
+             src: 'neighborhoods-parks',
+             tooltip: 'Neighborhood boundaries and parks as compiled by City of Detroit in concert with community groups.',
+             template: 'https://{switch:a,b,c,d}.tiles.mapbox.com/v4/jonahadkins.cjksmur6x34562qp9iv1u3ksf-54hev,jonahadkins.cjksmqxdx33jj2wp90xd9x2md-4e5y2/{z}/{x}/{y}.vector.pbf?access_token=pk.eyJ1Ijoiam9uYWhhZGtpbnMiLCJhIjoiRlVVVkx3VSJ9.9sdVEK_B_VkEXPjssU5MqA'
+           }, {
+             name: 'Detroit Composite POIs',
+             src: 'composite-poi',
+             tooltip: 'Fire Inspections, Business Licenses, and other public location data collated from the City of Detroit.',
+             template: 'https://{switch:a,b,c,d}.tiles.mapbox.com/v4/jonahadkins.cjksmm6a02sli31myxhsr7zf3-2sw8h/{z}/{x}/{y}.vector.pbf?access_token=pk.eyJ1Ijoiam9uYWhhZGtpbnMiLCJhIjoiRlVVVkx3VSJ9.9sdVEK_B_VkEXPjssU5MqA'
+           }, {
+             name: 'Detroit All-The-Places POIs',
+             src: 'alltheplaces-poi',
+             tooltip: 'Public domain business location data created by web scrapers.',
+             template: 'https://{switch:a,b,c,d}.tiles.mapbox.com/v4/jonahadkins.cjksmswgk340g2vo06p1w9w0j-8fjjc/{z}/{x}/{y}.vector.pbf?access_token=pk.eyJ1Ijoiam9uYWhhZGtpbnMiLCJhIjoiRlVVVkx3VSJ9.9sdVEK_B_VkEXPjssU5MqA'
+           }]; // Only show this if the map is around Detroit..
 
-         function entityPropertyForDisplay(wikidataEntity, propKey) {
-           if (!wikidataEntity[propKey]) return '';
-           var propObj = wikidataEntity[propKey];
-           var langKeys = Object.keys(propObj);
-           if (langKeys.length === 0) return ''; // sorted by priority, since we want to show the user's language first if possible
+           var detroit = geoExtent([-83.5, 42.1], [-82.8, 42.5]);
+           var showVectorItems = context.map().zoom() > 9 && detroit.contains(context.map().center());
+           var container = selection.selectAll('.vectortile-container').data(showVectorItems ? [0] : []);
+           container.exit().remove();
+           var containerEnter = container.enter().append('div').attr('class', 'vectortile-container');
+           containerEnter.append('h4').attr('class', 'vectortile-header').html('Detroit Vector Tiles (Beta)');
+           containerEnter.append('ul').attr('class', 'layer-list layer-list-vectortile');
+           containerEnter.append('div').attr('class', 'vectortile-footer').append('a').attr('target', '_blank').call(svgIcon('#iD-icon-out-link', 'inline')).attr('href', 'https://github.com/osmus/detroit-mapping-challenge').append('span').html('About these layers');
+           container = container.merge(containerEnter);
+           var ul = container.selectAll('.layer-list-vectortile');
+           var li = ul.selectAll('.list-item').data(vtData);
+           li.exit().remove();
+           var liEnter = li.enter().append('li').attr('class', function (d) {
+             return 'list-item list-item-' + d.src;
+           });
+           var labelEnter = liEnter.append('label').each(function (d) {
+             select(this).call(uiTooltip().title(d.tooltip).placement('top'));
+           });
+           labelEnter.append('input').attr('type', 'radio').attr('name', 'vectortile').on('change', selectVTLayer);
+           labelEnter.append('span').html(function (d) {
+             return d.name;
+           }); // Update
 
-           var langs = wikidata.languagesToQuery();
+           li.merge(liEnter).classed('active', isVTLayerSelected).selectAll('input').property('checked', isVTLayerSelected);
 
-           for (var i in langs) {
-             var lang = langs[i];
-             var valueObj = propObj[lang];
-             if (valueObj && valueObj.value && valueObj.value.length > 0) return valueObj.value;
-           } // default to any available value
+           function isVTLayerSelected(d) {
+             return dataLayer && dataLayer.template() === d.template;
+           }
 
+           function selectVTLayer(d3_event, d) {
+             corePreferences('settings-custom-data-url', d.template);
 
-           return propObj[langKeys[0]].value;
+             if (dataLayer) {
+               dataLayer.template(d.template, d.src);
+               dataLayer.enabled(true);
+             }
+           }
          }
 
-         wiki.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs;
-           _entityIDs = val;
-           return wiki;
-         };
-
-         wiki.focus = function () {
-           _searchInput.node().focus();
-         };
-
-         return utilRebind(wiki, dispatch$1, 'on');
-       }
-
-       function uiFieldWikipedia(field, context) {
-         var _arguments = arguments;
-         var dispatch$1 = dispatch('change');
-         var wikipedia = services.wikipedia;
-         var wikidata = services.wikidata;
-
-         var _langInput = select(null);
-
-         var _titleInput = select(null);
+         function drawCustomDataItems(selection) {
+           var dataLayer = layers.layer('data');
+           var hasData = dataLayer && dataLayer.hasData();
+           var showsData = hasData && dataLayer.enabled();
+           var ul = selection.selectAll('.layer-list-data').data(dataLayer ? [0] : []); // Exit
 
-         var _wikiURL = '';
+           ul.exit().remove(); // Enter
 
-         var _entityIDs;
+           var ulEnter = ul.enter().append('ul').attr('class', 'layer-list layer-list-data');
+           var liEnter = ulEnter.append('li').attr('class', 'list-item-data');
+           var labelEnter = liEnter.append('label').call(uiTooltip().title(_t.html('map_data.layers.custom.tooltip')).placement('top'));
+           labelEnter.append('input').attr('type', 'checkbox').on('change', function () {
+             toggleLayer('data');
+           });
+           labelEnter.append('span').html(_t.html('map_data.layers.custom.title'));
+           liEnter.append('button').attr('class', 'open-data-options').call(uiTooltip().title(_t.html('settings.custom_data.tooltip')).placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left')).on('click', function (d3_event) {
+             d3_event.preventDefault();
+             editCustom();
+           }).call(svgIcon('#iD-icon-more'));
+           liEnter.append('button').attr('class', 'zoom-to-data').call(uiTooltip().title(_t.html('map_data.layers.custom.zoom')).placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left')).on('click', function (d3_event) {
+             if (select(this).classed('disabled')) return;
+             d3_event.preventDefault();
+             d3_event.stopPropagation();
+             dataLayer.fitZoom();
+           }).call(svgIcon('#iD-icon-framed-dot', 'monochrome')); // Update
 
-         var _tags;
+           ul = ul.merge(ulEnter);
+           ul.selectAll('.list-item-data').classed('active', showsData).selectAll('label').classed('deemphasize', !hasData).selectAll('input').property('disabled', !hasData).property('checked', showsData);
+           ul.selectAll('button.zoom-to-data').classed('disabled', !hasData);
+         }
 
-         var _dataWikipedia = [];
-         _mainFileFetcher.get('wmf_sitematrix').then(function (d) {
-           _dataWikipedia = d;
-           if (_tags) updateForTags(_tags);
-         })["catch"](function () {
-           /* ignore */
-         });
-         var langCombo = uiCombobox(context, 'wikipedia-lang').fetcher(function (value, callback) {
-           var v = value.toLowerCase();
-           callback(_dataWikipedia.filter(function (d) {
-             return d[0].toLowerCase().indexOf(v) >= 0 || d[1].toLowerCase().indexOf(v) >= 0 || d[2].toLowerCase().indexOf(v) >= 0;
-           }).map(function (d) {
-             return {
-               value: d[1]
-             };
-           }));
-         });
-         var titleCombo = uiCombobox(context, 'wikipedia-title').fetcher(function (value, callback) {
-           if (!value) {
-             value = '';
+         function editCustom() {
+           context.container().call(settingsCustomData);
+         }
 
-             for (var i in _entityIDs) {
-               var entity = context.hasEntity(_entityIDs[i]);
+         function customChanged(d) {
+           var dataLayer = layers.layer('data');
 
-               if (entity.tags.name) {
-                 value = entity.tags.name;
-                 break;
-               }
-             }
+           if (d && d.url) {
+             dataLayer.url(d.url);
+           } else if (d && d.fileList) {
+             dataLayer.fileList(d.fileList);
            }
+         }
 
-           var searchfn = value.length > 7 ? wikipedia.search : wikipedia.suggestions;
-           searchfn(language()[2], value, function (query, data) {
-             callback(data.map(function (d) {
-               return {
-                 value: d
-               };
-             }));
-           });
-         });
-
-         function wiki(selection) {
-           var wrap = selection.selectAll('.form-field-input-wrap').data([0]);
-           wrap = wrap.enter().append('div').attr('class', "form-field-input-wrap form-field-input-".concat(field.type)).merge(wrap);
-           var langContainer = wrap.selectAll('.wiki-lang-container').data([0]);
-           langContainer = langContainer.enter().append('div').attr('class', 'wiki-lang-container').merge(langContainer);
-           _langInput = langContainer.selectAll('input.wiki-lang').data([0]);
-           _langInput = _langInput.enter().append('input').attr('type', 'text').attr('class', 'wiki-lang').attr('placeholder', _t('translate.localized_translation_language')).call(utilNoAuto).call(langCombo).merge(_langInput);
-
-           _langInput.on('blur', changeLang).on('change', changeLang);
-
-           var titleContainer = wrap.selectAll('.wiki-title-container').data([0]);
-           titleContainer = titleContainer.enter().append('div').attr('class', 'wiki-title-container').merge(titleContainer);
-           _titleInput = titleContainer.selectAll('input.wiki-title').data([0]);
-           _titleInput = _titleInput.enter().append('input').attr('type', 'text').attr('class', 'wiki-title').attr('id', field.domId).call(utilNoAuto).call(titleCombo).merge(_titleInput);
-
-           _titleInput.on('blur', function () {
-             change(true);
-           }).on('change', function () {
-             change(false);
+         function drawPanelItems(selection) {
+           var panelsListEnter = selection.selectAll('.md-extras-list').data([0]).enter().append('ul').attr('class', 'layer-list md-extras-list');
+           var historyPanelLabelEnter = panelsListEnter.append('li').attr('class', 'history-panel-toggle-item').append('label').call(uiTooltip().title(_t.html('map_data.history_panel.tooltip')).keys([uiCmd('⌘⇧' + _t('info_panels.history.key'))]).placement('top'));
+           historyPanelLabelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event) {
+             d3_event.preventDefault();
+             context.ui().info.toggle('history');
            });
-
-           var link = titleContainer.selectAll('.wiki-link').data([0]);
-           link = link.enter().append('button').attr('class', 'form-field-button wiki-link').attr('title', _t('icons.view_on', {
-             domain: 'wikipedia.org'
-           })).call(svgIcon('#iD-icon-out-link')).merge(link);
-           link.on('click', function (d3_event) {
+           historyPanelLabelEnter.append('span').html(_t.html('map_data.history_panel.title'));
+           var measurementPanelLabelEnter = panelsListEnter.append('li').attr('class', 'measurement-panel-toggle-item').append('label').call(uiTooltip().title(_t.html('map_data.measurement_panel.tooltip')).keys([uiCmd('⌘⇧' + _t('info_panels.measurement.key'))]).placement('top'));
+           measurementPanelLabelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event) {
              d3_event.preventDefault();
-             if (_wikiURL) window.open(_wikiURL, '_blank');
+             context.ui().info.toggle('measurement');
            });
+           measurementPanelLabelEnter.append('span').html(_t.html('map_data.measurement_panel.title'));
          }
 
-         function defaultLanguageInfo(skipEnglishFallback) {
-           var langCode = _mainLocalizer.languageCode().toLowerCase();
+         context.layers().on('change.uiSectionDataLayers', section.reRender);
+         context.map().on('move.uiSectionDataLayers', debounce(function () {
+           // Detroit layers may have moved in or out of view
+           window.requestIdleCallback(section.reRender);
+         }, 1000));
+         return section;
+       }
 
-           for (var i in _dataWikipedia) {
-             var d = _dataWikipedia[i]; // default to the language of iD's current locale
+       function uiSectionMapFeatures(context) {
+         var _features = context.features().keys();
 
-             if (d[2] === langCode) return d;
-           } // fallback to English
+         var section = uiSection('map-features', context).label(_t.html('map_data.map_features')).disclosureContent(renderDisclosureContent).expandedByDefault(false);
 
+         function renderDisclosureContent(selection) {
+           var container = selection.selectAll('.layer-feature-list-container').data([0]);
+           var containerEnter = container.enter().append('div').attr('class', 'layer-feature-list-container');
+           containerEnter.append('ul').attr('class', 'layer-list layer-feature-list');
+           var footer = containerEnter.append('div').attr('class', 'feature-list-links section-footer');
+           footer.append('a').attr('class', 'feature-list-link').attr('href', '#').html(_t.html('issues.disable_all')).on('click', function (d3_event) {
+             d3_event.preventDefault();
+             context.features().disableAll();
+           });
+           footer.append('a').attr('class', 'feature-list-link').attr('href', '#').html(_t.html('issues.enable_all')).on('click', function (d3_event) {
+             d3_event.preventDefault();
+             context.features().enableAll();
+           }); // Update
 
-           return skipEnglishFallback ? ['', '', ''] : ['English', 'English', 'en'];
+           container = container.merge(containerEnter);
+           container.selectAll('.layer-feature-list').call(drawListItems, _features, 'checkbox', 'feature', clickFeature, showsFeature);
          }
 
-         function language(skipEnglishFallback) {
-           var value = utilGetSetValue(_langInput).toLowerCase();
+         function drawListItems(selection, data, type, name, change, active) {
+           var items = selection.selectAll('li').data(data); // Exit
 
-           for (var i in _dataWikipedia) {
-             var d = _dataWikipedia[i]; // return the language already set in the UI, if supported
+           items.exit().remove(); // Enter
 
-             if (d[0].toLowerCase() === value || d[1].toLowerCase() === value || d[2] === value) return d;
-           } // fallback to English
+           var enter = items.enter().append('li').call(uiTooltip().title(function (d) {
+             var tip = _t.html(name + '.' + d + '.tooltip');
 
+             if (autoHiddenFeature(d)) {
+               var msg = showsLayer('osm') ? _t.html('map_data.autohidden') : _t.html('map_data.osmhidden');
+               tip += '<div>' + msg + '</div>';
+             }
 
-           return defaultLanguageInfo(skipEnglishFallback);
-         }
+             return tip;
+           }).placement('top'));
+           var label = enter.append('label');
+           label.append('input').attr('type', type).attr('name', name).on('change', change);
+           label.append('span').html(function (d) {
+             return _t.html(name + '.' + d + '.description');
+           }); // Update
 
-         function changeLang() {
-           utilGetSetValue(_langInput, language()[1]);
-           change(true);
+           items = items.merge(enter);
+           items.classed('active', active).selectAll('input').property('checked', active).property('indeterminate', autoHiddenFeature);
          }
 
-         function change(skipWikidata) {
-           var value = utilGetSetValue(_titleInput);
-           var m = value.match(/https?:\/\/([-a-z]+)\.wikipedia\.org\/(?:wiki|\1-[-a-z]+)\/([^#]+)(?:#(.+))?/);
-
-           var langInfo = m && _dataWikipedia.find(function (d) {
-             return m[1] === d[2];
-           });
+         function autoHiddenFeature(d) {
+           return context.features().autoHidden(d);
+         }
 
-           var syncTags = {};
+         function showsFeature(d) {
+           return context.features().enabled(d);
+         }
 
-           if (langInfo) {
-             var nativeLangName = langInfo[1]; // Normalize title http://www.mediawiki.org/wiki/API:Query#Title_normalization
+         function clickFeature(d3_event, d) {
+           context.features().toggle(d);
+         }
 
-             value = decodeURIComponent(m[2]).replace(/_/g, ' ');
+         function showsLayer(id) {
+           var layer = context.layers().layer(id);
+           return layer && layer.enabled();
+         } // add listeners
 
-             if (m[3]) {
-               var anchor; // try {
-               // leave this out for now - #6232
-               // Best-effort `anchordecode:` implementation
-               // anchor = decodeURIComponent(m[3].replace(/\.([0-9A-F]{2})/g, '%$1'));
-               // } catch (e) {
 
-               anchor = decodeURIComponent(m[3]); // }
+         context.features().on('change.map_features', section.reRender);
+         return section;
+       }
 
-               value += '#' + anchor.replace(/_/g, ' ');
-             }
+       function uiSectionMapStyleOptions(context) {
+         var section = uiSection('fill-area', context).label(_t.html('map_data.style_options')).disclosureContent(renderDisclosureContent).expandedByDefault(false);
 
-             value = value.slice(0, 1).toUpperCase() + value.slice(1);
-             utilGetSetValue(_langInput, nativeLangName);
-             utilGetSetValue(_titleInput, value);
-           }
+         function renderDisclosureContent(selection) {
+           var container = selection.selectAll('.layer-fill-list').data([0]);
+           container.enter().append('ul').attr('class', 'layer-list layer-fill-list').merge(container).call(drawListItems, context.map().areaFillOptions, 'radio', 'area_fill', setFill, isActiveFill);
+           var container2 = selection.selectAll('.layer-visual-diff-list').data([0]);
+           container2.enter().append('ul').attr('class', 'layer-list layer-visual-diff-list').merge(container2).call(drawListItems, ['highlight_edits'], 'checkbox', 'visual_diff', toggleHighlightEdited, function () {
+             return context.surface().classed('highlight-edited');
+           });
+         }
 
-           if (value) {
-             syncTags.wikipedia = context.cleanTagValue(language()[2] + ':' + value);
-           } else {
-             syncTags.wikipedia = undefined;
-           }
+         function drawListItems(selection, data, type, name, change, active) {
+           var items = selection.selectAll('li').data(data); // Exit
 
-           dispatch$1.call('change', this, syncTags);
-           if (skipWikidata || !value || !language()[2]) return; // attempt asynchronous update of wikidata tag..
+           items.exit().remove(); // Enter
 
-           var initGraph = context.graph();
-           var initEntityIDs = _entityIDs;
-           wikidata.itemsByTitle(language()[2], value, function (err, data) {
-             if (err || !data || !Object.keys(data).length) return; // If graph has changed, we can't apply this update.
+           var enter = items.enter().append('li').call(uiTooltip().title(function (d) {
+             return _t.html(name + '.' + d + '.tooltip');
+           }).keys(function (d) {
+             var key = d === 'wireframe' ? _t('area_fill.wireframe.key') : null;
+             if (d === 'highlight_edits') key = _t('map_data.highlight_edits.key');
+             return key ? [key] : null;
+           }).placement('top'));
+           var label = enter.append('label');
+           label.append('input').attr('type', type).attr('name', name).on('change', change);
+           label.append('span').html(function (d) {
+             return _t.html(name + '.' + d + '.description');
+           }); // Update
 
-             if (context.graph() !== initGraph) return;
-             var qids = Object.keys(data);
-             var value = qids && qids.find(function (id) {
-               return id.match(/^Q\d+$/);
-             });
-             var actions = initEntityIDs.map(function (entityID) {
-               var entity = context.entity(entityID).tags;
-               var currTags = Object.assign({}, entity); // shallow copy
+           items = items.merge(enter);
+           items.classed('active', active).selectAll('input').property('checked', active).property('indeterminate', false);
+         }
 
-               if (currTags.wikidata !== value) {
-                 currTags.wikidata = value;
-                 return actionChangeTags(entityID, currTags);
-               }
+         function isActiveFill(d) {
+           return context.map().activeAreaFill() === d;
+         }
 
-               return null;
-             }).filter(Boolean);
-             if (!actions.length) return; // Coalesce the update of wikidata tag into the previous tag change
+         function toggleHighlightEdited(d3_event) {
+           d3_event.preventDefault();
+           context.map().toggleHighlightEdited();
+         }
 
-             context.overwrite(function actionUpdateWikidataTags(graph) {
-               actions.forEach(function (action) {
-                 graph = action(graph);
-               });
-               return graph;
-             }, context.history().undoAnnotation()); // do not dispatch.call('change') here, because entity_editor
-             // changeTags() is not intended to be called asynchronously
-           });
+         function setFill(d3_event, d) {
+           context.map().activeAreaFill(d);
          }
 
-         wiki.tags = function (tags) {
-           _tags = tags;
-           updateForTags(tags);
-         };
+         context.map().on('changeHighlighting.ui_style, changeAreaFill.ui_style', section.reRender);
+         return section;
+       }
 
-         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`
+       function uiSectionPhotoOverlays(context) {
+         var layers = context.layers();
+         var section = uiSection('photo-overlays', context).label(_t.html('photo_overlays.title')).disclosureContent(renderDisclosureContent).expandedByDefault(false);
 
-           var m = value.match(/([^:]+):([^#]+)(?:#(.+))?/);
-           var tagLang = m && m[1];
-           var tagArticleTitle = m && m[2];
-           var anchor = m && m[3];
+         function renderDisclosureContent(selection) {
+           var container = selection.selectAll('.photo-overlay-container').data([0]);
+           container.enter().append('div').attr('class', 'photo-overlay-container').merge(container).call(drawPhotoItems).call(drawPhotoTypeItems).call(drawDateFilter).call(drawUsernameFilter);
+         }
 
-           var tagLangInfo = tagLang && _dataWikipedia.find(function (d) {
-             return tagLang === d[2];
-           }); // value in correct format
+         function drawPhotoItems(selection) {
+           var photoKeys = context.photos().overlayLayerIDs();
+           var photoLayers = layers.all().filter(function (obj) {
+             return photoKeys.indexOf(obj.id) !== -1;
+           });
+           var data = photoLayers.filter(function (obj) {
+             return obj.layer.supported();
+           });
 
+           function layerSupported(d) {
+             return d.layer && d.layer.supported();
+           }
 
-           if (tagLangInfo) {
-             var nativeLangName = tagLangInfo[1];
-             utilGetSetValue(_langInput, nativeLangName);
-             utilGetSetValue(_titleInput, tagArticleTitle + (anchor ? '#' + anchor : ''));
+           function layerEnabled(d) {
+             return layerSupported(d) && d.layer.enabled();
+           }
 
-             if (anchor) {
-               try {
-                 // Best-effort `anchorencode:` implementation
-                 anchor = encodeURIComponent(anchor.replace(/ /g, '_')).replace(/%/g, '.');
-               } catch (e) {
-                 anchor = anchor.replace(/ /g, '_');
-               }
+           var ul = selection.selectAll('.layer-list-photos').data([0]);
+           ul = ul.enter().append('ul').attr('class', 'layer-list layer-list-photos').merge(ul);
+           var li = ul.selectAll('.list-item-photos').data(data);
+           li.exit().remove();
+           var liEnter = li.enter().append('li').attr('class', function (d) {
+             var classes = 'list-item-photos list-item-' + d.id;
+
+             if (d.id === 'mapillary-signs' || d.id === 'mapillary-map-features') {
+               classes += ' indented';
              }
 
-             _wikiURL = 'https://' + tagLang + '.wikipedia.org/wiki/' + tagArticleTitle.replace(/ /g, '_') + (anchor ? '#' + anchor : ''); // unrecognized value format
-           } else {
-             utilGetSetValue(_titleInput, value);
+             return classes;
+           });
+           var labelEnter = liEnter.append('label').each(function (d) {
+             var titleID;
+             if (d.id === 'mapillary-signs') titleID = 'mapillary.signs.tooltip';else if (d.id === 'mapillary') titleID = 'mapillary_images.tooltip';else if (d.id === 'openstreetcam') titleID = 'openstreetcam_images.tooltip';else titleID = d.id.replace(/-/g, '_') + '.tooltip';
+             select(this).call(uiTooltip().title(_t.html(titleID)).placement('top'));
+           });
+           labelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event, d) {
+             toggleLayer(d.id);
+           });
+           labelEnter.append('span').html(function (d) {
+             var id = d.id;
+             if (id === 'mapillary-signs') id = 'photo_overlays.traffic_signs';
+             return _t.html(id.replace(/-/g, '_') + '.title');
+           }); // Update
 
-             if (value && value !== '') {
-               utilGetSetValue(_langInput, '');
-               var defaultLangInfo = defaultLanguageInfo();
-               _wikiURL = "https://".concat(defaultLangInfo[2], ".wikipedia.org/w/index.php?fulltext=1&search=").concat(value);
-             } else {
-               var shownOrDefaultLangInfo = language(true
-               /* skipEnglishFallback */
-               );
-               utilGetSetValue(_langInput, shownOrDefaultLangInfo[1]);
-               _wikiURL = '';
-             }
-           }
+           li.merge(liEnter).classed('active', layerEnabled).selectAll('input').property('checked', layerEnabled);
          }
 
-         wiki.entityIDs = function (val) {
-           if (!_arguments.length) return _entityIDs;
-           _entityIDs = val;
-           return wiki;
-         };
-
-         wiki.focus = function () {
-           _titleInput.node().focus();
-         };
+         function drawPhotoTypeItems(selection) {
+           var data = context.photos().allPhotoTypes();
 
-         return utilRebind(wiki, dispatch$1, 'on');
-       }
-       uiFieldWikipedia.supportsMultiselection = false;
+           function typeEnabled(d) {
+             return context.photos().showsPhotoType(d);
+           }
 
-       var uiFields = {
-         access: uiFieldAccess,
-         address: uiFieldAddress,
-         check: uiFieldCheck,
-         combo: uiFieldCombo,
-         cycleway: uiFieldCycleway,
-         defaultCheck: uiFieldCheck,
-         email: uiFieldText,
-         identifier: uiFieldText,
-         lanes: uiFieldLanes,
-         localized: uiFieldLocalized,
-         maxspeed: uiFieldMaxspeed,
-         manyCombo: uiFieldCombo,
-         multiCombo: uiFieldCombo,
-         networkCombo: uiFieldCombo,
-         number: uiFieldText,
-         onewayCheck: uiFieldCheck,
-         radio: uiFieldRadio,
-         restrictions: uiFieldRestrictions,
-         semiCombo: uiFieldCombo,
-         structureRadio: uiFieldRadio,
-         tel: uiFieldText,
-         text: uiFieldText,
-         textarea: uiFieldTextarea,
-         typeCombo: uiFieldCombo,
-         url: uiFieldText,
-         wikidata: uiFieldWikidata,
-         wikipedia: uiFieldWikipedia
-       };
+           var ul = selection.selectAll('.layer-list-photo-types').data([0]);
+           ul.exit().remove();
+           ul = ul.enter().append('ul').attr('class', 'layer-list layer-list-photo-types').merge(ul);
+           var li = ul.selectAll('.list-item-photo-types').data(context.photos().shouldFilterByPhotoType() ? data : []);
+           li.exit().remove();
+           var liEnter = li.enter().append('li').attr('class', function (d) {
+             return 'list-item-photo-types list-item-' + d;
+           });
+           var labelEnter = liEnter.append('label').each(function (d) {
+             select(this).call(uiTooltip().title(_t.html('photo_overlays.photo_type.' + d + '.tooltip')).placement('top'));
+           });
+           labelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event, d) {
+             context.photos().togglePhotoType(d);
+           });
+           labelEnter.append('span').html(function (d) {
+             return _t.html('photo_overlays.photo_type.' + d + '.title');
+           }); // Update
 
-       function uiField(context, presetField, entityIDs, options) {
-         options = Object.assign({
-           show: true,
-           wrap: true,
-           remove: true,
-           revert: true,
-           info: true
-         }, options);
-         var dispatch$1 = dispatch('change', 'revert');
-         var field = Object.assign({}, presetField); // shallow copy
+           li.merge(liEnter).classed('active', typeEnabled).selectAll('input').property('checked', typeEnabled);
+         }
 
-         field.domId = utilUniqueDomId('form-field-' + field.safeid);
-         var _show = options.show;
-         var _state = '';
-         var _tags = {};
-         var _locked = false;
+         function drawDateFilter(selection) {
+           var data = context.photos().dateFilters();
 
-         var _lockedTip = uiTooltip().title(_t.html('inspector.lock.suggestion', {
-           label: field.label
-         })).placement('bottom');
+           function filterEnabled(d) {
+             return context.photos().dateFilterValue(d);
+           }
 
-         field.keys = field.keys || [field.key]; // only create the fields that are actually being shown
+           var ul = selection.selectAll('.layer-list-date-filter').data([0]);
+           ul.exit().remove();
+           ul = ul.enter().append('ul').attr('class', 'layer-list layer-list-date-filter').merge(ul);
+           var li = ul.selectAll('.list-item-date-filter').data(context.photos().shouldFilterByDate() ? data : []);
+           li.exit().remove();
+           var liEnter = li.enter().append('li').attr('class', 'list-item-date-filter');
+           var labelEnter = liEnter.append('label').each(function (d) {
+             select(this).call(uiTooltip().title(_t.html('photo_overlays.date_filter.' + d + '.tooltip')).placement('top'));
+           });
+           labelEnter.append('span').html(function (d) {
+             return _t.html('photo_overlays.date_filter.' + d + '.title');
+           });
+           labelEnter.append('input').attr('type', 'date').attr('class', 'list-item-input').attr('placeholder', _t('units.year_month_day')).call(utilNoAuto).each(function (d) {
+             utilGetSetValue(select(this), context.photos().dateFilterValue(d) || '');
+           }).on('change', function (d3_event, d) {
+             var value = utilGetSetValue(select(this)).trim();
+             context.photos().setDateFilter(d, value, true); // reload the displayed dates
 
-         if (_show && !field.impl) {
-           createField();
-         } // Creates the field.. This is done lazily,
-         // once we know that the field will be shown.
+             li.selectAll('input').each(function (d) {
+               utilGetSetValue(select(this), context.photos().dateFilterValue(d) || '');
+             });
+           });
+           li = li.merge(liEnter).classed('active', filterEnabled);
+         }
 
+         function drawUsernameFilter(selection) {
+           function filterEnabled() {
+             return context.photos().usernames();
+           }
 
-         function createField() {
-           field.impl = uiFields[field.type](field, context).on('change', function (t, onInput) {
-             dispatch$1.call('change', field, t, onInput);
+           var ul = selection.selectAll('.layer-list-username-filter').data([0]);
+           ul.exit().remove();
+           ul = ul.enter().append('ul').attr('class', 'layer-list layer-list-username-filter').merge(ul);
+           var li = ul.selectAll('.list-item-username-filter').data(context.photos().shouldFilterByUsername() ? ['username-filter'] : []);
+           li.exit().remove();
+           var liEnter = li.enter().append('li').attr('class', 'list-item-username-filter');
+           var labelEnter = liEnter.append('label').each(function () {
+             select(this).call(uiTooltip().title(_t.html('photo_overlays.username_filter.tooltip')).placement('top'));
            });
+           labelEnter.append('span').html(_t.html('photo_overlays.username_filter.title'));
+           labelEnter.append('input').attr('type', 'text').attr('class', 'list-item-input').call(utilNoAuto).property('value', usernameValue).on('change', function () {
+             var value = select(this).property('value');
+             context.photos().setUsernameFilter(value, true);
+             select(this).property('value', usernameValue);
+           });
+           li.merge(liEnter).classed('active', filterEnabled);
 
-           if (entityIDs) {
-             field.entityIDs = entityIDs; // if this field cares about the entities, pass them along
-
-             if (field.impl.entityIDs) {
-               field.impl.entityIDs(entityIDs);
-             }
+           function usernameValue() {
+             var usernames = context.photos().usernames();
+             if (usernames) return usernames.join('; ');
+             return usernames;
            }
          }
 
-         function isModified() {
-           if (!entityIDs || !entityIDs.length) return false;
-           return entityIDs.some(function (entityID) {
-             var original = context.graph().base().entities[entityID];
-             var latest = context.graph().entity(entityID);
-             return field.keys.some(function (key) {
-               return original ? latest.tags[key] !== original.tags[key] : latest.tags[key];
-             });
-           });
+         function toggleLayer(which) {
+           setLayer(which, !showsLayer(which));
          }
 
-         function tagsContainFieldKey() {
-           return field.keys.some(function (key) {
-             if (field.type === 'multiCombo') {
-               for (var tagKey in _tags) {
-                 if (tagKey.indexOf(key) === 0) {
-                   return true;
-                 }
-               }
+         function showsLayer(which) {
+           var layer = layers.layer(which);
 
-               return false;
-             }
+           if (layer) {
+             return layer.enabled();
+           }
 
-             return _tags[key] !== undefined;
-           });
+           return false;
          }
 
-         function revert(d3_event, d) {
-           d3_event.stopPropagation();
-           d3_event.preventDefault();
-           if (!entityIDs || _locked) return;
-           dispatch$1.call('revert', d, d.keys);
+         function setLayer(which, enabled) {
+           var layer = layers.layer(which);
+
+           if (layer) {
+             layer.enabled(enabled);
+           }
          }
 
-         function remove(d3_event, d) {
-           d3_event.stopPropagation();
-           d3_event.preventDefault();
-           if (_locked) return;
-           var t = {};
-           d.keys.forEach(function (key) {
-             t[key] = undefined;
+         context.layers().on('change.uiSectionPhotoOverlays', section.reRender);
+         context.photos().on('change.uiSectionPhotoOverlays', section.reRender);
+         return section;
+       }
+
+       function uiPaneMapData(context) {
+         var mapDataPane = uiPane('map-data', context).key(_t('map_data.key')).label(_t.html('map_data.title')).description(_t.html('map_data.description')).iconName('iD-icon-data').sections([uiSectionDataLayers(context), uiSectionPhotoOverlays(context), uiSectionMapStyleOptions(context), uiSectionMapFeatures(context)]);
+         return mapDataPane;
+       }
+
+       function uiSectionPrivacy(context) {
+         var section = uiSection('preferences-third-party', context).label(_t.html('preferences.privacy.title')).disclosureContent(renderDisclosureContent);
+
+         var _showThirdPartyIcons = corePreferences('preferences.privacy.thirdpartyicons') || 'true';
+
+         function renderDisclosureContent(selection) {
+           // enter
+           var privacyOptionsListEnter = selection.selectAll('.privacy-options-list').data([0]).enter().append('ul').attr('class', 'layer-list privacy-options-list');
+           var thirdPartyIconsEnter = privacyOptionsListEnter.append('li').attr('class', 'privacy-third-party-icons-item').append('label').call(uiTooltip().title(_t.html('preferences.privacy.third_party_icons.tooltip')).placement('bottom'));
+           thirdPartyIconsEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event) {
+             d3_event.preventDefault();
+             _showThirdPartyIcons = _showThirdPartyIcons === 'true' ? 'false' : 'true';
+             corePreferences('preferences.privacy.thirdpartyicons', _showThirdPartyIcons);
+             update();
            });
-           dispatch$1.call('change', d, t);
+           thirdPartyIconsEnter.append('span').html(_t.html('preferences.privacy.third_party_icons.description')); // Privacy Policy link
+
+           selection.selectAll('.privacy-link').data([0]).enter().append('div').attr('class', 'privacy-link').append('a').attr('target', '_blank').call(svgIcon('#iD-icon-out-link', 'inline')).attr('href', 'https://github.com/openstreetmap/iD/blob/release/PRIVACY.md').append('span').html(_t.html('preferences.privacy.privacy_link'));
+           update();
+
+           function update() {
+             selection.selectAll('.privacy-third-party-icons-item').classed('active', _showThirdPartyIcons === 'true').select('input').property('checked', _showThirdPartyIcons === 'true');
+           }
          }
 
-         field.render = function (selection) {
-           var container = selection.selectAll('.form-field').data([field]); // Enter
+         return section;
+       }
 
-           var enter = container.enter().append('div').attr('class', function (d) {
-             return 'form-field form-field-' + d.safeid;
-           }).classed('nowrap', !options.wrap);
+       function uiPanePreferences(context) {
+         var preferencesPane = uiPane('preferences', context).key(_t('preferences.key')).label(_t.html('preferences.title')).description(_t.html('preferences.description')).iconName('fas-user-cog').sections([uiSectionPrivacy(context)]);
+         return preferencesPane;
+       }
 
-           if (options.wrap) {
-             var labelEnter = enter.append('label').attr('class', 'field-label').attr('for', function (d) {
-               return d.domId;
+       function uiInit(context) {
+         var _initCounter = 0;
+         var _needWidth = {};
+
+         var _lastPointerType;
+
+         function render(container) {
+           container.on('click.ui', function (d3_event) {
+             // we're only concerned with the primary mouse button
+             if (d3_event.button !== 0) return;
+             if (!d3_event.composedPath) return; // some targets have default click events we don't want to override
+
+             var isOkayTarget = d3_event.composedPath().some(function (node) {
+               // we only care about element nodes
+               return node.nodeType === 1 && ( // clicking <input> focuses it and/or changes a value
+               node.nodeName === 'INPUT' || // clicking <label> affects its <input> by default
+               node.nodeName === 'LABEL' || // clicking <a> opens a hyperlink by default
+               node.nodeName === 'A');
              });
-             var textEnter = labelEnter.append('span').attr('class', 'label-text');
-             textEnter.append('span').attr('class', 'label-textvalue').html(function (d) {
-               return d.label();
+             if (isOkayTarget) return; // disable double-tap-to-zoom on touchscreens
+
+             d3_event.preventDefault();
+           });
+           var detected = utilDetect(); // only WebKit supports gesture events
+
+           if ('GestureEvent' in window && // Listening for gesture events on iOS 13.4+ breaks double-tapping,
+           // but we only need to do this on desktop Safari anyway. – #7694
+           !detected.isMobileWebKit) {
+             // On iOS we disable pinch-to-zoom of the UI via the `touch-action`
+             // CSS property, but on desktop Safari we need to manually cancel the
+             // default gesture events.
+             container.on('gesturestart.ui gesturechange.ui gestureend.ui', function (d3_event) {
+               // disable pinch-to-zoom of the UI via multitouch trackpads on macOS Safari
+               d3_event.preventDefault();
              });
-             textEnter.append('span').attr('class', 'label-textannotation');
+           }
 
-             if (options.remove) {
-               labelEnter.append('button').attr('class', 'remove-icon').attr('title', _t('icons.remove')).call(svgIcon('#iD-operation-delete'));
-             }
+           if ('PointerEvent' in window) {
+             select(window).on('pointerdown.ui pointerup.ui', function (d3_event) {
+               var pointerType = d3_event.pointerType || 'mouse';
 
-             if (options.revert) {
-               labelEnter.append('button').attr('class', 'modified-icon').attr('title', _t('icons.undo')).call(svgIcon(_mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-redo' : '#iD-icon-undo'));
-             }
-           } // Update
+               if (_lastPointerType !== pointerType) {
+                 _lastPointerType = pointerType;
+                 container.attr('pointer', pointerType);
+               }
+             }, true);
+           } else {
+             _lastPointerType = 'mouse';
+             container.attr('pointer', 'mouse');
+           }
 
+           container.attr('lang', _mainLocalizer.localeCode()).attr('dir', _mainLocalizer.textDirection()); // setup fullscreen keybindings (no button shown at this time)
 
-           container = container.merge(enter);
-           container.select('.field-label > .remove-icon') // propagate bound data
-           .on('click', remove);
-           container.select('.field-label > .modified-icon') // propagate bound data
-           .on('click', revert);
-           container.each(function (d) {
-             var selection = select(this);
+           container.call(uiFullScreen(context));
+           var map = context.map();
+           map.redrawEnable(false); // don't draw until we've set zoom/lat/long
 
-             if (!d.impl) {
-               createField();
-             }
+           map.on('hitMinZoom.ui', function () {
+             ui.flash.iconName('#iD-icon-no').label(_t.html('cannot_zoom'))();
+           });
+           container.append('svg').attr('id', 'ideditor-defs').call(ui.svgDefs);
+           container.append('div').attr('class', 'sidebar').call(ui.sidebar);
+           var content = container.append('div').attr('class', 'main-content active'); // Top toolbar
 
-             var reference, help; // instantiate field help
+           content.append('div').attr('class', 'top-toolbar-wrap').append('div').attr('class', 'top-toolbar fillD').call(uiTopToolbar(context));
+           content.append('div').attr('class', 'main-map').attr('dir', 'ltr').call(map);
+           var overMap = content.append('div').attr('class', 'over-map'); // HACK: Mobile Safari 14 likes to select anything selectable when long-
+           // pressing, even if it's not targeted. This conflicts with long-pressing
+           // to show the edit menu. We add a selectable offscreen element as the first
+           // child to trick Safari into not showing the selection UI.
 
-             if (options.wrap && field.type === 'restrictions') {
-               help = uiFieldHelp(context, 'restrictions');
-             } // instantiate tag reference
+           overMap.append('div').attr('class', 'select-trap').text('t');
+           overMap.call(uiMapInMap(context)).call(uiNotice(context));
+           overMap.append('div').attr('class', 'spinner').call(uiSpinner(context)); // Map controls
 
+           var controls = overMap.append('div').attr('class', 'map-controls');
+           controls.append('div').attr('class', 'map-control zoombuttons').call(uiZoom(context));
+           controls.append('div').attr('class', 'map-control zoom-to-selection-control').call(uiZoomToSelection(context));
+           controls.append('div').attr('class', 'map-control geolocate-control').call(uiGeolocate(context)); // Add panes
+           // This should happen after map is initialized, as some require surface()
 
-             if (options.wrap && options.info) {
-               var referenceKey = d.key || '';
+           var panes = overMap.append('div').attr('class', 'map-panes');
+           var uiPanes = [uiPaneBackground(context), uiPaneMapData(context), uiPaneIssues(context), uiPanePreferences(context), uiPaneHelp(context)];
+           uiPanes.forEach(function (pane) {
+             controls.append('div').attr('class', 'map-control map-pane-control ' + pane.id + '-control').call(pane.renderToggleButton);
+             panes.call(pane.renderPane);
+           });
+           ui.info = uiInfo(context);
+           overMap.call(ui.info);
+           overMap.append('div').attr('class', 'photoviewer').classed('al', true) // 'al'=left,  'ar'=right
+           .classed('hide', true).call(ui.photoviewer);
+           overMap.append('div').attr('class', 'attribution-wrap').attr('dir', 'ltr').call(uiAttribution(context)); // Add footer
 
-               if (d.type === 'multiCombo') {
-                 // lookup key without the trailing ':'
-                 referenceKey = referenceKey.replace(/:$/, '');
-               }
+           var about = content.append('div').attr('class', 'map-footer');
+           about.append('div').attr('class', 'api-status').call(uiStatus(context));
+           var footer = about.append('div').attr('class', 'map-footer-bar fillD');
+           footer.append('div').attr('class', 'flash-wrap footer-hide');
+           var footerWrap = footer.append('div').attr('class', 'main-footer-wrap footer-show');
+           footerWrap.append('div').attr('class', 'scale-block').call(uiScale(context));
+           var aboutList = footerWrap.append('div').attr('class', 'info-block').append('ul').attr('class', 'map-footer-list');
+           aboutList.append('li').attr('class', 'user-list').call(uiContributors(context));
+           var apiConnections = context.apiConnections();
 
-               reference = uiTagReference(d.reference || {
-                 key: referenceKey
-               });
+           if (apiConnections && apiConnections.length > 1) {
+             aboutList.append('li').attr('class', 'source-switch').call(uiSourceSwitch(context).keys(apiConnections));
+           }
 
-               if (_state === 'hover') {
-                 reference.showing(false);
-               }
-             }
+           aboutList.append('li').attr('class', 'issues-info').call(uiIssuesInfo(context));
+           aboutList.append('li').attr('class', 'feature-warning').call(uiFeatureInfo(context));
+           var issueLinks = aboutList.append('li');
+           issueLinks.append('a').attr('target', '_blank').attr('href', 'https://github.com/openstreetmap/iD/issues').call(svgIcon('#iD-icon-bug', 'light')).call(uiTooltip().title(_t.html('report_a_bug')).placement('top'));
+           issueLinks.append('a').attr('target', '_blank').attr('href', 'https://github.com/openstreetmap/iD/blob/develop/CONTRIBUTING.md#translating').call(svgIcon('#iD-icon-translate', 'light')).call(uiTooltip().title(_t.html('help_translate')).placement('top'));
+           aboutList.append('li').attr('class', 'version').call(uiVersion(context));
 
-             selection.call(d.impl); // add field help components
+           if (!context.embed()) {
+             aboutList.call(uiAccount(context));
+           } // Setup map dimensions and move map to initial center/zoom.
+           // This should happen after .main-content and toolbars exist.
 
-             if (help) {
-               selection.call(help.body).select('.field-label').call(help.button);
-             } // add tag reference components
 
+           ui.onResize();
+           map.redrawEnable(true);
+           ui.hash = behaviorHash(context);
+           ui.hash();
 
-             if (reference) {
-               selection.call(reference.body).select('.field-label').call(reference.button);
-             }
+           if (!ui.hash.hadHash) {
+             map.centerZoom([0, 0], 2);
+           } // Bind events
 
-             d.impl.tags(_tags);
+
+           window.onbeforeunload = function () {
+             return context.save();
+           };
+
+           window.onunload = function () {
+             context.history().unlock();
+           };
+
+           select(window).on('resize.editor', function () {
+             ui.onResize();
            });
-           container.classed('locked', _locked).classed('modified', isModified()).classed('present', tagsContainFieldKey()); // show a tip and lock icon if the field is locked
+           var panPixels = 80;
+           context.keybinding().on('⌫', function (d3_event) {
+             d3_event.preventDefault();
+           }).on([_t('sidebar.key'), '`', '²', '@'], ui.sidebar.toggle) // #5663, #6864 - common QWERTY, AZERTY
+           .on('←', pan([panPixels, 0])).on('↑', pan([0, panPixels])).on('→', pan([-panPixels, 0])).on('↓', pan([0, -panPixels])).on(uiCmd('⌥←'), pan([map.dimensions()[0], 0])).on(uiCmd('⌥↑'), pan([0, map.dimensions()[1]])).on(uiCmd('⌥→'), pan([-map.dimensions()[0], 0])).on(uiCmd('⌥↓'), pan([0, -map.dimensions()[1]])).on(uiCmd('⌘' + _t('background.key')), function quickSwitch(d3_event) {
+             if (d3_event) {
+               d3_event.stopImmediatePropagation();
+               d3_event.preventDefault();
+             }
 
-           var annotation = container.selectAll('.field-label .label-textannotation');
-           var icon = annotation.selectAll('.icon').data(_locked ? [0] : []);
-           icon.exit().remove();
-           icon.enter().append('svg').attr('class', 'icon').append('use').attr('xlink:href', '#fas-lock');
-           container.call(_locked ? _lockedTip : _lockedTip.destroy);
-         };
+             var previousBackground = context.background().findSource(corePreferences('background-last-used-toggle'));
 
-         field.state = function (val) {
-           if (!arguments.length) return _state;
-           _state = val;
-           return field;
-         };
+             if (previousBackground) {
+               var currentBackground = context.background().baseLayerSource();
+               corePreferences('background-last-used-toggle', currentBackground.id);
+               corePreferences('background-last-used', previousBackground.id);
+               context.background().baseLayerSource(previousBackground);
+             }
+           }).on(_t('area_fill.wireframe.key'), function toggleWireframe(d3_event) {
+             d3_event.preventDefault();
+             d3_event.stopPropagation();
+             context.map().toggleWireframe();
+           }).on(uiCmd('⌥' + _t('area_fill.wireframe.key')), function toggleOsmData(d3_event) {
+             d3_event.preventDefault();
+             d3_event.stopPropagation(); // Don't allow layer changes while drawing - #6584
 
-         field.tags = function (val) {
-           if (!arguments.length) return _tags;
-           _tags = val;
+             var mode = context.mode();
+             if (mode && /^draw/.test(mode.id)) return;
+             var layer = context.layers().layer('osm');
 
-           if (tagsContainFieldKey() && !_show) {
-             // always show a field if it has a value to display
-             _show = true;
+             if (layer) {
+               layer.enabled(!layer.enabled());
 
-             if (!field.impl) {
-               createField();
+               if (!layer.enabled()) {
+                 context.enter(modeBrowse(context));
+               }
+             }
+           }).on(_t('map_data.highlight_edits.key'), function toggleHighlightEdited(d3_event) {
+             d3_event.preventDefault();
+             context.map().toggleHighlightEdited();
+           });
+           context.on('enter.editor', function (entered) {
+             container.classed('mode-' + entered.id, true);
+           }).on('exit.editor', function (exited) {
+             container.classed('mode-' + exited.id, false);
+           });
+           context.enter(modeBrowse(context));
+
+           if (!_initCounter++) {
+             if (!ui.hash.startWalkthrough) {
+               context.container().call(uiSplash(context)).call(uiRestore(context));
              }
+
+             context.container().call(ui.shortcuts);
            }
 
-           return field;
-         };
+           var osm = context.connection();
+           var auth = uiLoading(context).message(_t.html('loading_auth')).blocking(true);
 
-         field.locked = function (val) {
-           if (!arguments.length) return _locked;
-           _locked = val;
-           return field;
-         };
+           if (osm && auth) {
+             osm.on('authLoading.ui', function () {
+               context.container().call(auth);
+             }).on('authDone.ui', function () {
+               auth.close();
+             });
+           }
 
-         field.show = function () {
-           _show = true;
+           _initCounter++;
 
-           if (!field.impl) {
-             createField();
+           if (ui.hash.startWalkthrough) {
+             ui.hash.startWalkthrough = false;
+             context.container().call(uiIntro(context));
            }
 
-           if (field["default"] && field.key && _tags[field.key] !== field["default"]) {
-             var t = {};
-             t[field.key] = field["default"];
-             dispatch$1.call('change', this, t);
+           function pan(d) {
+             return function (d3_event) {
+               if (d3_event.shiftKey) return;
+               if (context.container().select('.combobox').size()) return;
+               d3_event.preventDefault();
+               context.map().pan(d, 100);
+             };
            }
-         }; // A shown field has a visible UI, a non-shown field is in the 'Add field' dropdown
+         }
 
+         var ui = {};
 
-         field.isShown = function () {
-           return _show;
-         }; // An allowed field can appear in the UI or in the 'Add field' dropdown.
-         // A non-allowed field is hidden from the user altogether
+         var _loadPromise; // renders the iD interface into the container node
 
 
-         field.isAllowed = function () {
-           if (entityIDs && entityIDs.length > 1 && uiFields[field.type].supportsMultiselection === false) return false;
-           if (field.geometry && !entityIDs.every(function (entityID) {
-             return field.matchGeometry(context.graph().geometry(entityID));
-           })) return false;
+         ui.ensureLoaded = function () {
+           if (_loadPromise) return _loadPromise;
+           return _loadPromise = Promise.all([// must have strings and presets before loading the UI
+           _mainLocalizer.ensureLoaded(), _mainPresetIndex.ensureLoaded()]).then(function () {
+             if (!context.container().empty()) render(context.container());
+           })["catch"](function (err) {
+             return console.error(err);
+           }); // eslint-disable-line
+         }; // `ui.restart()` will destroy and rebuild the entire iD interface,
+         // for example to switch the locale while iD is running.
 
-           if (field.countryCodes || field.notCountryCodes) {
-             var extent = combinedEntityExtent();
-             if (!extent) return true;
-             var center = extent.center();
-             var countryCode = iso1A2Code(center);
-             if (!countryCode) return false;
-             countryCode = countryCode.toLowerCase();
 
-             if (field.countryCodes && field.countryCodes.indexOf(countryCode) === -1) {
-               return false;
-             }
+         ui.restart = function () {
+           context.keybinding().clear();
+           _loadPromise = null;
+           context.container().selectAll('*').remove();
+           ui.ensureLoaded();
+         };
 
-             if (field.notCountryCodes && field.notCountryCodes.indexOf(countryCode) !== -1) {
-               return false;
-             }
-           }
+         ui.lastPointerType = function () {
+           return _lastPointerType;
+         };
 
-           var prerequisiteTag = field.prerequisiteTag;
+         ui.svgDefs = svgDefs(context);
+         ui.flash = uiFlash(context);
+         ui.sidebar = uiSidebar(context);
+         ui.photoviewer = uiPhotoviewer(context);
+         ui.shortcuts = uiShortcuts(context);
 
-           if (entityIDs && !tagsContainFieldKey() && // ignore tagging prerequisites if a value is already present
-           prerequisiteTag) {
-             if (!entityIDs.every(function (entityID) {
-               var entity = context.graph().entity(entityID);
+         ui.onResize = function (withPan) {
+           var map = context.map(); // Recalc dimensions of map and sidebar.. (`true` = force recalc)
+           // This will call `getBoundingClientRect` and trigger reflow,
+           //  but the values will be cached for later use.
 
-               if (prerequisiteTag.key) {
-                 var value = entity.tags[prerequisiteTag.key];
-                 if (!value) return false;
+           var mapDimensions = utilGetDimensions(context.container().select('.main-content'), true);
+           utilGetDimensions(context.container().select('.sidebar'), true);
 
-                 if (prerequisiteTag.valueNot) {
-                   return prerequisiteTag.valueNot !== value;
-                 }
+           if (withPan !== undefined) {
+             map.redrawEnable(false);
+             map.pan(withPan);
+             map.redrawEnable(true);
+           }
 
-                 if (prerequisiteTag.value) {
-                   return prerequisiteTag.value === value;
-                 }
-               } else if (prerequisiteTag.keyNot) {
-                 if (entity.tags[prerequisiteTag.keyNot]) return false;
-               }
+           map.dimensions(mapDimensions);
+           ui.photoviewer.onMapResize(); // check if header or footer have overflowed
 
-               return true;
-             })) return false;
+           ui.checkOverflow('.top-toolbar');
+           ui.checkOverflow('.map-footer-bar'); // Use outdated code so it works on Explorer
+
+           var resizeWindowEvent = document.createEvent('Event');
+           resizeWindowEvent.initEvent('resizeWindow', true, true);
+           document.dispatchEvent(resizeWindowEvent);
+         }; // Call checkOverflow when resizing or whenever the contents change.
+
+
+         ui.checkOverflow = function (selector, reset) {
+           if (reset) {
+             delete _needWidth[selector];
            }
 
-           return true;
-         };
+           var selection = context.container().select(selector);
+           if (selection.empty()) return;
+           var scrollWidth = selection.property('scrollWidth');
+           var clientWidth = selection.property('clientWidth');
+           var needed = _needWidth[selector] || scrollWidth;
 
-         field.focus = function () {
-           if (field.impl) {
-             field.impl.focus();
+           if (scrollWidth > clientWidth) {
+             // overflow happening
+             selection.classed('narrow', true);
+
+             if (!_needWidth[selector]) {
+               _needWidth[selector] = scrollWidth;
+             }
+           } else if (scrollWidth >= needed) {
+             selection.classed('narrow', false);
            }
          };
 
-         function combinedEntityExtent() {
-           return entityIDs && entityIDs.length && entityIDs.reduce(function (extent, entityID) {
-             var entity = context.graph().entity(entityID);
-             return extent.extend(entity.extent(context.graph()));
-           }, geoExtent());
-         }
+         ui.togglePanes = function (showPane) {
+           var hidePanes = context.container().selectAll('.map-pane.shown');
+           var side = _mainLocalizer.textDirection() === 'ltr' ? 'right' : 'left';
+           hidePanes.classed('shown', false).classed('hide', true);
+           context.container().selectAll('.map-pane-control button').classed('active', false);
 
-         return utilRebind(field, dispatch$1, 'on');
-       }
+           if (showPane) {
+             hidePanes.classed('shown', false).classed('hide', true).style(side, '-500px');
+             context.container().selectAll('.' + showPane.attr('pane') + '-control button').classed('active', true);
+             showPane.classed('shown', true).classed('hide', false);
 
-       function uiFormFields(context) {
-         var moreCombo = uiCombobox(context, 'more-fields').minItems(1);
-         var _fieldsArr = [];
-         var _lastPlaceholder = '';
-         var _state = '';
-         var _klass = '';
+             if (hidePanes.empty()) {
+               showPane.style(side, '-500px').transition().duration(200).style(side, '0px');
+             } else {
+               showPane.style(side, '0px');
+             }
+           } else {
+             hidePanes.classed('shown', true).classed('hide', false).style(side, '0px').transition().duration(200).style(side, '-500px').on('end', function () {
+               select(this).classed('shown', false).classed('hide', true);
+             });
+           }
+         };
 
-         function formFields(selection) {
-           var allowedFields = _fieldsArr.filter(function (field) {
-             return field.isAllowed();
-           });
+         var _editMenu = uiEditMenu(context);
 
-           var shown = allowedFields.filter(function (field) {
-             return field.isShown();
-           });
-           var notShown = allowedFields.filter(function (field) {
-             return !field.isShown();
-           });
-           var container = selection.selectAll('.form-fields-container').data([0]);
-           container = container.enter().append('div').attr('class', 'form-fields-container ' + (_klass || '')).merge(container);
-           var fields = container.selectAll('.wrap-form-field').data(shown, function (d) {
-             return d.id + (d.entityIDs ? d.entityIDs.join() : '');
-           });
-           fields.exit().remove(); // Enter
+         ui.editMenu = function () {
+           return _editMenu;
+         };
 
-           var enter = fields.enter().append('div').attr('class', function (d) {
-             return 'wrap-form-field wrap-form-field-' + d.safeid;
-           }); // Update
+         ui.showEditMenu = function (anchorPoint, triggerType, operations) {
+           // remove any displayed menu
+           ui.closeEditMenu();
+           if (!operations && context.mode().operations) operations = context.mode().operations();
+           if (!operations || !operations.length) return; // disable menu if in wide selection, for example
 
-           fields = fields.merge(enter);
-           fields.order().each(function (d) {
-             select(this).call(d.render);
-           });
-           var titles = [];
-           var moreFields = notShown.map(function (field) {
-             var title = field.title();
-             titles.push(title);
-             var terms = field.terms();
-             if (field.key) terms.push(field.key);
-             if (field.keys) terms = terms.concat(field.keys);
-             return {
-               display: field.label(),
-               value: title,
-               title: title,
-               field: field,
-               terms: terms
-             };
-           });
-           var placeholder = titles.slice(0, 3).join(', ') + (titles.length > 3 ? '…' : '');
-           var more = selection.selectAll('.more-fields').data(_state === 'hover' || moreFields.length === 0 ? [] : [0]);
-           more.exit().remove();
-           var moreEnter = more.enter().append('div').attr('class', 'more-fields').append('label');
-           moreEnter.append('span').html(_t.html('inspector.add_fields'));
-           more = moreEnter.merge(more);
-           var input = more.selectAll('.value').data([0]);
-           input.exit().remove();
-           input = input.enter().append('input').attr('class', 'value').attr('type', 'text').attr('placeholder', placeholder).call(utilNoAuto).merge(input);
-           input.call(utilGetSetValue, '').call(moreCombo.data(moreFields).on('accept', function (d) {
-             if (!d) return; // user entered something that was not matched
+           if (!context.map().editableDataEnabled()) return;
+           var surfaceNode = context.surface().node();
 
-             var field = d.field;
-             field.show();
-             selection.call(formFields); // rerender
+           if (surfaceNode.focus) {
+             // FF doesn't support it
+             // focus the surface or else clicking off the menu may not trigger modeBrowse
+             surfaceNode.focus();
+           }
 
-             field.focus();
-           })); // avoid updating placeholder excessively (triggers style recalc)
+           operations.forEach(function (operation) {
+             if (operation.point) operation.point(anchorPoint);
+           });
 
-           if (_lastPlaceholder !== placeholder) {
-             input.attr('placeholder', placeholder);
-             _lastPlaceholder = placeholder;
-           }
-         }
+           _editMenu.anchorLoc(anchorPoint).triggerType(triggerType).operations(operations); // render the menu
 
-         formFields.fieldsArr = function (val) {
-           if (!arguments.length) return _fieldsArr;
-           _fieldsArr = val || [];
-           return formFields;
-         };
 
-         formFields.state = function (val) {
-           if (!arguments.length) return _state;
-           _state = val;
-           return formFields;
+           context.map().supersurface.call(_editMenu);
          };
 
-         formFields.klass = function (val) {
-           if (!arguments.length) return _klass;
-           _klass = val;
-           return formFields;
+         ui.closeEditMenu = function () {
+           // remove any existing menu no matter how it was added
+           context.map().supersurface.select('.edit-menu').remove();
          };
 
-         return formFields;
-       }
+         var _saveLoading = select(null);
 
-       function uiSectionPresetFields(context) {
-         var section = uiSection('preset-fields', context).label(_t.html('inspector.fields')).disclosureContent(renderDisclosureContent);
-         var dispatch$1 = dispatch('change', 'revert');
-         var formFields = uiFormFields(context);
+         context.uploader().on('saveStarted.ui', function () {
+           _saveLoading = uiLoading(context).message(_t.html('save.uploading')).blocking(true);
+           context.container().call(_saveLoading); // block input during upload
+         }).on('saveEnded.ui', function () {
+           _saveLoading.close();
 
-         var _state;
+           _saveLoading = select(null);
+         });
+         return ui;
+       }
 
-         var _fieldsArr;
+       function coreContext() {
+         var _this = this;
 
-         var _presets = [];
+         var dispatch = dispatch$8('enter', 'exit', 'change');
+         var context = utilRebind({}, dispatch, 'on');
 
-         var _tags;
+         var _deferred = new Set();
 
-         var _entityIDs;
+         context.version = '2.20.0';
+         context.privacyVersion = '20201202'; // iD will alter the hash so cache the parameters intended to setup the session
 
-         function renderDisclosureContent(selection) {
-           if (!_fieldsArr) {
-             var graph = context.graph();
-             var geometries = Object.keys(_entityIDs.reduce(function (geoms, entityID) {
-               geoms[graph.entity(entityID).geometry(graph)] = true;
-               return geoms;
-             }, {}));
-             var presetsManager = _mainPresetIndex;
-             var allFields = [];
-             var allMoreFields = [];
-             var sharedTotalFields;
+         context.initialHashParams = window.location.hash ? utilStringQs(window.location.hash) : {};
+         context.isFirstSession = !corePreferences('sawSplash') && !corePreferences('sawPrivacyVersion');
+         /* Changeset */
+         // An osmChangeset object. Not loaded until needed.
 
-             _presets.forEach(function (preset) {
-               var fields = preset.fields();
-               var moreFields = preset.moreFields();
-               allFields = utilArrayUnion(allFields, fields);
-               allMoreFields = utilArrayUnion(allMoreFields, moreFields);
+         context.changeset = null;
+         var _defaultChangesetComment = context.initialHashParams.comment;
+         var _defaultChangesetSource = context.initialHashParams.source;
+         var _defaultChangesetHashtags = context.initialHashParams.hashtags;
 
-               if (!sharedTotalFields) {
-                 sharedTotalFields = utilArrayUnion(fields, moreFields);
-               } else {
-                 sharedTotalFields = sharedTotalFields.filter(function (field) {
-                   return fields.indexOf(field) !== -1 || moreFields.indexOf(field) !== -1;
-                 });
-               }
-             });
+         context.defaultChangesetComment = function (val) {
+           if (!arguments.length) return _defaultChangesetComment;
+           _defaultChangesetComment = val;
+           return context;
+         };
 
-             var sharedFields = allFields.filter(function (field) {
-               return sharedTotalFields.indexOf(field) !== -1;
-             });
-             var sharedMoreFields = allMoreFields.filter(function (field) {
-               return sharedTotalFields.indexOf(field) !== -1;
-             });
-             _fieldsArr = [];
-             sharedFields.forEach(function (field) {
-               if (field.matchAllGeometry(geometries)) {
-                 _fieldsArr.push(uiField(context, field, _entityIDs));
-               }
-             });
-             var singularEntity = _entityIDs.length === 1 && graph.hasEntity(_entityIDs[0]);
+         context.defaultChangesetSource = function (val) {
+           if (!arguments.length) return _defaultChangesetSource;
+           _defaultChangesetSource = val;
+           return context;
+         };
 
-             if (singularEntity && singularEntity.isHighwayIntersection(graph) && presetsManager.field('restrictions')) {
-               _fieldsArr.push(uiField(context, presetsManager.field('restrictions'), _entityIDs));
-             }
+         context.defaultChangesetHashtags = function (val) {
+           if (!arguments.length) return _defaultChangesetHashtags;
+           _defaultChangesetHashtags = val;
+           return context;
+         };
+         /* Document title */
 
-             var additionalFields = utilArrayUnion(sharedMoreFields, presetsManager.universal());
-             additionalFields.sort(function (field1, field2) {
-               return field1.label().localeCompare(field2.label(), _mainLocalizer.localeCode());
-             });
-             additionalFields.forEach(function (field) {
-               if (sharedFields.indexOf(field) === -1 && field.matchAllGeometry(geometries)) {
-                 _fieldsArr.push(uiField(context, field, _entityIDs, {
-                   show: false
-                 }));
-               }
-             });
+         /* (typically shown as the label for the browser window/tab) */
+         // If true, iD will update the title based on what the user is doing
 
-             _fieldsArr.forEach(function (field) {
-               field.on('change', function (t, onInput) {
-                 dispatch$1.call('change', field, _entityIDs, t, onInput);
-               }).on('revert', function (keys) {
-                 dispatch$1.call('revert', field, keys);
-               });
-             });
-           }
 
-           _fieldsArr.forEach(function (field) {
-             field.state(_state).tags(_tags);
-           });
+         var _setsDocumentTitle = true;
 
-           selection.call(formFields.fieldsArr(_fieldsArr).state(_state).klass('grouped-items-area'));
-           selection.selectAll('.wrap-form-field input').on('keydown', function (d3_event) {
-             // if user presses enter, and combobox is not active, accept edits..
-             if (d3_event.keyCode === 13 && // ↩ Return
-             context.container().select('.combobox').empty()) {
-               context.enter(modeBrowse(context));
-             }
-           });
-         }
+         context.setsDocumentTitle = function (val) {
+           if (!arguments.length) return _setsDocumentTitle;
+           _setsDocumentTitle = val;
+           return context;
+         }; // The part of the title that is always the same
 
-         section.presets = function (val) {
-           if (!arguments.length) return _presets;
 
-           if (!_presets || !val || !utilArrayIdentical(_presets, val)) {
-             _presets = val;
-             _fieldsArr = null;
-           }
+         var _documentTitleBase = document.title;
 
-           return section;
+         context.documentTitleBase = function (val) {
+           if (!arguments.length) return _documentTitleBase;
+           _documentTitleBase = val;
+           return context;
          };
+         /* User interface and keybinding */
 
-         section.state = function (val) {
-           if (!arguments.length) return _state;
-           _state = val;
-           return section;
-         };
 
-         section.tags = function (val) {
-           if (!arguments.length) return _tags;
-           _tags = val; // Don't reset _fieldsArr here.
+         var _ui;
 
-           return section;
+         context.ui = function () {
+           return _ui;
          };
 
-         section.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs;
+         context.lastPointerType = function () {
+           return _ui.lastPointerType();
+         };
 
-           if (!val || !_entityIDs || !utilArrayIdentical(_entityIDs, val)) {
-             _entityIDs = val;
-             _fieldsArr = null;
-           }
+         var _keybinding = utilKeybinding('context');
 
-           return section;
+         context.keybinding = function () {
+           return _keybinding;
          };
 
-         return utilRebind(section, dispatch$1, 'on');
-       }
+         select(document).call(_keybinding);
+         /* Straight accessors. Avoid using these if you can. */
+         // Instantiate the connection here because it doesn't require passing in
+         // `context` and it's needed for pre-init calls like `preauth`
 
-       function uiSectionRawMemberEditor(context) {
-         var section = uiSection('raw-member-editor', context).shouldDisplay(function () {
-           if (!_entityIDs || _entityIDs.length !== 1) return false;
-           var entity = context.hasEntity(_entityIDs[0]);
-           return entity && entity.type === 'relation';
-         }).label(function () {
-           var entity = context.hasEntity(_entityIDs[0]);
-           if (!entity) return '';
-           var gt = entity.members.length > _maxMembers ? '>' : '';
-           var count = gt + entity.members.slice(0, _maxMembers).length;
-           return _t('inspector.title_count', {
-             title: _t.html('inspector.members'),
-             count: count
-           });
-         }).disclosureContent(renderDisclosureContent);
-         var taginfo = services.taginfo;
+         var _connection = services.osm;
 
-         var _entityIDs;
+         var _history;
 
-         var _maxMembers = 1000;
+         var _validator;
 
-         function downloadMember(d3_event, d) {
-           d3_event.preventDefault(); // display the loading indicator
+         var _uploader;
 
-           select(this.parentNode).classed('tag-reference-loading', true);
-           context.loadEntity(d.id, function () {
-             section.reRender();
-           });
-         }
+         context.connection = function () {
+           return _connection;
+         };
 
-         function zoomToMember(d3_event, d) {
-           d3_event.preventDefault();
-           var entity = context.entity(d.id);
-           context.map().zoomToEase(entity); // highlight the feature in case it wasn't previously on-screen
+         context.history = function () {
+           return _history;
+         };
 
-           utilHighlightEntities([d.id], true, context);
-         }
+         context.validator = function () {
+           return _validator;
+         };
 
-         function selectMember(d3_event, d) {
-           d3_event.preventDefault(); // remove the hover-highlight styling
+         context.uploader = function () {
+           return _uploader;
+         };
+         /* Connection */
 
-           utilHighlightEntities([d.id], false, context);
-           var entity = context.entity(d.id);
-           var mapExtent = context.map().extent();
 
-           if (!entity.intersects(mapExtent, context.graph())) {
-             // zoom to the entity if its extent is not visible now
-             context.map().zoomToEase(entity);
+         context.preauth = function (options) {
+           if (_connection) {
+             _connection["switch"](options);
            }
 
-           context.enter(modeSelect(context, [d.id]));
-         }
-
-         function changeRole(d3_event, d) {
-           var oldRole = d.role;
-           var newRole = context.cleanRelationRole(select(this).property('value'));
-
-           if (oldRole !== newRole) {
-             var member = {
-               id: d.id,
-               type: d.type,
-               role: newRole
-             };
-             context.perform(actionChangeMember(d.relation.id, member, d.index), _t('operations.change_role.annotation', {
-               n: 1
-             }));
-             context.validator().validate();
-           }
-         }
+           return context;
+         };
+         /* connection options for source switcher (optional) */
 
-         function deleteMember(d3_event, d) {
-           // remove the hover-highlight styling
-           utilHighlightEntities([d.id], false, context);
-           context.perform(actionDeleteMember(d.relation.id, d.index), _t('operations.delete_member.annotation', {
-             n: 1
-           }));
 
-           if (!context.hasEntity(d.relation.id)) {
-             // Removing the last member will also delete the relation.
-             // If this happens we need to exit the selection mode
-             context.enter(modeBrowse(context));
-           } else {
-             // Changing the mode also runs `validate`, but otherwise we need to
-             // rerun it manually
-             context.validator().validate();
-           }
-         }
+         var _apiConnections;
 
-         function renderDisclosureContent(selection) {
-           var entityID = _entityIDs[0];
-           var memberships = [];
-           var entity = context.entity(entityID);
-           entity.members.slice(0, _maxMembers).forEach(function (member, index) {
-             memberships.push({
-               index: index,
-               id: member.id,
-               type: member.type,
-               role: member.role,
-               relation: entity,
-               member: context.hasEntity(member.id),
-               domId: utilUniqueDomId(entityID + '-member-' + index)
-             });
-           });
-           var list = selection.selectAll('.member-list').data([0]);
-           list = list.enter().append('ul').attr('class', 'member-list').merge(list);
-           var items = list.selectAll('li').data(memberships, function (d) {
-             return osmEntity.key(d.relation) + ',' + d.index + ',' + (d.member ? osmEntity.key(d.member) : 'incomplete');
-           });
-           items.exit().each(unbind).remove();
-           var itemsEnter = items.enter().append('li').attr('class', 'member-row form-field').classed('member-incomplete', function (d) {
-             return !d.member;
-           });
-           itemsEnter.each(function (d) {
-             var item = select(this);
-             var label = item.append('label').attr('class', 'field-label').attr('for', d.domId);
+         context.apiConnections = function (val) {
+           if (!arguments.length) return _apiConnections;
+           _apiConnections = val;
+           return context;
+         }; // A string or array or locale codes to prefer over the browser's settings
 
-             if (d.member) {
-               // highlight the member feature in the map while hovering on the list item
-               item.on('mouseover', function () {
-                 utilHighlightEntities([d.id], true, context);
-               }).on('mouseout', function () {
-                 utilHighlightEntities([d.id], false, context);
-               });
-               var labelLink = label.append('span').attr('class', 'label-text').append('a').attr('href', '#').on('click', selectMember);
-               labelLink.append('span').attr('class', 'member-entity-type').html(function (d) {
-                 var matched = _mainPresetIndex.match(d.member, context.graph());
-                 return matched && matched.name() || utilDisplayType(d.member.id);
-               });
-               labelLink.append('span').attr('class', 'member-entity-name').html(function (d) {
-                 return utilDisplayName(d.member);
-               });
-               label.append('button').attr('title', _t('icons.remove')).attr('class', 'remove member-delete').call(svgIcon('#iD-operation-delete'));
-               label.append('button').attr('class', 'member-zoom').attr('title', _t('icons.zoom_to')).call(svgIcon('#iD-icon-framed-dot', 'monochrome')).on('click', zoomToMember);
-             } else {
-               var labelText = label.append('span').attr('class', 'label-text');
-               labelText.append('span').attr('class', 'member-entity-type').html(_t.html('inspector.' + d.type, {
-                 id: d.id
-               }));
-               labelText.append('span').attr('class', 'member-entity-name').html(_t.html('inspector.incomplete', {
-                 id: d.id
-               }));
-               label.append('button').attr('class', 'member-download').attr('title', _t('icons.download')).call(svgIcon('#iD-icon-load')).on('click', downloadMember);
-             }
-           });
-           var wrapEnter = itemsEnter.append('div').attr('class', 'form-field-input-wrap form-field-input-member');
-           wrapEnter.append('input').attr('class', 'member-role').attr('id', function (d) {
-             return d.domId;
-           }).property('type', 'text').attr('placeholder', _t('inspector.role')).call(utilNoAuto);
 
-           if (taginfo) {
-             wrapEnter.each(bindTypeahead);
-           } // update
+         context.locale = function (locale) {
+           if (!arguments.length) return _mainLocalizer.localeCode();
+           _mainLocalizer.preferredLocaleCodes(locale);
+           return context;
+         };
 
+         function afterLoad(cid, callback) {
+           return function (err, result) {
+             if (err) {
+               // 400 Bad Request, 401 Unauthorized, 403 Forbidden..
+               if (err.status === 400 || err.status === 401 || err.status === 403) {
+                 if (_connection) {
+                   _connection.logout();
+                 }
+               }
 
-           items = items.merge(itemsEnter).order();
-           items.select('input.member-role').property('value', function (d) {
-             return d.role;
-           }).on('blur', changeRole).on('change', changeRole);
-           items.select('button.member-delete').on('click', deleteMember);
-           var dragOrigin, targetIndex;
-           items.call(d3_drag().on('start', function (d3_event) {
-             dragOrigin = {
-               x: d3_event.x,
-               y: d3_event.y
-             };
-             targetIndex = null;
-           }).on('drag', function (d3_event) {
-             var x = d3_event.x - dragOrigin.x,
-                 y = d3_event.y - dragOrigin.y;
-             if (!select(this).classed('dragging') && // don't display drag until dragging beyond a distance threshold
-             Math.sqrt(Math.pow(x, 2) + Math.pow(y, 2)) <= 5) return;
-             var index = items.nodes().indexOf(this);
-             select(this).classed('dragging', true);
-             targetIndex = null;
-             selection.selectAll('li.member-row').style('transform', function (d2, index2) {
-               var node = select(this).node();
+               if (typeof callback === 'function') {
+                 callback(err);
+               }
 
-               if (index === index2) {
-                 return 'translate(' + x + 'px, ' + y + 'px)';
-               } else if (index2 > index && d3_event.y > node.offsetTop) {
-                 if (targetIndex === null || index2 > targetIndex) {
-                   targetIndex = index2;
-                 }
+               return;
+             } else if (_connection && _connection.getConnectionId() !== cid) {
+               if (typeof callback === 'function') {
+                 callback({
+                   message: 'Connection Switched',
+                   status: -1
+                 });
+               }
 
-                 return 'translateY(-100%)';
-               } else if (index2 < index && d3_event.y < node.offsetTop + node.offsetHeight) {
-                 if (targetIndex === null || index2 < targetIndex) {
-                   targetIndex = index2;
-                 }
+               return;
+             } else {
+               _history.merge(result.data, result.extent);
 
-                 return 'translateY(100%)';
+               if (typeof callback === 'function') {
+                 callback(err, result);
                }
 
-               return null;
-             });
-           }).on('end', function (d3_event, d) {
-             if (!select(this).classed('dragging')) return;
-             var index = items.nodes().indexOf(this);
-             select(this).classed('dragging', false);
-             selection.selectAll('li.member-row').style('transform', null);
+               return;
+             }
+           };
+         }
 
-             if (targetIndex !== null) {
-               // dragged to a new position, reorder
-               context.perform(actionMoveMember(d.relation.id, index, targetIndex), _t('operations.reorder_members.annotation'));
-               context.validator().validate();
+         context.loadTiles = function (projection, callback) {
+           var handle = window.requestIdleCallback(function () {
+             _deferred["delete"](handle);
+
+             if (_connection && context.editableDataEnabled()) {
+               var cid = _connection.getConnectionId();
+
+               _connection.loadTiles(projection, afterLoad(cid, callback));
              }
-           }));
+           });
 
-           function bindTypeahead(d) {
-             var row = select(this);
-             var role = row.selectAll('input.member-role');
-             var origValue = role.property('value');
+           _deferred.add(handle);
+         };
 
-             function sort(value, data) {
-               var sameletter = [];
-               var other = [];
+         context.loadTileAtLoc = function (loc, callback) {
+           var handle = window.requestIdleCallback(function () {
+             _deferred["delete"](handle);
 
-               for (var i = 0; i < data.length; i++) {
-                 if (data[i].value.substring(0, value.length) === value) {
-                   sameletter.push(data[i]);
-                 } else {
-                   other.push(data[i]);
-                 }
-               }
+             if (_connection && context.editableDataEnabled()) {
+               var cid = _connection.getConnectionId();
 
-               return sameletter.concat(other);
+               _connection.loadTileAtLoc(loc, afterLoad(cid, callback));
              }
+           });
 
-             role.call(uiCombobox(context, 'member-role').fetcher(function (role, callback) {
-               // The `geometry` param is used in the `taginfo.js` interface for
-               // filtering results, as a key into the `tag_members_fractions`
-               // object.  If we don't know the geometry because the member is
-               // not yet downloaded, it's ok to guess based on type.
-               var geometry;
+           _deferred.add(handle);
+         }; // Download the full entity and its parent relations. The callback may be called multiple times.
 
-               if (d.member) {
-                 geometry = context.graph().geometry(d.member.id);
-               } else if (d.type === 'relation') {
-                 geometry = 'relation';
-               } else if (d.type === 'way') {
-                 geometry = 'line';
-               } else {
-                 geometry = 'point';
-               }
 
-               var rtype = entity.tags.type;
-               taginfo.roles({
-                 debounce: true,
-                 rtype: rtype || '',
-                 geometry: geometry,
-                 query: role
-               }, function (err, data) {
-                 if (!err) callback(sort(role, data));
-               });
-             }).on('cancel', function () {
-               role.property('value', origValue);
-             }));
-           }
+         context.loadEntity = function (entityID, callback) {
+           if (_connection) {
+             var cid = _connection.getConnectionId();
 
-           function unbind() {
-             var row = select(this);
-             row.selectAll('input.member-role').call(uiCombobox.off, context);
-           }
-         }
+             _connection.loadEntity(entityID, afterLoad(cid, callback)); // We need to fetch the parent relations separately.
 
-         section.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs;
-           _entityIDs = val;
-           return section;
+
+             _connection.loadEntityRelations(entityID, afterLoad(cid, callback));
+           }
          };
 
-         return section;
-       }
+         context.zoomToEntity = function (entityID, zoomTo) {
+           // be sure to load the entity even if we're not going to zoom to it
+           context.loadEntity(entityID, function (err, result) {
+             if (err) return;
 
-       function actionDeleteMembers(relationId, memberIndexes) {
-         return function (graph) {
-           // Remove the members in descending order so removals won't shift what members
-           // are at the remaining indexes
-           memberIndexes.sort(function (a, b) {
-             return b - a;
+             if (zoomTo !== false) {
+               var entity = result.data.find(function (e) {
+                 return e.id === entityID;
+               });
+
+               if (entity) {
+                 _map.zoomTo(entity);
+               }
+             }
            });
 
-           for (var i in memberIndexes) {
-             graph = actionDeleteMember(relationId, memberIndexes[i])(graph);
-           }
+           _map.on('drawn.zoomToEntity', function () {
+             if (!context.hasEntity(entityID)) return;
 
-           return graph;
-         };
-       }
+             _map.on('drawn.zoomToEntity', null);
 
-       function uiSectionRawMembershipEditor(context) {
-         var section = uiSection('raw-membership-editor', context).shouldDisplay(function () {
-           return _entityIDs && _entityIDs.length;
-         }).label(function () {
-           var parents = getSharedParentRelations();
-           var gt = parents.length > _maxMemberships ? '>' : '';
-           var count = gt + parents.slice(0, _maxMemberships).length;
-           return _t('inspector.title_count', {
-             title: _t.html('inspector.relations'),
-             count: count
+             context.on('enter.zoomToEntity', null);
+             context.enter(modeSelect(context, [entityID]));
            });
-         }).disclosureContent(renderDisclosureContent);
-         var taginfo = services.taginfo;
-         var nearbyCombo = uiCombobox(context, 'parent-relation').minItems(1).fetcher(fetchNearbyRelations).itemsMouseEnter(function (d3_event, d) {
-           if (d.relation) utilHighlightEntities([d.relation.id], true, context);
-         }).itemsMouseLeave(function (d3_event, d) {
-           if (d.relation) utilHighlightEntities([d.relation.id], false, context);
-         });
-         var _inChange = false;
-         var _entityIDs = [];
 
-         var _showBlank;
-
-         var _maxMemberships = 1000;
+           context.on('enter.zoomToEntity', function () {
+             if (_mode.id !== 'browse') {
+               _map.on('drawn.zoomToEntity', null);
 
-         function getSharedParentRelations() {
-           var parents = [];
+               context.on('enter.zoomToEntity', null);
+             }
+           });
+         };
 
-           for (var i = 0; i < _entityIDs.length; i++) {
-             var entity = context.graph().hasEntity(_entityIDs[i]);
-             if (!entity) continue;
+         var _minEditableZoom = 16;
 
-             if (i === 0) {
-               parents = context.graph().parentRelations(entity);
-             } else {
-               parents = utilArrayIntersection(parents, context.graph().parentRelations(entity));
-             }
+         context.minEditableZoom = function (val) {
+           if (!arguments.length) return _minEditableZoom;
+           _minEditableZoom = val;
 
-             if (!parents.length) break;
+           if (_connection) {
+             _connection.tileZoom(val);
            }
 
-           return parents;
-         }
+           return context;
+         }; // String length limits in Unicode characters, not JavaScript UTF-16 code units
 
-         function getMemberships() {
-           var memberships = [];
-           var relations = getSharedParentRelations().slice(0, _maxMemberships);
-           var isMultiselect = _entityIDs.length > 1;
-           var i, relation, membership, index, member, indexedMember;
 
-           for (i = 0; i < relations.length; i++) {
-             relation = relations[i];
-             membership = {
-               relation: relation,
-               members: [],
-               hash: osmEntity.key(relation)
-             };
+         context.maxCharsForTagKey = function () {
+           return 255;
+         };
 
-             for (index = 0; index < relation.members.length; index++) {
-               member = relation.members[index];
+         context.maxCharsForTagValue = function () {
+           return 255;
+         };
 
-               if (_entityIDs.indexOf(member.id) !== -1) {
-                 indexedMember = Object.assign({}, member, {
-                   index: index
-                 });
-                 membership.members.push(indexedMember);
-                 membership.hash += ',' + index.toString();
+         context.maxCharsForRelationRole = function () {
+           return 255;
+         };
 
-                 if (!isMultiselect) {
-                   // For single selections, list one entry per membership per relation.
-                   // For multiselections, list one entry per relation.
-                   memberships.push(membership);
-                   membership = {
-                     relation: relation,
-                     members: [],
-                     hash: osmEntity.key(relation)
-                   };
-                 }
-               }
-             }
+         function cleanOsmString(val, maxChars) {
+           // be lenient with input
+           if (val === undefined || val === null) {
+             val = '';
+           } else {
+             val = val.toString();
+           } // remove whitespace
 
-             if (membership.members.length) memberships.push(membership);
-           }
 
-           memberships.forEach(function (membership) {
-             membership.domId = utilUniqueDomId('membership-' + membership.relation.id);
-             var roles = [];
-             membership.members.forEach(function (member) {
-               if (roles.indexOf(member.role) === -1) roles.push(member.role);
-             });
-             membership.role = roles.length === 1 ? roles[0] : roles;
-           });
-           return memberships;
-         }
+           val = val.trim(); // use the canonical form of the string
 
-         function selectRelation(d3_event, d) {
-           d3_event.preventDefault(); // remove the hover-highlight styling
+           if (val.normalize) val = val.normalize('NFC'); // trim to the number of allowed characters
 
-           utilHighlightEntities([d.relation.id], false, context);
-           context.enter(modeSelect(context, [d.relation.id]));
+           return utilUnicodeCharsTruncated(val, maxChars);
          }
 
-         function zoomToRelation(d3_event, d) {
-           d3_event.preventDefault();
-           var entity = context.entity(d.relation.id);
-           context.map().zoomToEase(entity); // highlight the relation in case it wasn't previously on-screen
-
-           utilHighlightEntities([d.relation.id], true, context);
-         }
+         context.cleanTagKey = function (val) {
+           return cleanOsmString(val, context.maxCharsForTagKey());
+         };
 
-         function changeRole(d3_event, d) {
-           if (d === 0) return; // called on newrow (shouldn't happen)
+         context.cleanTagValue = function (val) {
+           return cleanOsmString(val, context.maxCharsForTagValue());
+         };
 
-           if (_inChange) return; // avoid accidental recursive call #5731
+         context.cleanRelationRole = function (val) {
+           return cleanOsmString(val, context.maxCharsForRelationRole());
+         };
+         /* History */
 
-           var newRole = context.cleanRelationRole(select(this).property('value'));
-           if (!newRole.trim() && typeof d.role !== 'string') return;
-           var membersToUpdate = d.members.filter(function (member) {
-             return member.role !== newRole;
-           });
 
-           if (membersToUpdate.length) {
-             _inChange = true;
-             context.perform(function actionChangeMemberRoles(graph) {
-               membersToUpdate.forEach(function (member) {
-                 var newMember = Object.assign({}, member, {
-                   role: newRole
-                 });
-                 delete newMember.index;
-                 graph = actionChangeMember(d.relation.id, newMember, member.index)(graph);
-               });
-               return graph;
-             }, _t('operations.change_role.annotation', {
-               n: membersToUpdate.length
-             }));
-             context.validator().validate();
-           }
+         var _inIntro = false;
 
-           _inChange = false;
-         }
+         context.inIntro = function (val) {
+           if (!arguments.length) return _inIntro;
+           _inIntro = val;
+           return context;
+         }; // Immediately save the user's history to localstorage, if possible
+         // This is called someteimes, but also on the `window.onbeforeunload` handler
 
-         function addMembership(d, role) {
-           this.blur(); // avoid keeping focus on the button
 
-           _showBlank = false;
+         context.save = function () {
+           // no history save, no message onbeforeunload
+           if (_inIntro || context.container().select('.modal').size()) return;
+           var canSave;
 
-           function actionAddMembers(relationId, ids, role) {
-             return function (graph) {
-               for (var i in ids) {
-                 var member = {
-                   id: ids[i],
-                   type: graph.entity(ids[i]).type,
-                   role: role
-                 };
-                 graph = actionAddMember(relationId, member)(graph);
-               }
+           if (_mode && _mode.id === 'save') {
+             canSave = false; // Attempt to prevent user from creating duplicate changes - see #5200
 
-               return graph;
-             };
-           }
+             if (services.osm && services.osm.isChangesetInflight()) {
+               _history.clearSaved();
 
-           if (d.relation) {
-             context.perform(actionAddMembers(d.relation.id, _entityIDs, role), _t('operations.add_member.annotation', {
-               n: _entityIDs.length
-             }));
-             context.validator().validate();
+               return;
+             }
            } else {
-             var relation = osmRelation();
-             context.perform(actionAddEntity(relation), actionAddMembers(relation.id, _entityIDs, role), _t('operations.add.annotation.relation')); // changing the mode also runs `validate`
+             canSave = context.selectedIDs().every(function (id) {
+               var entity = context.hasEntity(id);
+               return entity && !entity.isDegenerate();
+             });
+           }
 
-             context.enter(modeSelect(context, [relation.id]).newFeature(true));
+           if (canSave) {
+             _history.save();
            }
-         }
 
-         function deleteMembership(d3_event, d) {
-           this.blur(); // avoid keeping focus on the button
+           if (_history.hasChanges()) {
+             return _t('save.unsaved_changes');
+           }
+         }; // Debounce save, since it's a synchronous localStorage write,
+         // and history changes can happen frequently (e.g. when dragging).
 
-           if (d === 0) return; // called on newrow (shouldn't happen)
-           // remove the hover-highlight styling
 
-           utilHighlightEntities([d.relation.id], false, context);
-           var indexes = d.members.map(function (member) {
-             return member.index;
-           });
-           context.perform(actionDeleteMembers(d.relation.id, indexes), _t('operations.delete_member.annotation', {
-             n: _entityIDs.length
-           }));
-           context.validator().validate();
-         }
+         context.debouncedSave = debounce(context.save, 350);
 
-         function fetchNearbyRelations(q, callback) {
-           var newRelation = {
-             relation: null,
-             value: _t('inspector.new_relation'),
-             display: _t.html('inspector.new_relation')
+         function withDebouncedSave(fn) {
+           return function () {
+             var result = fn.apply(_history, arguments);
+             context.debouncedSave();
+             return result;
            };
-           var entityID = _entityIDs[0];
-           var result = [];
-           var graph = context.graph();
-
-           function baseDisplayLabel(entity) {
-             var matched = _mainPresetIndex.match(entity, graph);
-             var presetName = matched && matched.name() || _t('inspector.relation');
-             var entityName = utilDisplayName(entity) || '';
-             return presetName + ' ' + entityName;
-           }
+         }
+         /* Graph */
 
-           var explicitRelation = q && context.hasEntity(q.toLowerCase());
 
-           if (explicitRelation && explicitRelation.type === 'relation' && explicitRelation.id !== entityID) {
-             // loaded relation is specified explicitly, only show that
-             result.push({
-               relation: explicitRelation,
-               value: baseDisplayLabel(explicitRelation) + ' ' + explicitRelation.id
-             });
-           } else {
-             context.history().intersects(context.map().extent()).forEach(function (entity) {
-               if (entity.type !== 'relation' || entity.id === entityID) return;
-               var value = baseDisplayLabel(entity);
-               if (q && (value + ' ' + entity.id).toLowerCase().indexOf(q.toLowerCase()) === -1) return;
-               result.push({
-                 relation: entity,
-                 value: value
-               });
-             });
-             result.sort(function (a, b) {
-               return osmRelation.creationOrder(a.relation, b.relation);
-             }); // Dedupe identical names by appending relation id - see #2891
+         context.hasEntity = function (id) {
+           return _history.graph().hasEntity(id);
+         };
 
-             var dupeGroups = Object.values(utilArrayGroupBy(result, 'value')).filter(function (v) {
-               return v.length > 1;
-             });
-             dupeGroups.forEach(function (group) {
-               group.forEach(function (obj) {
-                 obj.value += ' ' + obj.relation.id;
-               });
-             });
-           }
+         context.entity = function (id) {
+           return _history.graph().entity(id);
+         };
+         /* Modes */
 
-           result.forEach(function (obj) {
-             obj.title = obj.value;
-           });
-           result.unshift(newRelation);
-           callback(result);
-         }
 
-         function renderDisclosureContent(selection) {
-           var memberships = getMemberships();
-           var list = selection.selectAll('.member-list').data([0]);
-           list = list.enter().append('ul').attr('class', 'member-list').merge(list);
-           var items = list.selectAll('li.member-row-normal').data(memberships, function (d) {
-             return d.hash;
-           });
-           items.exit().each(unbind).remove(); // Enter
+         var _mode;
 
-           var itemsEnter = items.enter().append('li').attr('class', 'member-row member-row-normal form-field'); // highlight the relation in the map while hovering on the list item
+         context.mode = function () {
+           return _mode;
+         };
 
-           itemsEnter.on('mouseover', function (d3_event, d) {
-             utilHighlightEntities([d.relation.id], true, context);
-           }).on('mouseout', function (d3_event, d) {
-             utilHighlightEntities([d.relation.id], false, context);
-           });
-           var labelEnter = itemsEnter.append('label').attr('class', 'field-label').attr('for', function (d) {
-             return d.domId;
-           });
-           var labelLink = labelEnter.append('span').attr('class', 'label-text').append('a').attr('href', '#').on('click', selectRelation);
-           labelLink.append('span').attr('class', 'member-entity-type').html(function (d) {
-             var matched = _mainPresetIndex.match(d.relation, context.graph());
-             return matched && matched.name() || _t('inspector.relation');
-           });
-           labelLink.append('span').attr('class', 'member-entity-name').html(function (d) {
-             return utilDisplayName(d.relation);
-           });
-           labelEnter.append('button').attr('class', 'remove member-delete').call(svgIcon('#iD-operation-delete')).on('click', deleteMembership);
-           labelEnter.append('button').attr('class', 'member-zoom').attr('title', _t('icons.zoom_to')).call(svgIcon('#iD-icon-framed-dot', 'monochrome')).on('click', zoomToRelation);
-           var wrapEnter = itemsEnter.append('div').attr('class', 'form-field-input-wrap form-field-input-member');
-           wrapEnter.append('input').attr('class', 'member-role').attr('id', function (d) {
-             return d.domId;
-           }).property('type', 'text').property('value', function (d) {
-             return typeof d.role === 'string' ? d.role : '';
-           }).attr('title', function (d) {
-             return Array.isArray(d.role) ? d.role.filter(Boolean).join('\n') : d.role;
-           }).attr('placeholder', function (d) {
-             return Array.isArray(d.role) ? _t('inspector.multiple_roles') : _t('inspector.role');
-           }).classed('mixed', function (d) {
-             return Array.isArray(d.role);
-           }).call(utilNoAuto).on('blur', changeRole).on('change', changeRole);
+         context.enter = function (newMode) {
+           if (_mode) {
+             _mode.exit();
 
-           if (taginfo) {
-             wrapEnter.each(bindTypeahead);
+             dispatch.call('exit', _this, _mode);
            }
 
-           var newMembership = list.selectAll('.member-row-new').data(_showBlank ? [0] : []); // Exit
+           _mode = newMode;
 
-           newMembership.exit().remove(); // Enter
+           _mode.enter();
 
-           var newMembershipEnter = newMembership.enter().append('li').attr('class', 'member-row member-row-new form-field');
-           var newLabelEnter = newMembershipEnter.append('label').attr('class', 'field-label');
-           newLabelEnter.append('input').attr('placeholder', _t('inspector.choose_relation')).attr('type', 'text').attr('class', 'member-entity-input').call(utilNoAuto);
-           newLabelEnter.append('button').attr('class', 'remove member-delete').call(svgIcon('#iD-operation-delete')).on('click', function () {
-             list.selectAll('.member-row-new').remove();
-           });
-           var newWrapEnter = newMembershipEnter.append('div').attr('class', 'form-field-input-wrap form-field-input-member');
-           newWrapEnter.append('input').attr('class', 'member-role').property('type', 'text').attr('placeholder', _t('inspector.role')).call(utilNoAuto); // Update
+           dispatch.call('enter', _this, _mode);
+         };
 
-           newMembership = newMembership.merge(newMembershipEnter);
-           newMembership.selectAll('.member-entity-input').on('blur', cancelEntity) // if it wasn't accepted normally, cancel it
-           .call(nearbyCombo.on('accept', acceptEntity).on('cancel', cancelEntity)); // Container for the Add button
+         context.selectedIDs = function () {
+           return _mode && _mode.selectedIDs && _mode.selectedIDs() || [];
+         };
 
-           var addRow = selection.selectAll('.add-row').data([0]); // enter
+         context.activeID = function () {
+           return _mode && _mode.activeID && _mode.activeID();
+         };
 
-           var addRowEnter = addRow.enter().append('div').attr('class', 'add-row');
-           var addRelationButton = addRowEnter.append('button').attr('class', 'add-relation');
-           addRelationButton.call(svgIcon('#iD-icon-plus', 'light'));
-           addRelationButton.call(uiTooltip().title(_t.html('inspector.add_to_relation')).placement(_mainLocalizer.textDirection() === 'ltr' ? 'right' : 'left'));
-           addRowEnter.append('div').attr('class', 'space-value'); // preserve space
+         var _selectedNoteID;
 
-           addRowEnter.append('div').attr('class', 'space-buttons'); // preserve space
-           // update
+         context.selectedNoteID = function (noteID) {
+           if (!arguments.length) return _selectedNoteID;
+           _selectedNoteID = noteID;
+           return context;
+         }; // NOTE: Don't change the name of this until UI v3 is merged
 
-           addRow = addRow.merge(addRowEnter);
-           addRow.select('.add-relation').on('click', function () {
-             _showBlank = true;
-             section.reRender();
-             list.selectAll('.member-entity-input').node().focus();
-           });
 
-           function acceptEntity(d) {
-             if (!d) {
-               cancelEntity();
-               return;
-             } // remove hover-higlighting
+         var _selectedErrorID;
 
+         context.selectedErrorID = function (errorID) {
+           if (!arguments.length) return _selectedErrorID;
+           _selectedErrorID = errorID;
+           return context;
+         };
+         /* Behaviors */
 
-             if (d.relation) utilHighlightEntities([d.relation.id], false, context);
-             var role = context.cleanRelationRole(list.selectAll('.member-row-new .member-role').property('value'));
-             addMembership(d, role);
-           }
 
-           function cancelEntity() {
-             var input = newMembership.selectAll('.member-entity-input');
-             input.property('value', ''); // remove hover-higlighting
+         context.install = function (behavior) {
+           return context.surface().call(behavior);
+         };
 
-             context.surface().selectAll('.highlighted').classed('highlighted', false);
-           }
+         context.uninstall = function (behavior) {
+           return context.surface().call(behavior.off);
+         };
+         /* Copy/Paste */
 
-           function bindTypeahead(d) {
-             var row = select(this);
-             var role = row.selectAll('input.member-role');
-             var origValue = role.property('value');
 
-             function sort(value, data) {
-               var sameletter = [];
-               var other = [];
+         var _copyGraph;
 
-               for (var i = 0; i < data.length; i++) {
-                 if (data[i].value.substring(0, value.length) === value) {
-                   sameletter.push(data[i]);
-                 } else {
-                   other.push(data[i]);
-                 }
-               }
+         context.copyGraph = function () {
+           return _copyGraph;
+         };
 
-               return sameletter.concat(other);
-             }
+         var _copyIDs = [];
 
-             role.call(uiCombobox(context, 'member-role').fetcher(function (role, callback) {
-               var rtype = d.relation.tags.type;
-               taginfo.roles({
-                 debounce: true,
-                 rtype: rtype || '',
-                 geometry: context.graph().geometry(_entityIDs[0]),
-                 query: role
-               }, function (err, data) {
-                 if (!err) callback(sort(role, data));
-               });
-             }).on('cancel', function () {
-               role.property('value', origValue);
-             }));
-           }
+         context.copyIDs = function (val) {
+           if (!arguments.length) return _copyIDs;
+           _copyIDs = val;
+           _copyGraph = _history.graph();
+           return context;
+         };
 
-           function unbind() {
-             var row = select(this);
-             row.selectAll('input.member-role').call(uiCombobox.off, context);
-           }
-         }
+         var _copyLonLat;
 
-         section.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs;
-           _entityIDs = val;
-           _showBlank = false;
-           return section;
+         context.copyLonLat = function (val) {
+           if (!arguments.length) return _copyLonLat;
+           _copyLonLat = val;
+           return context;
          };
+         /* Background */
 
-         return section;
-       }
 
-       function uiSectionSelectionList(context) {
-         var _selectedIDs = [];
-         var section = uiSection('selected-features', context).shouldDisplay(function () {
-           return _selectedIDs.length > 1;
-         }).label(function () {
-           return _t('inspector.title_count', {
-             title: _t.html('inspector.features'),
-             count: _selectedIDs.length
-           });
-         }).disclosureContent(renderDisclosureContent);
-         context.history().on('change.selectionList', function (difference) {
-           if (difference) {
-             section.reRender();
-           }
-         });
+         var _background;
 
-         section.entityIDs = function (val) {
-           if (!arguments.length) return _selectedIDs;
-           _selectedIDs = val;
-           return section;
+         context.background = function () {
+           return _background;
          };
+         /* Features */
 
-         function selectEntity(d3_event, entity) {
-           context.enter(modeSelect(context, [entity.id]));
-         }
 
-         function deselectEntity(d3_event, entity) {
-           var selectedIDs = _selectedIDs.slice();
+         var _features;
 
-           var index = selectedIDs.indexOf(entity.id);
+         context.features = function () {
+           return _features;
+         };
 
-           if (index > -1) {
-             selectedIDs.splice(index, 1);
-             context.enter(modeSelect(context, selectedIDs));
-           }
-         }
+         context.hasHiddenConnections = function (id) {
+           var graph = _history.graph();
 
-         function renderDisclosureContent(selection) {
-           var list = selection.selectAll('.feature-list').data([0]);
-           list = list.enter().append('ul').attr('class', 'feature-list').merge(list);
+           var entity = graph.entity(id);
+           return _features.hasHiddenConnections(entity, graph);
+         };
+         /* Photos */
 
-           var entities = _selectedIDs.map(function (id) {
-             return context.hasEntity(id);
-           }).filter(Boolean);
 
-           var items = list.selectAll('.feature-list-item').data(entities, osmEntity.key);
-           items.exit().remove(); // Enter
+         var _photos;
 
-           var enter = items.enter().append('li').attr('class', 'feature-list-item').each(function (d) {
-             select(this).on('mouseover', function () {
-               utilHighlightEntities([d.id], true, context);
-             }).on('mouseout', function () {
-               utilHighlightEntities([d.id], false, context);
-             });
-           });
-           var label = enter.append('button').attr('class', 'label').on('click', selectEntity);
-           label.append('span').attr('class', 'entity-geom-icon').call(svgIcon('', 'pre-text'));
-           label.append('span').attr('class', 'entity-type');
-           label.append('span').attr('class', 'entity-name');
-           enter.append('button').attr('class', 'close').attr('title', _t('icons.deselect')).on('click', deselectEntity).call(svgIcon('#iD-icon-close')); // Update
+         context.photos = function () {
+           return _photos;
+         };
+         /* Map */
 
-           items = items.merge(enter);
-           items.selectAll('.entity-geom-icon use').attr('href', function () {
-             var entity = this.parentNode.parentNode.__data__;
-             return '#iD-icon-' + entity.geometry(context.graph());
-           });
-           items.selectAll('.entity-type').html(function (entity) {
-             return _mainPresetIndex.match(entity, context.graph()).name();
-           });
-           items.selectAll('.entity-name').html(function (d) {
-             // fetch latest entity
-             var entity = context.entity(d.id);
-             return utilDisplayName(entity);
-           });
-         }
 
-         return section;
-       }
+         var _map;
 
-       function uiEntityEditor(context) {
-         var dispatch$1 = dispatch('choose');
-         var _state = 'select';
-         var _coalesceChanges = false;
-         var _modified = false;
+         context.map = function () {
+           return _map;
+         };
 
-         var _base;
+         context.layers = function () {
+           return _map.layers();
+         };
 
-         var _entityIDs;
+         context.surface = function () {
+           return _map.surface;
+         };
 
-         var _activePresets = [];
+         context.editableDataEnabled = function () {
+           return _map.editableDataEnabled();
+         };
 
-         var _newFeature;
+         context.surfaceRect = function () {
+           return _map.surface.node().getBoundingClientRect();
+         };
 
-         var _sections;
+         context.editable = function () {
+           // don't allow editing during save
+           var mode = context.mode();
+           if (!mode || mode.id === 'save') return false;
+           return _map.editableDataEnabled();
+         };
+         /* Debug */
 
-         function entityEditor(selection) {
-           var combinedTags = utilCombinedTags(_entityIDs, context.graph()); // Header
 
-           var header = selection.selectAll('.header').data([0]); // Enter
+         var _debugFlags = {
+           tile: false,
+           // tile boundaries
+           collision: false,
+           // label collision bounding boxes
+           imagery: false,
+           // imagery bounding polygons
+           target: false,
+           // touch targets
+           downloaded: false // downloaded data from osm
 
-           var headerEnter = header.enter().append('div').attr('class', 'header fillL');
-           headerEnter.append('button').attr('class', 'preset-reset preset-choose').call(svgIcon(_mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-forward' : '#iD-icon-backward'));
-           headerEnter.append('button').attr('class', 'close').on('click', function () {
-             context.enter(modeBrowse(context));
-           }).call(svgIcon(_modified ? '#iD-icon-apply' : '#iD-icon-close'));
-           headerEnter.append('h3'); // Update
+         };
 
-           header = header.merge(headerEnter);
-           header.selectAll('h3').html(_entityIDs.length === 1 ? _t.html('inspector.edit') : _t.html('inspector.edit_features'));
-           header.selectAll('.preset-reset').on('click', function () {
-             dispatch$1.call('choose', this, _activePresets);
-           }); // Body
+         context.debugFlags = function () {
+           return _debugFlags;
+         };
 
-           var body = selection.selectAll('.inspector-body').data([0]); // Enter
+         context.getDebug = function (flag) {
+           return flag && _debugFlags[flag];
+         };
 
-           var bodyEnter = body.enter().append('div').attr('class', 'entity-editor inspector-body sep-top'); // Update
+         context.setDebug = function (flag, val) {
+           if (arguments.length === 1) val = true;
+           _debugFlags[flag] = val;
+           dispatch.call('change');
+           return context;
+         };
+         /* Container */
 
-           body = body.merge(bodyEnter);
 
-           if (!_sections) {
-             _sections = [uiSectionSelectionList(context), uiSectionFeatureType(context).on('choose', function (presets) {
-               dispatch$1.call('choose', this, presets);
-             }), uiSectionEntityIssues(context), uiSectionPresetFields(context).on('change', changeTags).on('revert', revertTags), uiSectionRawTagEditor('raw-tag-editor', context).on('change', changeTags), uiSectionRawMemberEditor(context), uiSectionRawMembershipEditor(context)];
-           }
+         var _container = select(null);
 
-           _sections.forEach(function (section) {
-             if (section.entityIDs) {
-               section.entityIDs(_entityIDs);
-             }
+         context.container = function (val) {
+           if (!arguments.length) return _container;
+           _container = val;
 
-             if (section.presets) {
-               section.presets(_activePresets);
-             }
+           _container.classed('ideditor', true);
 
-             if (section.tags) {
-               section.tags(combinedTags);
-             }
+           return context;
+         };
 
-             if (section.state) {
-               section.state(_state);
-             }
+         context.containerNode = function (val) {
+           if (!arguments.length) return context.container().node();
+           context.container(select(val));
+           return context;
+         };
 
-             body.call(section.render);
-           });
+         var _embed;
 
-           context.history().on('change.entity-editor', historyChanged);
+         context.embed = function (val) {
+           if (!arguments.length) return _embed;
+           _embed = val;
+           return context;
+         };
+         /* Assets */
 
-           function historyChanged(difference) {
-             if (selection.selectAll('.entity-editor').empty()) return;
-             if (_state === 'hide') return;
-             var significant = !difference || difference.didChange.properties || difference.didChange.addition || difference.didChange.deletion;
-             if (!significant) return;
-             _entityIDs = _entityIDs.filter(context.hasEntity);
-             if (!_entityIDs.length) return;
-             var priorActivePreset = _activePresets.length === 1 && _activePresets[0];
-             loadActivePresets();
-             var graph = context.graph();
-             entityEditor.modified(_base !== graph);
-             entityEditor(selection);
 
-             if (priorActivePreset && _activePresets.length === 1 && priorActivePreset !== _activePresets[0]) {
-               // flash the button to indicate the preset changed
-               context.container().selectAll('.entity-editor button.preset-reset .label').style('background-color', '#fff').transition().duration(750).style('background-color', null);
-             }
-           }
-         } // Tag changes that fire on input can all get coalesced into a single
-         // history operation when the user leaves the field.  #2342
-         // Use explicit entityIDs in case the selection changes before the event is fired.
+         var _assetPath = '';
 
+         context.assetPath = function (val) {
+           if (!arguments.length) return _assetPath;
+           _assetPath = val;
+           _mainFileFetcher.assetPath(val);
+           return context;
+         };
 
-         function changeTags(entityIDs, changed, onInput) {
-           var actions = [];
+         var _assetMap = {};
 
-           for (var i in entityIDs) {
-             var entityID = entityIDs[i];
-             var entity = context.entity(entityID);
-             var tags = Object.assign({}, entity.tags); // shallow copy
+         context.assetMap = function (val) {
+           if (!arguments.length) return _assetMap;
+           _assetMap = val;
+           _mainFileFetcher.assetMap(val);
+           return context;
+         };
 
-             for (var k in changed) {
-               if (!k) continue;
-               var v = changed[k];
+         context.asset = function (val) {
+           if (/^http(s)?:\/\//i.test(val)) return val;
+           var filename = _assetPath + val;
+           return _assetMap[filename] || filename;
+         };
 
-               if (v !== undefined || tags.hasOwnProperty(k)) {
-                 tags[k] = v;
-               }
-             }
+         context.imagePath = function (val) {
+           return context.asset("img/".concat(val));
+         };
+         /* reset (aka flush) */
 
-             if (!onInput) {
-               tags = utilCleanTags(tags);
-             }
 
-             if (!fastDeepEqual(entity.tags, tags)) {
-               actions.push(actionChangeTags(entityID, tags));
+         context.reset = context.flush = function () {
+           context.debouncedSave.cancel();
+           Array.from(_deferred).forEach(function (handle) {
+             window.cancelIdleCallback(handle);
+
+             _deferred["delete"](handle);
+           });
+           Object.values(services).forEach(function (service) {
+             if (service && typeof service.reset === 'function') {
+               service.reset(context);
              }
-           }
+           });
+           context.changeset = null;
 
-           if (actions.length) {
-             var combinedAction = function combinedAction(graph) {
-               actions.forEach(function (action) {
-                 graph = action(graph);
-               });
-               return graph;
-             };
+           _validator.reset();
 
-             var annotation = _t('operations.change_tags.annotation');
+           _features.reset();
 
-             if (_coalesceChanges) {
-               context.overwrite(combinedAction, annotation);
-             } else {
-               context.perform(combinedAction, annotation);
-               _coalesceChanges = !!onInput;
-             }
-           } // if leaving field (blur event), rerun validation
+           _history.reset();
 
+           _uploader.reset(); // don't leave stale state in the inspector
 
-           if (!onInput) {
-             context.validator().validate();
-           }
-         }
 
-         function revertTags(keys) {
-           var actions = [];
+           context.container().select('.inspector-wrap *').remove();
+           return context;
+         };
+         /* Projections */
 
-           for (var i in _entityIDs) {
-             var entityID = _entityIDs[i];
-             var original = context.graph().base().entities[entityID];
-             var changed = {};
 
-             for (var j in keys) {
-               var key = keys[j];
-               changed[key] = original ? original.tags[key] : undefined;
-             }
+         context.projection = geoRawMercator();
+         context.curtainProjection = geoRawMercator();
+         /* Init */
 
-             var entity = context.entity(entityID);
-             var tags = Object.assign({}, entity.tags); // shallow copy
+         context.init = function () {
+           instantiateInternal();
+           initializeDependents();
+           return context; // Load variables and properties. No property of `context` should be accessed
+           // until this is complete since load statuses are indeterminate. The order
+           // of instantiation shouldn't matter.
 
-             for (var k in changed) {
-               if (!k) continue;
-               var v = changed[k];
+           function instantiateInternal() {
+             _history = coreHistory(context);
+             context.graph = _history.graph;
+             context.pauseChangeDispatch = _history.pauseChangeDispatch;
+             context.resumeChangeDispatch = _history.resumeChangeDispatch;
+             context.perform = withDebouncedSave(_history.perform);
+             context.replace = withDebouncedSave(_history.replace);
+             context.pop = withDebouncedSave(_history.pop);
+             context.overwrite = withDebouncedSave(_history.overwrite);
+             context.undo = withDebouncedSave(_history.undo);
+             context.redo = withDebouncedSave(_history.redo);
+             _validator = coreValidator(context);
+             _uploader = coreUploader(context);
+             _background = rendererBackground(context);
+             _features = rendererFeatures(context);
+             _map = rendererMap(context);
+             _photos = rendererPhotos(context);
+             _ui = uiInit(context);
+           } // Set up objects that might need to access properties of `context`. The order
+           // might matter if dependents make calls to each other. Be wary of async calls.
 
-               if (v !== undefined || tags.hasOwnProperty(k)) {
-                 tags[k] = v;
-               }
+
+           function initializeDependents() {
+             if (context.initialHashParams.presets) {
+               _mainPresetIndex.addablePresetIDs(new Set(context.initialHashParams.presets.split(',')));
              }
 
-             tags = utilCleanTags(tags);
+             if (context.initialHashParams.locale) {
+               _mainLocalizer.preferredLocaleCodes(context.initialHashParams.locale);
+             } // kick off some async work
 
-             if (!fastDeepEqual(entity.tags, tags)) {
-               actions.push(actionChangeTags(entityID, tags));
-             }
-           }
 
-           if (actions.length) {
-             var combinedAction = function combinedAction(graph) {
-               actions.forEach(function (action) {
-                 graph = action(graph);
-               });
-               return graph;
-             };
+             _mainLocalizer.ensureLoaded();
 
-             var annotation = _t('operations.change_tags.annotation');
+             _background.ensureLoaded();
 
-             if (_coalesceChanges) {
-               context.overwrite(combinedAction, annotation);
-             } else {
-               context.perform(combinedAction, annotation);
-               _coalesceChanges = false;
-             }
-           }
+             _mainPresetIndex.ensureLoaded();
+             Object.values(services).forEach(function (service) {
+               if (service && typeof service.init === 'function') {
+                 service.init();
+               }
+             });
 
-           context.validator().validate();
-         }
+             _map.init();
 
-         entityEditor.modified = function (val) {
-           if (!arguments.length) return _modified;
-           _modified = val;
-           return entityEditor;
-         };
+             _validator.init();
 
-         entityEditor.state = function (val) {
-           if (!arguments.length) return _state;
-           _state = val;
-           return entityEditor;
-         };
+             _features.init();
 
-         entityEditor.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs; // always reload these even if the entityIDs are unchanged, since we
-           // could be reselecting after something like dragging a node
+             if (services.maprules && context.initialHashParams.maprules) {
+               d3_json(context.initialHashParams.maprules).then(function (mapcss) {
+                 services.maprules.init();
+                 mapcss.forEach(function (mapcssSelector) {
+                   return services.maprules.addRule(mapcssSelector);
+                 });
+               })["catch"](function () {
+                 /* ignore */
+               });
+             } // if the container isn't available, e.g. when testing, don't load the UI
 
-           _base = context.graph();
-           _coalesceChanges = false;
-           if (val && _entityIDs && utilArrayIdentical(_entityIDs, val)) return entityEditor; // exit early if no change
 
-           _entityIDs = val;
-           loadActivePresets(true);
-           return entityEditor.modified(false);
+             if (!context.container().empty()) {
+               _ui.ensureLoaded().then(function () {
+                 _photos.init();
+               });
+             }
+           }
          };
 
-         entityEditor.newFeature = function (val) {
-           if (!arguments.length) return _newFeature;
-           _newFeature = val;
-           return entityEditor;
-         };
+         return context;
+       }
 
-         function loadActivePresets(isForNewSelection) {
-           var graph = context.graph();
-           var counts = {};
+       // NSI contains the most correct tagging for many commonly mapped features.
+       // See https://github.com/osmlab/name-suggestion-index  and  https://nsi.guide
+       // DATA
 
-           for (var i in _entityIDs) {
-             var entity = graph.hasEntity(_entityIDs[i]);
-             if (!entity) return;
-             var match = _mainPresetIndex.match(entity, graph);
-             if (!counts[match.id]) counts[match.id] = 0;
-             counts[match.id] += 1;
-           }
+       var _nsiStatus = 'loading'; // 'loading', 'ok', 'failed'
 
-           var matches = Object.keys(counts).sort(function (p1, p2) {
-             return counts[p2] - counts[p1];
-           }).map(function (pID) {
-             return _mainPresetIndex.item(pID);
-           });
+       var _nsi = {}; // Sometimes we can upgrade a feature tagged like `building=yes` to a better tag.
 
-           if (!isForNewSelection) {
-             // A "weak" preset doesn't set any tags. (e.g. "Address")
-             var weakPreset = _activePresets.length === 1 && !_activePresets[0].isFallback() && Object.keys(_activePresets[0].addTags || {}).length === 0; // Don't replace a weak preset with a fallback preset (e.g. "Point")
+       var buildingPreset = {
+         'building/commercial': true,
+         'building/government': true,
+         'building/hotel': true,
+         'building/retail': true,
+         'building/office': true,
+         'building/supermarket': true,
+         'building/yes': true
+       }; // Exceptions to the namelike regexes.
+       // Usually a tag suffix contains a language code like `name:en`, `name:ru`
+       // but we want to exclude things like `operator:type`, `name:etymology`, etc..
 
-             if (weakPreset && matches.length === 1 && matches[0].isFallback()) return;
-           }
+       var notNames = /:(colou?r|type|forward|backward|left|right|etymology|pronunciation|wikipedia)$/i; // Exceptions to the branchlike regexes
 
-           entityEditor.presets(matches);
-         }
+       var notBranches = /(coop|express|wireless|factory|outlet)/i; // PRIVATE FUNCTIONS
+       // `setNsiSources()`
+       // Adds the sources to iD's filemap so we can start downloading data.
+       //
 
-         entityEditor.presets = function (val) {
-           if (!arguments.length) return _activePresets; // don't reload the same preset
+       function setNsiSources() {
+         var nsiVersion = packageJSON.devDependencies['name-suggestion-index'];
+         var v = vparse(nsiVersion);
+         var vMinor = "".concat(v.major, ".").concat(v.minor);
+         var sources = {
+           'nsi_data': "https://cdn.jsdelivr.net/npm/name-suggestion-index@".concat(vMinor, "/dist/nsi.min.json"),
+           'nsi_dissolved': "https://cdn.jsdelivr.net/npm/name-suggestion-index@".concat(vMinor, "/dist/dissolved.min.json"),
+           'nsi_features': "https://cdn.jsdelivr.net/npm/name-suggestion-index@".concat(vMinor, "/dist/featureCollection.min.json"),
+           'nsi_generics': "https://cdn.jsdelivr.net/npm/name-suggestion-index@".concat(vMinor, "/dist/genericWords.min.json"),
+           'nsi_presets': "https://cdn.jsdelivr.net/npm/name-suggestion-index@".concat(vMinor, "/dist/presets/nsi-id-presets.min.json"),
+           'nsi_replacements': "https://cdn.jsdelivr.net/npm/name-suggestion-index@".concat(vMinor, "/dist/replacements.min.json"),
+           'nsi_trees': "https://cdn.jsdelivr.net/npm/name-suggestion-index@".concat(vMinor, "/dist/trees.min.json")
+         };
+         var fileMap = _mainFileFetcher.fileMap();
+
+         for (var k in sources) {
+           fileMap[k] = sources[k];
+         }
+       } // `loadNsiPresets()`
+       //  Returns a Promise fulfilled when the presets have been downloaded and merged into iD.
+       //
 
-           if (!utilArrayIdentical(val, _activePresets)) {
-             _activePresets = val;
-           }
 
-           return entityEditor;
-         };
+       function loadNsiPresets() {
+         return Promise.all([_mainFileFetcher.get('nsi_presets'), _mainFileFetcher.get('nsi_features')]).then(function (vals) {
+           // Add `suggestion=true` to all the nsi presets
+           // The preset json schema doesn't include it, but the iD code still uses it
+           Object.values(vals[0].presets).forEach(function (preset) {
+             return preset.suggestion = true;
+           });
+           _mainPresetIndex.merge({
+             presets: vals[0].presets,
+             featureCollection: vals[1]
+           });
+         });
+       } // `loadNsiData()`
+       //  Returns a Promise fulfilled when the other data have been downloaded and processed
+       //
 
-         return utilRebind(entityEditor, dispatch$1, 'on');
-       }
 
-       function uiPresetList(context) {
-         var dispatch$1 = dispatch('cancel', 'choose');
+       function loadNsiData() {
+         return Promise.all([_mainFileFetcher.get('nsi_data'), _mainFileFetcher.get('nsi_dissolved'), _mainFileFetcher.get('nsi_replacements'), _mainFileFetcher.get('nsi_trees')]).then(function (vals) {
+           _nsi = {
+             data: vals[0].nsi,
+             // the raw name-suggestion-index data
+             dissolved: vals[1].dissolved,
+             // list of dissolved items
+             replacements: vals[2].replacements,
+             // trivial old->new qid replacements
+             trees: vals[3].trees,
+             // metadata about trees, main tags
+             kvt: new Map(),
+             // Map (k -> Map (v -> t) )
+             qids: new Map(),
+             // Map (wd/wp tag values -> qids)
+             ids: new Map() // Map (id -> NSI item)
 
-         var _entityIDs;
+           };
+           _nsi.matcher = new Matcher();
 
-         var _currentPresets;
+           _nsi.matcher.buildMatchIndex(_nsi.data);
 
-         var _autofocus = false;
+           _nsi.matcher.buildLocationIndex(_nsi.data, _mainLocations.loco());
 
-         function presetList(selection) {
-           if (!_entityIDs) return;
-           var presets = _mainPresetIndex.matchAllGeometry(entityGeometries());
-           selection.html('');
-           var messagewrap = selection.append('div').attr('class', 'header fillL');
-           var message = messagewrap.append('h3').html(_t.html('inspector.choose'));
-           messagewrap.append('button').attr('class', 'preset-choose').on('click', function () {
-             dispatch$1.call('cancel', this);
-           }).call(svgIcon(_mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-backward' : '#iD-icon-forward'));
+           Object.keys(_nsi.data).forEach(function (tkv) {
+             var category = _nsi.data[tkv];
+             var parts = tkv.split('/', 3); // tkv = "tree/key/value"
 
-           function initialKeydown(d3_event) {
-             // hack to let delete shortcut work when search is autofocused
-             if (search.property('value').length === 0 && (d3_event.keyCode === utilKeybinding.keyCodes['⌫'] || d3_event.keyCode === utilKeybinding.keyCodes['⌦'])) {
-               d3_event.preventDefault();
-               d3_event.stopPropagation();
-               operationDelete(context, _entityIDs)(); // hack to let undo work when search is autofocused
-             } else if (search.property('value').length === 0 && (d3_event.ctrlKey || d3_event.metaKey) && d3_event.keyCode === utilKeybinding.keyCodes.z) {
-               d3_event.preventDefault();
-               d3_event.stopPropagation();
-               context.undo();
-             } else if (!d3_event.ctrlKey && !d3_event.metaKey) {
-               // don't check for delete/undo hack on future keydown events
-               select(this).on('keydown', keydown);
-               keydown.call(this, d3_event);
-             }
-           }
+             var t = parts[0];
+             var k = parts[1];
+             var v = parts[2]; // Build a reverse index of keys -> values -> trees present in the name-suggestion-index
+             // Collect primary keys  (e.g. "amenity", "craft", "shop", "man_made", "route", etc)
+             // "amenity": {
+             //   "restaurant": "brands"
+             // }
 
-           function keydown(d3_event) {
-             // down arrow
-             if (d3_event.keyCode === utilKeybinding.keyCodes['↓'] && // if insertion point is at the end of the string
-             search.node().selectionStart === search.property('value').length) {
-               d3_event.preventDefault();
-               d3_event.stopPropagation(); // move focus to the first item in the preset list
+             var vmap = _nsi.kvt.get(k);
 
-               var buttons = list.selectAll('.preset-list-button');
-               if (!buttons.empty()) buttons.nodes()[0].focus();
+             if (!vmap) {
+               vmap = new Map();
+
+               _nsi.kvt.set(k, vmap);
              }
-           }
 
-           function keypress(d3_event) {
-             // enter
-             var value = search.property('value');
+             vmap.set(v, t);
+             var tree = _nsi.trees[t]; // e.g. "brands", "operators"
 
-             if (d3_event.keyCode === 13 && // ↩ Return
-             value.length) {
-               list.selectAll('.preset-list-item:first-child').each(function (d) {
-                 d.choose.call(this);
-               });
-             }
-           }
+             var mainTag = tree.mainTag; // e.g. "brand:wikidata", "operator:wikidata", etc
 
-           function inputevent() {
-             var value = search.property('value');
-             list.classed('filtered', value.length);
-             var extent = combinedEntityExtent();
-             var results, messageText;
+             var items = category.items || [];
+             items.forEach(function (item) {
+               // Remember some useful things for later, cache NSI id -> item
+               item.tkv = tkv;
+               item.mainTag = mainTag;
 
-             if (value.length && extent) {
-               var center = extent.center();
-               var countryCode = iso1A2Code(center);
-               results = presets.search(value, entityGeometries()[0], countryCode && countryCode.toLowerCase());
-               messageText = _t('inspector.results', {
-                 n: results.collection.length,
-                 search: value
-               });
-             } else {
-               results = _mainPresetIndex.defaults(entityGeometries()[0], 36, !context.inIntro());
-               messageText = _t('inspector.choose');
-             }
+               _nsi.ids.set(item.id, item); // Cache Wikidata/Wikipedia values -> qid, for #6416
 
-             list.call(drawList, results);
-             message.html(messageText);
-           }
 
-           var searchWrap = selection.append('div').attr('class', 'search-header');
-           searchWrap.call(svgIcon('#iD-icon-search', 'pre-text'));
-           var search = searchWrap.append('input').attr('class', 'preset-search-input').attr('placeholder', _t('inspector.search')).attr('type', 'search').call(utilNoAuto).on('keydown', initialKeydown).on('keypress', keypress).on('input', inputevent);
+               var wd = item.tags[mainTag];
+               var wp = item.tags[mainTag.replace('wikidata', 'wikipedia')];
+               if (wd) _nsi.qids.set(wd, wd);
+               if (wp && wd) _nsi.qids.set(wp, wd);
+             });
+           });
+         });
+       } // `gatherKVs()`
+       // Gather all the k/v pairs that we will run through the NSI matcher.
+       // An OSM tags object can contain anything, but only a few tags will be interesting to NSI.
+       //
+       // This function will return the interesting tag pairs like:
+       //   "amenity/restaurant", "man_made/flagpole"
+       // and fallbacks like
+       //   "amenity/yes"
+       // excluding things like
+       //   "highway", "surface", "ref", etc.
+       //
+       // Arguments
+       //   `tags`: `Object` containing the feature's OSM tags
+       // Returns
+       //   `Object` containing kv pairs to test:
+       //   {
+       //     'primary': Set(),
+       //     'alternate': Set()
+       //   }
+       //
 
-           if (_autofocus) {
-             search.node().focus(); // Safari 14 doesn't always like to focus immediately,
-             // so try again on the next pass
 
-             setTimeout(function () {
-               search.node().focus();
-             }, 0);
+       function gatherKVs(tags) {
+         var primary = new Set();
+         var alternate = new Set();
+         Object.keys(tags).forEach(function (osmkey) {
+           var osmvalue = tags[osmkey];
+           if (!osmvalue) return;
+
+           var vmap = _nsi.kvt.get(osmkey);
+
+           if (!vmap) return;
+
+           if (osmvalue !== 'yes') {
+             primary.add("".concat(osmkey, "/").concat(osmvalue));
+           } else {
+             alternate.add("".concat(osmkey, "/").concat(osmvalue));
            }
+         }); // Can we try a generic building fallback match? - See #6122, #7197
+         // Only try this if we do a preset match and find nothing else remarkable about that building.
+         // For example, a way with `building=yes` + `name=Westfield` may be a Westfield department store.
+         // But a way with `building=yes` + `name=Westfield` + `public_transport=station` is a train station for a town named "Westfield"
 
-           var listWrap = selection.append('div').attr('class', 'inspector-body');
-           var list = listWrap.append('div').attr('class', 'preset-list').call(drawList, _mainPresetIndex.defaults(entityGeometries()[0], 36, !context.inIntro()));
-           context.features().on('change.preset-list', updateForFeatureHiddenState);
+         var preset = _mainPresetIndex.matchTags(tags, 'area');
+
+         if (buildingPreset[preset.id]) {
+           alternate.add('building/yes');
          }
 
-         function drawList(list, presets) {
-           presets = presets.matchAllGeometry(entityGeometries());
-           var collection = presets.collection.reduce(function (collection, preset) {
-             if (!preset) return collection;
+         return {
+           primary: primary,
+           alternate: alternate
+         };
+       } // `identifyTree()`
+       // NSI has a concept of trees: "brands", "operators", "flags", "transit".
+       // The tree determines things like which tags are namelike, and which tags hold important wikidata.
+       // This takes an Object of tags and tries to identify what tree to use.
+       //
+       // Arguments
+       //   `tags`: `Object` containing the feature's OSM tags
+       // Returns
+       //   `string` the name of the tree if known
+       //   or 'unknown' if it could match several trees (e.g. amenity/yes)
+       //   or null if no match
+       //
 
-             if (preset.members) {
-               if (preset.members.collection.filter(function (preset) {
-                 return preset.addable();
-               }).length > 1) {
-                 collection.push(CategoryItem(preset));
-               }
-             } else if (preset.addable()) {
-               collection.push(PresetItem(preset));
-             }
 
-             return collection;
-           }, []);
-           var items = list.selectAll('.preset-list-item').data(collection, function (d) {
-             return d.preset.id;
-           });
-           items.order();
-           items.exit().remove();
-           items.enter().append('div').attr('class', function (item) {
-             return 'preset-list-item preset-' + item.preset.id.replace('/', '-');
-           }).classed('current', function (item) {
-             return _currentPresets.indexOf(item.preset) !== -1;
-           }).each(function (item) {
-             select(this).call(item);
-           }).style('opacity', 0).transition().style('opacity', 1);
-           updateForFeatureHiddenState();
-         }
+       function identifyTree(tags) {
+         var unknown;
+         var t; // Check all tags
 
-         function itemKeydown(d3_event) {
-           // the actively focused item
-           var item = select(this.closest('.preset-list-item'));
-           var parentItem = select(item.node().parentNode.closest('.preset-list-item')); // arrow down, move focus to the next, lower item
+         Object.keys(tags).forEach(function (osmkey) {
+           if (t) return; // found already
 
-           if (d3_event.keyCode === utilKeybinding.keyCodes['↓']) {
-             d3_event.preventDefault();
-             d3_event.stopPropagation(); // the next item in the list at the same level
+           var osmvalue = tags[osmkey];
+           if (!osmvalue) return;
 
-             var nextItem = select(item.node().nextElementSibling); // if there is no next item in this list
+           var vmap = _nsi.kvt.get(osmkey);
 
-             if (nextItem.empty()) {
-               // if there is a parent item
-               if (!parentItem.empty()) {
-                 // the item is the last item of a sublist,
-                 // select the next item at the parent level
-                 nextItem = select(parentItem.node().nextElementSibling);
-               } // if the focused item is expanded
+           if (!vmap) return; // this key is not in nsi
 
-             } else if (select(this).classed('expanded')) {
-               // select the first subitem instead
-               nextItem = item.select('.subgrid .preset-list-item:first-child');
-             }
+           if (osmvalue === 'yes') {
+             unknown = 'unknown';
+           } else {
+             t = vmap.get(osmvalue);
+           }
+         });
+         return t || unknown || null;
+       } // `gatherNames()`
+       // Gather all the namelike values that we will run through the NSI matcher.
+       // It will gather values primarily from tags `name`, `name:ru`, `flag:name`
+       //  and fallback to alternate tags like `brand`, `brand:ru`, `alt_name`
+       //
+       // Arguments
+       //   `tags`: `Object` containing the feature's OSM tags
+       // Returns
+       //   `Object` containing namelike values to test:
+       //   {
+       //     'primary': Set(),
+       //     'fallbacks': Set()
+       //   }
+       //
 
-             if (!nextItem.empty()) {
-               // focus on the next item
-               nextItem.select('.preset-list-button').node().focus();
-             } // arrow up, move focus to the previous, higher item
 
-           } else if (d3_event.keyCode === utilKeybinding.keyCodes['↑']) {
-             d3_event.preventDefault();
-             d3_event.stopPropagation(); // the previous item in the list at the same level
+       function gatherNames(tags) {
+         var empty = {
+           primary: new Set(),
+           alternate: new Set()
+         };
+         var primary = new Set();
+         var alternate = new Set();
+         var foundSemi = false;
+         var testNameFragments = false;
+         var patterns; // Patterns for matching OSM keys that might contain namelike values.
+         // These roughly correspond to the "trees" concept in name-suggestion-index,
 
-             var previousItem = select(item.node().previousElementSibling); // if there is no previous item in this list
+         var t = identifyTree(tags);
+         if (!t) return empty;
 
-             if (previousItem.empty()) {
-               // if there is a parent item
-               if (!parentItem.empty()) {
-                 // the item is the first subitem of a sublist select the parent item
-                 previousItem = parentItem;
-               } // if the previous item is expanded
+         if (t === 'transit') {
+           patterns = {
+             primary: /^network$/i,
+             alternate: /^(operator|operator:\w+|network:\w+|\w+_name|\w+_name:\w+)$/i
+           };
+         } else if (t === 'flags') {
+           patterns = {
+             primary: /^(flag:name|flag:name:\w+)$/i,
+             alternate: /^(flag|flag:\w+|subject|subject:\w+)$/i // note: no `country`, we special-case it below
 
-             } else if (previousItem.select('.preset-list-button').classed('expanded')) {
-               // select the last subitem of the sublist of the previous item
-               previousItem = previousItem.select('.subgrid .preset-list-item:last-child');
-             }
+           };
+         } else if (t === 'brands') {
+           testNameFragments = true;
+           patterns = {
+             primary: /^(name|name:\w+)$/i,
+             alternate: /^(brand|brand:\w+|operator|operator:\w+|\w+_name|\w+_name:\w+)/i
+           };
+         } else if (t === 'operators') {
+           testNameFragments = true;
+           patterns = {
+             primary: /^(name|name:\w+|operator|operator:\w+)$/i,
+             alternate: /^(brand|brand:\w+|\w+_name|\w+_name:\w+)/i
+           };
+         } else {
+           // unknown/multiple
+           testNameFragments = true;
+           patterns = {
+             primary: /^(name|name:\w+)$/i,
+             alternate: /^(brand|brand:\w+|network|network:\w+|operator|operator:\w+|\w+_name|\w+_name:\w+)/i
+           };
+         } // Test `name` fragments, longest to shortest, to fit them into a "Name Branch" pattern.
+         // e.g. "TUI ReiseCenter - Neuss Innenstadt" -> ["TUI", "ReiseCenter", "Neuss", "Innenstadt"]
 
-             if (!previousItem.empty()) {
-               // focus on the previous item
-               previousItem.select('.preset-list-button').node().focus();
-             } else {
-               // the focus is at the top of the list, move focus back to the search field
-               var search = select(this.closest('.preset-list-pane')).select('.preset-search-input');
-               search.node().focus();
-             } // arrow left, move focus to the parent item if there is one
 
-           } else if (d3_event.keyCode === utilKeybinding.keyCodes[_mainLocalizer.textDirection() === 'rtl' ? '→' : '←']) {
-             d3_event.preventDefault();
-             d3_event.stopPropagation(); // if there is a parent item, focus on the parent item
+         if (tags.name && testNameFragments) {
+           var nameParts = tags.name.split(/[\s\-\/,.]/);
 
-             if (!parentItem.empty()) {
-               parentItem.select('.preset-list-button').node().focus();
-             } // arrow right, choose this item
+           for (var split = nameParts.length; split > 0; split--) {
+             var name = nameParts.slice(0, split).join(' '); // e.g. "TUI ReiseCenter"
 
-           } else if (d3_event.keyCode === utilKeybinding.keyCodes[_mainLocalizer.textDirection() === 'rtl' ? '←' : '→']) {
-             d3_event.preventDefault();
-             d3_event.stopPropagation();
-             item.datum().choose.call(select(this).node());
+             primary.add(name);
            }
-         }
+         } // Check all tags
 
-         function CategoryItem(preset) {
-           var box,
-               sublist,
-               shown = false;
 
-           function item(selection) {
-             var wrap = selection.append('div').attr('class', 'preset-list-button-wrap category');
+         Object.keys(tags).forEach(function (osmkey) {
+           var osmvalue = tags[osmkey];
+           if (!osmvalue) return;
 
-             function click() {
-               var isExpanded = select(this).classed('expanded');
-               var iconName = isExpanded ? _mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-backward' : '#iD-icon-forward' : '#iD-icon-down';
-               select(this).classed('expanded', !isExpanded);
-               select(this).selectAll('div.label-inner svg.icon use').attr('href', iconName);
-               item.choose();
+           if (isNamelike(osmkey, 'primary')) {
+             if (/;/.test(osmvalue)) {
+               foundSemi = true;
+             } else {
+               primary.add(osmvalue);
+               alternate["delete"](osmvalue);
              }
+           } else if (!primary.has(osmvalue) && isNamelike(osmkey, 'alternate')) {
+             if (/;/.test(osmvalue)) {
+               foundSemi = true;
+             } else {
+               alternate.add(osmvalue);
+             }
+           }
+         }); // For flags only, fallback to `country` tag only if no other namelike values were found.
+         // See https://github.com/openstreetmap/iD/pull/8305#issuecomment-769174070
 
-             var geometries = entityGeometries();
-             var button = wrap.append('button').attr('class', 'preset-list-button').classed('expanded', false).call(uiPresetIcon().geometry(geometries.length === 1 && geometries[0]).preset(preset)).on('click', click).on('keydown', function (d3_event) {
-               // right arrow, expand the focused item
-               if (d3_event.keyCode === utilKeybinding.keyCodes[_mainLocalizer.textDirection() === 'rtl' ? '←' : '→']) {
-                 d3_event.preventDefault();
-                 d3_event.stopPropagation(); // if the item isn't expanded
-
-                 if (!select(this).classed('expanded')) {
-                   // toggle expansion (expand the item)
-                   click.call(this, d3_event);
-                 } // left arrow, collapse the focused item
-
-               } else if (d3_event.keyCode === utilKeybinding.keyCodes[_mainLocalizer.textDirection() === 'rtl' ? '→' : '←']) {
-                 d3_event.preventDefault();
-                 d3_event.stopPropagation(); // if the item is expanded
+         if (tags.man_made === 'flagpole' && !primary.size && !alternate.size && !!tags.country) {
+           var osmvalue = tags.country;
 
-                 if (select(this).classed('expanded')) {
-                   // toggle expansion (collapse the item)
-                   click.call(this, d3_event);
-                 }
-               } else {
-                 itemKeydown.call(this, d3_event);
-               }
-             });
-             var label = button.append('div').attr('class', 'label').append('div').attr('class', 'label-inner');
-             label.append('div').attr('class', 'namepart').call(svgIcon(_mainLocalizer.textDirection() === 'rtl' ? '#iD-icon-backward' : '#iD-icon-forward', 'inline')).append('span').html(function () {
-               return preset.nameLabel() + '&hellip;';
-             });
-             box = selection.append('div').attr('class', 'subgrid').style('max-height', '0px').style('opacity', 0);
-             box.append('div').attr('class', 'arrow');
-             sublist = box.append('div').attr('class', 'preset-list fillL3');
+           if (/;/.test(osmvalue)) {
+             foundSemi = true;
+           } else {
+             alternate.add(osmvalue);
            }
+         } // If any namelike value contained a semicolon, return empty set and don't try matching anything.
 
-           item.choose = function () {
-             if (!box || !sublist) return;
 
-             if (shown) {
-               shown = false;
-               box.transition().duration(200).style('opacity', '0').style('max-height', '0px').style('padding-bottom', '0px');
-             } else {
-               shown = true;
-               var members = preset.members.matchAllGeometry(entityGeometries());
-               sublist.call(drawList, members);
-               box.transition().duration(200).style('opacity', '1').style('max-height', 200 + members.collection.length * 190 + 'px').style('padding-bottom', '10px');
-             }
+         if (foundSemi) {
+           return empty;
+         } else {
+           return {
+             primary: primary,
+             alternate: alternate
            };
+         }
 
-           item.preset = preset;
-           return item;
+         function isNamelike(osmkey, which) {
+           return patterns[which].test(osmkey) && !notNames.test(osmkey);
          }
+       } // `gatherTuples()`
+       // Generate all combinations of [key,value,name] that we want to test.
+       // This prioritizes them so that the primary name and k/v pairs go first
+       //
+       // Arguments
+       //   `tryKVs`: `Object` containing primary and alternate k/v pairs to test
+       //   `tryNames`: `Object` containing primary and alternate names to test
+       // Returns
+       //   `Array`: tuple objects ordered by priority
+       //
 
-         function PresetItem(preset) {
-           function item(selection) {
-             var wrap = selection.append('div').attr('class', 'preset-list-button-wrap');
-             var geometries = entityGeometries();
-             var button = wrap.append('button').attr('class', 'preset-list-button').call(uiPresetIcon().geometry(geometries.length === 1 && geometries[0]).preset(preset)).on('click', item.choose).on('keydown', itemKeydown);
-             var label = button.append('div').attr('class', 'label').append('div').attr('class', 'label-inner');
-             var nameparts = [preset.nameLabel(), preset.subtitleLabel()].filter(Boolean);
-             label.selectAll('.namepart').data(nameparts).enter().append('div').attr('class', 'namepart').html(function (d) {
-               return d;
-             });
-             wrap.call(item.reference.button);
-             selection.call(item.reference.body);
-           }
 
-           item.choose = function () {
-             if (select(this).classed('disabled')) return;
+       function gatherTuples(tryKVs, tryNames) {
+         var tuples = [];
+         ['primary', 'alternate'].forEach(function (whichName) {
+           // test names longest to shortest
+           var arr = Array.from(tryNames[whichName]).sort(function (a, b) {
+             return b.length - a.length;
+           });
+           arr.forEach(function (n) {
+             ['primary', 'alternate'].forEach(function (whichKV) {
+               tryKVs[whichKV].forEach(function (kv) {
+                 var parts = kv.split('/', 2);
+                 var k = parts[0];
+                 var v = parts[1];
+                 tuples.push({
+                   k: k,
+                   v: v,
+                   n: n
+                 });
+               });
+             });
+           });
+         });
+         return tuples;
+       } // `_upgradeTags()`
+       // Try to match a feature to a canonical record in name-suggestion-index
+       // and upgrade the tags to match.
+       //
+       // Arguments
+       //   `tags`: `Object` containing the feature's OSM tags
+       //   `loc`: Location where this feature exists, as a [lon, lat]
+       // Returns
+       //   `Object`: The tags the the feature should have, or `null` if no changes needed
+       //
 
-             if (!context.inIntro()) {
-               _mainPresetIndex.setMostRecent(preset, entityGeometries()[0]);
-             }
 
-             context.perform(function (graph) {
-               for (var i in _entityIDs) {
-                 var entityID = _entityIDs[i];
-                 var oldPreset = _mainPresetIndex.match(graph.entity(entityID), graph);
-                 graph = actionChangePreset(entityID, oldPreset, preset)(graph);
-               }
+       function _upgradeTags(tags, loc) {
+         var newTags = Object.assign({}, tags); // shallow copy
 
-               return graph;
-             }, _t('operations.change_tags.annotation'));
-             context.validator().validate(); // rerun validation
+         var changed = false; // Before anything, perform trivial Wikipedia/Wikidata replacements
 
-             dispatch$1.call('choose', this, preset);
-           };
+         Object.keys(newTags).forEach(function (osmkey) {
+           var matchTag = osmkey.match(/^(\w+:)?wikidata$/);
 
-           item.help = function (d3_event) {
-             d3_event.stopPropagation();
-             item.reference.toggle();
-           };
+           if (matchTag) {
+             // Look at '*:wikidata' tags
+             var prefix = matchTag[1] || '';
+             var wd = newTags[osmkey];
+             var replace = _nsi.replacements[wd]; // If it matches a QID in the replacement list...
 
-           item.preset = preset;
-           item.reference = uiTagReference(preset.reference());
-           return item;
-         }
+             if (replace && replace.wikidata !== undefined) {
+               // replace or delete `*:wikidata` tag
+               changed = true;
 
-         function updateForFeatureHiddenState() {
-           if (!_entityIDs.every(context.hasEntity)) return;
-           var geometries = entityGeometries();
-           var button = context.container().selectAll('.preset-list .preset-list-button'); // remove existing tooltips
+               if (replace.wikidata) {
+                 newTags[osmkey] = replace.wikidata;
+               } else {
+                 delete newTags[osmkey];
+               }
+             }
 
-           button.call(uiTooltip().destroyAny);
-           button.each(function (item, index) {
-             var hiddenPresetFeaturesId;
+             if (replace && replace.wikipedia !== undefined) {
+               // replace or delete `*:wikipedia` tag
+               changed = true;
+               var wpkey = "".concat(prefix, "wikipedia");
 
-             for (var i in geometries) {
-               hiddenPresetFeaturesId = context.features().isHiddenPreset(item.preset, geometries[i]);
-               if (hiddenPresetFeaturesId) break;
+               if (replace.wikipedia) {
+                 newTags[wpkey] = replace.wikipedia;
+               } else {
+                 delete newTags[wpkey];
+               }
              }
+           }
+         }); // Gather key/value tag pairs to try to match
 
-             var isHiddenPreset = !context.inIntro() && !!hiddenPresetFeaturesId && (_currentPresets.length !== 1 || item.preset !== _currentPresets[0]);
-             select(this).classed('disabled', isHiddenPreset);
+         var tryKVs = gatherKVs(tags);
+         if (!tryKVs.primary.size && !tryKVs.alternate.size) return changed ? newTags : null; // Gather namelike tag values to try to match
 
-             if (isHiddenPreset) {
-               var isAutoHidden = context.features().autoHidden(hiddenPresetFeaturesId);
-               select(this).call(uiTooltip().title(_t.html('inspector.hidden_preset.' + (isAutoHidden ? 'zoom' : 'manual'), {
-                 features: _t.html('feature.' + hiddenPresetFeaturesId + '.description')
-               })).placement(index < 2 ? 'bottom' : 'top'));
-             }
-           });
-         }
+         var tryNames = gatherNames(tags); // Do `wikidata=*` or `wikipedia=*` tags identify this entity as a chain? - See #6416
+         // If so, these tags can be swapped to e.g. `brand:wikidata`/`brand:wikipedia`.
 
-         presetList.autofocus = function (val) {
-           if (!arguments.length) return _autofocus;
-           _autofocus = val;
-           return presetList;
-         };
+         var foundQID = _nsi.qids.get(tags.wikidata) || _nsi.qids.get(tags.wikipedia);
 
-         presetList.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs;
-           _entityIDs = val;
+         if (foundQID) tryNames.primary.add(foundQID); // matcher will recognize the Wikidata QID as name too
 
-           if (_entityIDs && _entityIDs.length) {
-             var presets = _entityIDs.map(function (entityID) {
-               return _mainPresetIndex.match(context.entity(entityID), context.graph());
-             });
+         if (!tryNames.primary.size && !tryNames.alternate.size) return changed ? newTags : null; // Order the [key,value,name] tuples - test primary before alternate
 
-             presetList.presets(presets);
-           }
+         var tuples = gatherTuples(tryKVs, tryNames);
 
-           return presetList;
-         };
+         var _loop = function _loop(i) {
+           var tuple = tuples[i];
 
-         presetList.presets = function (val) {
-           if (!arguments.length) return _currentPresets;
-           _currentPresets = val;
-           return presetList;
-         };
+           var hits = _nsi.matcher.match(tuple.k, tuple.v, tuple.n, loc); // Attempt to match an item in NSI
 
-         function entityGeometries() {
-           var counts = {};
 
-           for (var i in _entityIDs) {
-             var entityID = _entityIDs[i];
-             var entity = context.entity(entityID);
-             var geometry = entity.geometry(context.graph()); // Treat entities on addr:interpolation lines as points, not vertices (#3241)
+           if (!hits || !hits.length) return "continue"; // no match, try next tuple
 
-             if (geometry === 'vertex' && entity.isOnAddressLine(context.graph())) {
-               geometry = 'point';
-             }
+           if (hits[0].match !== 'primary' && hits[0].match !== 'alternate') return "break"; // a generic match, stop looking
+           // A match may contain multiple results, the first one is likely the best one for this location
+           // e.g. `['pfk-a54c14', 'kfc-1ff19c', 'kfc-658eea']`
 
-             if (!counts[geometry]) counts[geometry] = 0;
-             counts[geometry] += 1;
-           }
+           var itemID = void 0,
+               item = void 0;
 
-           return Object.keys(counts).sort(function (geom1, geom2) {
-             return counts[geom2] - counts[geom1];
-           });
-         }
+           for (var j = 0; j < hits.length; j++) {
+             var hit = hits[j];
+             itemID = hit.itemID;
+             if (_nsi.dissolved[itemID]) continue; // Don't upgrade to a dissolved item
 
-         function combinedEntityExtent() {
-           return _entityIDs.reduce(function (extent, entityID) {
-             var entity = context.graph().entity(entityID);
-             return extent.extend(entity.extent(context.graph()));
-           }, geoExtent());
-         }
+             item = _nsi.ids.get(itemID);
+             if (!item) continue;
+             var mainTag = item.mainTag; // e.g. `brand:wikidata`
 
-         return utilRebind(presetList, dispatch$1, 'on');
-       }
+             var itemQID = item.tags[mainTag]; // e.g. `brand:wikidata` qid
 
-       function uiInspector(context) {
-         var presetList = uiPresetList(context);
-         var entityEditor = uiEntityEditor(context);
-         var wrap = select(null),
-             presetPane = select(null),
-             editorPane = select(null);
-         var _state = 'select';
+             var notQID = newTags["not:".concat(mainTag)]; // e.g. `not:brand:wikidata` qid
 
-         var _entityIDs;
+             if ( // Exceptions, skip this hit
+             !itemQID || itemQID === notQID || // No `*:wikidata` or matched a `not:*:wikidata`
+             newTags.office && !item.tags.office // feature may be a corporate office for a brand? - #6416
+             ) {
+                 item = null;
+                 continue; // continue looking
+               } else {
+                 break; // use `item`
+               }
+           } // Can't use any of these hits, try next tuple..
 
-         var _newFeature = false;
 
-         function inspector(selection) {
-           presetList.entityIDs(_entityIDs).autofocus(_newFeature).on('choose', inspector.setPreset).on('cancel', function () {
-             inspector.setPreset();
-           });
-           entityEditor.state(_state).entityIDs(_entityIDs).on('choose', inspector.showList);
-           wrap = selection.selectAll('.panewrap').data([0]);
-           var enter = wrap.enter().append('div').attr('class', 'panewrap');
-           enter.append('div').attr('class', 'preset-list-pane pane');
-           enter.append('div').attr('class', 'entity-editor-pane pane');
-           wrap = wrap.merge(enter);
-           presetPane = wrap.selectAll('.preset-list-pane');
-           editorPane = wrap.selectAll('.entity-editor-pane');
+           if (!item) return "continue"; // At this point we have matched a canonical item and can suggest tag upgrades..
 
-           function shouldDefaultToPresetList() {
-             // always show the inspector on hover
-             if (_state !== 'select') return false; // can only change preset on single selection
+           var tkv = item.tkv;
+           var parts = tkv.split('/', 3); // tkv = "tree/key/value"
 
-             if (_entityIDs.length !== 1) return false;
-             var entityID = _entityIDs[0];
-             var entity = context.hasEntity(entityID);
-             if (!entity) return false; // default to inspector if there are already tags
+           var k = parts[1];
+           var v = parts[2];
+           var category = _nsi.data[tkv];
+           var properties = category.properties || {}; // Preserve some tags that we specifically don't want NSI to overwrite. ('^name', sometimes)
 
-             if (entity.hasNonGeometryTags()) return false; // prompt to select preset if feature is new and untagged
+           var preserveTags = item.preserveTags || properties.preserveTags || [];
+           var regexes = preserveTags.map(function (s) {
+             return new RegExp(s, 'i');
+           });
+           regexes.push(/^building$/i, /^takeaway$/i);
+           var keepTags = {};
+           Object.keys(newTags).forEach(function (osmkey) {
+             if (regexes.some(function (regex) {
+               return regex.test(osmkey);
+             })) {
+               keepTags[osmkey] = newTags[osmkey];
+             }
+           }); // Remove any primary tags ("amenity", "craft", "shop", "man_made", "route", etc)
+           // with a value like `amenity=yes` or `shop=yes`
 
-             if (_newFeature) return true; // all existing features except vertices should default to inspector
+           _nsi.kvt.forEach(function (vmap, k) {
+             if (newTags[k] === 'yes') delete newTags[k];
+           }); // Replace mistagged `wikidata`/`wikipedia` with e.g. `brand:wikidata`/`brand:wikipedia`
 
-             if (entity.geometry(context.graph()) !== 'vertex') return false; // show vertex relations if any
 
-             if (context.graph().parentRelations(entity).length) return false; // show vertex issues if there are any
+           if (foundQID) {
+             delete newTags.wikipedia;
+             delete newTags.wikidata;
+           } // Do the tag upgrade
 
-             if (context.validator().getEntityIssues(entityID).length) return false; // show turn retriction editor for junction vertices
 
-             if (entity.isHighwayIntersection(context.graph())) return false; // otherwise show preset list for uninteresting vertices
+           Object.assign(newTags, item.tags, keepTags); // Special `branch` splitting rules - IF..
+           // - NSI is suggesting to replace `name`, AND
+           // - `branch` doesn't already contain something, AND
+           // - original name has not moved to an alternate name (e.g. "Dunkin' Donuts" -> "Dunkin'"), AND
+           // - original name is "some name" + "some stuff", THEN
+           // consider splitting `name` into `name`/`branch`..
 
-             return true;
-           }
+           var origName = tags.name;
+           var newName = newTags.name;
 
-           if (shouldDefaultToPresetList()) {
-             wrap.style('right', '-100%');
-             editorPane.classed('hide', true);
-             presetPane.classed('hide', false).call(presetList);
-           } else {
-             wrap.style('right', '0%');
-             presetPane.classed('hide', true);
-             editorPane.classed('hide', false).call(entityEditor);
-           }
+           if (newName && origName && newName !== origName && !newTags.branch) {
+             var newNames = gatherNames(newTags);
+             var newSet = new Set([].concat(_toConsumableArray(newNames.primary), _toConsumableArray(newNames.alternate)));
+             var isMoved = newSet.has(origName); // another tag holds the original name now
 
-           var footer = selection.selectAll('.footer').data([0]);
-           footer = footer.enter().append('div').attr('class', 'footer').merge(footer);
-           footer.call(uiViewOnOSM(context).what(context.hasEntity(_entityIDs.length === 1 && _entityIDs[0])));
-         }
+             if (!isMoved) {
+               // Test name fragments, longest to shortest, to fit them into a "Name Branch" pattern.
+               // e.g. "TUI ReiseCenter - Neuss Innenstadt" -> ["TUI", "ReiseCenter", "Neuss", "Innenstadt"]
+               var nameParts = origName.split(/[\s\-\/,.]/);
 
-         inspector.showList = function (presets) {
-           presetPane.classed('hide', false);
-           wrap.transition().styleTween('right', function () {
-             return interpolate('0%', '-100%');
-           }).on('end', function () {
-             editorPane.classed('hide', true);
-           });
+               for (var split = nameParts.length; split > 0; split--) {
+                 var name = nameParts.slice(0, split).join(' '); // e.g. "TUI ReiseCenter"
 
-           if (presets) {
-             presetList.presets(presets);
-           }
+                 var branch = nameParts.slice(split).join(' '); // e.g. "Neuss Innenstadt"
 
-           presetPane.call(presetList.autofocus(true));
-         };
+                 var nameHits = _nsi.matcher.match(k, v, name, loc);
 
-         inspector.setPreset = function (preset) {
-           // upon setting multipolygon, go to the area preset list instead of the editor
-           if (preset && preset.id === 'type/multipolygon') {
-             presetPane.call(presetList.autofocus(true));
-           } else {
-             editorPane.classed('hide', false);
-             wrap.transition().styleTween('right', function () {
-               return interpolate('-100%', '0%');
-             }).on('end', function () {
-               presetPane.classed('hide', true);
-             });
+                 if (!nameHits || !nameHits.length) continue; // no match, try next name fragment
 
-             if (preset) {
-               entityEditor.presets([preset]);
-             }
+                 if (nameHits.some(function (hit) {
+                   return hit.itemID === itemID;
+                 })) {
+                   // matched the name fragment to the same itemID above
+                   if (branch) {
+                     if (notBranches.test(branch)) {
+                       // "branch" was detected but is noise ("factory outlet", etc)
+                       newTags.name = origName; // Leave `name` alone, this part of the name may be significant..
+                     } else {
+                       var branchHits = _nsi.matcher.match(k, v, branch, loc);
+
+                       if (branchHits && branchHits.length) {
+                         // if "branch" matched something else in NSI..
+                         if (branchHits[0].match === 'primary' || branchHits[0].match === 'alternate') {
+                           // if another brand! (e.g. "KFC - Taco Bell"?)
+                           return {
+                             v: null
+                           }; //   bail out - can't suggest tags in this case
+                         } // else a generic (e.g. "gas", "cafe") - ignore
+
+                       } else {
+                         // "branch" is not noise and not something in NSI
+                         newTags.branch = branch; // Stick it in the `branch` tag..
+                       }
+                     }
+                   }
 
-             editorPane.call(entityEditor);
+                   break;
+                 }
+               }
+             }
            }
-         };
-
-         inspector.state = function (val) {
-           if (!arguments.length) return _state;
-           _state = val;
-           entityEditor.state(_state); // remove any old field help overlay that might have gotten attached to the inspector
 
-           context.container().selectAll('.field-help-body').remove();
-           return inspector;
+           return {
+             v: newTags
+           };
          };
 
-         inspector.entityIDs = function (val) {
-           if (!arguments.length) return _entityIDs;
-           _entityIDs = val;
-           return inspector;
-         };
+         for (var i = 0; i < tuples.length; i++) {
+           var _ret = _loop(i);
 
-         inspector.newFeature = function (val) {
-           if (!arguments.length) return _newFeature;
-           _newFeature = val;
-           return inspector;
-         };
+           if (_ret === "continue") continue;
+           if (_ret === "break") break;
+           if (_typeof(_ret) === "object") return _ret.v;
+         }
 
-         return inspector;
-       }
+         return changed ? newTags : null;
+       } // `_isGenericName()`
+       // Is the `name` tag generic?
+       //
+       // Arguments
+       //   `tags`: `Object` containing the feature's OSM tags
+       // Returns
+       //   `true` if it is generic, `false` if not
+       //
 
-       function uiSidebar(context) {
-         var inspector = uiInspector(context);
-         var dataEditor = uiDataEditor(context);
-         var noteEditor = uiNoteEditor(context);
-         var improveOsmEditor = uiImproveOsmEditor(context);
-         var keepRightEditor = uiKeepRightEditor(context);
-         var osmoseEditor = uiOsmoseEditor(context);
 
-         var _current;
+       function _isGenericName(tags) {
+         var n = tags.name;
+         if (!n) return false; // tryNames just contains the `name` tag value and nothing else
 
-         var _wasData = false;
-         var _wasNote = false;
-         var _wasQaItem = false; // use pointer events on supported platforms; fallback to mouse events
+         var tryNames = {
+           primary: new Set([n]),
+           alternate: new Set()
+         }; // Gather key/value tag pairs to try to match
 
-         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
+         var tryKVs = gatherKVs(tags);
+         if (!tryKVs.primary.size && !tryKVs.alternate.size) return false; // Order the [key,value,name] tuples - test primary before alternate
 
-         function sidebar(selection) {
-           var container = context.container();
-           var minWidth = 240;
-           var sidebarWidth;
-           var containerWidth;
-           var dragOffset; // Set the initial width constraints
+         var tuples = gatherTuples(tryKVs, tryNames);
 
-           selection.style('min-width', minWidth + 'px').style('max-width', '400px').style('width', '33.3333%');
-           var resizer = selection.append('div').attr('class', 'sidebar-resizer').on(_pointerPrefix + 'down.sidebar-resizer', pointerdown);
-           var downPointerId, lastClientX, containerLocGetter;
+         for (var i = 0; i < tuples.length; i++) {
+           var tuple = tuples[i];
 
-           function pointerdown(d3_event) {
-             if (downPointerId) return;
-             if ('button' in d3_event && d3_event.button !== 0) return;
-             downPointerId = d3_event.pointerId || 'mouse';
-             lastClientX = d3_event.clientX;
-             containerLocGetter = utilFastMouse(container.node()); // offset from edge of sidebar-resizer
+           var hits = _nsi.matcher.match(tuple.k, tuple.v, tuple.n); // Attempt to match an item in NSI
+           // If we get a `excludeGeneric` hit, this is a generic name.
 
-             dragOffset = utilFastMouse(resizer.node())(d3_event)[0] - 1;
-             sidebarWidth = selection.node().getBoundingClientRect().width;
-             containerWidth = container.node().getBoundingClientRect().width;
-             var widthPct = sidebarWidth / containerWidth * 100;
-             selection.style('width', widthPct + '%') // lock in current width
-             .style('max-width', '85%'); // but allow larger widths
 
-             resizer.classed('dragging', true);
-             select(window).on('touchmove.sidebar-resizer', function (d3_event) {
-               // disable page scrolling while resizing on touch input
-               d3_event.preventDefault();
-             }, {
-               passive: false
-             }).on(_pointerPrefix + 'move.sidebar-resizer', pointermove).on(_pointerPrefix + 'up.sidebar-resizer pointercancel.sidebar-resizer', pointerup);
-           }
+           if (hits && hits.length && hits[0].match === 'excludeGeneric') return true;
+         }
 
-           function pointermove(d3_event) {
-             if (downPointerId !== (d3_event.pointerId || 'mouse')) return;
-             d3_event.preventDefault();
-             var dx = d3_event.clientX - lastClientX;
-             lastClientX = d3_event.clientX;
-             var isRTL = _mainLocalizer.textDirection() === 'rtl';
-             var scaleX = isRTL ? 0 : 1;
-             var xMarginProperty = isRTL ? 'margin-right' : 'margin-left';
-             var x = containerLocGetter(d3_event)[0] - dragOffset;
-             sidebarWidth = isRTL ? containerWidth - x : x;
-             var isCollapsed = selection.classed('collapsed');
-             var shouldCollapse = sidebarWidth < minWidth;
-             selection.classed('collapsed', shouldCollapse);
+         return false;
+       } // PUBLIC INTERFACE
 
-             if (shouldCollapse) {
-               if (!isCollapsed) {
-                 selection.style(xMarginProperty, '-400px').style('width', '400px');
-                 context.ui().onResize([(sidebarWidth - dx) * scaleX, 0]);
-               }
-             } else {
-               var widthPct = sidebarWidth / containerWidth * 100;
-               selection.style(xMarginProperty, null).style('width', widthPct + '%');
 
-               if (isCollapsed) {
-                 context.ui().onResize([-sidebarWidth * scaleX, 0]);
-               } else {
-                 context.ui().onResize([-dx * scaleX, 0]);
-               }
-             }
-           }
+       var serviceNsi = {
+         // `init()`
+         // On init, start preparing the name-suggestion-index
+         //
+         init: function init() {
+           // Note: service.init is called immediately after the presetManager has started loading its data.
+           // We expect to chain onto an unfulfilled promise here.
+           setNsiSources();
+           _mainPresetIndex.ensureLoaded().then(function () {
+             return loadNsiPresets();
+           }).then(function () {
+             return delay(100);
+           }) // wait briefly for locationSets to enter the locationManager queue
+           .then(function () {
+             return _mainLocations.mergeLocationSets([]);
+           }) // wait for locationSets to resolve
+           .then(function () {
+             return loadNsiData();
+           }).then(function () {
+             return _nsiStatus = 'ok';
+           })["catch"](function () {
+             return _nsiStatus = 'failed';
+           });
 
-           function pointerup(d3_event) {
-             if (downPointerId !== (d3_event.pointerId || 'mouse')) return;
-             downPointerId = null;
-             resizer.classed('dragging', false);
-             select(window).on('touchmove.sidebar-resizer', null).on(_pointerPrefix + 'move.sidebar-resizer', null).on(_pointerPrefix + 'up.sidebar-resizer pointercancel.sidebar-resizer', null);
+           function delay(msec) {
+             return new Promise(function (resolve) {
+               window.setTimeout(resolve, msec);
+             });
            }
+         },
+         // `reset()`
+         // Reset is called when user saves data to OSM (does nothing here)
+         //
+         reset: function reset() {},
+         // `status()`
+         // To let other code know how it's going...
+         //
+         // Returns
+         //   `String`: 'loading', 'ok', 'failed'
+         //
+         status: function status() {
+           return _nsiStatus;
+         },
+         // `isGenericName()`
+         // Is the `name` tag generic?
+         //
+         // Arguments
+         //   `tags`: `Object` containing the feature's OSM tags
+         // Returns
+         //   `true` if it is generic, `false` if not
+         //
+         isGenericName: function isGenericName(tags) {
+           return _isGenericName(tags);
+         },
+         // `upgradeTags()`
+         // Suggest tag upgrades.
+         // This function will not modify the input tags, it makes a copy.
+         //
+         // Arguments
+         //   `tags`: `Object` containing the feature's OSM tags
+         //   `loc`: Location where this feature exists, as a [lon, lat]
+         // Returns
+         //   `Object`: The tags the the feature should have, or `null` if no change
+         //
+         upgradeTags: function upgradeTags(tags, loc) {
+           return _upgradeTags(tags, loc);
+         },
+         // `cache()`
+         // Direct access to the NSI cache, useful for testing or breaking things
+         //
+         // Returns
+         //   `Object`: the internal NSI cache
+         //
+         cache: function cache() {
+           return _nsi;
+         }
+       };
 
-           var featureListWrap = selection.append('div').attr('class', 'feature-list-pane').call(uiFeatureList(context));
-           var inspectorWrap = selection.append('div').attr('class', 'inspector-hidden inspector-wrap');
+       var apibase$1 = 'https://openstreetcam.org';
+       var maxResults$1 = 1000;
+       var tileZoom$1 = 14;
+       var tiler$3 = utilTiler().zoomExtent([tileZoom$1, tileZoom$1]).skipNullIsland(true);
+       var dispatch$3 = dispatch$8('loadedImages');
+       var imgZoom = d3_zoom().extent([[0, 0], [320, 240]]).translateExtent([[0, 0], [320, 240]]).scaleExtent([1, 15]);
 
-           var hoverModeSelect = function hoverModeSelect(targets) {
-             context.container().selectAll('.feature-list-item button').classed('hover', false);
+       var _oscCache;
 
-             if (context.selectedIDs().length > 1 && targets && targets.length) {
-               var elements = context.container().selectAll('.feature-list-item button').filter(function (node) {
-                 return targets.indexOf(node) !== -1;
-               });
+       var _oscSelectedImage;
 
-               if (!elements.empty()) {
-                 elements.classed('hover', true);
-               }
-             }
-           };
+       var _loadViewerPromise$1;
 
-           sidebar.hoverModeSelect = throttle(hoverModeSelect, 200);
+       function abortRequest$3(controller) {
+         controller.abort();
+       }
 
-           function hover(targets) {
-             var datum = targets && targets.length && targets[0];
+       function maxPageAtZoom(z) {
+         if (z < 15) return 2;
+         if (z === 15) return 5;
+         if (z === 16) return 10;
+         if (z === 17) return 20;
+         if (z === 18) return 40;
+         if (z > 18) return 80;
+       }
 
-             if (datum && datum.__featurehash__) {
-               // hovering on data
-               _wasData = true;
-               sidebar.show(dataEditor.datum(datum));
-               selection.selectAll('.sidebar-component').classed('inspector-hover', true);
-             } else if (datum instanceof osmNote) {
-               if (context.mode().id === 'drag-note') return;
-               _wasNote = true;
-               var osm = services.osm;
+       function loadTiles$1(which, url, projection) {
+         var currZoom = Math.floor(geoScaleToZoom(projection.scale()));
+         var tiles = tiler$3.getTiles(projection); // abort inflight requests that are no longer needed
 
-               if (osm) {
-                 datum = osm.getNote(datum.id); // marker may contain stale data - get latest
-               }
+         var cache = _oscCache[which];
+         Object.keys(cache.inflight).forEach(function (k) {
+           var wanted = tiles.find(function (tile) {
+             return k.indexOf(tile.id + ',') === 0;
+           });
 
-               sidebar.show(noteEditor.note(datum));
-               selection.selectAll('.sidebar-component').classed('inspector-hover', true);
-             } else if (datum instanceof QAItem) {
-               _wasQaItem = true;
-               var errService = services[datum.service];
+           if (!wanted) {
+             abortRequest$3(cache.inflight[k]);
+             delete cache.inflight[k];
+           }
+         });
+         tiles.forEach(function (tile) {
+           loadNextTilePage$1(which, currZoom, url, tile);
+         });
+       }
 
-               if (errService) {
-                 // marker may contain stale data - get latest
-                 datum = errService.getError(datum.id);
-               } // Currently only three possible services
+       function loadNextTilePage$1(which, currZoom, url, tile) {
+         var cache = _oscCache[which];
+         var bbox = tile.extent.bbox();
+         var maxPages = maxPageAtZoom(currZoom);
+         var nextPage = cache.nextPage[tile.id] || 1;
+         var params = utilQsString({
+           ipp: maxResults$1,
+           page: nextPage,
+           // client_id: clientId,
+           bbTopLeft: [bbox.maxY, bbox.minX].join(','),
+           bbBottomRight: [bbox.minY, bbox.maxX].join(',')
+         }, true);
+         if (nextPage > maxPages) return;
+         var id = tile.id + ',' + String(nextPage);
+         if (cache.loaded[id] || cache.inflight[id]) return;
+         var controller = new AbortController();
+         cache.inflight[id] = controller;
+         var options = {
+           method: 'POST',
+           signal: controller.signal,
+           body: params,
+           headers: {
+             'Content-Type': 'application/x-www-form-urlencoded'
+           }
+         };
+         d3_json(url, options).then(function (data) {
+           cache.loaded[id] = true;
+           delete cache.inflight[id];
 
+           if (!data || !data.currentPageItems || !data.currentPageItems.length) {
+             throw new Error('No Data');
+           }
 
-               var errEditor;
+           var features = data.currentPageItems.map(function (item) {
+             var loc = [+item.lng, +item.lat];
+             var d;
 
-               if (datum.service === 'keepRight') {
-                 errEditor = keepRightEditor;
-               } else if (datum.service === 'osmose') {
-                 errEditor = osmoseEditor;
-               } else {
-                 errEditor = improveOsmEditor;
-               }
+             if (which === 'images') {
+               d = {
+                 loc: loc,
+                 key: item.id,
+                 ca: +item.heading,
+                 captured_at: item.shot_date || item.date_added,
+                 captured_by: item.username,
+                 imagePath: item.lth_name,
+                 sequence_id: item.sequence_id,
+                 sequence_index: +item.sequence_index
+               }; // cache sequence info
 
-               context.container().selectAll('.qaItem.' + datum.service).classed('hover', function (d) {
-                 return d.id === datum.id;
-               });
-               sidebar.show(errEditor.error(datum));
-               selection.selectAll('.sidebar-component').classed('inspector-hover', true);
-             } else if (!_current && datum instanceof osmEntity) {
-               featureListWrap.classed('inspector-hidden', true);
-               inspectorWrap.classed('inspector-hidden', false).classed('inspector-hover', true);
+               var seq = _oscCache.sequences[d.sequence_id];
 
-               if (!inspector.entityIDs() || !utilArrayIdentical(inspector.entityIDs(), [datum.id]) || inspector.state() !== 'hover') {
-                 inspector.state('hover').entityIDs([datum.id]).newFeature(false);
-                 inspectorWrap.call(inspector);
+               if (!seq) {
+                 seq = {
+                   rotation: 0,
+                   images: []
+                 };
+                 _oscCache.sequences[d.sequence_id] = seq;
                }
-             } else if (!_current) {
-               featureListWrap.classed('inspector-hidden', false);
-               inspectorWrap.classed('inspector-hidden', true);
-               inspector.state('hide');
-             } else if (_wasData || _wasNote || _wasQaItem) {
-               _wasNote = false;
-               _wasData = false;
-               _wasQaItem = false;
-               context.container().selectAll('.note').classed('hover', false);
-               context.container().selectAll('.qaItem').classed('hover', false);
-               sidebar.hide();
+
+               seq.images[d.sequence_index] = d;
+               _oscCache.images.forImageKey[d.key] = d; // cache imageKey -> image
              }
+
+             return {
+               minX: loc[0],
+               minY: loc[1],
+               maxX: loc[0],
+               maxY: loc[1],
+               data: d
+             };
+           });
+           cache.rtree.load(features);
+
+           if (data.currentPageItems.length === maxResults$1) {
+             // more pages to load
+             cache.nextPage[tile.id] = nextPage + 1;
+             loadNextTilePage$1(which, currZoom, url, tile);
+           } else {
+             cache.nextPage[tile.id] = Infinity; // no more pages to load
            }
 
-           sidebar.hover = throttle(hover, 200);
+           if (which === 'images') {
+             dispatch$3.call('loadedImages');
+           }
+         })["catch"](function () {
+           cache.loaded[id] = true;
+           delete cache.inflight[id];
+         });
+       } // partition viewport into higher zoom tiles
 
-           sidebar.intersects = function (extent) {
-             var rect = selection.node().getBoundingClientRect();
-             return extent.intersects([context.projection.invert([0, rect.height]), context.projection.invert([rect.width, 0])]);
-           };
 
-           sidebar.select = function (ids, newFeature) {
-             sidebar.hide();
+       function partitionViewport$1(projection) {
+         var z = geoScaleToZoom(projection.scale());
+         var z2 = Math.ceil(z * 2) / 2 + 2.5; // round to next 0.5 and add 2.5
 
-             if (ids && ids.length) {
-               var entity = ids.length === 1 && context.entity(ids[0]);
+         var tiler = utilTiler().zoomExtent([z2, z2]);
+         return tiler.getTiles(projection).map(function (tile) {
+           return tile.extent;
+         });
+       } // no more than `limit` results per partition.
 
-               if (entity && newFeature && selection.classed('collapsed')) {
-                 // uncollapse the sidebar
-                 var extent = entity.extent(context.graph());
-                 sidebar.expand(sidebar.intersects(extent));
-               }
 
-               featureListWrap.classed('inspector-hidden', true);
-               inspectorWrap.classed('inspector-hidden', false).classed('inspector-hover', false); // reload the UI even if the ids are the same since the entities
-               // themselves may have changed
+       function searchLimited$1(limit, projection, rtree) {
+         limit = limit || 5;
+         return partitionViewport$1(projection).reduce(function (result, extent) {
+           var found = rtree.search(extent.bbox()).slice(0, limit).map(function (d) {
+             return d.data;
+           });
+           return found.length ? result.concat(found) : result;
+         }, []);
+       }
 
-               inspector.state('select').entityIDs(ids).newFeature(newFeature);
-               inspectorWrap.call(inspector);
-             } else {
-               inspector.state('hide');
-             }
-           };
+       var serviceOpenstreetcam = {
+         init: function init() {
+           if (!_oscCache) {
+             this.reset();
+           }
 
-           sidebar.showPresetList = function () {
-             inspector.showList();
-           };
+           this.event = utilRebind(this, dispatch$3, 'on');
+         },
+         reset: function reset() {
+           if (_oscCache) {
+             Object.values(_oscCache.images.inflight).forEach(abortRequest$3);
+           }
 
-           sidebar.show = function (component, element) {
-             featureListWrap.classed('inspector-hidden', true);
-             inspectorWrap.classed('inspector-hidden', true);
-             if (_current) _current.remove();
-             _current = selection.append('div').attr('class', 'sidebar-component').call(component, element);
+           _oscCache = {
+             images: {
+               inflight: {},
+               loaded: {},
+               nextPage: {},
+               rtree: new RBush(),
+               forImageKey: {}
+             },
+             sequences: {}
            };
+           _oscSelectedImage = null;
+         },
+         images: function images(projection) {
+           var limit = 5;
+           return searchLimited$1(limit, projection, _oscCache.images.rtree);
+         },
+         sequences: function sequences(projection) {
+           var viewport = projection.clipExtent();
+           var min = [viewport[0][0], viewport[1][1]];
+           var max = [viewport[1][0], viewport[0][1]];
+           var bbox = geoExtent(projection.invert(min), projection.invert(max)).bbox();
+           var sequenceKeys = {}; // all sequences for images in viewport
 
-           sidebar.hide = function () {
-             featureListWrap.classed('inspector-hidden', false);
-             inspectorWrap.classed('inspector-hidden', true);
-             if (_current) _current.remove();
-             _current = null;
-           };
+           _oscCache.images.rtree.search(bbox).forEach(function (d) {
+             sequenceKeys[d.data.sequence_id] = true;
+           }); // make linestrings from those sequences
 
-           sidebar.expand = function (moveMap) {
-             if (selection.classed('collapsed')) {
-               sidebar.toggle(moveMap);
-             }
-           };
 
-           sidebar.collapse = function (moveMap) {
-             if (!selection.classed('collapsed')) {
-               sidebar.toggle(moveMap);
+           var lineStrings = [];
+           Object.keys(sequenceKeys).forEach(function (sequenceKey) {
+             var seq = _oscCache.sequences[sequenceKey];
+             var images = seq && seq.images;
+
+             if (images) {
+               lineStrings.push({
+                 type: 'LineString',
+                 coordinates: images.map(function (d) {
+                   return d.loc;
+                 }).filter(Boolean),
+                 properties: {
+                   captured_at: images[0] ? images[0].captured_at : null,
+                   captured_by: images[0] ? images[0].captured_by : null,
+                   key: sequenceKey
+                 }
+               });
              }
-           };
+           });
+           return lineStrings;
+         },
+         cachedImage: function cachedImage(imageKey) {
+           return _oscCache.images.forImageKey[imageKey];
+         },
+         loadImages: function loadImages(projection) {
+           var url = apibase$1 + '/1.0/list/nearby-photos/';
+           loadTiles$1('images', url, projection);
+         },
+         ensureViewerLoaded: function ensureViewerLoaded(context) {
+           if (_loadViewerPromise$1) return _loadViewerPromise$1; // add osc-wrapper
 
-           sidebar.toggle = function (moveMap) {
-             // Don't allow sidebar to toggle when the user is in the walkthrough.
-             if (context.inIntro()) return;
-             var isCollapsed = selection.classed('collapsed');
-             var isCollapsing = !isCollapsed;
-             var isRTL = _mainLocalizer.textDirection() === 'rtl';
-             var scaleX = isRTL ? 0 : 1;
-             var xMarginProperty = isRTL ? 'margin-right' : 'margin-left';
-             sidebarWidth = selection.node().getBoundingClientRect().width; // switch from % to px
+           var wrap = context.container().select('.photoviewer').selectAll('.osc-wrapper').data([0]);
+           var that = this;
+           var wrapEnter = wrap.enter().append('div').attr('class', 'photo-wrapper osc-wrapper').classed('hide', true).call(imgZoom.on('zoom', zoomPan)).on('dblclick.zoom', null);
+           wrapEnter.append('div').attr('class', 'photo-attribution fillD');
+           var controlsEnter = wrapEnter.append('div').attr('class', 'photo-controls-wrap').append('div').attr('class', 'photo-controls');
+           controlsEnter.append('button').on('click.back', step(-1)).html('◄');
+           controlsEnter.append('button').on('click.rotate-ccw', rotate(-90)).html('⤿');
+           controlsEnter.append('button').on('click.rotate-cw', rotate(90)).html('⤾');
+           controlsEnter.append('button').on('click.forward', step(1)).html('►');
+           wrapEnter.append('div').attr('class', 'osc-image-wrap'); // Register viewer resize handler
+
+           context.ui().photoviewer.on('resize.openstreetcam', function (dimensions) {
+             imgZoom = d3_zoom().extent([[0, 0], dimensions]).translateExtent([[0, 0], dimensions]).scaleExtent([1, 15]).on('zoom', zoomPan);
+           });
+
+           function zoomPan(d3_event) {
+             var t = d3_event.transform;
+             context.container().select('.photoviewer .osc-image-wrap').call(utilSetTransform, t.x, t.y, t.k);
+           }
+
+           function rotate(deg) {
+             return function () {
+               if (!_oscSelectedImage) return;
+               var sequenceKey = _oscSelectedImage.sequence_id;
+               var sequence = _oscCache.sequences[sequenceKey];
+               if (!sequence) return;
+               var r = sequence.rotation || 0;
+               r += deg;
+               if (r > 180) r -= 360;
+               if (r < -180) r += 360;
+               sequence.rotation = r;
+               var wrap = context.container().select('.photoviewer .osc-wrapper');
+               wrap.transition().duration(100).call(imgZoom.transform, identity$2);
+               wrap.selectAll('.osc-image').transition().duration(100).style('transform', 'rotate(' + r + 'deg)');
+             };
+           }
+
+           function step(stepBy) {
+             return function () {
+               if (!_oscSelectedImage) return;
+               var sequenceKey = _oscSelectedImage.sequence_id;
+               var sequence = _oscCache.sequences[sequenceKey];
+               if (!sequence) return;
+               var nextIndex = _oscSelectedImage.sequence_index + stepBy;
+               var nextImage = sequence.images[nextIndex];
+               if (!nextImage) return;
+               context.map().centerEase(nextImage.loc);
+               that.selectImage(context, nextImage.key);
+             };
+           } // don't need any async loading so resolve immediately
+
+
+           _loadViewerPromise$1 = Promise.resolve();
+           return _loadViewerPromise$1;
+         },
+         showViewer: function showViewer(context) {
+           var viewer = context.container().select('.photoviewer').classed('hide', false);
+           var isHidden = viewer.selectAll('.photo-wrapper.osc-wrapper.hide').size();
+
+           if (isHidden) {
+             viewer.selectAll('.photo-wrapper:not(.osc-wrapper)').classed('hide', true);
+             viewer.selectAll('.photo-wrapper.osc-wrapper').classed('hide', false);
+           }
+
+           return this;
+         },
+         hideViewer: function hideViewer(context) {
+           _oscSelectedImage = null;
+           this.updateUrlImage(null);
+           var viewer = context.container().select('.photoviewer');
+           if (!viewer.empty()) viewer.datum(null);
+           viewer.classed('hide', true).selectAll('.photo-wrapper').classed('hide', true);
+           context.container().selectAll('.viewfield-group, .sequence, .icon-sign').classed('currentView', false);
+           return this.setStyles(context, null, true);
+         },
+         selectImage: function selectImage(context, imageKey) {
+           var d = this.cachedImage(imageKey);
+           _oscSelectedImage = d;
+           this.updateUrlImage(imageKey);
+           var viewer = context.container().select('.photoviewer');
+           if (!viewer.empty()) viewer.datum(d);
+           this.setStyles(context, null, true);
+           context.container().selectAll('.icon-sign').classed('currentView', false);
+           if (!d) return this;
+           var wrap = context.container().select('.photoviewer .osc-wrapper');
+           var imageWrap = wrap.selectAll('.osc-image-wrap');
+           var attribution = wrap.selectAll('.photo-attribution').html('');
+           wrap.transition().duration(100).call(imgZoom.transform, identity$2);
+           imageWrap.selectAll('.osc-image').remove();
 
-             selection.style('width', sidebarWidth + 'px');
-             var startMargin, endMargin, lastMargin;
+           if (d) {
+             var sequence = _oscCache.sequences[d.sequence_id];
+             var r = sequence && sequence.rotation || 0;
+             imageWrap.append('img').attr('class', 'osc-image').attr('src', apibase$1 + '/' + d.imagePath).style('transform', 'rotate(' + r + 'deg)');
 
-             if (isCollapsing) {
-               startMargin = lastMargin = 0;
-               endMargin = -sidebarWidth;
-             } else {
-               startMargin = lastMargin = -sidebarWidth;
-               endMargin = 0;
+             if (d.captured_by) {
+               attribution.append('a').attr('class', 'captured_by').attr('target', '_blank').attr('href', 'https://openstreetcam.org/user/' + encodeURIComponent(d.captured_by)).html('@' + d.captured_by);
+               attribution.append('span').html('|');
              }
 
-             if (!isCollapsing) {
-               // unhide the sidebar's content before it transitions onscreen
-               selection.classed('collapsed', isCollapsing);
+             if (d.captured_at) {
+               attribution.append('span').attr('class', 'captured_at').html(localeDateString(d.captured_at));
+               attribution.append('span').html('|');
              }
 
-             selection.transition().style(xMarginProperty, endMargin + 'px').tween('panner', function () {
-               var i = d3_interpolateNumber(startMargin, endMargin);
-               return function (t) {
-                 var dx = lastMargin - Math.round(i(t));
-                 lastMargin = lastMargin - dx;
-                 context.ui().onResize(moveMap ? undefined : [dx * scaleX, 0]);
-               };
-             }).on('end', function () {
-               if (isCollapsing) {
-                 // hide the sidebar's content after it transitions offscreen
-                 selection.classed('collapsed', isCollapsing);
-               } // switch back from px to %
+             attribution.append('a').attr('class', 'image-link').attr('target', '_blank').attr('href', 'https://openstreetcam.org/details/' + d.sequence_id + '/' + d.sequence_index).html('openstreetcam.org');
+           }
 
+           return this;
 
-               if (!isCollapsing) {
-                 var containerWidth = container.node().getBoundingClientRect().width;
-                 var widthPct = sidebarWidth / containerWidth * 100;
-                 selection.style(xMarginProperty, null).style('width', widthPct + '%');
-               }
-             });
-           }; // toggle the sidebar collapse when double-clicking the resizer
+           function localeDateString(s) {
+             if (!s) return null;
+             var options = {
+               day: 'numeric',
+               month: 'short',
+               year: 'numeric'
+             };
+             var d = new Date(s);
+             if (isNaN(d.getTime())) return null;
+             return d.toLocaleDateString(_mainLocalizer.localeCode(), options);
+           }
+         },
+         getSelectedImage: function getSelectedImage() {
+           return _oscSelectedImage;
+         },
+         getSequenceKeyForImage: function getSequenceKeyForImage(d) {
+           return d && d.sequence_id;
+         },
+         // Updates the currently highlighted sequence and selected bubble.
+         // Reset is only necessary when interacting with the viewport because
+         // this implicitly changes the currently selected bubble/sequence
+         setStyles: function setStyles(context, hovered, reset) {
+           if (reset) {
+             // reset all layers
+             context.container().selectAll('.viewfield-group').classed('highlighted', false).classed('hovered', false).classed('currentView', false);
+             context.container().selectAll('.sequence').classed('highlighted', false).classed('currentView', false);
+           }
 
+           var hoveredImageKey = hovered && hovered.key;
+           var hoveredSequenceKey = this.getSequenceKeyForImage(hovered);
+           var hoveredSequence = hoveredSequenceKey && _oscCache.sequences[hoveredSequenceKey];
+           var hoveredImageKeys = hoveredSequence && hoveredSequence.images.map(function (d) {
+             return d.key;
+           }) || [];
+           var viewer = context.container().select('.photoviewer');
+           var selected = viewer.empty() ? undefined : viewer.datum();
+           var selectedImageKey = selected && selected.key;
+           var selectedSequenceKey = this.getSequenceKeyForImage(selected);
+           var selectedSequence = selectedSequenceKey && _oscCache.sequences[selectedSequenceKey];
+           var selectedImageKeys = selectedSequence && selectedSequence.images.map(function (d) {
+             return d.key;
+           }) || []; // highlight sibling viewfields on either the selected or the hovered sequences
 
-           resizer.on('dblclick', function (d3_event) {
-             d3_event.preventDefault();
+           var highlightedImageKeys = utilArrayUnion(hoveredImageKeys, selectedImageKeys);
+           context.container().selectAll('.layer-openstreetcam .viewfield-group').classed('highlighted', function (d) {
+             return highlightedImageKeys.indexOf(d.key) !== -1;
+           }).classed('hovered', function (d) {
+             return d.key === hoveredImageKey;
+           }).classed('currentView', function (d) {
+             return d.key === selectedImageKey;
+           });
+           context.container().selectAll('.layer-openstreetcam .sequence').classed('highlighted', function (d) {
+             return d.properties.key === hoveredSequenceKey;
+           }).classed('currentView', function (d) {
+             return d.properties.key === selectedSequenceKey;
+           }); // update viewfields if needed
 
-             if (d3_event.sourceEvent) {
-               d3_event.sourceEvent.preventDefault();
-             }
+           context.container().selectAll('.layer-openstreetcam .viewfield-group .viewfield').attr('d', viewfieldPath);
 
-             sidebar.toggle();
-           }); // ensure hover sidebar is closed when zooming out beyond editable zoom
+           function viewfieldPath() {
+             var d = this.parentNode.__data__;
 
-           context.map().on('crossEditableZoom.sidebar', function (within) {
-             if (!within && !selection.select('.inspector-hover').empty()) {
-               hover([]);
+             if (d.pano && d.key !== selectedImageKey) {
+               return 'M 8,13 m -10,0 a 10,10 0 1,0 20,0 a 10,10 0 1,0 -20,0';
+             } else {
+               return 'M 6,9 C 8,8.4 8,8.4 10,9 L 16,-2 C 12,-5 4,-5 0,-2 z';
              }
-           });
-         }
-
-         sidebar.showPresetList = function () {};
-
-         sidebar.hover = function () {};
+           }
 
-         sidebar.hover.cancel = function () {};
+           return this;
+         },
+         updateUrlImage: function updateUrlImage(imageKey) {
+           if (!window.mocha) {
+             var hash = utilStringQs(window.location.hash);
 
-         sidebar.intersects = function () {};
+             if (imageKey) {
+               hash.photo = 'openstreetcam/' + imageKey;
+             } else {
+               delete hash.photo;
+             }
 
-         sidebar.select = function () {};
+             window.location.replace('#' + utilQsString(hash, true));
+           }
+         },
+         cache: function cache() {
+           return _oscCache;
+         }
+       };
 
-         sidebar.show = function () {};
+       var hashes = createCommonjsModule(function (module, exports) {
+         (function () {
+           var Hashes;
 
-         sidebar.hide = function () {};
+           function utf8Encode(str) {
+             var x,
+                 y,
+                 output = '',
+                 i = -1,
+                 l;
 
-         sidebar.expand = function () {};
+             if (str && str.length) {
+               l = str.length;
 
-         sidebar.collapse = function () {};
+               while ((i += 1) < l) {
+                 /* Decode utf-16 surrogate pairs */
+                 x = str.charCodeAt(i);
+                 y = i + 1 < l ? str.charCodeAt(i + 1) : 0;
 
-         sidebar.toggle = function () {};
+                 if (0xD800 <= x && x <= 0xDBFF && 0xDC00 <= y && y <= 0xDFFF) {
+                   x = 0x10000 + ((x & 0x03FF) << 10) + (y & 0x03FF);
+                   i += 1;
+                 }
+                 /* Encode output as utf-8 */
 
-         return sidebar;
-       }
 
-       function uiSourceSwitch(context) {
-         var keys;
+                 if (x <= 0x7F) {
+                   output += String.fromCharCode(x);
+                 } else if (x <= 0x7FF) {
+                   output += String.fromCharCode(0xC0 | x >>> 6 & 0x1F, 0x80 | x & 0x3F);
+                 } else if (x <= 0xFFFF) {
+                   output += String.fromCharCode(0xE0 | x >>> 12 & 0x0F, 0x80 | x >>> 6 & 0x3F, 0x80 | x & 0x3F);
+                 } else if (x <= 0x1FFFFF) {
+                   output += String.fromCharCode(0xF0 | x >>> 18 & 0x07, 0x80 | x >>> 12 & 0x3F, 0x80 | x >>> 6 & 0x3F, 0x80 | x & 0x3F);
+                 }
+               }
+             }
 
-         function click(d3_event) {
-           d3_event.preventDefault();
-           var osm = context.connection();
-           if (!osm) return;
-           if (context.inIntro()) return;
-           if (context.history().hasChanges() && !window.confirm(_t('source_switch.lose_changes'))) return;
-           var isLive = select(this).classed('live');
-           isLive = !isLive;
-           context.enter(modeBrowse(context));
-           context.history().clearSaved(); // remove saved history
+             return output;
+           }
 
-           context.flush(); // remove stored data
+           function utf8Decode(str) {
+             var i,
+                 ac,
+                 c1,
+                 c2,
+                 c3,
+                 arr = [],
+                 l;
+             i = ac = c1 = c2 = c3 = 0;
 
-           select(this).html(isLive ? _t.html('source_switch.live') : _t.html('source_switch.dev')).classed('live', isLive).classed('chip', isLive);
-           osm["switch"](isLive ? keys[0] : keys[1]); // switch connection (warning: dispatches 'change' event)
-         }
+             if (str && str.length) {
+               l = str.length;
+               str += '';
 
-         var sourceSwitch = function sourceSwitch(selection) {
-           selection.append('a').attr('href', '#').html(_t.html('source_switch.live')).attr('class', 'live chip').on('click', click);
-         };
+               while (i < l) {
+                 c1 = str.charCodeAt(i);
+                 ac += 1;
 
-         sourceSwitch.keys = function (_) {
-           if (!arguments.length) return keys;
-           keys = _;
-           return sourceSwitch;
-         };
+                 if (c1 < 128) {
+                   arr[ac] = String.fromCharCode(c1);
+                   i += 1;
+                 } else if (c1 > 191 && c1 < 224) {
+                   c2 = str.charCodeAt(i + 1);
+                   arr[ac] = String.fromCharCode((c1 & 31) << 6 | c2 & 63);
+                   i += 2;
+                 } else {
+                   c2 = str.charCodeAt(i + 1);
+                   c3 = str.charCodeAt(i + 2);
+                   arr[ac] = String.fromCharCode((c1 & 15) << 12 | (c2 & 63) << 6 | c3 & 63);
+                   i += 3;
+                 }
+               }
+             }
 
-         return sourceSwitch;
-       }
+             return arr.join('');
+           }
+           /**
+            * Add integers, wrapping at 2^32. This uses 16-bit operations internally
+            * to work around bugs in some JS interpreters.
+            */
 
-       function uiSpinner(context) {
-         var osm = context.connection();
-         return function (selection) {
-           var img = selection.append('img').attr('src', context.imagePath('loader-black.gif')).style('opacity', 0);
 
-           if (osm) {
-             osm.on('loading.spinner', function () {
-               img.transition().style('opacity', 1);
-             }).on('loaded.spinner', function () {
-               img.transition().style('opacity', 0);
-             });
+           function safe_add(x, y) {
+             var lsw = (x & 0xFFFF) + (y & 0xFFFF),
+                 msw = (x >> 16) + (y >> 16) + (lsw >> 16);
+             return msw << 16 | lsw & 0xFFFF;
            }
-         };
-       }
-
-       function uiSplash(context) {
-         return function (selection) {
-           // Exception - if there are restorable changes, skip this splash screen.
-           // This is because we currently only support one `uiModal` at a time
-           //  and we need to show them `uiRestore`` instead of this one.
-           if (context.history().hasRestorableChanges()) return; // If user has not seen this version of the privacy policy, show the splash again.
+           /**
+            * Bitwise rotate a 32-bit number to the left.
+            */
 
-           var updateMessage = '';
-           var sawPrivacyVersion = corePreferences('sawPrivacyVersion');
-           var showSplash = !corePreferences('sawSplash');
 
-           if (sawPrivacyVersion !== context.privacyVersion) {
-             updateMessage = _t('splash.privacy_update');
-             showSplash = true;
+           function bit_rol(num, cnt) {
+             return num << cnt | num >>> 32 - cnt;
            }
+           /**
+            * Convert a raw string to a hex string
+            */
 
-           if (!showSplash) return;
-           corePreferences('sawSplash', true);
-           corePreferences('sawPrivacyVersion', context.privacyVersion); // fetch intro graph data now, while user is looking at the splash screen
-
-           _mainFileFetcher.get('intro_graph');
-           var modalSelection = uiModal(selection);
-           modalSelection.select('.modal').attr('class', 'modal-splash modal');
-           var introModal = modalSelection.select('.content').append('div').attr('class', 'fillL');
-           introModal.append('div').attr('class', 'modal-section').append('h3').html(_t.html('splash.welcome'));
-           var modalSection = introModal.append('div').attr('class', 'modal-section');
-           modalSection.append('p').html(_t.html('splash.text', {
-             version: context.version,
-             website: '<a target="_blank" href="http://ideditor.blog/">ideditor.blog</a>',
-             github: '<a target="_blank" href="https://github.com/openstreetmap/iD">github.com</a>'
-           }));
-           modalSection.append('p').html(_t.html('splash.privacy', {
-             updateMessage: updateMessage,
-             privacyLink: '<a target="_blank" href="https://github.com/openstreetmap/iD/blob/release/PRIVACY.md">' + _t('splash.privacy_policy') + '</a>'
-           }));
-           var buttonWrap = introModal.append('div').attr('class', 'modal-actions');
-           var walkthrough = buttonWrap.append('button').attr('class', 'walkthrough').on('click', function () {
-             context.container().call(uiIntro(context));
-             modalSelection.close();
-           });
-           walkthrough.append('svg').attr('class', 'logo logo-walkthrough').append('use').attr('xlink:href', '#iD-logo-walkthrough');
-           walkthrough.append('div').html(_t.html('splash.walkthrough'));
-           var startEditing = buttonWrap.append('button').attr('class', 'start-editing').on('click', modalSelection.close);
-           startEditing.append('svg').attr('class', 'logo logo-features').append('use').attr('xlink:href', '#iD-logo-features');
-           startEditing.append('div').html(_t.html('splash.start'));
-           modalSelection.select('button.close').attr('class', 'hide');
-         };
-       }
 
-       function uiStatus(context) {
-         var osm = context.connection();
-         return function (selection) {
-           if (!osm) return;
+           function rstr2hex(input, hexcase) {
+             var hex_tab = hexcase ? '0123456789ABCDEF' : '0123456789abcdef',
+                 output = '',
+                 x,
+                 i = 0,
+                 l = input.length;
 
-           function update(err, apiStatus) {
-             selection.html('');
+             for (; i < l; i += 1) {
+               x = input.charCodeAt(i);
+               output += hex_tab.charAt(x >>> 4 & 0x0F) + hex_tab.charAt(x & 0x0F);
+             }
 
-             if (err) {
-               if (apiStatus === 'connectionSwitched') {
-                 // if the connection was just switched, we can't rely on
-                 // the status (we're getting the status of the previous api)
-                 return;
-               } else if (apiStatus === 'rateLimited') {
-                 selection.html(_t.html('osm_api_status.message.rateLimit')).append('a').attr('href', '#').attr('class', 'api-status-login').attr('target', '_blank').call(svgIcon('#iD-icon-out-link', 'inline')).append('span').html(_t.html('login')).on('click.login', function (d3_event) {
-                   d3_event.preventDefault();
-                   osm.authenticate();
-                 });
-               } else {
-                 // don't allow retrying too rapidly
-                 var throttledRetry = throttle(function () {
-                   // try loading the visible tiles
-                   context.loadTiles(context.projection); // manually reload the status too in case all visible tiles were already loaded
+             return output;
+           }
+           /**
+            * Convert an array of big-endian words to a string
+            */
 
-                   osm.reloadApiStatus();
-                 }, 2000); // eslint-disable-next-line no-warning-comments
-                 // TODO: nice messages for different error types
 
+           function binb2rstr(input) {
+             var i,
+                 l = input.length * 32,
+                 output = '';
 
-                 selection.html(_t.html('osm_api_status.message.error') + ' ').append('a').attr('href', '#') // let the user manually retry their connection directly
-                 .html(_t.html('osm_api_status.retry')).on('click.retry', function (d3_event) {
-                   d3_event.preventDefault();
-                   throttledRetry();
-                 });
-               }
-             } else if (apiStatus === 'readonly') {
-               selection.html(_t.html('osm_api_status.message.readonly'));
-             } else if (apiStatus === 'offline') {
-               selection.html(_t.html('osm_api_status.message.offline'));
+             for (i = 0; i < l; i += 8) {
+               output += String.fromCharCode(input[i >> 5] >>> 24 - i % 32 & 0xFF);
              }
 
-             selection.attr('class', 'api-status ' + (err ? 'error' : apiStatus));
+             return output;
            }
+           /**
+            * Convert an array of little-endian words to a string
+            */
 
-           osm.on('apiStatusChange.uiStatus', update); // reload the status periodically regardless of other factors
-
-           window.setInterval(function () {
-             osm.reloadApiStatus();
-           }, 90000); // load the initial status in case no OSM data was loaded yet
 
-           osm.reloadApiStatus();
-         };
-       }
+           function binl2rstr(input) {
+             var i,
+                 l = input.length * 32,
+                 output = '';
 
-       function modeDrawArea(context, wayID, startGraph, button) {
-         var mode = {
-           button: button,
-           id: 'draw-area'
-         };
-         var behavior = behaviorDrawWay(context, wayID, mode, startGraph).on('rejectedSelfIntersection.modeDrawArea', function () {
-           context.ui().flash.iconName('#iD-icon-no').label(_t('self_intersection.error.areas'))();
-         });
-         mode.wayID = wayID;
+             for (i = 0; i < l; i += 8) {
+               output += String.fromCharCode(input[i >> 5] >>> i % 32 & 0xFF);
+             }
 
-         mode.enter = function () {
-           context.install(behavior);
-         };
+             return output;
+           }
+           /**
+            * Convert a raw string to an array of little-endian words
+            * Characters >255 have their high-byte silently ignored.
+            */
 
-         mode.exit = function () {
-           context.uninstall(behavior);
-         };
 
-         mode.selectedIDs = function () {
-           return [wayID];
-         };
+           function rstr2binl(input) {
+             var i,
+                 l = input.length * 8,
+                 output = Array(input.length >> 2),
+                 lo = output.length;
 
-         mode.activeID = function () {
-           return behavior && behavior.activeID() || [];
-         };
+             for (i = 0; i < lo; i += 1) {
+               output[i] = 0;
+             }
 
-         return mode;
-       }
+             for (i = 0; i < l; i += 8) {
+               output[i >> 5] |= (input.charCodeAt(i / 8) & 0xFF) << i % 32;
+             }
 
-       function modeAddArea(context, mode) {
-         mode.id = 'add-area';
-         var behavior = behaviorAddWay(context).on('start', start).on('startFromWay', startFromWay).on('startFromNode', startFromNode);
-         var defaultTags = {
-           area: 'yes'
-         };
-         if (mode.preset) defaultTags = mode.preset.setTags(defaultTags, 'area');
+             return output;
+           }
+           /**
+            * Convert a raw string to an array of big-endian words
+            * Characters >255 have their high-byte silently ignored.
+            */
 
-         function actionClose(wayId) {
-           return function (graph) {
-             return graph.replace(graph.entity(wayId).close());
-           };
-         }
 
-         function start(loc) {
-           var startGraph = context.graph();
-           var node = osmNode({
-             loc: loc
-           });
-           var way = osmWay({
-             tags: defaultTags
-           });
-           context.perform(actionAddEntity(node), actionAddEntity(way), actionAddVertex(way.id, node.id), actionClose(way.id));
-           context.enter(modeDrawArea(context, way.id, startGraph, mode.button));
-         }
+           function rstr2binb(input) {
+             var i,
+                 l = input.length * 8,
+                 output = Array(input.length >> 2),
+                 lo = output.length;
 
-         function startFromWay(loc, edge) {
-           var startGraph = context.graph();
-           var node = osmNode({
-             loc: loc
-           });
-           var way = osmWay({
-             tags: defaultTags
-           });
-           context.perform(actionAddEntity(node), actionAddEntity(way), actionAddVertex(way.id, node.id), actionClose(way.id), actionAddMidpoint({
-             loc: loc,
-             edge: edge
-           }, node));
-           context.enter(modeDrawArea(context, way.id, startGraph, mode.button));
-         }
+             for (i = 0; i < lo; i += 1) {
+               output[i] = 0;
+             }
 
-         function startFromNode(node) {
-           var startGraph = context.graph();
-           var way = osmWay({
-             tags: defaultTags
-           });
-           context.perform(actionAddEntity(way), actionAddVertex(way.id, node.id), actionClose(way.id));
-           context.enter(modeDrawArea(context, way.id, startGraph, mode.button));
-         }
+             for (i = 0; i < l; i += 8) {
+               output[i >> 5] |= (input.charCodeAt(i / 8) & 0xFF) << 24 - i % 32;
+             }
 
-         mode.enter = function () {
-           context.install(behavior);
-         };
+             return output;
+           }
+           /**
+            * Convert a raw string to an arbitrary string encoding
+            */
 
-         mode.exit = function () {
-           context.uninstall(behavior);
-         };
 
-         return mode;
-       }
+           function rstr2any(input, encoding) {
+             var divisor = encoding.length,
+                 remainders = Array(),
+                 i,
+                 q,
+                 x,
+                 ld,
+                 quotient,
+                 dividend,
+                 output,
+                 full_length;
+             /* Convert to an array of 16-bit big-endian values, forming the dividend */
 
-       function modeAddLine(context, mode) {
-         mode.id = 'add-line';
-         var behavior = behaviorAddWay(context).on('start', start).on('startFromWay', startFromWay).on('startFromNode', startFromNode);
-         var defaultTags = {};
-         if (mode.preset) defaultTags = mode.preset.setTags(defaultTags, 'line');
+             dividend = Array(Math.ceil(input.length / 2));
+             ld = dividend.length;
 
-         function start(loc) {
-           var startGraph = context.graph();
-           var node = osmNode({
-             loc: loc
-           });
-           var way = osmWay({
-             tags: defaultTags
-           });
-           context.perform(actionAddEntity(node), actionAddEntity(way), actionAddVertex(way.id, node.id));
-           context.enter(modeDrawLine(context, way.id, startGraph, mode.button));
-         }
+             for (i = 0; i < ld; i += 1) {
+               dividend[i] = input.charCodeAt(i * 2) << 8 | input.charCodeAt(i * 2 + 1);
+             }
+             /**
+              * Repeatedly perform a long division. The binary array forms the dividend,
+              * the length of the encoding is the divisor. Once computed, the quotient
+              * forms the dividend for the next step. We stop when the dividend is zerHashes.
+              * All remainders are stored for later use.
+              */
 
-         function startFromWay(loc, edge) {
-           var startGraph = context.graph();
-           var node = osmNode({
-             loc: loc
-           });
-           var way = osmWay({
-             tags: defaultTags
-           });
-           context.perform(actionAddEntity(node), actionAddEntity(way), actionAddVertex(way.id, node.id), actionAddMidpoint({
-             loc: loc,
-             edge: edge
-           }, node));
-           context.enter(modeDrawLine(context, way.id, startGraph, mode.button));
-         }
 
-         function startFromNode(node) {
-           var startGraph = context.graph();
-           var way = osmWay({
-             tags: defaultTags
-           });
-           context.perform(actionAddEntity(way), actionAddVertex(way.id, node.id));
-           context.enter(modeDrawLine(context, way.id, startGraph, mode.button));
-         }
+             while (dividend.length > 0) {
+               quotient = Array();
+               x = 0;
 
-         mode.enter = function () {
-           context.install(behavior);
-         };
+               for (i = 0; i < dividend.length; i += 1) {
+                 x = (x << 16) + dividend[i];
+                 q = Math.floor(x / divisor);
+                 x -= q * divisor;
 
-         mode.exit = function () {
-           context.uninstall(behavior);
-         };
+                 if (quotient.length > 0 || q > 0) {
+                   quotient[quotient.length] = q;
+                 }
+               }
 
-         return mode;
-       }
+               remainders[remainders.length] = x;
+               dividend = quotient;
+             }
+             /* Convert the remainders to the output string */
 
-       function modeAddPoint(context, mode) {
-         mode.id = 'add-point';
-         var behavior = behaviorDraw(context).on('click', add).on('clickWay', addWay).on('clickNode', addNode).on('cancel', cancel).on('finish', cancel);
-         var defaultTags = {};
-         if (mode.preset) defaultTags = mode.preset.setTags(defaultTags, 'point');
 
-         function add(loc) {
-           var node = osmNode({
-             loc: loc,
-             tags: defaultTags
-           });
-           context.perform(actionAddEntity(node), _t('operations.add.annotation.point'));
-           enterSelectMode(node);
-         }
+             output = '';
 
-         function addWay(loc, edge) {
-           var node = osmNode({
-             tags: defaultTags
-           });
-           context.perform(actionAddMidpoint({
-             loc: loc,
-             edge: edge
-           }, node), _t('operations.add.annotation.vertex'));
-           enterSelectMode(node);
-         }
+             for (i = remainders.length - 1; i >= 0; i--) {
+               output += encoding.charAt(remainders[i]);
+             }
+             /* Append leading zero equivalents */
 
-         function enterSelectMode(node) {
-           context.enter(modeSelect(context, [node.id]).newFeature(true));
-         }
 
-         function addNode(node) {
-           if (Object.keys(defaultTags).length === 0) {
-             enterSelectMode(node);
-             return;
-           }
+             full_length = Math.ceil(input.length * 8 / (Math.log(encoding.length) / Math.log(2)));
 
-           var tags = Object.assign({}, node.tags); // shallow copy
+             for (i = output.length; i < full_length; i += 1) {
+               output = encoding[0] + output;
+             }
 
-           for (var key in defaultTags) {
-             tags[key] = defaultTags[key];
+             return output;
            }
+           /**
+            * Convert a raw string to a base-64 string
+            */
 
-           context.perform(actionChangeTags(node.id, tags), _t('operations.add.annotation.point'));
-           enterSelectMode(node);
-         }
-
-         function cancel() {
-           context.enter(modeBrowse(context));
-         }
 
-         mode.enter = function () {
-           context.install(behavior);
-         };
+           function rstr2b64(input, b64pad) {
+             var tab = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/',
+                 output = '',
+                 len = input.length,
+                 i,
+                 j,
+                 triplet;
+             b64pad = b64pad || '=';
 
-         mode.exit = function () {
-           context.uninstall(behavior);
-         };
+             for (i = 0; i < len; i += 3) {
+               triplet = input.charCodeAt(i) << 16 | (i + 1 < len ? input.charCodeAt(i + 1) << 8 : 0) | (i + 2 < len ? input.charCodeAt(i + 2) : 0);
 
-         return mode;
-       }
+               for (j = 0; j < 4; j += 1) {
+                 if (i * 8 + j * 6 > input.length * 8) {
+                   output += b64pad;
+                 } else {
+                   output += tab.charAt(triplet >>> 6 * (3 - j) & 0x3F);
+                 }
+               }
+             }
 
-       function modeAddNote(context) {
-         var mode = {
-           id: 'add-note',
-           button: 'note',
-           description: _t.html('modes.add_note.description'),
-           key: _t('modes.add_note.key')
-         };
-         var behavior = behaviorDraw(context).on('click', add).on('cancel', cancel).on('finish', cancel);
+             return output;
+           }
 
-         function add(loc) {
-           var osm = services.osm;
-           if (!osm) return;
-           var note = osmNote({
-             loc: loc,
-             status: 'open',
-             comments: []
-           });
-           osm.replaceNote(note); // force a reraw (there is no history change that would otherwise do this)
+           Hashes = {
+             /**
+              * @property {String} version
+              * @readonly
+              */
+             VERSION: '1.0.6',
 
-           context.map().pan([0, 0]);
-           context.selectedNoteID(note.id).enter(modeSelectNote(context, note.id).newFeature(true));
-         }
+             /**
+              * @member Hashes
+              * @class Base64
+              * @constructor
+              */
+             Base64: function Base64() {
+               // private properties
+               var tab = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/',
+                   pad = '=',
+                   // URL encoding support @todo
+               utf8 = true; // by default enable UTF-8 support encoding
+               // public method for encoding
 
-         function cancel() {
-           context.enter(modeBrowse(context));
-         }
+               this.encode = function (input) {
+                 var i,
+                     j,
+                     triplet,
+                     output = '',
+                     len = input.length;
+                 pad = pad || '=';
+                 input = utf8 ? utf8Encode(input) : input;
 
-         mode.enter = function () {
-           context.install(behavior);
-         };
+                 for (i = 0; i < len; i += 3) {
+                   triplet = input.charCodeAt(i) << 16 | (i + 1 < len ? input.charCodeAt(i + 1) << 8 : 0) | (i + 2 < len ? input.charCodeAt(i + 2) : 0);
 
-         mode.exit = function () {
-           context.uninstall(behavior);
-         };
+                   for (j = 0; j < 4; j += 1) {
+                     if (i * 8 + j * 6 > len * 8) {
+                       output += pad;
+                     } else {
+                       output += tab.charAt(triplet >>> 6 * (3 - j) & 0x3F);
+                     }
+                   }
+                 }
 
-         return mode;
-       }
+                 return output;
+               }; // public method for decoding
 
-       function uiConflicts(context) {
-         var dispatch$1 = dispatch('cancel', 'save');
-         var keybinding = utilKeybinding('conflicts');
 
-         var _origChanges;
+               this.decode = function (input) {
+                 // var b64 = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';
+                 var i,
+                     o1,
+                     o2,
+                     o3,
+                     h1,
+                     h2,
+                     h3,
+                     h4,
+                     bits,
+                     ac,
+                     dec = '',
+                     arr = [];
 
-         var _conflictList;
+                 if (!input) {
+                   return input;
+                 }
 
-         var _shownConflictIndex;
+                 i = ac = 0;
+                 input = input.replace(new RegExp('\\' + pad, 'gi'), ''); // use '='
+                 //input += '';
 
-         function keybindingOn() {
-           select(document).call(keybinding.on('⎋', cancel, true));
-         }
+                 do {
+                   // unpack four hexets into three octets using index points in b64
+                   h1 = tab.indexOf(input.charAt(i += 1));
+                   h2 = tab.indexOf(input.charAt(i += 1));
+                   h3 = tab.indexOf(input.charAt(i += 1));
+                   h4 = tab.indexOf(input.charAt(i += 1));
+                   bits = h1 << 18 | h2 << 12 | h3 << 6 | h4;
+                   o1 = bits >> 16 & 0xff;
+                   o2 = bits >> 8 & 0xff;
+                   o3 = bits & 0xff;
+                   ac += 1;
 
-         function keybindingOff() {
-           select(document).call(keybinding.unbind);
-         }
+                   if (h3 === 64) {
+                     arr[ac] = String.fromCharCode(o1);
+                   } else if (h4 === 64) {
+                     arr[ac] = String.fromCharCode(o1, o2);
+                   } else {
+                     arr[ac] = String.fromCharCode(o1, o2, o3);
+                   }
+                 } while (i < input.length);
 
-         function tryAgain() {
-           keybindingOff();
-           dispatch$1.call('save');
-         }
+                 dec = arr.join('');
+                 dec = utf8 ? utf8Decode(dec) : dec;
+                 return dec;
+               }; // set custom pad string
 
-         function cancel() {
-           keybindingOff();
-           dispatch$1.call('cancel');
-         }
 
-         function conflicts(selection) {
-           keybindingOn();
-           var headerEnter = selection.selectAll('.header').data([0]).enter().append('div').attr('class', 'header fillL');
-           headerEnter.append('button').attr('class', 'fr').on('click', cancel).call(svgIcon('#iD-icon-close'));
-           headerEnter.append('h3').html(_t.html('save.conflict.header'));
-           var bodyEnter = selection.selectAll('.body').data([0]).enter().append('div').attr('class', 'body fillL');
-           var conflictsHelpEnter = bodyEnter.append('div').attr('class', 'conflicts-help').html(_t.html('save.conflict.help')); // Download changes link
+               this.setPad = function (str) {
+                 pad = str || pad;
+                 return this;
+               }; // set custom tab string characters
 
-           var detected = utilDetect();
-           var changeset = new osmChangeset();
-           delete changeset.id; // Export without changeset_id
 
-           var data = JXON.stringify(changeset.osmChangeJXON(_origChanges));
-           var blob = new Blob([data], {
-             type: 'text/xml;charset=utf-8;'
-           });
-           var fileName = 'changes.osc';
-           var linkEnter = conflictsHelpEnter.selectAll('.download-changes').append('a').attr('class', 'download-changes');
+               this.setTab = function (str) {
+                 tab = str || tab;
+                 return this;
+               };
 
-           if (detected.download) {
-             // All except IE11 and Edge
-             linkEnter // download the data as a file
-             .attr('href', window.URL.createObjectURL(blob)).attr('download', fileName);
-           } else {
-             // IE11 and Edge
-             linkEnter // open data uri in a new tab
-             .attr('target', '_blank').on('click.download', function () {
-               navigator.msSaveBlob(blob, fileName);
-             });
-           }
+               this.setUTF8 = function (bool) {
+                 if (typeof bool === 'boolean') {
+                   utf8 = bool;
+                 }
 
-           linkEnter.call(svgIcon('#iD-icon-load', 'inline')).append('span').html(_t.html('save.conflict.download_changes'));
-           bodyEnter.append('div').attr('class', 'conflict-container fillL3').call(showConflict, 0);
-           bodyEnter.append('div').attr('class', 'conflicts-done').attr('opacity', 0).style('display', 'none').html(_t.html('save.conflict.done'));
-           var buttonsEnter = bodyEnter.append('div').attr('class', 'buttons col12 joined conflicts-buttons');
-           buttonsEnter.append('button').attr('disabled', _conflictList.length > 1).attr('class', 'action conflicts-button col6').html(_t.html('save.title')).on('click.try_again', tryAgain);
-           buttonsEnter.append('button').attr('class', 'secondary-action conflicts-button col6').html(_t.html('confirm.cancel')).on('click.cancel', cancel);
-         }
+                 return this;
+               };
+             },
 
-         function showConflict(selection, index) {
-           index = utilWrap(index, _conflictList.length);
-           _shownConflictIndex = index;
-           var parent = select(selection.node().parentNode); // enable save button if this is the last conflict being reviewed..
+             /**
+              * CRC-32 calculation
+              * @member Hashes
+              * @method CRC32
+              * @static
+              * @param {String} str Input String
+              * @return {String}
+              */
+             CRC32: function CRC32(str) {
+               var crc = 0,
+                   x = 0,
+                   y = 0,
+                   table,
+                   i,
+                   iTop;
+               str = utf8Encode(str);
+               table = ['00000000 77073096 EE0E612C 990951BA 076DC419 706AF48F E963A535 9E6495A3 0EDB8832 ', '79DCB8A4 E0D5E91E 97D2D988 09B64C2B 7EB17CBD E7B82D07 90BF1D91 1DB71064 6AB020F2 F3B97148 ', '84BE41DE 1ADAD47D 6DDDE4EB F4D4B551 83D385C7 136C9856 646BA8C0 FD62F97A 8A65C9EC 14015C4F ', '63066CD9 FA0F3D63 8D080DF5 3B6E20C8 4C69105E D56041E4 A2677172 3C03E4D1 4B04D447 D20D85FD ', 'A50AB56B 35B5A8FA 42B2986C DBBBC9D6 ACBCF940 32D86CE3 45DF5C75 DCD60DCF ABD13D59 26D930AC ', '51DE003A C8D75180 BFD06116 21B4F4B5 56B3C423 CFBA9599 B8BDA50F 2802B89E 5F058808 C60CD9B2 ', 'B10BE924 2F6F7C87 58684C11 C1611DAB B6662D3D 76DC4190 01DB7106 98D220BC EFD5102A 71B18589 ', '06B6B51F 9FBFE4A5 E8B8D433 7807C9A2 0F00F934 9609A88E E10E9818 7F6A0DBB 086D3D2D 91646C97 ', 'E6635C01 6B6B51F4 1C6C6162 856530D8 F262004E 6C0695ED 1B01A57B 8208F4C1 F50FC457 65B0D9C6 ', '12B7E950 8BBEB8EA FCB9887C 62DD1DDF 15DA2D49 8CD37CF3 FBD44C65 4DB26158 3AB551CE A3BC0074 ', 'D4BB30E2 4ADFA541 3DD895D7 A4D1C46D D3D6F4FB 4369E96A 346ED9FC AD678846 DA60B8D0 44042D73 ', '33031DE5 AA0A4C5F DD0D7CC9 5005713C 270241AA BE0B1010 C90C2086 5768B525 206F85B3 B966D409 ', 'CE61E49F 5EDEF90E 29D9C998 B0D09822 C7D7A8B4 59B33D17 2EB40D81 B7BD5C3B C0BA6CAD EDB88320 ', '9ABFB3B6 03B6E20C 74B1D29A EAD54739 9DD277AF 04DB2615 73DC1683 E3630B12 94643B84 0D6D6A3E ', '7A6A5AA8 E40ECF0B 9309FF9D 0A00AE27 7D079EB1 F00F9344 8708A3D2 1E01F268 6906C2FE F762575D ', '806567CB 196C3671 6E6B06E7 FED41B76 89D32BE0 10DA7A5A 67DD4ACC F9B9DF6F 8EBEEFF9 17B7BE43 ', '60B08ED5 D6D6A3E8 A1D1937E 38D8C2C4 4FDFF252 D1BB67F1 A6BC5767 3FB506DD 48B2364B D80D2BDA ', 'AF0A1B4C 36034AF6 41047A60 DF60EFC3 A867DF55 316E8EEF 4669BE79 CB61B38C BC66831A 256FD2A0 ', '5268E236 CC0C7795 BB0B4703 220216B9 5505262F C5BA3BBE B2BD0B28 2BB45A92 5CB36A04 C2D7FFA7 ', 'B5D0CF31 2CD99E8B 5BDEAE1D 9B64C2B0 EC63F226 756AA39C 026D930A 9C0906A9 EB0E363F 72076785 ', '05005713 95BF4A82 E2B87A14 7BB12BAE 0CB61B38 92D28E9B E5D5BE0D 7CDCEFB7 0BDBDF21 86D3D2D4 ', 'F1D4E242 68DDB3F8 1FDA836E 81BE16CD F6B9265B 6FB077E1 18B74777 88085AE6 FF0F6A70 66063BCA ', '11010B5C 8F659EFF F862AE69 616BFFD3 166CCF45 A00AE278 D70DD2EE 4E048354 3903B3C2 A7672661 ', 'D06016F7 4969474D 3E6E77DB AED16A4A D9D65ADC 40DF0B66 37D83BF0 A9BCAE53 DEBB9EC5 47B2CF7F ', '30B5FFE9 BDBDF21C CABAC28A 53B39330 24B4A3A6 BAD03605 CDD70693 54DE5729 23D967BF B3667A2E ', 'C4614AB8 5D681B02 2A6F2B94 B40BBE37 C30C8EA1 5A05DF1B 2D02EF8D'].join('');
+               crc = crc ^ -1;
 
-           if (index === _conflictList.length - 1) {
-             window.setTimeout(function () {
-               parent.select('.conflicts-button').attr('disabled', null);
-               parent.select('.conflicts-done').transition().attr('opacity', 1).style('display', 'block');
-             }, 250);
-           }
+               for (i = 0, iTop = str.length; i < iTop; i += 1) {
+                 y = (crc ^ str.charCodeAt(i)) & 0xFF;
+                 x = '0x' + table.substr(y * 9, 8);
+                 crc = crc >>> 8 ^ x;
+               } // always return a positive number (that's what >>> 0 does)
 
-           var conflict = selection.selectAll('.conflict').data([_conflictList[index]]);
-           conflict.exit().remove();
-           var conflictEnter = conflict.enter().append('div').attr('class', 'conflict');
-           conflictEnter.append('h4').attr('class', 'conflict-count').html(_t.html('save.conflict.count', {
-             num: index + 1,
-             total: _conflictList.length
-           }));
-           conflictEnter.append('a').attr('class', 'conflict-description').attr('href', '#').html(function (d) {
-             return d.name;
-           }).on('click', function (d3_event, d) {
-             d3_event.preventDefault();
-             zoomToEntity(d.id);
-           });
-           var details = conflictEnter.append('div').attr('class', 'conflict-detail-container');
-           details.append('ul').attr('class', 'conflict-detail-list').selectAll('li').data(function (d) {
-             return d.details || [];
-           }).enter().append('li').attr('class', 'conflict-detail-item').html(function (d) {
-             return d;
-           });
-           details.append('div').attr('class', 'conflict-choices').call(addChoices);
-           details.append('div').attr('class', 'conflict-nav-buttons joined cf').selectAll('button').data(['previous', 'next']).enter().append('button').html(function (d) {
-             return _t.html('save.conflict.' + d);
-           }).attr('class', 'conflict-nav-button action col6').attr('disabled', function (d, i) {
-             return i === 0 && index === 0 || i === 1 && index === _conflictList.length - 1 || null;
-           }).on('click', function (d3_event, d) {
-             d3_event.preventDefault();
-             var container = parent.selectAll('.conflict-container');
-             var sign = d === 'previous' ? -1 : 1;
-             container.selectAll('.conflict').remove();
-             container.call(showConflict, index + sign);
-           });
-         }
 
-         function addChoices(selection) {
-           var choices = selection.append('ul').attr('class', 'layer-list').selectAll('li').data(function (d) {
-             return d.choices || [];
-           }); // enter
+               return (crc ^ -1) >>> 0;
+             },
 
-           var choicesEnter = choices.enter().append('li').attr('class', 'layer');
-           var labelEnter = choicesEnter.append('label');
-           labelEnter.append('input').attr('type', 'radio').attr('name', function (d) {
-             return d.id;
-           }).on('change', function (d3_event, d) {
-             var ul = this.parentNode.parentNode.parentNode;
-             ul.__data__.chosen = d.id;
-             choose(d3_event, ul, d);
-           });
-           labelEnter.append('span').html(function (d) {
-             return d.text;
-           }); // update
+             /**
+              * @member Hashes
+              * @class MD5
+              * @constructor
+              * @param {Object} [config]
+              *
+              * A JavaScript implementation of the RSA Data Security, Inc. MD5 Message
+              * Digest Algorithm, as defined in RFC 1321.
+              * Version 2.2 Copyright (C) Paul Johnston 1999 - 2009
+              * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
+              * See <http://pajhome.org.uk/crypt/md5> for more infHashes.
+              */
+             MD5: function MD5(options) {
+               /**
+                * Private config properties. You may need to tweak these to be compatible with
+                * the server-side, but the defaults work in most cases.
+                * See {@link Hashes.MD5#method-setUpperCase} and {@link Hashes.SHA1#method-setUpperCase}
+                */
+               var hexcase = options && typeof options.uppercase === 'boolean' ? options.uppercase : false,
+                   // hexadecimal output case format. false - lowercase; true - uppercase
+               b64pad = options && typeof options.pad === 'string' ? options.pad : '=',
+                   // base-64 pad character. Defaults to '=' for strict RFC compliance
+               utf8 = options && typeof options.utf8 === 'boolean' ? options.utf8 : true; // enable/disable utf8 encoding
+               // privileged (public) methods
 
-           choicesEnter.merge(choices).each(function (d) {
-             var ul = this.parentNode;
+               this.hex = function (s) {
+                 return rstr2hex(rstr(s), hexcase);
+               };
 
-             if (ul.__data__.chosen === d.id) {
-               choose(null, ul, d);
-             }
-           });
-         }
+               this.b64 = function (s) {
+                 return rstr2b64(rstr(s), b64pad);
+               };
 
-         function choose(d3_event, ul, datum) {
-           if (d3_event) d3_event.preventDefault();
-           select(ul).selectAll('li').classed('active', function (d) {
-             return d === datum;
-           }).selectAll('input').property('checked', function (d) {
-             return d === datum;
-           });
-           var extent = geoExtent();
-           var entity;
-           entity = context.graph().hasEntity(datum.id);
-           if (entity) extent._extend(entity.extent(context.graph()));
-           datum.action();
-           entity = context.graph().hasEntity(datum.id);
-           if (entity) extent._extend(entity.extent(context.graph()));
-           zoomToEntity(datum.id, extent);
-         }
+               this.any = function (s, e) {
+                 return rstr2any(rstr(s), e);
+               };
 
-         function zoomToEntity(id, extent) {
-           context.surface().selectAll('.hover').classed('hover', false);
-           var entity = context.graph().hasEntity(id);
+               this.raw = function (s) {
+                 return rstr(s);
+               };
 
-           if (entity) {
-             if (extent) {
-               context.map().trimmedExtent(extent);
-             } else {
-               context.map().zoomToEase(entity);
-             }
+               this.hex_hmac = function (k, d) {
+                 return rstr2hex(rstr_hmac(k, d), hexcase);
+               };
 
-             context.surface().selectAll(utilEntityOrMemberSelector([entity.id], context.graph())).classed('hover', true);
-           }
-         } // The conflict list should be an array of objects like:
-         // {
-         //     id: id,
-         //     name: entityName(local),
-         //     details: merge.conflicts(),
-         //     chosen: 1,
-         //     choices: [
-         //         choice(id, keepMine, forceLocal),
-         //         choice(id, keepTheirs, forceRemote)
-         //     ]
-         // }
+               this.b64_hmac = function (k, d) {
+                 return rstr2b64(rstr_hmac(k, d), b64pad);
+               };
 
+               this.any_hmac = function (k, d, e) {
+                 return rstr2any(rstr_hmac(k, d), e);
+               };
+               /**
+                * Perform a simple self-test to see if the VM is working
+                * @return {String} Hexadecimal hash sample
+                */
 
-         conflicts.conflictList = function (_) {
-           if (!arguments.length) return _conflictList;
-           _conflictList = _;
-           return conflicts;
-         };
 
-         conflicts.origChanges = function (_) {
-           if (!arguments.length) return _origChanges;
-           _origChanges = _;
-           return conflicts;
-         };
+               this.vm_test = function () {
+                 return hex('abc').toLowerCase() === '900150983cd24fb0d6963f7d28e17f72';
+               };
+               /**
+                * Enable/disable uppercase hexadecimal returned string
+                * @param {Boolean}
+                * @return {Object} this
+                */
 
-         conflicts.shownEntityIds = function () {
-           if (_conflictList && typeof _shownConflictIndex === 'number') {
-             return [_conflictList[_shownConflictIndex].id];
-           }
 
-           return [];
-         };
+               this.setUpperCase = function (a) {
+                 if (typeof a === 'boolean') {
+                   hexcase = a;
+                 }
 
-         return utilRebind(conflicts, dispatch$1, 'on');
-       }
+                 return this;
+               };
+               /**
+                * Defines a base64 pad string
+                * @param {String} Pad
+                * @return {Object} this
+                */
 
-       function uiConfirm(selection) {
-         var modalSelection = uiModal(selection);
-         modalSelection.select('.modal').classed('modal-alert', true);
-         var section = modalSelection.select('.content');
-         section.append('div').attr('class', 'modal-section header');
-         section.append('div').attr('class', 'modal-section message-text');
-         var buttons = section.append('div').attr('class', 'modal-section buttons cf');
 
-         modalSelection.okButton = function () {
-           buttons.append('button').attr('class', 'button ok-button action').on('click.confirm', function () {
-             modalSelection.remove();
-           }).html(_t.html('confirm.okay')).node().focus();
-           return modalSelection;
-         };
+               this.setPad = function (a) {
+                 b64pad = a || b64pad;
+                 return this;
+               };
+               /**
+                * Defines a base64 pad string
+                * @param {Boolean}
+                * @return {Object} [this]
+                */
 
-         return modalSelection;
-       }
 
-       function uiChangesetEditor(context) {
-         var dispatch$1 = dispatch('change');
-         var formFields = uiFormFields(context);
-         var commentCombo = uiCombobox(context, 'comment').caseSensitive(true);
+               this.setUTF8 = function (a) {
+                 if (typeof a === 'boolean') {
+                   utf8 = a;
+                 }
 
-         var _fieldsArr;
+                 return this;
+               }; // private methods
 
-         var _tags;
+               /**
+                * Calculate the MD5 of a raw string
+                */
 
-         var _changesetID;
 
-         function changesetEditor(selection) {
-           render(selection);
-         }
+               function rstr(s) {
+                 s = utf8 ? utf8Encode(s) : s;
+                 return binl2rstr(binl(rstr2binl(s), s.length * 8));
+               }
+               /**
+                * Calculate the HMAC-MD5, of a key and some data (raw strings)
+                */
 
-         function render(selection) {
-           var initial = false;
 
-           if (!_fieldsArr) {
-             initial = true;
-             var presets = _mainPresetIndex;
-             _fieldsArr = [uiField(context, presets.field('comment'), null, {
-               show: true,
-               revert: false
-             }), uiField(context, presets.field('source'), null, {
-               show: false,
-               revert: false
-             }), uiField(context, presets.field('hashtags'), null, {
-               show: false,
-               revert: false
-             })];
+               function rstr_hmac(key, data) {
+                 var bkey, ipad, opad, hash, i;
+                 key = utf8 ? utf8Encode(key) : key;
+                 data = utf8 ? utf8Encode(data) : data;
+                 bkey = rstr2binl(key);
 
-             _fieldsArr.forEach(function (field) {
-               field.on('change', function (t, onInput) {
-                 dispatch$1.call('change', field, undefined, t, onInput);
-               });
-             });
-           }
+                 if (bkey.length > 16) {
+                   bkey = binl(bkey, key.length * 8);
+                 }
 
-           _fieldsArr.forEach(function (field) {
-             field.tags(_tags);
-           });
+                 ipad = Array(16), opad = Array(16);
 
-           selection.call(formFields.fieldsArr(_fieldsArr));
+                 for (i = 0; i < 16; i += 1) {
+                   ipad[i] = bkey[i] ^ 0x36363636;
+                   opad[i] = bkey[i] ^ 0x5C5C5C5C;
+                 }
 
-           if (initial) {
-             var commentField = selection.select('.form-field-comment textarea');
-             var commentNode = commentField.node();
+                 hash = binl(ipad.concat(rstr2binl(data)), 512 + data.length * 8);
+                 return binl2rstr(binl(opad.concat(hash), 512 + 128));
+               }
+               /**
+                * Calculate the MD5 of an array of little-endian words, and a bit length.
+                */
 
-             if (commentNode) {
-               commentNode.focus();
-               commentNode.select();
-             } // trigger a 'blur' event so that comment field can be cleaned
-             // and checked for hashtags, even if retrieved from localstorage
 
+               function binl(x, len) {
+                 var i,
+                     olda,
+                     oldb,
+                     oldc,
+                     oldd,
+                     a = 1732584193,
+                     b = -271733879,
+                     c = -1732584194,
+                     d = 271733878;
+                 /* append padding */
 
-             utilTriggerEvent(commentField, 'blur');
-             var osm = context.connection();
+                 x[len >> 5] |= 0x80 << len % 32;
+                 x[(len + 64 >>> 9 << 4) + 14] = len;
 
-             if (osm) {
-               osm.userChangesets(function (err, changesets) {
-                 if (err) return;
-                 var comments = changesets.map(function (changeset) {
-                   var comment = changeset.tags.comment;
-                   return comment ? {
-                     title: comment,
-                     value: comment
-                   } : null;
-                 }).filter(Boolean);
-                 commentField.call(commentCombo.data(utilArrayUniqBy(comments, 'title')));
-               });
-             }
-           } // Add warning if comment mentions Google
+                 for (i = 0; i < x.length; i += 16) {
+                   olda = a;
+                   oldb = b;
+                   oldc = c;
+                   oldd = d;
+                   a = md5_ff(a, b, c, d, x[i + 0], 7, -680876936);
+                   d = md5_ff(d, a, b, c, x[i + 1], 12, -389564586);
+                   c = md5_ff(c, d, a, b, x[i + 2], 17, 606105819);
+                   b = md5_ff(b, c, d, a, x[i + 3], 22, -1044525330);
+                   a = md5_ff(a, b, c, d, x[i + 4], 7, -176418897);
+                   d = md5_ff(d, a, b, c, x[i + 5], 12, 1200080426);
+                   c = md5_ff(c, d, a, b, x[i + 6], 17, -1473231341);
+                   b = md5_ff(b, c, d, a, x[i + 7], 22, -45705983);
+                   a = md5_ff(a, b, c, d, x[i + 8], 7, 1770035416);
+                   d = md5_ff(d, a, b, c, x[i + 9], 12, -1958414417);
+                   c = md5_ff(c, d, a, b, x[i + 10], 17, -42063);
+                   b = md5_ff(b, c, d, a, x[i + 11], 22, -1990404162);
+                   a = md5_ff(a, b, c, d, x[i + 12], 7, 1804603682);
+                   d = md5_ff(d, a, b, c, x[i + 13], 12, -40341101);
+                   c = md5_ff(c, d, a, b, x[i + 14], 17, -1502002290);
+                   b = md5_ff(b, c, d, a, x[i + 15], 22, 1236535329);
+                   a = md5_gg(a, b, c, d, x[i + 1], 5, -165796510);
+                   d = md5_gg(d, a, b, c, x[i + 6], 9, -1069501632);
+                   c = md5_gg(c, d, a, b, x[i + 11], 14, 643717713);
+                   b = md5_gg(b, c, d, a, x[i + 0], 20, -373897302);
+                   a = md5_gg(a, b, c, d, x[i + 5], 5, -701558691);
+                   d = md5_gg(d, a, b, c, x[i + 10], 9, 38016083);
+                   c = md5_gg(c, d, a, b, x[i + 15], 14, -660478335);
+                   b = md5_gg(b, c, d, a, x[i + 4], 20, -405537848);
+                   a = md5_gg(a, b, c, d, x[i + 9], 5, 568446438);
+                   d = md5_gg(d, a, b, c, x[i + 14], 9, -1019803690);
+                   c = md5_gg(c, d, a, b, x[i + 3], 14, -187363961);
+                   b = md5_gg(b, c, d, a, x[i + 8], 20, 1163531501);
+                   a = md5_gg(a, b, c, d, x[i + 13], 5, -1444681467);
+                   d = md5_gg(d, a, b, c, x[i + 2], 9, -51403784);
+                   c = md5_gg(c, d, a, b, x[i + 7], 14, 1735328473);
+                   b = md5_gg(b, c, d, a, x[i + 12], 20, -1926607734);
+                   a = md5_hh(a, b, c, d, x[i + 5], 4, -378558);
+                   d = md5_hh(d, a, b, c, x[i + 8], 11, -2022574463);
+                   c = md5_hh(c, d, a, b, x[i + 11], 16, 1839030562);
+                   b = md5_hh(b, c, d, a, x[i + 14], 23, -35309556);
+                   a = md5_hh(a, b, c, d, x[i + 1], 4, -1530992060);
+                   d = md5_hh(d, a, b, c, x[i + 4], 11, 1272893353);
+                   c = md5_hh(c, d, a, b, x[i + 7], 16, -155497632);
+                   b = md5_hh(b, c, d, a, x[i + 10], 23, -1094730640);
+                   a = md5_hh(a, b, c, d, x[i + 13], 4, 681279174);
+                   d = md5_hh(d, a, b, c, x[i + 0], 11, -358537222);
+                   c = md5_hh(c, d, a, b, x[i + 3], 16, -722521979);
+                   b = md5_hh(b, c, d, a, x[i + 6], 23, 76029189);
+                   a = md5_hh(a, b, c, d, x[i + 9], 4, -640364487);
+                   d = md5_hh(d, a, b, c, x[i + 12], 11, -421815835);
+                   c = md5_hh(c, d, a, b, x[i + 15], 16, 530742520);
+                   b = md5_hh(b, c, d, a, x[i + 2], 23, -995338651);
+                   a = md5_ii(a, b, c, d, x[i + 0], 6, -198630844);
+                   d = md5_ii(d, a, b, c, x[i + 7], 10, 1126891415);
+                   c = md5_ii(c, d, a, b, x[i + 14], 15, -1416354905);
+                   b = md5_ii(b, c, d, a, x[i + 5], 21, -57434055);
+                   a = md5_ii(a, b, c, d, x[i + 12], 6, 1700485571);
+                   d = md5_ii(d, a, b, c, x[i + 3], 10, -1894986606);
+                   c = md5_ii(c, d, a, b, x[i + 10], 15, -1051523);
+                   b = md5_ii(b, c, d, a, x[i + 1], 21, -2054922799);
+                   a = md5_ii(a, b, c, d, x[i + 8], 6, 1873313359);
+                   d = md5_ii(d, a, b, c, x[i + 15], 10, -30611744);
+                   c = md5_ii(c, d, a, b, x[i + 6], 15, -1560198380);
+                   b = md5_ii(b, c, d, a, x[i + 13], 21, 1309151649);
+                   a = md5_ii(a, b, c, d, x[i + 4], 6, -145523070);
+                   d = md5_ii(d, a, b, c, x[i + 11], 10, -1120210379);
+                   c = md5_ii(c, d, a, b, x[i + 2], 15, 718787259);
+                   b = md5_ii(b, c, d, a, x[i + 9], 21, -343485551);
+                   a = safe_add(a, olda);
+                   b = safe_add(b, oldb);
+                   c = safe_add(c, oldc);
+                   d = safe_add(d, oldd);
+                 }
 
+                 return Array(a, b, c, d);
+               }
+               /**
+                * These functions implement the four basic operations the algorithm uses.
+                */
 
-           var hasGoogle = _tags.comment.match(/google/i);
 
-           var commentWarning = selection.select('.form-field-comment').selectAll('.comment-warning').data(hasGoogle ? [0] : []);
-           commentWarning.exit().transition().duration(200).style('opacity', 0).remove();
-           var commentEnter = commentWarning.enter().insert('div', '.tag-reference-body').attr('class', 'field-warning comment-warning').style('opacity', 0);
-           commentEnter.append('a').attr('target', '_blank').call(svgIcon('#iD-icon-alert', 'inline')).attr('href', _t('commit.google_warning_link')).append('span').html(_t.html('commit.google_warning'));
-           commentEnter.transition().duration(200).style('opacity', 1);
-         }
+               function md5_cmn(q, a, b, x, s, t) {
+                 return safe_add(bit_rol(safe_add(safe_add(a, q), safe_add(x, t)), s), b);
+               }
 
-         changesetEditor.tags = function (_) {
-           if (!arguments.length) return _tags;
-           _tags = _; // Don't reset _fieldsArr here.
+               function md5_ff(a, b, c, d, x, s, t) {
+                 return md5_cmn(b & c | ~b & d, a, b, x, s, t);
+               }
 
-           return changesetEditor;
-         };
+               function md5_gg(a, b, c, d, x, s, t) {
+                 return md5_cmn(b & d | c & ~d, a, b, x, s, t);
+               }
 
-         changesetEditor.changesetID = function (_) {
-           if (!arguments.length) return _changesetID;
-           if (_changesetID === _) return changesetEditor;
-           _changesetID = _;
-           _fieldsArr = null;
-           return changesetEditor;
-         };
+               function md5_hh(a, b, c, d, x, s, t) {
+                 return md5_cmn(b ^ c ^ d, a, b, x, s, t);
+               }
 
-         return utilRebind(changesetEditor, dispatch$1, 'on');
-       }
+               function md5_ii(a, b, c, d, x, s, t) {
+                 return md5_cmn(c ^ (b | ~d), a, b, x, s, t);
+               }
+             },
 
-       function uiSectionChanges(context) {
-         var detected = utilDetect();
-         var _discardTags = {};
-         _mainFileFetcher.get('discarded').then(function (d) {
-           _discardTags = d;
-         })["catch"](function () {
-           /* ignore */
-         });
-         var section = uiSection('changes-list', context).label(function () {
-           var history = context.history();
-           var summary = history.difference().summary();
-           return _t('inspector.title_count', {
-             title: _t.html('commit.changes'),
-             count: summary.length
-           });
-         }).disclosureContent(renderDisclosureContent);
+             /**
+              * @member Hashes
+              * @class Hashes.SHA1
+              * @param {Object} [config]
+              * @constructor
+              *
+              * A JavaScript implementation of the Secure Hash Algorithm, SHA-1, as defined in FIPS 180-1
+              * Version 2.2 Copyright Paul Johnston 2000 - 2009.
+              * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
+              * See http://pajhome.org.uk/crypt/md5 for details.
+              */
+             SHA1: function SHA1(options) {
+               /**
+                * Private config properties. You may need to tweak these to be compatible with
+                * the server-side, but the defaults work in most cases.
+                * See {@link Hashes.MD5#method-setUpperCase} and {@link Hashes.SHA1#method-setUpperCase}
+                */
+               var hexcase = options && typeof options.uppercase === 'boolean' ? options.uppercase : false,
+                   // hexadecimal output case format. false - lowercase; true - uppercase
+               b64pad = options && typeof options.pad === 'string' ? options.pad : '=',
+                   // base-64 pad character. Defaults to '=' for strict RFC compliance
+               utf8 = options && typeof options.utf8 === 'boolean' ? options.utf8 : true; // enable/disable utf8 encoding
+               // public methods
 
-         function renderDisclosureContent(selection) {
-           var history = context.history();
-           var summary = history.difference().summary();
-           var container = selection.selectAll('.commit-section').data([0]);
-           var containerEnter = container.enter().append('div').attr('class', 'commit-section');
-           containerEnter.append('ul').attr('class', 'changeset-list');
-           container = containerEnter.merge(container);
-           var items = container.select('ul').selectAll('li').data(summary);
-           var itemsEnter = items.enter().append('li').attr('class', 'change-item');
-           var buttons = itemsEnter.append('button').on('mouseover', mouseover).on('mouseout', mouseout).on('click', click);
-           buttons.each(function (d) {
-             select(this).call(svgIcon('#iD-icon-' + d.entity.geometry(d.graph), 'pre-text ' + d.changeType));
-           });
-           buttons.append('span').attr('class', 'change-type').html(function (d) {
-             return _t.html('commit.' + d.changeType) + ' ';
-           });
-           buttons.append('strong').attr('class', 'entity-type').html(function (d) {
-             var matched = _mainPresetIndex.match(d.entity, d.graph);
-             return matched && matched.name() || utilDisplayType(d.entity.id);
-           });
-           buttons.append('span').attr('class', 'entity-name').html(function (d) {
-             var name = utilDisplayName(d.entity) || '',
-                 string = '';
+               this.hex = function (s) {
+                 return rstr2hex(rstr(s), hexcase);
+               };
 
-             if (name !== '') {
-               string += ':';
-             }
+               this.b64 = function (s) {
+                 return rstr2b64(rstr(s), b64pad);
+               };
 
-             return string += ' ' + name;
-           });
-           items = itemsEnter.merge(items); // Download changeset link
+               this.any = function (s, e) {
+                 return rstr2any(rstr(s), e);
+               };
 
-           var changeset = new osmChangeset().update({
-             id: undefined
-           });
-           var changes = history.changes(actionDiscardTags(history.difference(), _discardTags));
-           delete changeset.id; // Export without chnageset_id
+               this.raw = function (s) {
+                 return rstr(s);
+               };
 
-           var data = JXON.stringify(changeset.osmChangeJXON(changes));
-           var blob = new Blob([data], {
-             type: 'text/xml;charset=utf-8;'
-           });
-           var fileName = 'changes.osc';
-           var linkEnter = container.selectAll('.download-changes').data([0]).enter().append('a').attr('class', 'download-changes');
+               this.hex_hmac = function (k, d) {
+                 return rstr2hex(rstr_hmac(k, d));
+               };
 
-           if (detected.download) {
-             // All except IE11 and Edge
-             linkEnter // download the data as a file
-             .attr('href', window.URL.createObjectURL(blob)).attr('download', fileName);
-           } else {
-             // IE11 and Edge
-             linkEnter // open data uri in a new tab
-             .attr('target', '_blank').on('click.download', function () {
-               navigator.msSaveBlob(blob, fileName);
-             });
-           }
+               this.b64_hmac = function (k, d) {
+                 return rstr2b64(rstr_hmac(k, d), b64pad);
+               };
 
-           linkEnter.call(svgIcon('#iD-icon-load', 'inline')).append('span').html(_t.html('commit.download_changes'));
+               this.any_hmac = function (k, d, e) {
+                 return rstr2any(rstr_hmac(k, d), e);
+               };
+               /**
+                * Perform a simple self-test to see if the VM is working
+                * @return {String} Hexadecimal hash sample
+                * @public
+                */
 
-           function mouseover(d) {
-             if (d.entity) {
-               context.surface().selectAll(utilEntityOrMemberSelector([d.entity.id], context.graph())).classed('hover', true);
-             }
-           }
 
-           function mouseout() {
-             context.surface().selectAll('.hover').classed('hover', false);
-           }
+               this.vm_test = function () {
+                 return hex('abc').toLowerCase() === '900150983cd24fb0d6963f7d28e17f72';
+               };
+               /**
+                * @description Enable/disable uppercase hexadecimal returned string
+                * @param {boolean}
+                * @return {Object} this
+                * @public
+                */
 
-           function click(d3_event, change) {
-             if (change.changeType !== 'deleted') {
-               var entity = change.entity;
-               context.map().zoomToEase(entity);
-               context.surface().selectAll(utilEntityOrMemberSelector([entity.id], context.graph())).classed('hover', true);
-             }
-           }
-         }
 
-         return section;
-       }
+               this.setUpperCase = function (a) {
+                 if (typeof a === 'boolean') {
+                   hexcase = a;
+                 }
 
-       function uiCommitWarnings(context) {
-         function commitWarnings(selection) {
-           var issuesBySeverity = context.validator().getIssuesBySeverity({
-             what: 'edited',
-             where: 'all',
-             includeDisabledRules: true
-           });
+                 return this;
+               };
+               /**
+                * @description Defines a base64 pad string
+                * @param {string} Pad
+                * @return {Object} this
+                * @public
+                */
 
-           for (var severity in issuesBySeverity) {
-             var issues = issuesBySeverity[severity];
-             var section = severity + '-section';
-             var issueItem = severity + '-item';
-             var container = selection.selectAll('.' + section).data(issues.length ? [0] : []);
-             container.exit().remove();
-             var containerEnter = container.enter().append('div').attr('class', 'modal-section ' + section + ' fillL2');
-             containerEnter.append('h3').html(severity === 'warning' ? _t.html('commit.warnings') : _t.html('commit.errors'));
-             containerEnter.append('ul').attr('class', 'changeset-list');
-             container = containerEnter.merge(container);
-             var items = container.select('ul').selectAll('li').data(issues, function (d) {
-               return d.id;
-             });
-             items.exit().remove();
-             var itemsEnter = items.enter().append('li').attr('class', issueItem);
-             var buttons = itemsEnter.append('button').on('mouseover', function (d3_event, d) {
-               if (d.entityIds) {
-                 context.surface().selectAll(utilEntityOrMemberSelector(d.entityIds, context.graph())).classed('hover', true);
-               }
-             }).on('mouseout', function () {
-               context.surface().selectAll('.hover').classed('hover', false);
-             }).on('click', function (d3_event, d) {
-               context.validator().focusIssue(d);
-             });
-             buttons.call(svgIcon('#iD-icon-alert', 'pre-text'));
-             buttons.append('strong').attr('class', 'issue-message');
-             buttons.filter(function (d) {
-               return d.tooltip;
-             }).call(uiTooltip().title(function (d) {
-               return d.tooltip;
-             }).placement('top'));
-             items = itemsEnter.merge(items);
-             items.selectAll('.issue-message').html(function (d) {
-               return d.message(context);
-             });
-           }
-         }
 
-         return commitWarnings;
-       }
+               this.setPad = function (a) {
+                 b64pad = a || b64pad;
+                 return this;
+               };
+               /**
+                * @description Defines a base64 pad string
+                * @param {boolean}
+                * @return {Object} this
+                * @public
+                */
 
-       var readOnlyTags = [/^changesets_count$/, /^created_by$/, /^ideditor:/, /^imagery_used$/, /^host$/, /^locale$/, /^warnings:/, /^resolved:/, /^closed:note$/, /^closed:keepright$/, /^closed:improveosm:/, /^closed:osmose:/]; // treat most punctuation (except -, _, +, &) as hashtag delimiters - #4398
-       // from https://stackoverflow.com/a/25575009
 
-       var hashtagRegex = /(#[^\u2000-\u206F\u2E00-\u2E7F\s\\'!"#$%()*,.\/:;<=>?@\[\]^`{|}~]+)/g;
-       function uiCommit(context) {
-         var dispatch$1 = dispatch('cancel');
+               this.setUTF8 = function (a) {
+                 if (typeof a === 'boolean') {
+                   utf8 = a;
+                 }
+
+                 return this;
+               }; // private methods
 
-         var _userDetails;
+               /**
+                * Calculate the SHA-512 of a raw string
+                */
 
-         var _selection;
 
-         var changesetEditor = uiChangesetEditor(context).on('change', changeTags);
-         var rawTagEditor = uiSectionRawTagEditor('changeset-tag-editor', context).on('change', changeTags).readOnlyTags(readOnlyTags);
-         var commitChanges = uiSectionChanges(context);
-         var commitWarnings = uiCommitWarnings(context);
+               function rstr(s) {
+                 s = utf8 ? utf8Encode(s) : s;
+                 return binb2rstr(binb(rstr2binb(s), s.length * 8));
+               }
+               /**
+                * Calculate the HMAC-SHA1 of a key and some data (raw strings)
+                */
 
-         function commit(selection) {
-           _selection = selection; // Initialize changeset if one does not exist yet.
 
-           if (!context.changeset) initChangeset();
-           loadDerivedChangesetTags();
-           selection.call(render);
-         }
+               function rstr_hmac(key, data) {
+                 var bkey, ipad, opad, i, hash;
+                 key = utf8 ? utf8Encode(key) : key;
+                 data = utf8 ? utf8Encode(data) : data;
+                 bkey = rstr2binb(key);
 
-         function initChangeset() {
-           // expire stored comment, hashtags, source after cutoff datetime - #3947 #4899
-           var commentDate = +corePreferences('commentDate') || 0;
-           var currDate = Date.now();
-           var cutoff = 2 * 86400 * 1000; // 2 days
+                 if (bkey.length > 16) {
+                   bkey = binb(bkey, key.length * 8);
+                 }
 
-           if (commentDate > currDate || currDate - commentDate > cutoff) {
-             corePreferences('comment', null);
-             corePreferences('hashtags', null);
-             corePreferences('source', null);
-           } // load in explicitly-set values, if any
+                 ipad = Array(16), opad = Array(16);
 
+                 for (i = 0; i < 16; i += 1) {
+                   ipad[i] = bkey[i] ^ 0x36363636;
+                   opad[i] = bkey[i] ^ 0x5C5C5C5C;
+                 }
 
-           if (context.defaultChangesetComment()) {
-             corePreferences('comment', context.defaultChangesetComment());
-             corePreferences('commentDate', Date.now());
-           }
+                 hash = binb(ipad.concat(rstr2binb(data)), 512 + data.length * 8);
+                 return binb2rstr(binb(opad.concat(hash), 512 + 160));
+               }
+               /**
+                * Calculate the SHA-1 of an array of big-endian words, and a bit length
+                */
 
-           if (context.defaultChangesetSource()) {
-             corePreferences('source', context.defaultChangesetSource());
-             corePreferences('commentDate', Date.now());
-           }
 
-           if (context.defaultChangesetHashtags()) {
-             corePreferences('hashtags', context.defaultChangesetHashtags());
-             corePreferences('commentDate', Date.now());
-           }
+               function binb(x, len) {
+                 var i,
+                     j,
+                     t,
+                     olda,
+                     oldb,
+                     oldc,
+                     oldd,
+                     olde,
+                     w = Array(80),
+                     a = 1732584193,
+                     b = -271733879,
+                     c = -1732584194,
+                     d = 271733878,
+                     e = -1009589776;
+                 /* append padding */
 
-           var detected = utilDetect();
-           var tags = {
-             comment: corePreferences('comment') || '',
-             created_by: context.cleanTagValue('iD ' + context.version),
-             host: context.cleanTagValue(detected.host),
-             locale: context.cleanTagValue(_mainLocalizer.localeCode())
-           }; // call findHashtags initially - this will remove stored
-           // hashtags if any hashtags are found in the comment - #4304
+                 x[len >> 5] |= 0x80 << 24 - len % 32;
+                 x[(len + 64 >> 9 << 4) + 15] = len;
 
-           findHashtags(tags, true);
-           var hashtags = corePreferences('hashtags');
+                 for (i = 0; i < x.length; i += 16) {
+                   olda = a;
+                   oldb = b;
+                   oldc = c;
+                   oldd = d;
+                   olde = e;
 
-           if (hashtags) {
-             tags.hashtags = hashtags;
-           }
+                   for (j = 0; j < 80; j += 1) {
+                     if (j < 16) {
+                       w[j] = x[i + j];
+                     } else {
+                       w[j] = bit_rol(w[j - 3] ^ w[j - 8] ^ w[j - 14] ^ w[j - 16], 1);
+                     }
 
-           var source = corePreferences('source');
+                     t = safe_add(safe_add(bit_rol(a, 5), sha1_ft(j, b, c, d)), safe_add(safe_add(e, w[j]), sha1_kt(j)));
+                     e = d;
+                     d = c;
+                     c = bit_rol(b, 30);
+                     b = a;
+                     a = t;
+                   }
 
-           if (source) {
-             tags.source = source;
-           }
+                   a = safe_add(a, olda);
+                   b = safe_add(b, oldb);
+                   c = safe_add(c, oldc);
+                   d = safe_add(d, oldd);
+                   e = safe_add(e, olde);
+                 }
 
-           var photoOverlaysUsed = context.history().photoOverlaysUsed();
+                 return Array(a, b, c, d, e);
+               }
+               /**
+                * Perform the appropriate triplet combination function for the current
+                * iteration
+                */
 
-           if (photoOverlaysUsed.length) {
-             var sources = (tags.source || '').split(';'); // include this tag for any photo layer
 
-             if (sources.indexOf('streetlevel imagery') === -1) {
-               sources.push('streetlevel imagery');
-             } // add the photo overlays used during editing as sources
+               function sha1_ft(t, b, c, d) {
+                 if (t < 20) {
+                   return b & c | ~b & d;
+                 }
 
+                 if (t < 40) {
+                   return b ^ c ^ d;
+                 }
 
-             photoOverlaysUsed.forEach(function (photoOverlay) {
-               if (sources.indexOf(photoOverlay) === -1) {
-                 sources.push(photoOverlay);
+                 if (t < 60) {
+                   return b & c | b & d | c & d;
+                 }
+
+                 return b ^ c ^ d;
                }
-             });
-             tags.source = context.cleanTagValue(sources.join(';'));
-           }
+               /**
+                * Determine the appropriate additive constant for the current iteration
+                */
 
-           context.changeset = new osmChangeset({
-             tags: tags
-           });
-         } // Calculates read-only metadata tags based on the user's editing session and applies
-         // them to the changeset.
 
+               function sha1_kt(t) {
+                 return t < 20 ? 1518500249 : t < 40 ? 1859775393 : t < 60 ? -1894007588 : -899497514;
+               }
+             },
 
-         function loadDerivedChangesetTags() {
-           var osm = context.connection();
-           if (!osm) return;
-           var tags = Object.assign({}, context.changeset.tags); // shallow copy
-           // assign tags for imagery used
+             /**
+              * @class Hashes.SHA256
+              * @param {config}
+              *
+              * A JavaScript implementation of the Secure Hash Algorithm, SHA-256, as defined in FIPS 180-2
+              * Version 2.2 Copyright Angel Marin, Paul Johnston 2000 - 2009.
+              * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
+              * See http://pajhome.org.uk/crypt/md5 for details.
+              * Also http://anmar.eu.org/projects/jssha2/
+              */
+             SHA256: function SHA256(options) {
+               /**
+                * Private properties configuration variables. You may need to tweak these to be compatible with
+                * the server-side, but the defaults work in most cases.
+                * @see this.setUpperCase() method
+                * @see this.setPad() method
+                */
+               options && typeof options.uppercase === 'boolean' ? options.uppercase : false;
+                   var // hexadecimal output case format. false - lowercase; true - uppercase  */
+               b64pad = options && typeof options.pad === 'string' ? options.pad : '=',
 
-           var imageryUsed = context.cleanTagValue(context.history().imageryUsed().join(';'));
-           tags.imagery_used = imageryUsed || 'None'; // assign tags for closed issues and notes
+               /* base-64 pad character. Default '=' for strict RFC compliance   */
+               utf8 = options && typeof options.utf8 === 'boolean' ? options.utf8 : true,
 
-           var osmClosed = osm.getClosedIDs();
-           var itemType;
+               /* enable/disable utf8 encoding */
+               sha256_K;
+               /* privileged (public) methods */
 
-           if (osmClosed.length) {
-             tags['closed:note'] = context.cleanTagValue(osmClosed.join(';'));
-           }
+               this.hex = function (s) {
+                 return rstr2hex(rstr(s, utf8));
+               };
 
-           if (services.keepRight) {
-             var krClosed = services.keepRight.getClosedIDs();
+               this.b64 = function (s) {
+                 return rstr2b64(rstr(s, utf8), b64pad);
+               };
 
-             if (krClosed.length) {
-               tags['closed:keepright'] = context.cleanTagValue(krClosed.join(';'));
-             }
-           }
+               this.any = function (s, e) {
+                 return rstr2any(rstr(s, utf8), e);
+               };
 
-           if (services.improveOSM) {
-             var iOsmClosed = services.improveOSM.getClosedCounts();
+               this.raw = function (s) {
+                 return rstr(s, utf8);
+               };
 
-             for (itemType in iOsmClosed) {
-               tags['closed:improveosm:' + itemType] = context.cleanTagValue(iOsmClosed[itemType].toString());
-             }
-           }
+               this.hex_hmac = function (k, d) {
+                 return rstr2hex(rstr_hmac(k, d));
+               };
 
-           if (services.osmose) {
-             var osmoseClosed = services.osmose.getClosedCounts();
+               this.b64_hmac = function (k, d) {
+                 return rstr2b64(rstr_hmac(k, d), b64pad);
+               };
 
-             for (itemType in osmoseClosed) {
-               tags['closed:osmose:' + itemType] = context.cleanTagValue(osmoseClosed[itemType].toString());
-             }
-           } // remove existing issue counts
+               this.any_hmac = function (k, d, e) {
+                 return rstr2any(rstr_hmac(k, d), e);
+               };
+               /**
+                * Perform a simple self-test to see if the VM is working
+                * @return {String} Hexadecimal hash sample
+                * @public
+                */
 
 
-           for (var key in tags) {
-             if (key.match(/(^warnings:)|(^resolved:)/)) {
-               delete tags[key];
-             }
-           }
+               this.vm_test = function () {
+                 return hex('abc').toLowerCase() === '900150983cd24fb0d6963f7d28e17f72';
+               };
+               /**
+                * Enable/disable uppercase hexadecimal returned string
+                * @param {boolean}
+                * @return {Object} this
+                * @public
+                */
 
-           function addIssueCounts(issues, prefix) {
-             var issuesByType = utilArrayGroupBy(issues, 'type');
 
-             for (var issueType in issuesByType) {
-               var issuesOfType = issuesByType[issueType];
+               this.setUpperCase = function (a) {
 
-               if (issuesOfType[0].subtype) {
-                 var issuesBySubtype = utilArrayGroupBy(issuesOfType, 'subtype');
+                 return this;
+               };
+               /**
+                * @description Defines a base64 pad string
+                * @param {string} Pad
+                * @return {Object} this
+                * @public
+                */
 
-                 for (var issueSubtype in issuesBySubtype) {
-                   var issuesOfSubtype = issuesBySubtype[issueSubtype];
-                   tags[prefix + ':' + issueType + ':' + issueSubtype] = context.cleanTagValue(issuesOfSubtype.length.toString());
-                 }
-               } else {
-                 tags[prefix + ':' + issueType] = context.cleanTagValue(issuesOfType.length.toString());
-               }
-             }
-           } // add counts of warnings generated by the user's edits
 
+               this.setPad = function (a) {
+                 b64pad = a || b64pad;
+                 return this;
+               };
+               /**
+                * Defines a base64 pad string
+                * @param {boolean}
+                * @return {Object} this
+                * @public
+                */
 
-           var warnings = context.validator().getIssuesBySeverity({
-             what: 'edited',
-             where: 'all',
-             includeIgnored: true,
-             includeDisabledRules: true
-           }).warning;
-           addIssueCounts(warnings, 'warnings'); // add counts of issues resolved by the user's edits
 
-           var resolvedIssues = context.validator().getResolvedIssues();
-           addIssueCounts(resolvedIssues, 'resolved');
-           context.changeset = context.changeset.update({
-             tags: tags
-           });
-         }
+               this.setUTF8 = function (a) {
+                 if (typeof a === 'boolean') {
+                   utf8 = a;
+                 }
 
-         function render(selection) {
-           var osm = context.connection();
-           if (!osm) return;
-           var header = selection.selectAll('.header').data([0]);
-           var headerTitle = header.enter().append('div').attr('class', 'header fillL');
-           headerTitle.append('div').append('h3').html(_t.html('commit.title'));
-           headerTitle.append('button').attr('class', 'close').on('click', function () {
-             dispatch$1.call('cancel', this);
-           }).call(svgIcon('#iD-icon-close'));
-           var body = selection.selectAll('.body').data([0]);
-           body = body.enter().append('div').attr('class', 'body').merge(body); // Changeset Section
+                 return this;
+               }; // private methods
 
-           var changesetSection = body.selectAll('.changeset-editor').data([0]);
-           changesetSection = changesetSection.enter().append('div').attr('class', 'modal-section changeset-editor').merge(changesetSection);
-           changesetSection.call(changesetEditor.changesetID(context.changeset.id).tags(context.changeset.tags)); // Warnings
+               /**
+                * Calculate the SHA-512 of a raw string
+                */
 
-           body.call(commitWarnings); // Upload Explanation
 
-           var saveSection = body.selectAll('.save-section').data([0]);
-           saveSection = saveSection.enter().append('div').attr('class', 'modal-section save-section fillL').merge(saveSection);
-           var prose = saveSection.selectAll('.commit-info').data([0]);
+               function rstr(s, utf8) {
+                 s = utf8 ? utf8Encode(s) : s;
+                 return binb2rstr(binb(rstr2binb(s), s.length * 8));
+               }
+               /**
+                * Calculate the HMAC-sha256 of a key and some data (raw strings)
+                */
 
-           if (prose.enter().size()) {
-             // first time, make sure to update user details in prose
-             _userDetails = null;
-           }
 
-           prose = prose.enter().append('p').attr('class', 'commit-info').html(_t.html('commit.upload_explanation')).merge(prose); // always check if this has changed, but only update prose.html()
-           // if needed, because it can trigger a style recalculation
+               function rstr_hmac(key, data) {
+                 key = utf8 ? utf8Encode(key) : key;
+                 data = utf8 ? utf8Encode(data) : data;
+                 var hash,
+                     i = 0,
+                     bkey = rstr2binb(key),
+                     ipad = Array(16),
+                     opad = Array(16);
 
-           osm.userDetails(function (err, user) {
-             if (err) return;
-             if (_userDetails === user) return; // no change
+                 if (bkey.length > 16) {
+                   bkey = binb(bkey, key.length * 8);
+                 }
 
-             _userDetails = user;
-             var userLink = select(document.createElement('div'));
+                 for (; i < 16; i += 1) {
+                   ipad[i] = bkey[i] ^ 0x36363636;
+                   opad[i] = bkey[i] ^ 0x5C5C5C5C;
+                 }
 
-             if (user.image_url) {
-               userLink.append('img').attr('src', user.image_url).attr('class', 'icon pre-text user-icon');
-             }
+                 hash = binb(ipad.concat(rstr2binb(data)), 512 + data.length * 8);
+                 return binb2rstr(binb(opad.concat(hash), 512 + 256));
+               }
+               /*
+                * Main sha256 function, with its support functions
+                */
 
-             userLink.append('a').attr('class', 'user-info').html(user.display_name).attr('href', osm.userURL(user.display_name)).attr('target', '_blank');
-             prose.html(_t.html('commit.upload_explanation_with_user', {
-               user: userLink.html()
-             }));
-           }); // Request Review
 
-           var requestReview = saveSection.selectAll('.request-review').data([0]); // Enter
+               function sha256_S(X, n) {
+                 return X >>> n | X << 32 - n;
+               }
 
-           var requestReviewEnter = requestReview.enter().append('div').attr('class', 'request-review');
-           var requestReviewDomId = utilUniqueDomId('commit-input-request-review');
-           var labelEnter = requestReviewEnter.append('label').attr('for', requestReviewDomId);
-           labelEnter.append('input').attr('type', 'checkbox').attr('id', requestReviewDomId);
-           labelEnter.append('span').html(_t.html('commit.request_review')); // Update
+               function sha256_R(X, n) {
+                 return X >>> n;
+               }
 
-           requestReview = requestReview.merge(requestReviewEnter);
-           var requestReviewInput = requestReview.selectAll('input').property('checked', isReviewRequested(context.changeset.tags)).on('change', toggleRequestReview); // Buttons
+               function sha256_Ch(x, y, z) {
+                 return x & y ^ ~x & z;
+               }
 
-           var buttonSection = saveSection.selectAll('.buttons').data([0]); // enter
+               function sha256_Maj(x, y, z) {
+                 return x & y ^ x & z ^ y & z;
+               }
 
-           var buttonEnter = buttonSection.enter().append('div').attr('class', 'buttons fillL');
-           buttonEnter.append('button').attr('class', 'secondary-action button cancel-button').append('span').attr('class', 'label').html(_t.html('commit.cancel'));
-           var uploadButton = buttonEnter.append('button').attr('class', 'action button save-button');
-           uploadButton.append('span').attr('class', 'label').html(_t.html('commit.save'));
-           var uploadBlockerTooltipText = getUploadBlockerMessage(); // update
+               function sha256_Sigma0256(x) {
+                 return sha256_S(x, 2) ^ sha256_S(x, 13) ^ sha256_S(x, 22);
+               }
 
-           buttonSection = buttonSection.merge(buttonEnter);
-           buttonSection.selectAll('.cancel-button').on('click.cancel', function () {
-             dispatch$1.call('cancel', this);
-           });
-           buttonSection.selectAll('.save-button').classed('disabled', uploadBlockerTooltipText !== null).on('click.save', function () {
-             if (!select(this).classed('disabled')) {
-               this.blur(); // avoid keeping focus on the button - #4641
+               function sha256_Sigma1256(x) {
+                 return sha256_S(x, 6) ^ sha256_S(x, 11) ^ sha256_S(x, 25);
+               }
 
-               for (var key in context.changeset.tags) {
-                 // remove any empty keys before upload
-                 if (!key) delete context.changeset.tags[key];
+               function sha256_Gamma0256(x) {
+                 return sha256_S(x, 7) ^ sha256_S(x, 18) ^ sha256_R(x, 3);
                }
 
-               context.uploader().save(context.changeset);
-             }
-           }); // remove any existing tooltip
+               function sha256_Gamma1256(x) {
+                 return sha256_S(x, 17) ^ sha256_S(x, 19) ^ sha256_R(x, 10);
+               }
 
-           uiTooltip().destroyAny(buttonSection.selectAll('.save-button'));
+               sha256_K = [1116352408, 1899447441, -1245643825, -373957723, 961987163, 1508970993, -1841331548, -1424204075, -670586216, 310598401, 607225278, 1426881987, 1925078388, -2132889090, -1680079193, -1046744716, -459576895, -272742522, 264347078, 604807628, 770255983, 1249150122, 1555081692, 1996064986, -1740746414, -1473132947, -1341970488, -1084653625, -958395405, -710438585, 113926993, 338241895, 666307205, 773529912, 1294757372, 1396182291, 1695183700, 1986661051, -2117940946, -1838011259, -1564481375, -1474664885, -1035236496, -949202525, -778901479, -694614492, -200395387, 275423344, 430227734, 506948616, 659060556, 883997877, 958139571, 1322822218, 1537002063, 1747873779, 1955562222, 2024104815, -2067236844, -1933114872, -1866530822, -1538233109, -1090935817, -965641998];
 
-           if (uploadBlockerTooltipText) {
-             buttonSection.selectAll('.save-button').call(uiTooltip().title(uploadBlockerTooltipText).placement('top'));
-           } // Raw Tag Editor
+               function binb(m, l) {
+                 var HASH = [1779033703, -1150833019, 1013904242, -1521486534, 1359893119, -1694144372, 528734635, 1541459225];
+                 var W = new Array(64);
+                 var a, b, c, d, e, f, g, h;
+                 var i, j, T1, T2;
+                 /* append padding */
 
+                 m[l >> 5] |= 0x80 << 24 - l % 32;
+                 m[(l + 64 >> 9 << 4) + 15] = l;
 
-           var tagSection = body.selectAll('.tag-section.raw-tag-editor').data([0]);
-           tagSection = tagSection.enter().append('div').attr('class', 'modal-section tag-section raw-tag-editor').merge(tagSection);
-           tagSection.call(rawTagEditor.tags(Object.assign({}, context.changeset.tags)) // shallow copy
-           .render);
-           var changesSection = body.selectAll('.commit-changes-section').data([0]);
-           changesSection = changesSection.enter().append('div').attr('class', 'modal-section commit-changes-section').merge(changesSection); // Change summary
+                 for (i = 0; i < m.length; i += 16) {
+                   a = HASH[0];
+                   b = HASH[1];
+                   c = HASH[2];
+                   d = HASH[3];
+                   e = HASH[4];
+                   f = HASH[5];
+                   g = HASH[6];
+                   h = HASH[7];
 
-           changesSection.call(commitChanges.render);
+                   for (j = 0; j < 64; j += 1) {
+                     if (j < 16) {
+                       W[j] = m[j + i];
+                     } else {
+                       W[j] = safe_add(safe_add(safe_add(sha256_Gamma1256(W[j - 2]), W[j - 7]), sha256_Gamma0256(W[j - 15])), W[j - 16]);
+                     }
 
-           function toggleRequestReview() {
-             var rr = requestReviewInput.property('checked');
-             updateChangeset({
-               review_requested: rr ? 'yes' : undefined
-             });
-             tagSection.call(rawTagEditor.tags(Object.assign({}, context.changeset.tags)) // shallow copy
-             .render);
-           }
-         }
+                     T1 = safe_add(safe_add(safe_add(safe_add(h, sha256_Sigma1256(e)), sha256_Ch(e, f, g)), sha256_K[j]), W[j]);
+                     T2 = safe_add(sha256_Sigma0256(a), sha256_Maj(a, b, c));
+                     h = g;
+                     g = f;
+                     f = e;
+                     e = safe_add(d, T1);
+                     d = c;
+                     c = b;
+                     b = a;
+                     a = safe_add(T1, T2);
+                   }
 
-         function getUploadBlockerMessage() {
-           var errors = context.validator().getIssuesBySeverity({
-             what: 'edited',
-             where: 'all'
-           }).error;
+                   HASH[0] = safe_add(a, HASH[0]);
+                   HASH[1] = safe_add(b, HASH[1]);
+                   HASH[2] = safe_add(c, HASH[2]);
+                   HASH[3] = safe_add(d, HASH[3]);
+                   HASH[4] = safe_add(e, HASH[4]);
+                   HASH[5] = safe_add(f, HASH[5]);
+                   HASH[6] = safe_add(g, HASH[6]);
+                   HASH[7] = safe_add(h, HASH[7]);
+                 }
 
-           if (errors.length) {
-             return _t('commit.outstanding_errors_message', {
-               count: errors.length
-             });
-           } else {
-             var hasChangesetComment = context.changeset && context.changeset.tags.comment && context.changeset.tags.comment.trim().length;
+                 return HASH;
+               }
+             },
 
-             if (!hasChangesetComment) {
-               return _t('commit.comment_needed_message');
-             }
-           }
+             /**
+              * @class Hashes.SHA512
+              * @param {config}
+              *
+              * A JavaScript implementation of the Secure Hash Algorithm, SHA-512, as defined in FIPS 180-2
+              * Version 2.2 Copyright Anonymous Contributor, Paul Johnston 2000 - 2009.
+              * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
+              * See http://pajhome.org.uk/crypt/md5 for details.
+              */
+             SHA512: function SHA512(options) {
+               /**
+                * Private properties configuration variables. You may need to tweak these to be compatible with
+                * the server-side, but the defaults work in most cases.
+                * @see this.setUpperCase() method
+                * @see this.setPad() method
+                */
+               options && typeof options.uppercase === 'boolean' ? options.uppercase : false;
 
-           return null;
-         }
+               var /* hexadecimal output case format. false - lowercase; true - uppercase  */
+               b64pad = options && typeof options.pad === 'string' ? options.pad : '=',
 
-         function changeTags(_, changed, onInput) {
-           if (changed.hasOwnProperty('comment')) {
-             if (changed.comment === undefined) {
-               changed.comment = '';
-             }
+               /* base-64 pad character. Default '=' for strict RFC compliance   */
+               utf8 = options && typeof options.utf8 === 'boolean' ? options.utf8 : true,
 
-             if (!onInput) {
-               corePreferences('comment', changed.comment);
-               corePreferences('commentDate', Date.now());
-             }
-           }
+               /* enable/disable utf8 encoding */
+               sha512_k;
+               /* privileged (public) methods */
 
-           if (changed.hasOwnProperty('source')) {
-             if (changed.source === undefined) {
-               corePreferences('source', null);
-             } else if (!onInput) {
-               corePreferences('source', changed.source);
-               corePreferences('commentDate', Date.now());
-             }
-           } // no need to update `prefs` for `hashtags` here since it's done in `updateChangeset`
+               this.hex = function (s) {
+                 return rstr2hex(rstr(s));
+               };
 
+               this.b64 = function (s) {
+                 return rstr2b64(rstr(s), b64pad);
+               };
 
-           updateChangeset(changed, onInput);
+               this.any = function (s, e) {
+                 return rstr2any(rstr(s), e);
+               };
 
-           if (_selection) {
-             _selection.call(render);
-           }
-         }
+               this.raw = function (s) {
+                 return rstr(s);
+               };
 
-         function findHashtags(tags, commentOnly) {
-           var detectedHashtags = commentHashtags();
+               this.hex_hmac = function (k, d) {
+                 return rstr2hex(rstr_hmac(k, d));
+               };
 
-           if (detectedHashtags.length) {
-             // always remove stored hashtags if there are hashtags in the comment - #4304
-             corePreferences('hashtags', null);
-           }
+               this.b64_hmac = function (k, d) {
+                 return rstr2b64(rstr_hmac(k, d), b64pad);
+               };
 
-           if (!detectedHashtags.length || !commentOnly) {
-             detectedHashtags = detectedHashtags.concat(hashtagHashtags());
-           }
+               this.any_hmac = function (k, d, e) {
+                 return rstr2any(rstr_hmac(k, d), e);
+               };
+               /**
+                * Perform a simple self-test to see if the VM is working
+                * @return {String} Hexadecimal hash sample
+                * @public
+                */
 
-           var allLowerCase = new Set();
-           return detectedHashtags.filter(function (hashtag) {
-             // Compare tags as lowercase strings, but keep original case tags
-             var lowerCase = hashtag.toLowerCase();
 
-             if (!allLowerCase.has(lowerCase)) {
-               allLowerCase.add(lowerCase);
-               return true;
-             }
+               this.vm_test = function () {
+                 return hex('abc').toLowerCase() === '900150983cd24fb0d6963f7d28e17f72';
+               };
+               /**
+                * @description Enable/disable uppercase hexadecimal returned string
+                * @param {boolean}
+                * @return {Object} this
+                * @public
+                */
 
-             return false;
-           }); // Extract hashtags from `comment`
 
-           function commentHashtags() {
-             var matches = (tags.comment || '').replace(/http\S*/g, '') // drop anything that looks like a URL - #4289
-             .match(hashtagRegex);
-             return matches || [];
-           } // Extract and clean hashtags from `hashtags`
+               this.setUpperCase = function (a) {
 
+                 return this;
+               };
+               /**
+                * @description Defines a base64 pad string
+                * @param {string} Pad
+                * @return {Object} this
+                * @public
+                */
 
-           function hashtagHashtags() {
-             var matches = (tags.hashtags || '').split(/[,;\s]+/).map(function (s) {
-               if (s[0] !== '#') {
-                 s = '#' + s;
-               } // prepend '#'
 
+               this.setPad = function (a) {
+                 b64pad = a || b64pad;
+                 return this;
+               };
+               /**
+                * @description Defines a base64 pad string
+                * @param {boolean}
+                * @return {Object} this
+                * @public
+                */
 
-               var matched = s.match(hashtagRegex);
-               return matched && matched[0];
-             }).filter(Boolean); // exclude falsy
 
-             return matches || [];
-           }
-         }
+               this.setUTF8 = function (a) {
+                 if (typeof a === 'boolean') {
+                   utf8 = a;
+                 }
 
-         function isReviewRequested(tags) {
-           var rr = tags.review_requested;
-           if (rr === undefined) return false;
-           rr = rr.trim().toLowerCase();
-           return !(rr === '' || rr === 'no');
-         }
+                 return this;
+               };
+               /* private methods */
 
-         function updateChangeset(changed, onInput) {
-           var tags = Object.assign({}, context.changeset.tags); // shallow copy
+               /**
+                * Calculate the SHA-512 of a raw string
+                */
 
-           Object.keys(changed).forEach(function (k) {
-             var v = changed[k];
-             k = context.cleanTagKey(k);
-             if (readOnlyTags.indexOf(k) !== -1) return;
 
-             if (v === undefined) {
-               delete tags[k];
-             } else if (onInput) {
-               tags[k] = v;
-             } else {
-               tags[k] = context.cleanTagValue(v);
-             }
-           });
+               function rstr(s) {
+                 s = utf8 ? utf8Encode(s) : s;
+                 return binb2rstr(binb(rstr2binb(s), s.length * 8));
+               }
+               /*
+                * Calculate the HMAC-SHA-512 of a key and some data (raw strings)
+                */
 
-           if (!onInput) {
-             // when changing the comment, override hashtags with any found in comment.
-             var commentOnly = changed.hasOwnProperty('comment') && changed.comment !== '';
-             var arr = findHashtags(tags, commentOnly);
 
-             if (arr.length) {
-               tags.hashtags = context.cleanTagValue(arr.join(';'));
-               corePreferences('hashtags', tags.hashtags);
-             } else {
-               delete tags.hashtags;
-               corePreferences('hashtags', null);
-             }
-           } // always update userdetails, just in case user reauthenticates as someone else
+               function rstr_hmac(key, data) {
+                 key = utf8 ? utf8Encode(key) : key;
+                 data = utf8 ? utf8Encode(data) : data;
+                 var hash,
+                     i = 0,
+                     bkey = rstr2binb(key),
+                     ipad = Array(32),
+                     opad = Array(32);
 
+                 if (bkey.length > 32) {
+                   bkey = binb(bkey, key.length * 8);
+                 }
 
-           if (_userDetails && _userDetails.changesets_count !== undefined) {
-             var changesetsCount = parseInt(_userDetails.changesets_count, 10) + 1; // #4283
+                 for (; i < 32; i += 1) {
+                   ipad[i] = bkey[i] ^ 0x36363636;
+                   opad[i] = bkey[i] ^ 0x5C5C5C5C;
+                 }
 
-             tags.changesets_count = String(changesetsCount); // first 100 edits - new user
+                 hash = binb(ipad.concat(rstr2binb(data)), 1024 + data.length * 8);
+                 return binb2rstr(binb(opad.concat(hash), 1024 + 512));
+               }
+               /**
+                * Calculate the SHA-512 of an array of big-endian dwords, and a bit length
+                */
 
-             if (changesetsCount <= 100) {
-               var s;
-               s = corePreferences('walkthrough_completed');
 
-               if (s) {
-                 tags['ideditor:walkthrough_completed'] = s;
-               }
+               function binb(x, len) {
+                 var j,
+                     i,
+                     l,
+                     W = new Array(80),
+                     hash = new Array(16),
+                     //Initial hash values
+                 H = [new int64(0x6a09e667, -205731576), new int64(-1150833019, -2067093701), new int64(0x3c6ef372, -23791573), new int64(-1521486534, 0x5f1d36f1), new int64(0x510e527f, -1377402159), new int64(-1694144372, 0x2b3e6c1f), new int64(0x1f83d9ab, -79577749), new int64(0x5be0cd19, 0x137e2179)],
+                     T1 = new int64(0, 0),
+                     T2 = new int64(0, 0),
+                     a = new int64(0, 0),
+                     b = new int64(0, 0),
+                     c = new int64(0, 0),
+                     d = new int64(0, 0),
+                     e = new int64(0, 0),
+                     f = new int64(0, 0),
+                     g = new int64(0, 0),
+                     h = new int64(0, 0),
+                     //Temporary variables not specified by the document
+                 s0 = new int64(0, 0),
+                     s1 = new int64(0, 0),
+                     Ch = new int64(0, 0),
+                     Maj = new int64(0, 0),
+                     r1 = new int64(0, 0),
+                     r2 = new int64(0, 0),
+                     r3 = new int64(0, 0);
 
-               s = corePreferences('walkthrough_progress');
+                 if (sha512_k === undefined) {
+                   //SHA512 constants
+                   sha512_k = [new int64(0x428a2f98, -685199838), new int64(0x71374491, 0x23ef65cd), new int64(-1245643825, -330482897), new int64(-373957723, -2121671748), new int64(0x3956c25b, -213338824), new int64(0x59f111f1, -1241133031), new int64(-1841331548, -1357295717), new int64(-1424204075, -630357736), new int64(-670586216, -1560083902), new int64(0x12835b01, 0x45706fbe), new int64(0x243185be, 0x4ee4b28c), new int64(0x550c7dc3, -704662302), new int64(0x72be5d74, -226784913), new int64(-2132889090, 0x3b1696b1), new int64(-1680079193, 0x25c71235), new int64(-1046744716, -815192428), new int64(-459576895, -1628353838), new int64(-272742522, 0x384f25e3), new int64(0xfc19dc6, -1953704523), new int64(0x240ca1cc, 0x77ac9c65), new int64(0x2de92c6f, 0x592b0275), new int64(0x4a7484aa, 0x6ea6e483), new int64(0x5cb0a9dc, -1119749164), new int64(0x76f988da, -2096016459), new int64(-1740746414, -295247957), new int64(-1473132947, 0x2db43210), new int64(-1341970488, -1728372417), new int64(-1084653625, -1091629340), new int64(-958395405, 0x3da88fc2), new int64(-710438585, -1828018395), new int64(0x6ca6351, -536640913), new int64(0x14292967, 0xa0e6e70), new int64(0x27b70a85, 0x46d22ffc), new int64(0x2e1b2138, 0x5c26c926), new int64(0x4d2c6dfc, 0x5ac42aed), new int64(0x53380d13, -1651133473), new int64(0x650a7354, -1951439906), new int64(0x766a0abb, 0x3c77b2a8), new int64(-2117940946, 0x47edaee6), new int64(-1838011259, 0x1482353b), new int64(-1564481375, 0x4cf10364), new int64(-1474664885, -1136513023), new int64(-1035236496, -789014639), new int64(-949202525, 0x654be30), new int64(-778901479, -688958952), new int64(-694614492, 0x5565a910), new int64(-200395387, 0x5771202a), new int64(0x106aa070, 0x32bbd1b8), new int64(0x19a4c116, -1194143544), new int64(0x1e376c08, 0x5141ab53), new int64(0x2748774c, -544281703), new int64(0x34b0bcb5, -509917016), new int64(0x391c0cb3, -976659869), new int64(0x4ed8aa4a, -482243893), new int64(0x5b9cca4f, 0x7763e373), new int64(0x682e6ff3, -692930397), new int64(0x748f82ee, 0x5defb2fc), new int64(0x78a5636f, 0x43172f60), new int64(-2067236844, -1578062990), new int64(-1933114872, 0x1a6439ec), new int64(-1866530822, 0x23631e28), new int64(-1538233109, -561857047), new int64(-1090935817, -1295615723), new int64(-965641998, -479046869), new int64(-903397682, -366583396), new int64(-779700025, 0x21c0c207), new int64(-354779690, -840897762), new int64(-176337025, -294727304), new int64(0x6f067aa, 0x72176fba), new int64(0xa637dc5, -1563912026), new int64(0x113f9804, -1090974290), new int64(0x1b710b35, 0x131c471b), new int64(0x28db77f5, 0x23047d84), new int64(0x32caab7b, 0x40c72493), new int64(0x3c9ebe0a, 0x15c9bebc), new int64(0x431d67c4, -1676669620), new int64(0x4cc5d4be, -885112138), new int64(0x597f299c, -60457430), new int64(0x5fcb6fab, 0x3ad6faec), new int64(0x6c44198c, 0x4a475817)];
+                 }
 
-               if (s) {
-                 tags['ideditor:walkthrough_progress'] = s;
-               }
+                 for (i = 0; i < 80; i += 1) {
+                   W[i] = new int64(0, 0);
+                 } // append padding to the source string. The format is described in the FIPS.
 
-               s = corePreferences('walkthrough_started');
 
-               if (s) {
-                 tags['ideditor:walkthrough_started'] = s;
-               }
-             }
-           } else {
-             delete tags.changesets_count;
-           }
+                 x[len >> 5] |= 0x80 << 24 - (len & 0x1f);
+                 x[(len + 128 >> 10 << 5) + 31] = len;
+                 l = x.length;
 
-           if (!fastDeepEqual(context.changeset.tags, tags)) {
-             context.changeset = context.changeset.update({
-               tags: tags
-             });
-           }
-         }
+                 for (i = 0; i < l; i += 32) {
+                   //32 dwords is the block size
+                   int64copy(a, H[0]);
+                   int64copy(b, H[1]);
+                   int64copy(c, H[2]);
+                   int64copy(d, H[3]);
+                   int64copy(e, H[4]);
+                   int64copy(f, H[5]);
+                   int64copy(g, H[6]);
+                   int64copy(h, H[7]);
 
-         commit.reset = function () {
-           context.changeset = null;
-         };
+                   for (j = 0; j < 16; j += 1) {
+                     W[j].h = x[i + 2 * j];
+                     W[j].l = x[i + 2 * j + 1];
+                   }
 
-         return utilRebind(commit, dispatch$1, 'on');
-       }
+                   for (j = 16; j < 80; j += 1) {
+                     //sigma1
+                     int64rrot(r1, W[j - 2], 19);
+                     int64revrrot(r2, W[j - 2], 29);
+                     int64shr(r3, W[j - 2], 6);
+                     s1.l = r1.l ^ r2.l ^ r3.l;
+                     s1.h = r1.h ^ r2.h ^ r3.h; //sigma0
 
-       var globalIsFinite = global_1.isFinite;
+                     int64rrot(r1, W[j - 15], 1);
+                     int64rrot(r2, W[j - 15], 8);
+                     int64shr(r3, W[j - 15], 7);
+                     s0.l = r1.l ^ r2.l ^ r3.l;
+                     s0.h = r1.h ^ r2.h ^ r3.h;
+                     int64add4(W[j], s1, W[j - 7], s0, W[j - 16]);
+                   }
 
-       // `Number.isFinite` method
-       // https://tc39.es/ecma262/#sec-number.isfinite
-       var numberIsFinite = Number.isFinite || function isFinite(it) {
-         return typeof it == 'number' && globalIsFinite(it);
-       };
+                   for (j = 0; j < 80; j += 1) {
+                     //Ch
+                     Ch.l = e.l & f.l ^ ~e.l & g.l;
+                     Ch.h = e.h & f.h ^ ~e.h & g.h; //Sigma1
 
-       // `Number.isFinite` method
-       // https://tc39.es/ecma262/#sec-number.isfinite
-       _export({ target: 'Number', stat: true }, { isFinite: numberIsFinite });
+                     int64rrot(r1, e, 14);
+                     int64rrot(r2, e, 18);
+                     int64revrrot(r3, e, 9);
+                     s1.l = r1.l ^ r2.l ^ r3.l;
+                     s1.h = r1.h ^ r2.h ^ r3.h; //Sigma0
 
-       var RADIUS = 6378137;
-       var FLATTENING = 1 / 298.257223563;
-       var POLAR_RADIUS$1 = 6356752.3142;
-       var wgs84 = {
-         RADIUS: RADIUS,
-         FLATTENING: FLATTENING,
-         POLAR_RADIUS: POLAR_RADIUS$1
-       };
+                     int64rrot(r1, a, 28);
+                     int64revrrot(r2, a, 2);
+                     int64revrrot(r3, a, 7);
+                     s0.l = r1.l ^ r2.l ^ r3.l;
+                     s0.h = r1.h ^ r2.h ^ r3.h; //Maj
 
-       var geometry_1 = geometry;
-       var ring = ringArea;
+                     Maj.l = a.l & b.l ^ a.l & c.l ^ b.l & c.l;
+                     Maj.h = a.h & b.h ^ a.h & c.h ^ b.h & c.h;
+                     int64add5(T1, h, s1, Ch, sha512_k[j], W[j]);
+                     int64add(T2, s0, Maj);
+                     int64copy(h, g);
+                     int64copy(g, f);
+                     int64copy(f, e);
+                     int64add(e, d, T1);
+                     int64copy(d, c);
+                     int64copy(c, b);
+                     int64copy(b, a);
+                     int64add(a, T1, T2);
+                   }
 
-       function geometry(_) {
-         var area = 0,
-             i;
+                   int64add(H[0], H[0], a);
+                   int64add(H[1], H[1], b);
+                   int64add(H[2], H[2], c);
+                   int64add(H[3], H[3], d);
+                   int64add(H[4], H[4], e);
+                   int64add(H[5], H[5], f);
+                   int64add(H[6], H[6], g);
+                   int64add(H[7], H[7], h);
+                 } //represent the hash as an array of 32-bit dwords
 
-         switch (_.type) {
-           case 'Polygon':
-             return polygonArea(_.coordinates);
 
-           case 'MultiPolygon':
-             for (i = 0; i < _.coordinates.length; i++) {
-               area += polygonArea(_.coordinates[i]);
-             }
+                 for (i = 0; i < 8; i += 1) {
+                   hash[2 * i] = H[i].h;
+                   hash[2 * i + 1] = H[i].l;
+                 }
 
-             return area;
+                 return hash;
+               } //A constructor for 64-bit numbers
 
-           case 'Point':
-           case 'MultiPoint':
-           case 'LineString':
-           case 'MultiLineString':
-             return 0;
 
-           case 'GeometryCollection':
-             for (i = 0; i < _.geometries.length; i++) {
-               area += geometry(_.geometries[i]);
-             }
+               function int64(h, l) {
+                 this.h = h;
+                 this.l = l; //this.toString = int64toString;
+               } //Copies src into dst, assuming both are 64-bit numbers
 
-             return area;
-         }
-       }
 
-       function polygonArea(coords) {
-         var area = 0;
+               function int64copy(dst, src) {
+                 dst.h = src.h;
+                 dst.l = src.l;
+               } //Right-rotates a 64-bit number by shift
+               //Won't handle cases of shift>=32
+               //The function revrrot() is for that
 
-         if (coords && coords.length > 0) {
-           area += Math.abs(ringArea(coords[0]));
 
-           for (var i = 1; i < coords.length; i++) {
-             area -= Math.abs(ringArea(coords[i]));
-           }
-         }
+               function int64rrot(dst, x, shift) {
+                 dst.l = x.l >>> shift | x.h << 32 - shift;
+                 dst.h = x.h >>> shift | x.l << 32 - shift;
+               } //Reverses the dwords of the source and then rotates right by shift.
+               //This is equivalent to rotation by 32+shift
 
-         return area;
-       }
-       /**
-        * Calculate the approximate area of the polygon were it projected onto
-        *     the earth.  Note that this area will be positive if ring is oriented
-        *     clockwise, otherwise it will be negative.
-        *
-        * Reference:
-        * Robert. G. Chamberlain and William H. Duquette, "Some Algorithms for
-        *     Polygons on a Sphere", JPL Publication 07-03, Jet Propulsion
-        *     Laboratory, Pasadena, CA, June 2007 http://trs-new.jpl.nasa.gov/dspace/handle/2014/40409
-        *
-        * Returns:
-        * {float} The approximate signed geodesic area of the polygon in square
-        *     meters.
-        */
 
+               function int64revrrot(dst, x, shift) {
+                 dst.l = x.h >>> shift | x.l << 32 - shift;
+                 dst.h = x.l >>> shift | x.h << 32 - shift;
+               } //Bitwise-shifts right a 64-bit number by shift
+               //Won't handle shift>=32, but it's never needed in SHA512
 
-       function ringArea(coords) {
-         var p1,
-             p2,
-             p3,
-             lowerIndex,
-             middleIndex,
-             upperIndex,
-             i,
-             area = 0,
-             coordsLength = coords.length;
 
-         if (coordsLength > 2) {
-           for (i = 0; i < coordsLength; i++) {
-             if (i === coordsLength - 2) {
-               // i = N-2
-               lowerIndex = coordsLength - 2;
-               middleIndex = coordsLength - 1;
-               upperIndex = 0;
-             } else if (i === coordsLength - 1) {
-               // i = N-1
-               lowerIndex = coordsLength - 1;
-               middleIndex = 0;
-               upperIndex = 1;
-             } else {
-               // i = 0 to N-3
-               lowerIndex = i;
-               middleIndex = i + 1;
-               upperIndex = i + 2;
-             }
+               function int64shr(dst, x, shift) {
+                 dst.l = x.l >>> shift | x.h << 32 - shift;
+                 dst.h = x.h >>> shift;
+               } //Adds two 64-bit numbers
+               //Like the original implementation, does not rely on 32-bit operations
 
-             p1 = coords[lowerIndex];
-             p2 = coords[middleIndex];
-             p3 = coords[upperIndex];
-             area += (rad(p3[0]) - rad(p1[0])) * Math.sin(rad(p2[1]));
-           }
 
-           area = area * wgs84.RADIUS * wgs84.RADIUS / 2;
-         }
+               function int64add(dst, x, y) {
+                 var w0 = (x.l & 0xffff) + (y.l & 0xffff);
+                 var w1 = (x.l >>> 16) + (y.l >>> 16) + (w0 >>> 16);
+                 var w2 = (x.h & 0xffff) + (y.h & 0xffff) + (w1 >>> 16);
+                 var w3 = (x.h >>> 16) + (y.h >>> 16) + (w2 >>> 16);
+                 dst.l = w0 & 0xffff | w1 << 16;
+                 dst.h = w2 & 0xffff | w3 << 16;
+               } //Same, except with 4 addends. Works faster than adding them one by one.
 
-         return area;
-       }
 
-       function rad(_) {
-         return _ * Math.PI / 180;
-       }
+               function int64add4(dst, a, b, c, d) {
+                 var w0 = (a.l & 0xffff) + (b.l & 0xffff) + (c.l & 0xffff) + (d.l & 0xffff);
+                 var w1 = (a.l >>> 16) + (b.l >>> 16) + (c.l >>> 16) + (d.l >>> 16) + (w0 >>> 16);
+                 var w2 = (a.h & 0xffff) + (b.h & 0xffff) + (c.h & 0xffff) + (d.h & 0xffff) + (w1 >>> 16);
+                 var w3 = (a.h >>> 16) + (b.h >>> 16) + (c.h >>> 16) + (d.h >>> 16) + (w2 >>> 16);
+                 dst.l = w0 & 0xffff | w1 << 16;
+                 dst.h = w2 & 0xffff | w3 << 16;
+               } //Same, except with 5 addends
 
-       var geojsonArea = {
-         geometry: geometry_1,
-         ring: ring
-       };
 
-       var validateCenter_1 = function validateCenter(center) {
-         var validCenterLengths = [2, 3];
+               function int64add5(dst, a, b, c, d, e) {
+                 var w0 = (a.l & 0xffff) + (b.l & 0xffff) + (c.l & 0xffff) + (d.l & 0xffff) + (e.l & 0xffff),
+                     w1 = (a.l >>> 16) + (b.l >>> 16) + (c.l >>> 16) + (d.l >>> 16) + (e.l >>> 16) + (w0 >>> 16),
+                     w2 = (a.h & 0xffff) + (b.h & 0xffff) + (c.h & 0xffff) + (d.h & 0xffff) + (e.h & 0xffff) + (w1 >>> 16),
+                     w3 = (a.h >>> 16) + (b.h >>> 16) + (c.h >>> 16) + (d.h >>> 16) + (e.h >>> 16) + (w2 >>> 16);
+                 dst.l = w0 & 0xffff | w1 << 16;
+                 dst.h = w2 & 0xffff | w3 << 16;
+               }
+             },
 
-         if (!Array.isArray(center) || !validCenterLengths.includes(center.length)) {
-           throw new Error("ERROR! Center has to be an array of length two or three");
-         }
+             /**
+              * @class Hashes.RMD160
+              * @constructor
+              * @param {Object} [config]
+              *
+              * A JavaScript implementation of the RIPEMD-160 Algorithm
+              * Version 2.2 Copyright Jeremy Lin, Paul Johnston 2000 - 2009.
+              * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
+              * See http://pajhome.org.uk/crypt/md5 for details.
+              * Also http://www.ocf.berkeley.edu/~jjlin/jsotp/
+              */
+             RMD160: function RMD160(options) {
+               /**
+                * Private properties configuration variables. You may need to tweak these to be compatible with
+                * the server-side, but the defaults work in most cases.
+                * @see this.setUpperCase() method
+                * @see this.setPad() method
+                */
+               options && typeof options.uppercase === 'boolean' ? options.uppercase : false;
 
-         var _center = _slicedToArray(center, 2),
-             lng = _center[0],
-             lat = _center[1];
+               var /* hexadecimal output case format. false - lowercase; true - uppercase  */
+               b64pad = options && typeof options.pad === 'string' ? options.pa : '=',
 
-         if (typeof lng !== "number" || typeof lat !== "number") {
-           throw new Error("ERROR! Longitude and Latitude has to be numbers but where ".concat(_typeof(lng), " and ").concat(_typeof(lat)));
-         }
+               /* base-64 pad character. Default '=' for strict RFC compliance   */
+               utf8 = options && typeof options.utf8 === 'boolean' ? options.utf8 : true,
 
-         if (lng > 180 || lng < -180) {
-           throw new Error("ERROR! Longitude has to be between -180 and 180 but was ".concat(lng));
-         }
+               /* enable/disable utf8 encoding */
+               rmd160_r1 = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 7, 4, 13, 1, 10, 6, 15, 3, 12, 0, 9, 5, 2, 14, 11, 8, 3, 10, 14, 4, 9, 15, 8, 1, 2, 7, 0, 6, 13, 11, 5, 12, 1, 9, 11, 10, 0, 8, 12, 4, 13, 3, 7, 15, 14, 5, 6, 2, 4, 0, 5, 9, 7, 12, 2, 10, 14, 1, 3, 8, 11, 6, 15, 13],
+                   rmd160_r2 = [5, 14, 7, 0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12, 6, 11, 3, 7, 0, 13, 5, 10, 14, 15, 8, 12, 4, 9, 1, 2, 15, 5, 1, 3, 7, 14, 6, 9, 11, 8, 12, 2, 10, 0, 4, 13, 8, 6, 4, 1, 3, 11, 15, 0, 5, 12, 2, 13, 9, 7, 10, 14, 12, 15, 10, 4, 1, 5, 8, 7, 6, 2, 13, 14, 0, 3, 9, 11],
+                   rmd160_s1 = [11, 14, 15, 12, 5, 8, 7, 9, 11, 13, 14, 15, 6, 7, 9, 8, 7, 6, 8, 13, 11, 9, 7, 15, 7, 12, 15, 9, 11, 7, 13, 12, 11, 13, 6, 7, 14, 9, 13, 15, 14, 8, 13, 6, 5, 12, 7, 5, 11, 12, 14, 15, 14, 15, 9, 8, 9, 14, 5, 6, 8, 6, 5, 12, 9, 15, 5, 11, 6, 8, 13, 12, 5, 12, 13, 14, 11, 8, 5, 6],
+                   rmd160_s2 = [8, 9, 9, 11, 13, 15, 15, 5, 7, 7, 8, 11, 14, 14, 12, 6, 9, 13, 15, 7, 12, 8, 9, 11, 7, 7, 12, 7, 6, 15, 13, 11, 9, 7, 15, 11, 8, 6, 6, 14, 12, 13, 5, 14, 13, 13, 7, 5, 15, 5, 8, 11, 14, 14, 6, 14, 6, 9, 12, 9, 12, 5, 15, 8, 8, 5, 12, 9, 12, 5, 14, 6, 8, 13, 6, 5, 15, 13, 11, 11];
+               /* privileged (public) methods */
 
-         if (lat > 90 || lat < -90) {
-           throw new Error("ERROR! Latitude has to be between -90 and 90 but was ".concat(lat));
-         }
-       };
+               this.hex = function (s) {
+                 return rstr2hex(rstr(s));
+               };
 
-       var validateCenter = {
-         validateCenter: validateCenter_1
-       };
+               this.b64 = function (s) {
+                 return rstr2b64(rstr(s), b64pad);
+               };
 
-       var validateRadius_1 = function validateRadius(radius) {
-         if (typeof radius !== "number") {
-           throw new Error("ERROR! Radius has to be a positive number but was: ".concat(_typeof(radius)));
-         }
+               this.any = function (s, e) {
+                 return rstr2any(rstr(s), e);
+               };
 
-         if (radius <= 0) {
-           throw new Error("ERROR! Radius has to be a positive number but was: ".concat(radius));
-         }
-       };
+               this.raw = function (s) {
+                 return rstr(s);
+               };
 
-       var validateRadius = {
-         validateRadius: validateRadius_1
-       };
+               this.hex_hmac = function (k, d) {
+                 return rstr2hex(rstr_hmac(k, d));
+               };
 
-       var validateNumberOfEdges_1 = function validateNumberOfEdges(numberOfEdges) {
-         if (typeof numberOfEdges !== "number") {
-           var ARGUMENT_TYPE = Array.isArray(numberOfEdges) ? "array" : _typeof(numberOfEdges);
-           throw new Error("ERROR! Number of edges has to be a number but was: ".concat(ARGUMENT_TYPE));
-         }
+               this.b64_hmac = function (k, d) {
+                 return rstr2b64(rstr_hmac(k, d), b64pad);
+               };
 
-         if (numberOfEdges < 3) {
-           throw new Error("ERROR! Number of edges has to be at least 3 but was: ".concat(numberOfEdges));
-         }
-       };
+               this.any_hmac = function (k, d, e) {
+                 return rstr2any(rstr_hmac(k, d), e);
+               };
+               /**
+                * Perform a simple self-test to see if the VM is working
+                * @return {String} Hexadecimal hash sample
+                * @public
+                */
 
-       var validateNumberOfEdges = {
-         validateNumberOfEdges: validateNumberOfEdges_1
-       };
 
-       var validateEarthRadius_1 = function validateEarthRadius(earthRadius) {
-         if (typeof earthRadius !== "number") {
-           var ARGUMENT_TYPE = Array.isArray(earthRadius) ? "array" : _typeof(earthRadius);
-           throw new Error("ERROR! Earth radius has to be a number but was: ".concat(ARGUMENT_TYPE));
-         }
+               this.vm_test = function () {
+                 return hex('abc').toLowerCase() === '900150983cd24fb0d6963f7d28e17f72';
+               };
+               /**
+                * @description Enable/disable uppercase hexadecimal returned string
+                * @param {boolean}
+                * @return {Object} this
+                * @public
+                */
 
-         if (earthRadius <= 0) {
-           throw new Error("ERROR! Earth radius has to be a positive number but was: ".concat(earthRadius));
-         }
-       };
 
-       var validateEarthRadius = {
-         validateEarthRadius: validateEarthRadius_1
-       };
+               this.setUpperCase = function (a) {
 
-       var validateBearing_1 = function validateBearing(bearing) {
-         if (typeof bearing !== "number") {
-           var ARGUMENT_TYPE = Array.isArray(bearing) ? "array" : _typeof(bearing);
-           throw new Error("ERROR! Bearing has to be a number but was: ".concat(ARGUMENT_TYPE));
-         }
-       };
+                 return this;
+               };
+               /**
+                * @description Defines a base64 pad string
+                * @param {string} Pad
+                * @return {Object} this
+                * @public
+                */
 
-       var validateBearing = {
-         validateBearing: validateBearing_1
-       };
 
-       var validateCenter$1 = validateCenter.validateCenter;
-       var validateRadius$1 = validateRadius.validateRadius;
-       var validateNumberOfEdges$1 = validateNumberOfEdges.validateNumberOfEdges;
-       var validateEarthRadius$1 = validateEarthRadius.validateEarthRadius;
-       var validateBearing$1 = validateBearing.validateBearing;
+               this.setPad = function (a) {
+                 if (typeof a !== 'undefined') {
+                   b64pad = a;
+                 }
 
-       function validateInput(_ref) {
-         var center = _ref.center,
-             radius = _ref.radius,
-             numberOfEdges = _ref.numberOfEdges,
-             earthRadius = _ref.earthRadius,
-             bearing = _ref.bearing;
-         validateCenter$1(center);
-         validateRadius$1(radius);
-         validateNumberOfEdges$1(numberOfEdges);
-         validateEarthRadius$1(earthRadius);
-         validateBearing$1(bearing);
-       }
-
-       var validateCenter_1$1 = validateCenter$1;
-       var validateRadius_1$1 = validateRadius$1;
-       var validateNumberOfEdges_1$1 = validateNumberOfEdges$1;
-       var validateEarthRadius_1$1 = validateEarthRadius$1;
-       var validateBearing_1$1 = validateBearing$1;
-       var validateInput_1 = validateInput;
-       var inputValidation = {
-         validateCenter: validateCenter_1$1,
-         validateRadius: validateRadius_1$1,
-         validateNumberOfEdges: validateNumberOfEdges_1$1,
-         validateEarthRadius: validateEarthRadius_1$1,
-         validateBearing: validateBearing_1$1,
-         validateInput: validateInput_1
-       };
+                 return this;
+               };
+               /**
+                * @description Defines a base64 pad string
+                * @param {boolean}
+                * @return {Object} this
+                * @public
+                */
 
-       var validateInput$1 = inputValidation.validateInput;
-       var defaultEarthRadius = 6378137; // equatorial Earth radius
 
-       function toRadians(angleInDegrees) {
-         return angleInDegrees * Math.PI / 180;
-       }
+               this.setUTF8 = function (a) {
+                 if (typeof a === 'boolean') {
+                   utf8 = a;
+                 }
 
-       function toDegrees(angleInRadians) {
-         return angleInRadians * 180 / Math.PI;
-       }
+                 return this;
+               };
+               /* private methods */
 
-       function offset(c1, distance, earthRadius, bearing) {
-         var lat1 = toRadians(c1[1]);
-         var lon1 = toRadians(c1[0]);
-         var dByR = distance / earthRadius;
-         var lat = Math.asin(Math.sin(lat1) * Math.cos(dByR) + Math.cos(lat1) * Math.sin(dByR) * Math.cos(bearing));
-         var lon = lon1 + Math.atan2(Math.sin(bearing) * Math.sin(dByR) * Math.cos(lat1), Math.cos(dByR) - Math.sin(lat1) * Math.sin(lat));
-         return [toDegrees(lon), toDegrees(lat)];
-       }
+               /**
+                * Calculate the rmd160 of a raw string
+                */
 
-       var circleToPolygon = function circleToPolygon(center, radius, options) {
-         var n = getNumberOfEdges(options);
-         var earthRadius = getEarthRadius(options);
-         var bearing = getBearing(options); // validateInput() throws error on invalid input and do nothing on valid input
 
-         validateInput$1({
-           center: center,
-           radius: radius,
-           numberOfEdges: n,
-           earthRadius: earthRadius,
-           bearing: bearing
-         });
-         var start = toRadians(bearing);
-         var coordinates = [];
+               function rstr(s) {
+                 s = utf8 ? utf8Encode(s) : s;
+                 return binl2rstr(binl(rstr2binl(s), s.length * 8));
+               }
+               /**
+                * Calculate the HMAC-rmd160 of a key and some data (raw strings)
+                */
 
-         for (var i = 0; i < n; ++i) {
-           coordinates.push(offset(center, radius, earthRadius, start + 2 * Math.PI * -i / n));
-         }
 
-         coordinates.push(coordinates[0]);
-         return {
-           type: "Polygon",
-           coordinates: [coordinates]
-         };
-       };
+               function rstr_hmac(key, data) {
+                 key = utf8 ? utf8Encode(key) : key;
+                 data = utf8 ? utf8Encode(data) : data;
+                 var i,
+                     hash,
+                     bkey = rstr2binl(key),
+                     ipad = Array(16),
+                     opad = Array(16);
 
-       function getNumberOfEdges(options) {
-         if (options === undefined) {
-           return 32;
-         } else if (isObjectNotArray(options)) {
-           var numberOfEdges = options.numberOfEdges;
-           return numberOfEdges === undefined ? 32 : numberOfEdges;
-         }
+                 if (bkey.length > 16) {
+                   bkey = binl(bkey, key.length * 8);
+                 }
 
-         return options;
-       }
+                 for (i = 0; i < 16; i += 1) {
+                   ipad[i] = bkey[i] ^ 0x36363636;
+                   opad[i] = bkey[i] ^ 0x5C5C5C5C;
+                 }
 
-       function getEarthRadius(options) {
-         if (options === undefined) {
-           return defaultEarthRadius;
-         } else if (isObjectNotArray(options)) {
-           var earthRadius = options.earthRadius;
-           return earthRadius === undefined ? defaultEarthRadius : earthRadius;
-         }
+                 hash = binl(ipad.concat(rstr2binl(data)), 512 + data.length * 8);
+                 return binl2rstr(binl(opad.concat(hash), 512 + 160));
+               }
+               /**
+                * Convert an array of little-endian words to a string
+                */
 
-         return defaultEarthRadius;
-       }
 
-       function getBearing(options) {
-         if (options === undefined) {
-           return 0;
-         } else if (isObjectNotArray(options)) {
-           var bearing = options.bearing;
-           return bearing === undefined ? 0 : bearing;
-         }
+               function binl2rstr(input) {
+                 var i,
+                     output = '',
+                     l = input.length * 32;
 
-         return 0;
-       }
+                 for (i = 0; i < l; i += 8) {
+                   output += String.fromCharCode(input[i >> 5] >>> i % 32 & 0xFF);
+                 }
 
-       function isObjectNotArray(argument) {
-         return _typeof(argument) === "object" && !Array.isArray(argument);
-       }
+                 return output;
+               }
+               /**
+                * Calculate the RIPE-MD160 of an array of little-endian words, and a bit length.
+                */
 
-       // `Number.EPSILON` constant
-       // https://tc39.es/ecma262/#sec-number.epsilon
-       _export({ target: 'Number', stat: true }, {
-         EPSILON: Math.pow(2, -52)
-       });
 
-       /**
-        * splaytree v3.1.0
-        * Fast Splay tree for Node and browser
-        *
-        * @author Alexander Milevski <info@w8r.name>
-        * @license MIT
-        * @preserve
-        */
-       var Node$1 =
-       /** @class */
-       function () {
-         function Node(key, data) {
-           this.next = null;
-           this.key = key;
-           this.data = data;
-           this.left = null;
-           this.right = null;
-         }
+               function binl(x, len) {
+                 var T,
+                     j,
+                     i,
+                     l,
+                     h0 = 0x67452301,
+                     h1 = 0xefcdab89,
+                     h2 = 0x98badcfe,
+                     h3 = 0x10325476,
+                     h4 = 0xc3d2e1f0,
+                     A1,
+                     B1,
+                     C1,
+                     D1,
+                     E1,
+                     A2,
+                     B2,
+                     C2,
+                     D2,
+                     E2;
+                 /* append padding */
 
-         return Node;
-       }();
-       /* follows "An implementation of top-down splaying"
-        * by D. Sleator <sleator@cs.cmu.edu> March 1992
-        */
+                 x[len >> 5] |= 0x80 << len % 32;
+                 x[(len + 64 >>> 9 << 4) + 14] = len;
+                 l = x.length;
 
+                 for (i = 0; i < l; i += 16) {
+                   A1 = A2 = h0;
+                   B1 = B2 = h1;
+                   C1 = C2 = h2;
+                   D1 = D2 = h3;
+                   E1 = E2 = h4;
 
-       function DEFAULT_COMPARE$1(a, b) {
-         return a > b ? 1 : a < b ? -1 : 0;
-       }
-       /**
-        * Simple top down splay, not requiring i to be in the tree t.
-        */
+                   for (j = 0; j <= 79; j += 1) {
+                     T = safe_add(A1, rmd160_f(j, B1, C1, D1));
+                     T = safe_add(T, x[i + rmd160_r1[j]]);
+                     T = safe_add(T, rmd160_K1(j));
+                     T = safe_add(bit_rol(T, rmd160_s1[j]), E1);
+                     A1 = E1;
+                     E1 = D1;
+                     D1 = bit_rol(C1, 10);
+                     C1 = B1;
+                     B1 = T;
+                     T = safe_add(A2, rmd160_f(79 - j, B2, C2, D2));
+                     T = safe_add(T, x[i + rmd160_r2[j]]);
+                     T = safe_add(T, rmd160_K2(j));
+                     T = safe_add(bit_rol(T, rmd160_s2[j]), E2);
+                     A2 = E2;
+                     E2 = D2;
+                     D2 = bit_rol(C2, 10);
+                     C2 = B2;
+                     B2 = T;
+                   }
 
+                   T = safe_add(h1, safe_add(C1, D2));
+                   h1 = safe_add(h2, safe_add(D1, E2));
+                   h2 = safe_add(h3, safe_add(E1, A2));
+                   h3 = safe_add(h4, safe_add(A1, B2));
+                   h4 = safe_add(h0, safe_add(B1, C2));
+                   h0 = T;
+                 }
 
-       function splay(i, t, comparator) {
-         var N = new Node$1(null, null);
-         var l = N;
-         var r = N;
+                 return [h0, h1, h2, h3, h4];
+               } // specific algorithm methods
 
-         while (true) {
-           var cmp = comparator(i, t.key); //if (i < t.key) {
 
-           if (cmp < 0) {
-             if (t.left === null) break; //if (i < t.left.key) {
+               function rmd160_f(j, x, y, z) {
+                 return 0 <= j && j <= 15 ? x ^ y ^ z : 16 <= j && j <= 31 ? x & y | ~x & z : 32 <= j && j <= 47 ? (x | ~y) ^ z : 48 <= j && j <= 63 ? x & z | y & ~z : 64 <= j && j <= 79 ? x ^ (y | ~z) : 'rmd160_f: j out of range';
+               }
 
-             if (comparator(i, t.left.key) < 0) {
-               var y = t.left;
-               /* rotate right */
+               function rmd160_K1(j) {
+                 return 0 <= j && j <= 15 ? 0x00000000 : 16 <= j && j <= 31 ? 0x5a827999 : 32 <= j && j <= 47 ? 0x6ed9eba1 : 48 <= j && j <= 63 ? 0x8f1bbcdc : 64 <= j && j <= 79 ? 0xa953fd4e : 'rmd160_K1: j out of range';
+               }
 
-               t.left = y.right;
-               y.right = t;
-               t = y;
-               if (t.left === null) break;
+               function rmd160_K2(j) {
+                 return 0 <= j && j <= 15 ? 0x50a28be6 : 16 <= j && j <= 31 ? 0x5c4dd124 : 32 <= j && j <= 47 ? 0x6d703ef3 : 48 <= j && j <= 63 ? 0x7a6d76e9 : 64 <= j && j <= 79 ? 0x00000000 : 'rmd160_K2: j out of range';
+               }
              }
+           }; // exposes Hashes
 
-             r.left = t;
-             /* link right */
+           (function (window, undefined$1) {
+             var freeExports = false;
 
-             r = t;
-             t = t.left; //} else if (i > t.key) {
-           } else if (cmp > 0) {
-             if (t.right === null) break; //if (i > t.right.key) {
+             {
+               freeExports = exports;
 
-             if (comparator(i, t.right.key) > 0) {
-               var y = t.right;
-               /* rotate left */
+               if (exports && _typeof(commonjsGlobal) === 'object' && commonjsGlobal && commonjsGlobal === commonjsGlobal.global) {
+                 window = commonjsGlobal;
+               }
+             }
 
-               t.right = y.left;
-               y.left = t;
-               t = y;
-               if (t.right === null) break;
+             if (typeof undefined$1 === 'function' && _typeof(undefined$1.amd) === 'object' && undefined$1.amd) {
+               // define as an anonymous module, so, through path mapping, it can be aliased
+               undefined$1(function () {
+                 return Hashes;
+               });
+             } else if (freeExports) {
+               // in Node.js or RingoJS v0.8.0+
+               if (module && module.exports === freeExports) {
+                 module.exports = Hashes;
+               } // in Narwhal or RingoJS v0.7.0-
+               else {
+                   freeExports.Hashes = Hashes;
+                 }
+             } else {
+               // in a browser or Rhino
+               window.Hashes = Hashes;
              }
+           })(this);
+         })(); // IIFE
 
-             l.right = t;
-             /* link left */
+       });
 
-             l = t;
-             t = t.right;
-           } else break;
-         }
-         /* assemble */
+       var sha1 = new hashes.SHA1(); // # xtend
 
+       var hasOwnProperty$1 = Object.prototype.hasOwnProperty;
 
-         l.right = t.left;
-         r.left = t.right;
-         t.left = N.right;
-         t.right = N.left;
-         return t;
-       }
+       function xtend() {
+         var target = {};
 
-       function insert(i, data, t, comparator) {
-         var node = new Node$1(i, data);
+         for (var i = 0; i < arguments.length; i++) {
+           var source = arguments[i];
 
-         if (t === null) {
-           node.left = node.right = null;
-           return node;
+           for (var key in source) {
+             if (hasOwnProperty$1.call(source, key)) {
+               target[key] = source[key];
+             }
+           }
          }
 
-         t = splay(i, t, comparator);
-         var cmp = comparator(i, t.key);
+         return target;
+       }
 
-         if (cmp < 0) {
-           node.left = t.left;
-           node.right = t;
-           t.left = null;
-         } else if (cmp >= 0) {
-           node.right = t.right;
-           node.left = t;
-           t.right = null;
-         }
+       var ohauth = {};
 
-         return node;
-       }
+       ohauth.qsString = function (obj) {
+         return Object.keys(obj).sort().map(function (key) {
+           return ohauth.percentEncode(key) + '=' + ohauth.percentEncode(obj[key]);
+         }).join('&');
+       };
 
-       function split$2(key, v, comparator) {
-         var left = null;
-         var right = null;
+       ohauth.stringQs = function (str) {
+         return str.split('&').filter(function (pair) {
+           return pair !== '';
+         }).reduce(function (obj, pair) {
+           var parts = pair.split('=');
+           obj[decodeURIComponent(parts[0])] = null === parts[1] ? '' : decodeURIComponent(parts[1]);
+           return obj;
+         }, {});
+       };
 
-         if (v) {
-           v = splay(key, v, comparator);
-           var cmp = comparator(v.key, key);
+       ohauth.rawxhr = function (method, url, data, headers, callback) {
+         var xhr = new XMLHttpRequest(),
+             twoHundred = /^20\d$/;
 
-           if (cmp === 0) {
-             left = v.left;
-             right = v.right;
-           } else if (cmp < 0) {
-             right = v.right;
-             v.right = null;
-             left = v;
-           } else {
-             left = v.left;
-             v.left = null;
-             right = v;
+         xhr.onreadystatechange = function () {
+           if (4 === xhr.readyState && 0 !== xhr.status) {
+             if (twoHundred.test(xhr.status)) callback(null, xhr);else return callback(xhr, null);
            }
-         }
-
-         return {
-           left: left,
-           right: right
          };
-       }
-
-       function merge$4(left, right, comparator) {
-         if (right === null) return left;
-         if (left === null) return right;
-         right = splay(left.key, right, comparator);
-         right.left = left;
-         return right;
-       }
-       /**
-        * Prints level of the tree
-        */
-
 
-       function printRow(root, prefix, isTail, out, printNode) {
-         if (root) {
-           out("" + prefix + (isTail ? '└── ' : '├── ') + printNode(root) + "\n");
-           var indent = prefix + (isTail ? '    ' : '│   ');
-           if (root.left) printRow(root.left, indent, false, out, printNode);
-           if (root.right) printRow(root.right, indent, true, out, printNode);
-         }
-       }
+         xhr.onerror = function (e) {
+           return callback(e, null);
+         };
 
-       var Tree =
-       /** @class */
-       function () {
-         function Tree(comparator) {
-           if (comparator === void 0) {
-             comparator = DEFAULT_COMPARE$1;
-           }
+         xhr.open(method, url, true);
 
-           this._root = null;
-           this._size = 0;
-           this._comparator = comparator;
-         }
-         /**
-          * Inserts a key, allows duplicates
-          */
+         for (var h in headers) {
+           xhr.setRequestHeader(h, headers[h]);
+         }
 
+         xhr.send(data);
+         return xhr;
+       };
 
-         Tree.prototype.insert = function (key, data) {
-           this._size++;
-           return this._root = insert(key, data, this._root, this._comparator);
+       ohauth.xhr = function (method, url, auth, data, options, callback) {
+         var headers = options && options.header || {
+           'Content-Type': 'application/x-www-form-urlencoded'
          };
-         /**
-          * Adds a key, if it is not present in the tree
-          */
-
+         headers.Authorization = 'OAuth ' + ohauth.authHeader(auth);
+         return ohauth.rawxhr(method, url, data, headers, callback);
+       };
 
-         Tree.prototype.add = function (key, data) {
-           var node = new Node$1(key, data);
+       ohauth.nonce = function () {
+         for (var o = ''; o.length < 6;) {
+           o += '0123456789ABCDEFGHIJKLMNOPQRSTUVWXTZabcdefghiklmnopqrstuvwxyz'[Math.floor(Math.random() * 61)];
+         }
 
-           if (this._root === null) {
-             node.left = node.right = null;
-             this._size++;
-             this._root = node;
-           }
+         return o;
+       };
 
-           var comparator = this._comparator;
-           var t = splay(key, this._root, comparator);
-           var cmp = comparator(key, t.key);
-           if (cmp === 0) this._root = t;else {
-             if (cmp < 0) {
-               node.left = t.left;
-               node.right = t;
-               t.left = null;
-             } else if (cmp > 0) {
-               node.right = t.right;
-               node.left = t;
-               t.right = null;
-             }
+       ohauth.authHeader = function (obj) {
+         return Object.keys(obj).sort().map(function (key) {
+           return encodeURIComponent(key) + '="' + encodeURIComponent(obj[key]) + '"';
+         }).join(', ');
+       };
 
-             this._size++;
-             this._root = node;
-           }
-           return this._root;
-         };
-         /**
-          * @param  {Key} key
-          * @return {Node|null}
-          */
+       ohauth.timestamp = function () {
+         return ~~(+new Date() / 1000);
+       };
 
+       ohauth.percentEncode = function (s) {
+         return encodeURIComponent(s).replace(/\!/g, '%21').replace(/\'/g, '%27').replace(/\*/g, '%2A').replace(/\(/g, '%28').replace(/\)/g, '%29');
+       };
 
-         Tree.prototype.remove = function (key) {
-           this._root = this._remove(key, this._root, this._comparator);
-         };
-         /**
-          * Deletes i from the tree if it's there
-          */
+       ohauth.baseString = function (method, url, params) {
+         if (params.oauth_signature) delete params.oauth_signature;
+         return [method, ohauth.percentEncode(url), ohauth.percentEncode(ohauth.qsString(params))].join('&');
+       };
 
+       ohauth.signature = function (oauth_secret, token_secret, baseString) {
+         return sha1.b64_hmac(ohauth.percentEncode(oauth_secret) + '&' + ohauth.percentEncode(token_secret), baseString);
+       };
+       /**
+        * Takes an options object for configuration (consumer_key,
+        * consumer_secret, version, signature_method, token, token_secret)
+        * and returns a function that generates the Authorization header
+        * for given data.
+        *
+        * The returned function takes these parameters:
+        * - method: GET/POST/...
+        * - uri: full URI with protocol, port, path and query string
+        * - extra_params: any extra parameters (that are passed in the POST data),
+        *   can be an object or a from-urlencoded string.
+        *
+        * Returned function returns full OAuth header with "OAuth" string in it.
+        */
 
-         Tree.prototype._remove = function (i, t, comparator) {
-           var x;
-           if (t === null) return null;
-           t = splay(i, t, comparator);
-           var cmp = comparator(i, t.key);
 
-           if (cmp === 0) {
-             /* found it */
-             if (t.left === null) {
-               x = t.right;
-             } else {
-               x = splay(i, t.left, comparator);
-               x.right = t.right;
-             }
+       ohauth.headerGenerator = function (options) {
+         options = options || {};
+         var consumer_key = options.consumer_key || '',
+             consumer_secret = options.consumer_secret || '',
+             signature_method = options.signature_method || 'HMAC-SHA1',
+             version = options.version || '1.0',
+             token = options.token || '',
+             token_secret = options.token_secret || '';
+         return function (method, uri, extra_params) {
+           method = method.toUpperCase();
 
-             this._size--;
-             return x;
+           if (typeof extra_params === 'string' && extra_params.length > 0) {
+             extra_params = ohauth.stringQs(extra_params);
            }
 
-           return t;
-           /* It wasn't there */
+           var uri_parts = uri.split('?', 2),
+               base_uri = uri_parts[0];
+           var query_params = uri_parts.length === 2 ? ohauth.stringQs(uri_parts[1]) : {};
+           var oauth_params = {
+             oauth_consumer_key: consumer_key,
+             oauth_signature_method: signature_method,
+             oauth_version: version,
+             oauth_timestamp: ohauth.timestamp(),
+             oauth_nonce: ohauth.nonce()
+           };
+           if (token) oauth_params.oauth_token = token;
+           var all_params = xtend({}, oauth_params, query_params, extra_params),
+               base_str = ohauth.baseString(method, base_uri, all_params);
+           oauth_params.oauth_signature = ohauth.signature(consumer_secret, token_secret, base_str);
+           return 'OAuth ' + ohauth.authHeader(oauth_params);
          };
-         /**
-          * Removes and returns the node with smallest key
-          */
+       };
 
+       var ohauth_1 = ohauth;
 
-         Tree.prototype.pop = function () {
-           var node = this._root;
+       var resolveUrl = createCommonjsModule(function (module, exports) {
+         // Copyright 2014 Simon Lydell
+         // X11 (“MIT”) Licensed. (See LICENSE.)
+         void function (root, factory) {
+           {
+             module.exports = factory();
+           }
+         }(commonjsGlobal, function () {
+           function resolveUrl()
+           /* ...urls */
+           {
+             var numUrls = arguments.length;
 
-           if (node) {
-             while (node.left) {
-               node = node.left;
+             if (numUrls === 0) {
+               throw new Error("resolveUrl requires at least one argument; got none.");
              }
 
-             this._root = splay(node.key, this._root, this._comparator);
-             this._root = this._remove(node.key, this._root, this._comparator);
-             return {
-               key: node.key,
-               data: node.data
-             };
-           }
+             var base = document.createElement("base");
+             base.href = arguments[0];
 
-           return null;
-         };
-         /**
-          * Find without splaying
-          */
+             if (numUrls === 1) {
+               return base.href;
+             }
 
+             var head = document.getElementsByTagName("head")[0];
+             head.insertBefore(base, head.firstChild);
+             var a = document.createElement("a");
+             var resolved;
 
-         Tree.prototype.findStatic = function (key) {
-           var current = this._root;
-           var compare = this._comparator;
+             for (var index = 1; index < numUrls; index++) {
+               a.href = arguments[index];
+               resolved = a.href;
+               base.href = resolved;
+             }
 
-           while (current) {
-             var cmp = compare(key, current.key);
-             if (cmp === 0) return current;else if (cmp < 0) current = current.left;else current = current.right;
+             head.removeChild(base);
+             return resolved;
            }
 
-           return null;
-         };
-
-         Tree.prototype.find = function (key) {
-           if (this._root) {
-             this._root = splay(key, this._root, this._comparator);
-             if (this._comparator(key, this._root.key) !== 0) return null;
-           }
+           return resolveUrl;
+         });
+       });
 
-           return this._root;
-         };
+       var assign = make_assign();
+       var create$1 = make_create();
+       var trim$1 = make_trim();
+       var Global$5 = typeof window !== 'undefined' ? window : commonjsGlobal;
+       var util = {
+         assign: assign,
+         create: create$1,
+         trim: trim$1,
+         bind: bind$1,
+         slice: slice$1,
+         each: each$7,
+         map: map,
+         pluck: pluck$1,
+         isList: isList$1,
+         isFunction: isFunction$1,
+         isObject: isObject$1,
+         Global: Global$5
+       };
 
-         Tree.prototype.contains = function (key) {
-           var current = this._root;
-           var compare = this._comparator;
+       function make_assign() {
+         if (Object.assign) {
+           return Object.assign;
+         } else {
+           return function shimAssign(obj, props1, props2, etc) {
+             for (var i = 1; i < arguments.length; i++) {
+               each$7(Object(arguments[i]), function (val, key) {
+                 obj[key] = val;
+               });
+             }
 
-           while (current) {
-             var cmp = compare(key, current.key);
-             if (cmp === 0) return true;else if (cmp < 0) current = current.left;else current = current.right;
-           }
+             return obj;
+           };
+         }
+       }
 
-           return false;
-         };
+       function make_create() {
+         if (Object.create) {
+           return function create(obj, assignProps1, assignProps2, etc) {
+             var assignArgsList = slice$1(arguments, 1);
+             return assign.apply(this, [Object.create(obj)].concat(assignArgsList));
+           };
+         } else {
+           var F = function F() {}; // eslint-disable-line no-inner-declarations
 
-         Tree.prototype.forEach = function (visitor, ctx) {
-           var current = this._root;
-           var Q = [];
-           /* Initialize stack s */
 
-           var done = false;
+           return function create(obj, assignProps1, assignProps2, etc) {
+             var assignArgsList = slice$1(arguments, 1);
+             F.prototype = obj;
+             return assign.apply(this, [new F()].concat(assignArgsList));
+           };
+         }
+       }
 
-           while (!done) {
-             if (current !== null) {
-               Q.push(current);
-               current = current.left;
-             } else {
-               if (Q.length !== 0) {
-                 current = Q.pop();
-                 visitor.call(ctx, current);
-                 current = current.right;
-               } else done = true;
-             }
-           }
+       function make_trim() {
+         if (String.prototype.trim) {
+           return function trim(str) {
+             return String.prototype.trim.call(str);
+           };
+         } else {
+           return function trim(str) {
+             return str.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, '');
+           };
+         }
+       }
 
-           return this;
+       function bind$1(obj, fn) {
+         return function () {
+           return fn.apply(obj, Array.prototype.slice.call(arguments, 0));
          };
-         /**
-          * Walk key range from `low` to `high`. Stops if `fn` returns a value.
-          */
+       }
 
+       function slice$1(arr, index) {
+         return Array.prototype.slice.call(arr, index || 0);
+       }
 
-         Tree.prototype.range = function (low, high, fn, ctx) {
-           var Q = [];
-           var compare = this._comparator;
-           var node = this._root;
-           var cmp;
+       function each$7(obj, fn) {
+         pluck$1(obj, function (val, key) {
+           fn(val, key);
+           return false;
+         });
+       }
 
-           while (Q.length !== 0 || node) {
-             if (node) {
-               Q.push(node);
-               node = node.left;
-             } else {
-               node = Q.pop();
-               cmp = compare(node.key, high);
+       function map(obj, fn) {
+         var res = isList$1(obj) ? [] : {};
+         pluck$1(obj, function (v, k) {
+           res[k] = fn(v, k);
+           return false;
+         });
+         return res;
+       }
 
-               if (cmp > 0) {
-                 break;
-               } else if (compare(node.key, low) >= 0) {
-                 if (fn.call(ctx, node)) return this; // stop if smth is returned
+       function pluck$1(obj, fn) {
+         if (isList$1(obj)) {
+           for (var i = 0; i < obj.length; i++) {
+             if (fn(obj[i], i)) {
+               return obj[i];
+             }
+           }
+         } else {
+           for (var key in obj) {
+             if (obj.hasOwnProperty(key)) {
+               if (fn(obj[key], key)) {
+                 return obj[key];
                }
-
-               node = node.right;
              }
            }
+         }
+       }
 
-           return this;
-         };
-         /**
-          * Returns array of keys
-          */
+       function isList$1(val) {
+         return val != null && typeof val != 'function' && typeof val.length == 'number';
+       }
 
+       function isFunction$1(val) {
+         return val && {}.toString.call(val) === '[object Function]';
+       }
 
-         Tree.prototype.keys = function () {
-           var keys = [];
-           this.forEach(function (_a) {
-             var key = _a.key;
-             return keys.push(key);
-           });
-           return keys;
-         };
-         /**
-          * Returns array of all the data in the nodes
-          */
+       function isObject$1(val) {
+         return val && {}.toString.call(val) === '[object Object]';
+       }
 
+       var slice = util.slice;
+       var pluck = util.pluck;
+       var each$6 = util.each;
+       var bind = util.bind;
+       var create = util.create;
+       var isList = util.isList;
+       var isFunction = util.isFunction;
+       var isObject = util.isObject;
+       var storeEngine = {
+         createStore: _createStore
+       };
+       var storeAPI = {
+         version: '2.0.12',
+         enabled: false,
+         // get returns the value of the given key. If that value
+         // is undefined, it returns optionalDefaultValue instead.
+         get: function get(key, optionalDefaultValue) {
+           var data = this.storage.read(this._namespacePrefix + key);
+           return this._deserialize(data, optionalDefaultValue);
+         },
+         // set will store the given value at key and returns value.
+         // Calling set with value === undefined is equivalent to calling remove.
+         set: function set(key, value) {
+           if (value === undefined) {
+             return this.remove(key);
+           }
 
-         Tree.prototype.values = function () {
-           var values = [];
-           this.forEach(function (_a) {
-             var data = _a.data;
-             return values.push(data);
+           this.storage.write(this._namespacePrefix + key, this._serialize(value));
+           return value;
+         },
+         // remove deletes the key and value stored at the given key.
+         remove: function remove(key) {
+           this.storage.remove(this._namespacePrefix + key);
+         },
+         // each will call the given callback once for each key-value pair
+         // in this store.
+         each: function each(callback) {
+           var self = this;
+           this.storage.each(function (val, namespacedKey) {
+             callback.call(self, self._deserialize(val), (namespacedKey || '').replace(self._namespaceRegexp, ''));
            });
-           return values;
-         };
+         },
+         // clearAll will remove all the stored key-value pairs in this store.
+         clearAll: function clearAll() {
+           this.storage.clearAll();
+         },
+         // additional functionality that can't live in plugins
+         // ---------------------------------------------------
+         // hasNamespace returns true if this store instance has the given namespace.
+         hasNamespace: function hasNamespace(namespace) {
+           return this._namespacePrefix == '__storejs_' + namespace + '_';
+         },
+         // createStore creates a store.js instance with the first
+         // functioning storage in the list of storage candidates,
+         // and applies the the given mixins to the instance.
+         createStore: function createStore() {
+           return _createStore.apply(this, arguments);
+         },
+         addPlugin: function addPlugin(plugin) {
+           this._addPlugin(plugin);
+         },
+         namespace: function namespace(_namespace) {
+           return _createStore(this.storage, this.plugins, _namespace);
+         }
+       };
 
-         Tree.prototype.min = function () {
-           if (this._root) return this.minNode(this._root).key;
-           return null;
-         };
+       function _warn() {
+         var _console = typeof console == 'undefined' ? null : console;
 
-         Tree.prototype.max = function () {
-           if (this._root) return this.maxNode(this._root).key;
-           return null;
-         };
+         if (!_console) {
+           return;
+         }
 
-         Tree.prototype.minNode = function (t) {
-           if (t === void 0) {
-             t = this._root;
-           }
+         var fn = _console.warn ? _console.warn : _console.log;
+         fn.apply(_console, arguments);
+       }
 
-           if (t) while (t.left) {
-             t = t.left;
-           }
-           return t;
-         };
+       function _createStore(storages, plugins, namespace) {
+         if (!namespace) {
+           namespace = '';
+         }
 
-         Tree.prototype.maxNode = function (t) {
-           if (t === void 0) {
-             t = this._root;
-           }
+         if (storages && !isList(storages)) {
+           storages = [storages];
+         }
 
-           if (t) while (t.right) {
-             t = t.right;
-           }
-           return t;
-         };
-         /**
-          * Returns node at given index
-          */
+         if (plugins && !isList(plugins)) {
+           plugins = [plugins];
+         }
 
+         var namespacePrefix = namespace ? '__storejs_' + namespace + '_' : '';
+         var namespaceRegexp = namespace ? new RegExp('^' + namespacePrefix) : null;
+         var legalNamespaces = /^[a-zA-Z0-9_\-]*$/; // alpha-numeric + underscore and dash
 
-         Tree.prototype.at = function (index) {
-           var current = this._root;
-           var done = false;
-           var i = 0;
-           var Q = [];
+         if (!legalNamespaces.test(namespace)) {
+           throw new Error('store.js namespaces can only have alphanumerics + underscores and dashes');
+         }
 
-           while (!done) {
-             if (current) {
-               Q.push(current);
-               current = current.left;
-             } else {
-               if (Q.length > 0) {
-                 current = Q.pop();
-                 if (i === index) return current;
-                 i++;
-                 current = current.right;
-               } else done = true;
+         var _privateStoreProps = {
+           _namespacePrefix: namespacePrefix,
+           _namespaceRegexp: namespaceRegexp,
+           _testStorage: function _testStorage(storage) {
+             try {
+               var testStr = '__storejs__test__';
+               storage.write(testStr, testStr);
+               var ok = storage.read(testStr) === testStr;
+               storage.remove(testStr);
+               return ok;
+             } catch (e) {
+               return false;
              }
-           }
+           },
+           _assignPluginFnProp: function _assignPluginFnProp(pluginFnProp, propName) {
+             var oldFn = this[propName];
+
+             this[propName] = function pluginFn() {
+               var args = slice(arguments, 0);
+               var self = this; // super_fn calls the old function which was overwritten by
+               // this mixin.
+
+               function super_fn() {
+                 if (!oldFn) {
+                   return;
+                 }
+
+                 each$6(arguments, function (arg, i) {
+                   args[i] = arg;
+                 });
+                 return oldFn.apply(self, args);
+               } // Give mixing function access to super_fn by prefixing all mixin function
+               // arguments with super_fn.
+
+
+               var newFnArgs = [super_fn].concat(args);
+               return pluginFnProp.apply(self, newFnArgs);
+             };
+           },
+           _serialize: function _serialize(obj) {
+             return JSON.stringify(obj);
+           },
+           _deserialize: function _deserialize(strVal, defaultVal) {
+             if (!strVal) {
+               return defaultVal;
+             } // It is possible that a raw string value has been previously stored
+             // in a storage without using store.js, meaning it will be a raw
+             // string value instead of a JSON serialized string. By defaulting
+             // to the raw string value in case of a JSON parse error, we allow
+             // for past stored values to be forwards-compatible with store.js
 
-           return null;
-         };
 
-         Tree.prototype.next = function (d) {
-           var root = this._root;
-           var successor = null;
+             var val = '';
 
-           if (d.right) {
-             successor = d.right;
+             try {
+               val = JSON.parse(strVal);
+             } catch (e) {
+               val = strVal;
+             }
 
-             while (successor.left) {
-               successor = successor.left;
+             return val !== undefined ? val : defaultVal;
+           },
+           _addStorage: function _addStorage(storage) {
+             if (this.enabled) {
+               return;
              }
 
-             return successor;
-           }
+             if (this._testStorage(storage)) {
+               this.storage = storage;
+               this.enabled = true;
+             }
+           },
+           _addPlugin: function _addPlugin(plugin) {
+             var self = this; // If the plugin is an array, then add all plugins in the array.
+             // This allows for a plugin to depend on other plugins.
 
-           var comparator = this._comparator;
+             if (isList(plugin)) {
+               each$6(plugin, function (plugin) {
+                 self._addPlugin(plugin);
+               });
+               return;
+             } // Keep track of all plugins we've seen so far, so that we
+             // don't add any of them twice.
 
-           while (root) {
-             var cmp = comparator(d.key, root.key);
-             if (cmp === 0) break;else if (cmp < 0) {
-               successor = root;
-               root = root.left;
-             } else root = root.right;
-           }
 
-           return successor;
-         };
+             var seenPlugin = pluck(this.plugins, function (seenPlugin) {
+               return plugin === seenPlugin;
+             });
 
-         Tree.prototype.prev = function (d) {
-           var root = this._root;
-           var predecessor = null;
+             if (seenPlugin) {
+               return;
+             }
 
-           if (d.left !== null) {
-             predecessor = d.left;
+             this.plugins.push(plugin); // Check that the plugin is properly formed
 
-             while (predecessor.right) {
-               predecessor = predecessor.right;
+             if (!isFunction(plugin)) {
+               throw new Error('Plugins must be function values that return objects');
              }
 
-             return predecessor;
-           }
+             var pluginProperties = plugin.call(this);
 
-           var comparator = this._comparator;
+             if (!isObject(pluginProperties)) {
+               throw new Error('Plugins must return an object of function properties');
+             } // Add the plugin function properties to this store instance.
 
-           while (root) {
-             var cmp = comparator(d.key, root.key);
-             if (cmp === 0) break;else if (cmp < 0) root = root.left;else {
-               predecessor = root;
-               root = root.right;
-             }
-           }
 
-           return predecessor;
-         };
+             each$6(pluginProperties, function (pluginFnProp, propName) {
+               if (!isFunction(pluginFnProp)) {
+                 throw new Error('Bad plugin property: ' + propName + ' from plugin ' + plugin.name + '. Plugins should only return functions.');
+               }
 
-         Tree.prototype.clear = function () {
-           this._root = null;
-           this._size = 0;
-           return this;
-         };
+               self._assignPluginFnProp(pluginFnProp, propName);
+             });
+           },
+           // Put deprecated properties in the private API, so as to not expose it to accidential
+           // discovery through inspection of the store object.
+           // Deprecated: addStorage
+           addStorage: function addStorage(storage) {
+             _warn('store.addStorage(storage) is deprecated. Use createStore([storages])');
 
-         Tree.prototype.toList = function () {
-           return toList(this._root);
+             this._addStorage(storage);
+           }
          };
-         /**
-          * Bulk-load items. Both array have to be same size
-          */
+         var store = create(_privateStoreProps, storeAPI, {
+           plugins: []
+         });
+         store.raw = {};
+         each$6(store, function (prop, propName) {
+           if (isFunction(prop)) {
+             store.raw[propName] = bind(store, prop);
+           }
+         });
+         each$6(storages, function (storage) {
+           store._addStorage(storage);
+         });
+         each$6(plugins, function (plugin) {
+           store._addPlugin(plugin);
+         });
+         return store;
+       }
 
+       var Global$4 = util.Global;
+       var localStorage_1 = {
+         name: 'localStorage',
+         read: read$5,
+         write: write$5,
+         each: each$5,
+         remove: remove$5,
+         clearAll: clearAll$5
+       };
 
-         Tree.prototype.load = function (keys, values, presort) {
-           if (values === void 0) {
-             values = [];
-           }
+       function localStorage$1() {
+         return Global$4.localStorage;
+       }
 
-           if (presort === void 0) {
-             presort = false;
-           }
+       function read$5(key) {
+         return localStorage$1().getItem(key);
+       }
 
-           var size = keys.length;
-           var comparator = this._comparator; // sort if needed
+       function write$5(key, data) {
+         return localStorage$1().setItem(key, data);
+       }
 
-           if (presort) sort$1(keys, values, 0, size - 1, comparator);
+       function each$5(fn) {
+         for (var i = localStorage$1().length - 1; i >= 0; i--) {
+           var key = localStorage$1().key(i);
+           fn(read$5(key), key);
+         }
+       }
 
-           if (this._root === null) {
-             // empty tree
-             this._root = loadRecursive$1(keys, values, 0, size);
-             this._size = size;
-           } else {
-             // that re-builds the whole tree from two in-order traversals
-             var mergedList = mergeLists(this.toList(), createList(keys, values), comparator);
-             size = this._size + size;
-             this._root = sortedListToBST({
-               head: mergedList
-             }, 0, size);
-           }
+       function remove$5(key) {
+         return localStorage$1().removeItem(key);
+       }
 
-           return this;
-         };
+       function clearAll$5() {
+         return localStorage$1().clear();
+       }
 
-         Tree.prototype.isEmpty = function () {
-           return this._root === null;
-         };
+       // versions 6 and 7, where no localStorage, etc
+       // is available.
 
-         Object.defineProperty(Tree.prototype, "size", {
-           get: function get() {
-             return this._size;
-           },
-           enumerable: true,
-           configurable: true
-         });
-         Object.defineProperty(Tree.prototype, "root", {
-           get: function get() {
-             return this._root;
-           },
-           enumerable: true,
-           configurable: true
-         });
+       var Global$3 = util.Global;
+       var oldFFGlobalStorage = {
+         name: 'oldFF-globalStorage',
+         read: read$4,
+         write: write$4,
+         each: each$4,
+         remove: remove$4,
+         clearAll: clearAll$4
+       };
+       var globalStorage = Global$3.globalStorage;
 
-         Tree.prototype.toString = function (printNode) {
-           if (printNode === void 0) {
-             printNode = function printNode(n) {
-               return String(n.key);
-             };
-           }
+       function read$4(key) {
+         return globalStorage[key];
+       }
 
-           var out = [];
-           printRow(this._root, '', true, function (v) {
-             return out.push(v);
-           }, printNode);
-           return out.join('');
-         };
+       function write$4(key, data) {
+         globalStorage[key] = data;
+       }
 
-         Tree.prototype.update = function (key, newKey, newData) {
-           var comparator = this._comparator;
+       function each$4(fn) {
+         for (var i = globalStorage.length - 1; i >= 0; i--) {
+           var key = globalStorage.key(i);
+           fn(globalStorage[key], key);
+         }
+       }
 
-           var _a = split$2(key, this._root, comparator),
-               left = _a.left,
-               right = _a.right;
+       function remove$4(key) {
+         return globalStorage.removeItem(key);
+       }
 
-           if (comparator(key, newKey) < 0) {
-             right = insert(newKey, newData, right, comparator);
-           } else {
-             left = insert(newKey, newData, left, comparator);
-           }
+       function clearAll$4() {
+         each$4(function (key, _) {
+           delete globalStorage[key];
+         });
+       }
 
-           this._root = merge$4(left, right, comparator);
-         };
+       // versions 6 and 7, where no localStorage, sessionStorage, etc
+       // is available.
 
-         Tree.prototype.split = function (key) {
-           return split$2(key, this._root, this._comparator);
-         };
+       var Global$2 = util.Global;
+       var oldIEUserDataStorage = {
+         name: 'oldIE-userDataStorage',
+         write: write$3,
+         read: read$3,
+         each: each$3,
+         remove: remove$3,
+         clearAll: clearAll$3
+       };
+       var storageName = 'storejs';
+       var doc$1 = Global$2.document;
 
-         return Tree;
-       }();
+       var _withStorageEl = _makeIEStorageElFunction();
 
-       function loadRecursive$1(keys, values, start, end) {
-         var size = end - start;
+       var disable = (Global$2.navigator ? Global$2.navigator.userAgent : '').match(/ (MSIE 8|MSIE 9|MSIE 10)\./); // MSIE 9.x, MSIE 10.x
 
-         if (size > 0) {
-           var middle = start + Math.floor(size / 2);
-           var key = keys[middle];
-           var data = values[middle];
-           var node = new Node$1(key, data);
-           node.left = loadRecursive$1(keys, values, start, middle);
-           node.right = loadRecursive$1(keys, values, middle + 1, end);
-           return node;
+       function write$3(unfixedKey, data) {
+         if (disable) {
+           return;
          }
 
-         return null;
-       }
+         var fixedKey = fixKey(unfixedKey);
 
-       function createList(keys, values) {
-         var head = new Node$1(null, null);
-         var p = head;
+         _withStorageEl(function (storageEl) {
+           storageEl.setAttribute(fixedKey, data);
+           storageEl.save(storageName);
+         });
+       }
 
-         for (var i = 0; i < keys.length; i++) {
-           p = p.next = new Node$1(keys[i], values[i]);
+       function read$3(unfixedKey) {
+         if (disable) {
+           return;
          }
 
-         p.next = null;
-         return head.next;
+         var fixedKey = fixKey(unfixedKey);
+         var res = null;
+
+         _withStorageEl(function (storageEl) {
+           res = storageEl.getAttribute(fixedKey);
+         });
+
+         return res;
        }
 
-       function toList(root) {
-         var current = root;
-         var Q = [];
-         var done = false;
-         var head = new Node$1(null, null);
-         var p = head;
+       function each$3(callback) {
+         _withStorageEl(function (storageEl) {
+           var attributes = storageEl.XMLDocument.documentElement.attributes;
 
-         while (!done) {
-           if (current) {
-             Q.push(current);
-             current = current.left;
-           } else {
-             if (Q.length > 0) {
-               current = p = p.next = Q.pop();
-               current = current.right;
-             } else done = true;
+           for (var i = attributes.length - 1; i >= 0; i--) {
+             var attr = attributes[i];
+             callback(storageEl.getAttribute(attr.name), attr.name);
            }
-         }
+         });
+       }
 
-         p.next = null; // that'll work even if the tree was empty
+       function remove$3(unfixedKey) {
+         var fixedKey = fixKey(unfixedKey);
 
-         return head.next;
+         _withStorageEl(function (storageEl) {
+           storageEl.removeAttribute(fixedKey);
+           storageEl.save(storageName);
+         });
        }
 
-       function sortedListToBST(list, start, end) {
-         var size = end - start;
+       function clearAll$3() {
+         _withStorageEl(function (storageEl) {
+           var attributes = storageEl.XMLDocument.documentElement.attributes;
+           storageEl.load(storageName);
 
-         if (size > 0) {
-           var middle = start + Math.floor(size / 2);
-           var left = sortedListToBST(list, start, middle);
-           var root = list.head;
-           root.left = left;
-           list.head = list.head.next;
-           root.right = sortedListToBST(list, middle + 1, end);
-           return root;
-         }
+           for (var i = attributes.length - 1; i >= 0; i--) {
+             storageEl.removeAttribute(attributes[i].name);
+           }
 
-         return null;
-       }
+           storageEl.save(storageName);
+         });
+       } // Helpers
+       //////////
+       // In IE7, keys cannot start with a digit or contain certain chars.
+       // See https://github.com/marcuswestin/store.js/issues/40
+       // See https://github.com/marcuswestin/store.js/issues/83
 
-       function mergeLists(l1, l2, compare) {
-         var head = new Node$1(null, null); // dummy
 
-         var p = head;
-         var p1 = l1;
-         var p2 = l2;
+       var forbiddenCharsRegex = new RegExp("[!\"#$%&'()*+,/\\\\:;<=>?@[\\]^`{|}~]", "g");
 
-         while (p1 !== null && p2 !== null) {
-           if (compare(p1.key, p2.key) < 0) {
-             p.next = p1;
-             p1 = p1.next;
-           } else {
-             p.next = p2;
-             p2 = p2.next;
-           }
+       function fixKey(key) {
+         return key.replace(/^\d/, '___$&').replace(forbiddenCharsRegex, '___');
+       }
 
-           p = p.next;
+       function _makeIEStorageElFunction() {
+         if (!doc$1 || !doc$1.documentElement || !doc$1.documentElement.addBehavior) {
+           return null;
          }
 
-         if (p1 !== null) {
-           p.next = p1;
-         } else if (p2 !== null) {
-           p.next = p2;
+         var scriptTag = 'script',
+             storageOwner,
+             storageContainer,
+             storageEl; // Since #userData storage applies only to specific paths, we need to
+         // somehow link our data to a specific path.  We choose /favicon.ico
+         // as a pretty safe option, since all browsers already make a request to
+         // this URL anyway and being a 404 will not hurt us here.  We wrap an
+         // iframe pointing to the favicon in an ActiveXObject(htmlfile) object
+         // (see: http://msdn.microsoft.com/en-us/library/aa752574(v=VS.85).aspx)
+         // since the iframe access rules appear to allow direct access and
+         // manipulation of the document element, even for a 404 page.  This
+         // document can be used instead of the current document (which would
+         // have been limited to the current path) to perform #userData storage.
+
+         try {
+           /* global ActiveXObject */
+           storageContainer = new ActiveXObject('htmlfile');
+           storageContainer.open();
+           storageContainer.write('<' + scriptTag + '>document.w=window</' + scriptTag + '><iframe src="/favicon.ico"></iframe>');
+           storageContainer.close();
+           storageOwner = storageContainer.w.frames[0].document;
+           storageEl = storageOwner.createElement('div');
+         } catch (e) {
+           // somehow ActiveXObject instantiation failed (perhaps some special
+           // security settings or otherwse), fall back to per-path storage
+           storageEl = doc$1.createElement('div');
+           storageOwner = doc$1.body;
          }
 
-         return head.next;
-       }
+         return function (storeFunction) {
+           var args = [].slice.call(arguments, 0);
+           args.unshift(storageEl); // See http://msdn.microsoft.com/en-us/library/ms531081(v=VS.85).aspx
+           // and http://msdn.microsoft.com/en-us/library/ms531424(v=VS.85).aspx
 
-       function sort$1(keys, values, left, right, compare) {
-         if (left >= right) return;
-         var pivot = keys[left + right >> 1];
-         var i = left - 1;
-         var j = right + 1;
+           storageOwner.appendChild(storageEl);
+           storageEl.addBehavior('#default#userData');
+           storageEl.load(storageName);
+           storeFunction.apply(this, args);
+           storageOwner.removeChild(storageEl);
+           return;
+         };
+       }
 
-         while (true) {
-           do {
-             i++;
-           } while (compare(keys[i], pivot) < 0);
+       // doesn't work but cookies do. This implementation is adopted from
+       // https://developer.mozilla.org/en-US/docs/Web/API/Storage/LocalStorage
 
-           do {
-             j--;
-           } while (compare(keys[j], pivot) > 0);
+       var Global$1 = util.Global;
+       var trim = util.trim;
+       var cookieStorage = {
+         name: 'cookieStorage',
+         read: read$2,
+         write: write$2,
+         each: each$2,
+         remove: remove$2,
+         clearAll: clearAll$2
+       };
+       var doc = Global$1.document;
 
-           if (i >= j) break;
-           var tmp = keys[i];
-           keys[i] = keys[j];
-           keys[j] = tmp;
-           tmp = values[i];
-           values[i] = values[j];
-           values[j] = tmp;
+       function read$2(key) {
+         if (!key || !_has(key)) {
+           return null;
          }
 
-         sort$1(keys, values, left, j, compare);
-         sort$1(keys, values, j + 1, right, compare);
+         var regexpStr = "(?:^|.*;\\s*)" + escape(key).replace(/[\-\.\+\*]/g, "\\$&") + "\\s*\\=\\s*((?:[^;](?!;))*[^;]?).*";
+         return unescape(doc.cookie.replace(new RegExp(regexpStr), "$1"));
        }
 
-       function _classCallCheck$1(instance, Constructor) {
-         if (!(instance instanceof Constructor)) {
-           throw new TypeError("Cannot call a class as a function");
+       function each$2(callback) {
+         var cookies = doc.cookie.split(/; ?/g);
+
+         for (var i = cookies.length - 1; i >= 0; i--) {
+           if (!trim(cookies[i])) {
+             continue;
+           }
+
+           var kvp = cookies[i].split('=');
+           var key = unescape(kvp[0]);
+           var val = unescape(kvp[1]);
+           callback(val, key);
          }
        }
 
-       function _defineProperties$1(target, props) {
-         for (var i = 0; i < props.length; i++) {
-           var descriptor = props[i];
-           descriptor.enumerable = descriptor.enumerable || false;
-           descriptor.configurable = true;
-           if ("value" in descriptor) descriptor.writable = true;
-           Object.defineProperty(target, descriptor.key, descriptor);
+       function write$2(key, data) {
+         if (!key) {
+           return;
          }
-       }
 
-       function _createClass$1(Constructor, protoProps, staticProps) {
-         if (protoProps) _defineProperties$1(Constructor.prototype, protoProps);
-         if (staticProps) _defineProperties$1(Constructor, staticProps);
-         return Constructor;
+         doc.cookie = escape(key) + "=" + escape(data) + "; expires=Tue, 19 Jan 2038 03:14:07 GMT; path=/";
        }
-       /**
-        * A bounding box has the format:
-        *
-        *  { ll: { x: xmin, y: ymin }, ur: { x: xmax, y: ymax } }
-        *
-        */
-
-
-       var isInBbox = function isInBbox(bbox, point) {
-         return bbox.ll.x <= point.x && point.x <= bbox.ur.x && bbox.ll.y <= point.y && point.y <= bbox.ur.y;
-       };
-       /* Returns either null, or a bbox (aka an ordered pair of points)
-        * If there is only one point of overlap, a bbox with identical points
-        * will be returned */
-
-
-       var getBboxOverlap = function getBboxOverlap(b1, b2) {
-         // check if the bboxes overlap at all
-         if (b2.ur.x < b1.ll.x || b1.ur.x < b2.ll.x || b2.ur.y < b1.ll.y || b1.ur.y < b2.ll.y) return null; // find the middle two X values
 
-         var lowerX = b1.ll.x < b2.ll.x ? b2.ll.x : b1.ll.x;
-         var upperX = b1.ur.x < b2.ur.x ? b1.ur.x : b2.ur.x; // find the middle two Y values
+       function remove$2(key) {
+         if (!key || !_has(key)) {
+           return;
+         }
 
-         var lowerY = b1.ll.y < b2.ll.y ? b2.ll.y : b1.ll.y;
-         var upperY = b1.ur.y < b2.ur.y ? b1.ur.y : b2.ur.y; // put those middle values together to get the overlap
+         doc.cookie = escape(key) + "=; expires=Thu, 01 Jan 1970 00:00:00 GMT; path=/";
+       }
 
-         return {
-           ll: {
-             x: lowerX,
-             y: lowerY
-           },
-           ur: {
-             x: upperX,
-             y: upperY
-           }
-         };
-       };
-       /* Javascript doesn't do integer math. Everything is
-        * floating point with percision Number.EPSILON.
-        *
-        * https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/EPSILON
-        */
+       function clearAll$2() {
+         each$2(function (_, key) {
+           remove$2(key);
+         });
+       }
 
+       function _has(key) {
+         return new RegExp("(?:^|;\\s*)" + escape(key).replace(/[\-\.\+\*]/g, "\\$&") + "\\s*\\=").test(doc.cookie);
+       }
 
-       var epsilon$2 = Number.EPSILON; // IE Polyfill
+       var Global = util.Global;
+       var sessionStorage_1 = {
+         name: 'sessionStorage',
+         read: read$1,
+         write: write$1,
+         each: each$1,
+         remove: remove$1,
+         clearAll: clearAll$1
+       };
 
-       if (epsilon$2 === undefined) epsilon$2 = Math.pow(2, -52);
-       var EPSILON_SQ = epsilon$2 * epsilon$2;
-       /* FLP comparator */
+       function sessionStorage() {
+         return Global.sessionStorage;
+       }
 
-       var cmp = function cmp(a, b) {
-         // check if they're both 0
-         if (-epsilon$2 < a && a < epsilon$2) {
-           if (-epsilon$2 < b && b < epsilon$2) {
-             return 0;
-           }
-         } // check if they're flp equal
+       function read$1(key) {
+         return sessionStorage().getItem(key);
+       }
 
+       function write$1(key, data) {
+         return sessionStorage().setItem(key, data);
+       }
 
-         var ab = a - b;
+       function each$1(fn) {
+         for (var i = sessionStorage().length - 1; i >= 0; i--) {
+           var key = sessionStorage().key(i);
+           fn(read$1(key), key);
+         }
+       }
 
-         if (ab * ab < EPSILON_SQ * a * b) {
-           return 0;
-         } // normal comparison
+       function remove$1(key) {
+         return sessionStorage().removeItem(key);
+       }
 
+       function clearAll$1() {
+         return sessionStorage().clear();
+       }
 
-         return a < b ? -1 : 1;
+       // memoryStorage is a useful last fallback to ensure that the store
+       // is functions (meaning store.get(), store.set(), etc will all function).
+       // However, stored values will not persist when the browser navigates to
+       // a new page or reloads the current page.
+       var memoryStorage_1 = {
+         name: 'memoryStorage',
+         read: read,
+         write: write,
+         each: each,
+         remove: remove,
+         clearAll: clearAll
        };
-       /**
-        * This class rounds incoming values sufficiently so that
-        * floating points problems are, for the most part, avoided.
-        *
-        * Incoming points are have their x & y values tested against
-        * all previously seen x & y values. If either is 'too close'
-        * to a previously seen value, it's value is 'snapped' to the
-        * previously seen value.
-        *
-        * All points should be rounded by this class before being
-        * stored in any data structures in the rest of this algorithm.
-        */
+       var memoryStorage = {};
 
+       function read(key) {
+         return memoryStorage[key];
+       }
 
-       var PtRounder = /*#__PURE__*/function () {
-         function PtRounder() {
-           _classCallCheck$1(this, PtRounder);
+       function write(key, data) {
+         memoryStorage[key] = data;
+       }
 
-           this.reset();
+       function each(callback) {
+         for (var key in memoryStorage) {
+           if (memoryStorage.hasOwnProperty(key)) {
+             callback(memoryStorage[key], key);
+           }
          }
+       }
 
-         _createClass$1(PtRounder, [{
-           key: "reset",
-           value: function reset() {
-             this.xRounder = new CoordRounder();
-             this.yRounder = new CoordRounder();
-           }
-         }, {
-           key: "round",
-           value: function round(x, y) {
-             return {
-               x: this.xRounder.round(x),
-               y: this.yRounder.round(y)
-             };
-           }
-         }]);
+       function remove(key) {
+         delete memoryStorage[key];
+       }
 
-         return PtRounder;
-       }();
+       function clearAll(key) {
+         memoryStorage = {};
+       }
 
-       var CoordRounder = /*#__PURE__*/function () {
-         function CoordRounder() {
-           _classCallCheck$1(this, CoordRounder);
+       var all = [// Listed in order of usage preference
+       localStorage_1, oldFFGlobalStorage, oldIEUserDataStorage, cookieStorage, sessionStorage_1, memoryStorage_1];
 
-           this.tree = new Tree(); // preseed with 0 so we don't end up with values < Number.EPSILON
+       /* eslint-disable */
+       //  json2.js
+       //  2016-10-28
+       //  Public Domain.
+       //  NO WARRANTY EXPRESSED OR IMPLIED. USE AT YOUR OWN RISK.
+       //  See http://www.JSON.org/js.html
+       //  This code should be minified before deployment.
+       //  See http://javascript.crockford.com/jsmin.html
+       //  USE YOUR OWN COPY. IT IS EXTREMELY UNWISE TO LOAD CODE FROM SERVERS YOU DO
+       //  NOT CONTROL.
+       //  This file creates a global JSON object containing two methods: stringify
+       //  and parse. This file provides the ES5 JSON capability to ES3 systems.
+       //  If a project might run on IE8 or earlier, then this file should be included.
+       //  This file does nothing on ES5 systems.
+       //      JSON.stringify(value, replacer, space)
+       //          value       any JavaScript value, usually an object or array.
+       //          replacer    an optional parameter that determines how object
+       //                      values are stringified for objects. It can be a
+       //                      function or an array of strings.
+       //          space       an optional parameter that specifies the indentation
+       //                      of nested structures. If it is omitted, the text will
+       //                      be packed without extra whitespace. If it is a number,
+       //                      it will specify the number of spaces to indent at each
+       //                      level. If it is a string (such as "\t" or "&nbsp;"),
+       //                      it contains the characters used to indent at each level.
+       //          This method produces a JSON text from a JavaScript value.
+       //          When an object value is found, if the object contains a toJSON
+       //          method, its toJSON method will be called and the result will be
+       //          stringified. A toJSON method does not serialize: it returns the
+       //          value represented by the name/value pair that should be serialized,
+       //          or undefined if nothing should be serialized. The toJSON method
+       //          will be passed the key associated with the value, and this will be
+       //          bound to the value.
+       //          For example, this would serialize Dates as ISO strings.
+       //              Date.prototype.toJSON = function (key) {
+       //                  function f(n) {
+       //                      // Format integers to have at least two digits.
+       //                      return (n < 10)
+       //                          ? "0" + n
+       //                          : n;
+       //                  }
+       //                  return this.getUTCFullYear()   + "-" +
+       //                       f(this.getUTCMonth() + 1) + "-" +
+       //                       f(this.getUTCDate())      + "T" +
+       //                       f(this.getUTCHours())     + ":" +
+       //                       f(this.getUTCMinutes())   + ":" +
+       //                       f(this.getUTCSeconds())   + "Z";
+       //              };
+       //          You can provide an optional replacer method. It will be passed the
+       //          key and value of each member, with this bound to the containing
+       //          object. The value that is returned from your method will be
+       //          serialized. If your method returns undefined, then the member will
+       //          be excluded from the serialization.
+       //          If the replacer parameter is an array of strings, then it will be
+       //          used to select the members to be serialized. It filters the results
+       //          such that only members with keys listed in the replacer array are
+       //          stringified.
+       //          Values that do not have JSON representations, such as undefined or
+       //          functions, will not be serialized. Such values in objects will be
+       //          dropped; in arrays they will be replaced with null. You can use
+       //          a replacer function to replace those with JSON values.
+       //          JSON.stringify(undefined) returns undefined.
+       //          The optional space parameter produces a stringification of the
+       //          value that is filled with line breaks and indentation to make it
+       //          easier to read.
+       //          If the space parameter is a non-empty string, then that string will
+       //          be used for indentation. If the space parameter is a number, then
+       //          the indentation will be that many spaces.
+       //          Example:
+       //          text = JSON.stringify(["e", {pluribus: "unum"}]);
+       //          // text is '["e",{"pluribus":"unum"}]'
+       //          text = JSON.stringify(["e", {pluribus: "unum"}], null, "\t");
+       //          // text is '[\n\t"e",\n\t{\n\t\t"pluribus": "unum"\n\t}\n]'
+       //          text = JSON.stringify([new Date()], function (key, value) {
+       //              return this[key] instanceof Date
+       //                  ? "Date(" + this[key] + ")"
+       //                  : value;
+       //          });
+       //          // text is '["Date(---current time---)"]'
+       //      JSON.parse(text, reviver)
+       //          This method parses a JSON text to produce an object or array.
+       //          It can throw a SyntaxError exception.
+       //          The optional reviver parameter is a function that can filter and
+       //          transform the results. It receives each of the keys and values,
+       //          and its return value is used instead of the original value.
+       //          If it returns what it received, then the structure is not modified.
+       //          If it returns undefined then the member is deleted.
+       //          Example:
+       //          // Parse the text. Values that look like ISO date strings will
+       //          // be converted to Date objects.
+       //          myData = JSON.parse(text, function (key, value) {
+       //              var a;
+       //              if (typeof value === "string") {
+       //                  a =
+       //   /^(\d{4})-(\d{2})-(\d{2})T(\d{2}):(\d{2}):(\d{2}(?:\.\d*)?)Z$/.exec(value);
+       //                  if (a) {
+       //                      return new Date(Date.UTC(+a[1], +a[2] - 1, +a[3], +a[4],
+       //                          +a[5], +a[6]));
+       //                  }
+       //              }
+       //              return value;
+       //          });
+       //          myData = JSON.parse('["Date(09/09/2001)"]', function (key, value) {
+       //              var d;
+       //              if (typeof value === "string" &&
+       //                      value.slice(0, 5) === "Date(" &&
+       //                      value.slice(-1) === ")") {
+       //                  d = new Date(value.slice(5, -1));
+       //                  if (d) {
+       //                      return d;
+       //                  }
+       //              }
+       //              return value;
+       //          });
+       //  This is a reference implementation. You are free to copy, modify, or
+       //  redistribute.
 
-           this.round(0);
-         } // Note: this can rounds input values backwards or forwards.
-         //       You might ask, why not restrict this to just rounding
-         //       forwards? Wouldn't that allow left endpoints to always
-         //       remain left endpoints during splitting (never change to
-         //       right). No - it wouldn't, because we snap intersections
-         //       to endpoints (to establish independence from the segment
-         //       angle for t-intersections).
+       /*jslint
+           eval, for, this
+       */
 
+       /*property
+           JSON, apply, call, charCodeAt, getUTCDate, getUTCFullYear, getUTCHours,
+           getUTCMinutes, getUTCMonth, getUTCSeconds, hasOwnProperty, join,
+           lastIndex, length, parse, prototype, push, replace, slice, stringify,
+           test, toJSON, toString, valueOf
+       */
+       // Create a JSON object only if one does not already exist. We create the
+       // methods in a closure to avoid creating global variables.
+       if ((typeof JSON === "undefined" ? "undefined" : _typeof(JSON)) !== "object") {
+         JSON = {};
+       }
 
-         _createClass$1(CoordRounder, [{
-           key: "round",
-           value: function round(coord) {
-             var node = this.tree.add(coord);
-             var prevNode = this.tree.prev(node);
+       (function () {
 
-             if (prevNode !== null && cmp(node.key, prevNode.key) === 0) {
-               this.tree.remove(coord);
-               return prevNode.key;
-             }
+         var rx_one = /^[\],:{}\s]*$/;
+         var rx_two = /\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g;
+         var rx_three = /"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g;
+         var rx_four = /(?:^|:|,)(?:\s*\[)+/g;
+         var rx_escapable = /[\\"\u0000-\u001f\u007f-\u009f\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g;
+         var rx_dangerous = /[\u0000\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g;
 
-             var nextNode = this.tree.next(node);
+         function f(n) {
+           // Format integers to have at least two digits.
+           return n < 10 ? "0" + n : n;
+         }
 
-             if (nextNode !== null && cmp(node.key, nextNode.key) === 0) {
-               this.tree.remove(coord);
-               return nextNode.key;
-             }
+         function this_value() {
+           return this.valueOf();
+         }
 
-             return coord;
-           }
-         }]);
+         if (typeof Date.prototype.toJSON !== "function") {
+           Date.prototype.toJSON = function () {
+             return isFinite(this.valueOf()) ? this.getUTCFullYear() + "-" + f(this.getUTCMonth() + 1) + "-" + f(this.getUTCDate()) + "T" + f(this.getUTCHours()) + ":" + f(this.getUTCMinutes()) + ":" + f(this.getUTCSeconds()) + "Z" : null;
+           };
 
-         return CoordRounder;
-       }(); // singleton available by import
+           Boolean.prototype.toJSON = this_value;
+           Number.prototype.toJSON = this_value;
+           String.prototype.toJSON = this_value;
+         }
 
+         var gap;
+         var indent;
+         var meta;
+         var rep;
 
-       var rounder = new PtRounder();
-       /* Cross Product of two vectors with first point at origin */
+         function quote(string) {
+           // If the string contains no control characters, no quote characters, and no
+           // backslash characters, then we can safely slap some quotes around it.
+           // Otherwise we must also replace the offending characters with safe escape
+           // sequences.
+           rx_escapable.lastIndex = 0;
+           return rx_escapable.test(string) ? "\"" + string.replace(rx_escapable, function (a) {
+             var c = meta[a];
+             return typeof c === "string" ? c : "\\u" + ("0000" + a.charCodeAt(0).toString(16)).slice(-4);
+           }) + "\"" : "\"" + string + "\"";
+         }
 
-       var crossProduct$1 = function crossProduct(a, b) {
-         return a.x * b.y - a.y * b.x;
-       };
-       /* Dot Product of two vectors with first point at origin */
+         function str(key, holder) {
+           // Produce a string from holder[key].
+           var i; // The loop counter.
 
+           var k; // The member key.
 
-       var dotProduct$1 = function dotProduct(a, b) {
-         return a.x * b.x + a.y * b.y;
-       };
-       /* Comparator for two vectors with same starting point */
+           var v; // The member value.
 
+           var length;
+           var mind = gap;
+           var partial;
+           var value = holder[key]; // If the value has a toJSON method, call it to obtain a replacement value.
 
-       var compareVectorAngles = function compareVectorAngles(basePt, endPt1, endPt2) {
-         var v1 = {
-           x: endPt1.x - basePt.x,
-           y: endPt1.y - basePt.y
-         };
-         var v2 = {
-           x: endPt2.x - basePt.x,
-           y: endPt2.y - basePt.y
-         };
-         var kross = crossProduct$1(v1, v2);
-         return cmp(kross, 0);
-       };
+           if (value && _typeof(value) === "object" && typeof value.toJSON === "function") {
+             value = value.toJSON(key);
+           } // If we were called with a replacer function, then call the replacer to
+           // obtain a replacement value.
 
-       var length = function length(v) {
-         return Math.sqrt(dotProduct$1(v, v));
-       };
-       /* Get the sine of the angle from pShared -> pAngle to pShaed -> pBase */
 
+           if (typeof rep === "function") {
+             value = rep.call(holder, key, value);
+           } // What happens next depends on the value's type.
 
-       var sineOfAngle = function sineOfAngle(pShared, pBase, pAngle) {
-         var vBase = {
-           x: pBase.x - pShared.x,
-           y: pBase.y - pShared.y
-         };
-         var vAngle = {
-           x: pAngle.x - pShared.x,
-           y: pAngle.y - pShared.y
-         };
-         return crossProduct$1(vAngle, vBase) / length(vAngle) / length(vBase);
-       };
-       /* Get the cosine of the angle from pShared -> pAngle to pShaed -> pBase */
 
+           switch (_typeof(value)) {
+             case "string":
+               return quote(value);
 
-       var cosineOfAngle = function cosineOfAngle(pShared, pBase, pAngle) {
-         var vBase = {
-           x: pBase.x - pShared.x,
-           y: pBase.y - pShared.y
-         };
-         var vAngle = {
-           x: pAngle.x - pShared.x,
-           y: pAngle.y - pShared.y
-         };
-         return dotProduct$1(vAngle, vBase) / length(vAngle) / length(vBase);
-       };
-       /* Get the x coordinate where the given line (defined by a point and vector)
-        * crosses the horizontal line with the given y coordiante.
-        * In the case of parrallel lines (including overlapping ones) returns null. */
+             case "number":
+               // JSON numbers must be finite. Encode non-finite numbers as null.
+               return isFinite(value) ? String(value) : "null";
 
+             case "boolean":
+             case "null":
+               // If the value is a boolean or null, convert it to a string. Note:
+               // typeof null does not produce "null". The case is included here in
+               // the remote chance that this gets fixed someday.
+               return String(value);
+             // If the type is "object", we might be dealing with an object or an array or
+             // null.
 
-       var horizontalIntersection = function horizontalIntersection(pt, v, y) {
-         if (v.y === 0) return null;
-         return {
-           x: pt.x + v.x / v.y * (y - pt.y),
-           y: y
-         };
-       };
-       /* Get the y coordinate where the given line (defined by a point and vector)
-        * crosses the vertical line with the given x coordiante.
-        * In the case of parrallel lines (including overlapping ones) returns null. */
+             case "object":
+               // Due to a specification blunder in ECMAScript, typeof null is "object",
+               // so watch out for that case.
+               if (!value) {
+                 return "null";
+               } // Make an array to hold the partial results of stringifying this object value.
 
 
-       var verticalIntersection = function verticalIntersection(pt, v, x) {
-         if (v.x === 0) return null;
-         return {
-           x: x,
-           y: pt.y + v.y / v.x * (x - pt.x)
-         };
-       };
-       /* Get the intersection of two lines, each defined by a base point and a vector.
-        * In the case of parrallel lines (including overlapping ones) returns null. */
+               gap += indent;
+               partial = []; // Is the value an array?
 
+               if (Object.prototype.toString.apply(value) === "[object Array]") {
+                 // The value is an array. Stringify every element. Use null as a placeholder
+                 // for non-JSON values.
+                 length = value.length;
 
-       var intersection$1 = function intersection(pt1, v1, pt2, v2) {
-         // take some shortcuts for vertical and horizontal lines
-         // this also ensures we don't calculate an intersection and then discover
-         // it's actually outside the bounding box of the line
-         if (v1.x === 0) return verticalIntersection(pt2, v2, pt1.x);
-         if (v2.x === 0) return verticalIntersection(pt1, v1, pt2.x);
-         if (v1.y === 0) return horizontalIntersection(pt2, v2, pt1.y);
-         if (v2.y === 0) return horizontalIntersection(pt1, v1, pt2.y); // General case for non-overlapping segments.
-         // This algorithm is based on Schneider and Eberly.
-         // http://www.cimec.org.ar/~ncalvo/Schneider_Eberly.pdf - pg 244
+                 for (i = 0; i < length; i += 1) {
+                   partial[i] = str(i, value) || "null";
+                 } // Join all of the elements together, separated with commas, and wrap them in
+                 // brackets.
 
-         var kross = crossProduct$1(v1, v2);
-         if (kross == 0) return null;
-         var ve = {
-           x: pt2.x - pt1.x,
-           y: pt2.y - pt1.y
-         };
-         var d1 = crossProduct$1(ve, v1) / kross;
-         var d2 = crossProduct$1(ve, v2) / kross; // take the average of the two calculations to minimize rounding error
 
-         var x1 = pt1.x + d2 * v1.x,
-             x2 = pt2.x + d1 * v2.x;
-         var y1 = pt1.y + d2 * v1.y,
-             y2 = pt2.y + d1 * v2.y;
-         var x = (x1 + x2) / 2;
-         var y = (y1 + y2) / 2;
-         return {
-           x: x,
-           y: y
-         };
-       };
+                 v = partial.length === 0 ? "[]" : gap ? "[\n" + gap + partial.join(",\n" + gap) + "\n" + mind + "]" : "[" + partial.join(",") + "]";
+                 gap = mind;
+                 return v;
+               } // If the replacer is an array, use it to select the members to be stringified.
 
-       var SweepEvent$1 = /*#__PURE__*/function () {
-         _createClass$1(SweepEvent, null, [{
-           key: "compare",
-           // for ordering sweep events in the sweep event queue
-           value: function compare(a, b) {
-             // favor event with a point that the sweep line hits first
-             var ptCmp = SweepEvent.comparePoints(a.point, b.point);
-             if (ptCmp !== 0) return ptCmp; // the points are the same, so link them if needed
 
-             if (a.point !== b.point) a.link(b); // favor right events over left
+               if (rep && _typeof(rep) === "object") {
+                 length = rep.length;
 
-             if (a.isLeft !== b.isLeft) return a.isLeft ? 1 : -1; // we have two matching left or right endpoints
-             // ordering of this case is the same as for their segments
+                 for (i = 0; i < length; i += 1) {
+                   if (typeof rep[i] === "string") {
+                     k = rep[i];
+                     v = str(k, value);
 
-             return Segment.compare(a.segment, b.segment);
-           } // for ordering points in sweep line order
+                     if (v) {
+                       partial.push(quote(k) + (gap ? ": " : ":") + v);
+                     }
+                   }
+                 }
+               } else {
+                 // Otherwise, iterate through all of the keys in the object.
+                 for (k in value) {
+                   if (Object.prototype.hasOwnProperty.call(value, k)) {
+                     v = str(k, value);
 
-         }, {
-           key: "comparePoints",
-           value: function comparePoints(aPt, bPt) {
-             if (aPt.x < bPt.x) return -1;
-             if (aPt.x > bPt.x) return 1;
-             if (aPt.y < bPt.y) return -1;
-             if (aPt.y > bPt.y) return 1;
-             return 0;
-           } // Warning: 'point' input will be modified and re-used (for performance)
+                     if (v) {
+                       partial.push(quote(k) + (gap ? ": " : ":") + v);
+                     }
+                   }
+                 }
+               } // Join all of the member texts together, separated with commas,
+               // and wrap them in braces.
 
-         }]);
 
-         function SweepEvent(point, isLeft) {
-           _classCallCheck$1(this, SweepEvent);
+               v = partial.length === 0 ? "{}" : gap ? "{\n" + gap + partial.join(",\n" + gap) + "\n" + mind + "}" : "{" + partial.join(",") + "}";
+               gap = mind;
+               return v;
+           }
+         } // If the JSON object does not yet have a stringify method, give it one.
+
+
+         if (typeof JSON.stringify !== "function") {
+           meta = {
+             // table of character substitutions
+             "\b": "\\b",
+             "\t": "\\t",
+             "\n": "\\n",
+             "\f": "\\f",
+             "\r": "\\r",
+             "\"": "\\\"",
+             "\\": "\\\\"
+           };
+
+           JSON.stringify = function (value, replacer, space) {
+             // The stringify method takes a value and an optional replacer, and an optional
+             // space parameter, and returns a JSON text. The replacer can be a function
+             // that can replace values, or an array of strings that will select the keys.
+             // A default replacer method can be provided. Use of the space parameter can
+             // produce text that is more easily readable.
+             var i;
+             gap = "";
+             indent = ""; // If the space parameter is a number, make an indent string containing that
+             // many spaces.
+
+             if (typeof space === "number") {
+               for (i = 0; i < space; i += 1) {
+                 indent += " ";
+               } // If the space parameter is a string, it will be used as the indent string.
+
+             } else if (typeof space === "string") {
+               indent = space;
+             } // If there is a replacer, it must be a function or an array.
+             // Otherwise, throw an error.
 
-           if (point.events === undefined) point.events = [this];else point.events.push(this);
-           this.point = point;
-           this.isLeft = isLeft; // this.segment, this.otherSE set by factory
-         }
 
-         _createClass$1(SweepEvent, [{
-           key: "link",
-           value: function link(other) {
-             if (other.point === this.point) {
-               throw new Error('Tried to link already linked events');
-             }
+             rep = replacer;
 
-             var otherEvents = other.point.events;
+             if (replacer && typeof replacer !== "function" && (_typeof(replacer) !== "object" || typeof replacer.length !== "number")) {
+               throw new Error("JSON.stringify");
+             } // Make a fake root object containing our value under the key of "".
+             // Return the result of stringifying the value.
 
-             for (var i = 0, iMax = otherEvents.length; i < iMax; i++) {
-               var evt = otherEvents[i];
-               this.point.events.push(evt);
-               evt.point = this.point;
-             }
 
-             this.checkForConsuming();
-           }
-           /* Do a pass over our linked events and check to see if any pair
-            * of segments match, and should be consumed. */
+             return str("", {
+               "": value
+             });
+           };
+         } // If the JSON object does not yet have a parse method, give it one.
 
-         }, {
-           key: "checkForConsuming",
-           value: function checkForConsuming() {
-             // FIXME: The loops in this method run O(n^2) => no good.
-             //        Maintain little ordered sweep event trees?
-             //        Can we maintaining an ordering that avoids the need
-             //        for the re-sorting with getLeftmostComparator in geom-out?
-             // Compare each pair of events to see if other events also match
-             var numEvents = this.point.events.length;
 
-             for (var i = 0; i < numEvents; i++) {
-               var evt1 = this.point.events[i];
-               if (evt1.segment.consumedBy !== undefined) continue;
+         if (typeof JSON.parse !== "function") {
+           JSON.parse = function (text, reviver) {
+             // The parse method takes a text and an optional reviver function, and returns
+             // a JavaScript value if the text is a valid JSON text.
+             var j;
 
-               for (var j = i + 1; j < numEvents; j++) {
-                 var evt2 = this.point.events[j];
-                 if (evt2.consumedBy !== undefined) continue;
-                 if (evt1.otherSE.point.events !== evt2.otherSE.point.events) continue;
-                 evt1.segment.consume(evt2.segment);
-               }
-             }
-           }
-         }, {
-           key: "getAvailableLinkedEvents",
-           value: function getAvailableLinkedEvents() {
-             // point.events is always of length 2 or greater
-             var events = [];
+             function walk(holder, key) {
+               // The walk method is used to recursively walk the resulting structure so
+               // that modifications can be made.
+               var k;
+               var v;
+               var value = holder[key];
 
-             for (var i = 0, iMax = this.point.events.length; i < iMax; i++) {
-               var evt = this.point.events[i];
+               if (value && _typeof(value) === "object") {
+                 for (k in value) {
+                   if (Object.prototype.hasOwnProperty.call(value, k)) {
+                     v = walk(value, k);
 
-               if (evt !== this && !evt.segment.ringOut && evt.segment.isInResult()) {
-                 events.push(evt);
+                     if (v !== undefined) {
+                       value[k] = v;
+                     } else {
+                       delete value[k];
+                     }
+                   }
+                 }
                }
-             }
 
-             return events;
-           }
-           /**
-            * Returns a comparator function for sorting linked events that will
-            * favor the event that will give us the smallest left-side angle.
-            * All ring construction starts as low as possible heading to the right,
-            * so by always turning left as sharp as possible we'll get polygons
-            * without uncessary loops & holes.
-            *
-            * The comparator function has a compute cache such that it avoids
-            * re-computing already-computed values.
-            */
+               return reviver.call(holder, key, value);
+             } // Parsing happens in four stages. In the first stage, we replace certain
+             // Unicode characters with escape sequences. JavaScript handles many characters
+             // incorrectly, either silently deleting them, or treating them as line endings.
 
-         }, {
-           key: "getLeftmostComparator",
-           value: function getLeftmostComparator(baseEvent) {
-             var _this = this;
 
-             var cache = new Map();
+             text = String(text);
+             rx_dangerous.lastIndex = 0;
 
-             var fillCache = function fillCache(linkedEvent) {
-               var nextEvent = linkedEvent.otherSE;
-               cache.set(linkedEvent, {
-                 sine: sineOfAngle(_this.point, baseEvent.point, nextEvent.point),
-                 cosine: cosineOfAngle(_this.point, baseEvent.point, nextEvent.point)
+             if (rx_dangerous.test(text)) {
+               text = text.replace(rx_dangerous, function (a) {
+                 return "\\u" + ("0000" + a.charCodeAt(0).toString(16)).slice(-4);
                });
-             };
+             } // In the second stage, we run the text against regular expressions that look
+             // for non-JSON patterns. We are especially concerned with "()" and "new"
+             // because they can cause invocation, and "=" because it can cause mutation.
+             // But just to be safe, we want to reject all unexpected forms.
+             // We split the second stage into 4 regexp operations in order to work around
+             // crippling inefficiencies in IE's and Safari's regexp engines. First we
+             // replace the JSON backslash pairs with "@" (a non-JSON character). Second, we
+             // replace all simple value tokens with "]" characters. Third, we delete all
+             // open brackets that follow a colon or comma or that begin the text. Finally,
+             // we look to see that the remaining characters are only whitespace or "]" or
+             // "," or ":" or "{" or "}". If that is so, then the text is safe for eval.
 
-             return function (a, b) {
-               if (!cache.has(a)) fillCache(a);
-               if (!cache.has(b)) fillCache(b);
 
-               var _cache$get = cache.get(a),
-                   asine = _cache$get.sine,
-                   acosine = _cache$get.cosine;
+             if (rx_one.test(text.replace(rx_two, "@").replace(rx_three, "]").replace(rx_four, ""))) {
+               // In the third stage we use the eval function to compile the text into a
+               // JavaScript structure. The "{" operator is subject to a syntactic ambiguity
+               // in JavaScript: it can begin a block or an object literal. We wrap the text
+               // in parens to eliminate the ambiguity.
+               j = eval("(" + text + ")"); // In the optional fourth stage, we recursively walk the new structure, passing
+               // each name/value pair to a reviver function for possible transformation.
 
-               var _cache$get2 = cache.get(b),
-                   bsine = _cache$get2.sine,
-                   bcosine = _cache$get2.cosine; // both on or above x-axis
+               return typeof reviver === "function" ? walk({
+                 "": j
+               }, "") : j;
+             } // If the text is not JSON parseable, then a SyntaxError is thrown.
 
 
-               if (asine >= 0 && bsine >= 0) {
-                 if (acosine < bcosine) return 1;
-                 if (acosine > bcosine) return -1;
-                 return 0;
-               } // both below x-axis
+             throw new SyntaxError("JSON.parse");
+           };
+         }
+       })();
 
+       var json2 = json2Plugin;
 
-               if (asine < 0 && bsine < 0) {
-                 if (acosine < bcosine) return -1;
-                 if (acosine > bcosine) return 1;
-                 return 0;
-               } // one above x-axis, one below
+       function json2Plugin() {
+         return {};
+       }
 
+       var plugins = [json2];
+       var store_legacy = storeEngine.createStore(all, plugins);
 
-               if (bsine < asine) return -1;
-               if (bsine > asine) return 1;
-               return 0;
-             };
-           }
-         }]);
+       var immutable = extend;
+       var hasOwnProperty = Object.prototype.hasOwnProperty;
 
-         return SweepEvent;
-       }(); // segments and sweep events when all else is identical
+       function extend() {
+         var target = {};
 
+         for (var i = 0; i < arguments.length; i++) {
+           var source = arguments[i];
 
-       var segmentId = 0;
+           for (var key in source) {
+             if (hasOwnProperty.call(source, key)) {
+               target[key] = source[key];
+             }
+           }
+         }
 
-       var Segment = /*#__PURE__*/function () {
-         _createClass$1(Segment, null, [{
-           key: "compare",
+         return target;
+       }
 
-           /* This compare() function is for ordering segments in the sweep
-            * line tree, and does so according to the following criteria:
-            *
-            * Consider the vertical line that lies an infinestimal step to the
-            * right of the right-more of the two left endpoints of the input
-            * segments. Imagine slowly moving a point up from negative infinity
-            * in the increasing y direction. Which of the two segments will that
-            * point intersect first? That segment comes 'before' the other one.
-            *
-            * If neither segment would be intersected by such a line, (if one
-            * or more of the segments are vertical) then the line to be considered
-            * is directly on the right-more of the two left inputs.
-            */
-           value: function compare(a, b) {
-             var alx = a.leftSE.point.x;
-             var blx = b.leftSE.point.x;
-             var arx = a.rightSE.point.x;
-             var brx = b.rightSE.point.x; // check if they're even in the same vertical plane
+       //
+       // This code is only compatible with IE10+ because the [XDomainRequest](http://bit.ly/LfO7xo)
+       // object, IE<10's idea of [CORS](http://en.wikipedia.org/wiki/Cross-origin_resource_sharing),
+       // does not support custom headers, which this uses everywhere.
 
-             if (brx < alx) return 1;
-             if (arx < blx) return -1;
-             var aly = a.leftSE.point.y;
-             var bly = b.leftSE.point.y;
-             var ary = a.rightSE.point.y;
-             var bry = b.rightSE.point.y; // is left endpoint of segment B the right-more?
 
-             if (alx < blx) {
-               // are the two segments in the same horizontal plane?
-               if (bly < aly && bly < ary) return 1;
-               if (bly > aly && bly > ary) return -1; // is the B left endpoint colinear to segment A?
+       var osmAuth = function osmAuth(o) {
+         var oauth = {}; // authenticated users will also have a request token secret, but it's
+         // not used in transactions with the server
 
-               var aCmpBLeft = a.comparePoint(b.leftSE.point);
-               if (aCmpBLeft < 0) return 1;
-               if (aCmpBLeft > 0) return -1; // is the A right endpoint colinear to segment B ?
+         oauth.authenticated = function () {
+           return !!(token('oauth_token') && token('oauth_token_secret'));
+         };
 
-               var bCmpARight = b.comparePoint(a.rightSE.point);
-               if (bCmpARight !== 0) return bCmpARight; // colinear segments, consider the one with left-more
-               // left endpoint to be first (arbitrary?)
+         oauth.logout = function () {
+           token('oauth_token', '');
+           token('oauth_token_secret', '');
+           token('oauth_request_token_secret', '');
+           return oauth;
+         }; // TODO: detect lack of click event
 
-               return -1;
-             } // is left endpoint of segment A the right-more?
 
+         oauth.authenticate = function (callback) {
+           if (oauth.authenticated()) return callback();
+           oauth.logout(); // ## Getting a request token
 
-             if (alx > blx) {
-               if (aly < bly && aly < bry) return -1;
-               if (aly > bly && aly > bry) return 1; // is the A left endpoint colinear to segment B?
+           var params = timenonce(getAuth(o)),
+               url = o.url + '/oauth/request_token';
+           params.oauth_signature = ohauth_1.signature(o.oauth_secret, '', ohauth_1.baseString('POST', url, params));
 
-               var bCmpALeft = b.comparePoint(a.leftSE.point);
-               if (bCmpALeft !== 0) return bCmpALeft; // is the B right endpoint colinear to segment A?
+           if (!o.singlepage) {
+             // Create a 600x550 popup window in the center of the screen
+             var w = 600,
+                 h = 550,
+                 settings = [['width', w], ['height', h], ['left', screen.width / 2 - w / 2], ['top', screen.height / 2 - h / 2]].map(function (x) {
+               return x.join('=');
+             }).join(','),
+                 popup = window.open('about:blank', 'oauth_window', settings);
+             oauth.popupWindow = popup;
 
-               var aCmpBRight = a.comparePoint(b.rightSE.point);
-               if (aCmpBRight < 0) return 1;
-               if (aCmpBRight > 0) return -1; // colinear segments, consider the one with left-more
-               // left endpoint to be first (arbitrary?)
+             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
+           // window is redirected to OSM's authorization page.
 
-               return 1;
-             } // if we get here, the two left endpoints are in the same
-             // vertical plane, ie alx === blx
-             // consider the lower left-endpoint to come first
 
+           ohauth_1.xhr('POST', url, params, null, {}, reqTokenDone);
+           o.loading();
 
-             if (aly < bly) return -1;
-             if (aly > bly) return 1; // left endpoints are identical
-             // check for colinearity by using the left-more right endpoint
-             // is the A right endpoint more left-more?
+           function reqTokenDone(err, xhr) {
+             o.done();
+             if (err) return callback(err);
+             var resp = ohauth_1.stringQs(xhr.response);
+             token('oauth_request_token_secret', resp.oauth_token_secret);
+             var authorize_url = o.url + '/oauth/authorize?' + ohauth_1.qsString({
+               oauth_token: resp.oauth_token,
+               oauth_callback: resolveUrl(o.landing)
+             });
 
-             if (arx < brx) {
-               var _bCmpARight = b.comparePoint(a.rightSE.point);
+             if (o.singlepage) {
+               location.href = authorize_url;
+             } else {
+               popup.location = authorize_url;
+             }
+           } // Called by a function in a landing page, in the popup window. The
+           // window closes itself.
 
-               if (_bCmpARight !== 0) return _bCmpARight;
-             } // is the B right endpoint more left-more?
 
+           window.authComplete = function (token) {
+             var oauth_token = ohauth_1.stringQs(token.split('?')[1]);
+             get_access_token(oauth_token.oauth_token);
+             delete window.authComplete;
+           }; // ## Getting an request token
+           //
+           // At this point we have an `oauth_token`, brought in from a function
+           // call on a landing page popup.
 
-             if (arx > brx) {
-               var _aCmpBRight = a.comparePoint(b.rightSE.point);
 
-               if (_aCmpBRight < 0) return 1;
-               if (_aCmpBRight > 0) return -1;
-             }
+           function get_access_token(oauth_token) {
+             var url = o.url + '/oauth/access_token',
+                 params = timenonce(getAuth(o)),
+                 request_token_secret = token('oauth_request_token_secret');
+             params.oauth_token = oauth_token;
+             params.oauth_signature = ohauth_1.signature(o.oauth_secret, request_token_secret, ohauth_1.baseString('POST', url, params)); // ## Getting an access token
+             //
+             // The final token required for authentication. At this point
+             // we have a `request token secret`
 
-             if (arx !== brx) {
-               // are these two [almost] vertical segments with opposite orientation?
-               // if so, the one with the lower right endpoint comes first
-               var ay = ary - aly;
-               var ax = arx - alx;
-               var by = bry - bly;
-               var bx = brx - blx;
-               if (ay > ax && by < bx) return 1;
-               if (ay < ax && by > bx) return -1;
-             } // we have colinear segments with matching orientation
-             // consider the one with more left-more right endpoint to be first
+             ohauth_1.xhr('POST', url, params, null, {}, accessTokenDone);
+             o.loading();
+           }
 
+           function accessTokenDone(err, xhr) {
+             o.done();
+             if (err) return callback(err);
+             var access_token = ohauth_1.stringQs(xhr.response);
+             token('oauth_token', access_token.oauth_token);
+             token('oauth_token_secret', access_token.oauth_token_secret);
+             callback(null, oauth);
+           }
+         };
 
-             if (arx > brx) return 1;
-             if (arx < brx) return -1; // if we get here, two two right endpoints are in the same
-             // vertical plane, ie arx === brx
-             // consider the lower right-endpoint to come first
+         oauth.bringPopupWindowToFront = function () {
+           var brougtPopupToFront = false;
 
-             if (ary < bry) return -1;
-             if (ary > bry) return 1; // right endpoints identical as well, so the segments are idential
-             // fall back on creation order as consistent tie-breaker
+           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)
+           }
 
-             if (a.id < b.id) return -1;
-             if (a.id > b.id) return 1; // identical segment, ie a === b
+           return brougtPopupToFront;
+         };
 
-             return 0;
-           }
-           /* Warning: a reference to ringWindings input will be stored,
-            *  and possibly will be later modified */
+         oauth.bootstrapToken = function (oauth_token, callback) {
+           // ## Getting an request token
+           // At this point we have an `oauth_token`, brought in from a function
+           // call on a landing page popup.
+           function get_access_token(oauth_token) {
+             var url = o.url + '/oauth/access_token',
+                 params = timenonce(getAuth(o)),
+                 request_token_secret = token('oauth_request_token_secret');
+             params.oauth_token = oauth_token;
+             params.oauth_signature = ohauth_1.signature(o.oauth_secret, request_token_secret, ohauth_1.baseString('POST', url, params)); // ## Getting an access token
+             // The final token required for authentication. At this point
+             // we have a `request token secret`
 
-         }]);
+             ohauth_1.xhr('POST', url, params, null, {}, accessTokenDone);
+             o.loading();
+           }
 
-         function Segment(leftSE, rightSE, rings, windings) {
-           _classCallCheck$1(this, Segment);
+           function accessTokenDone(err, xhr) {
+             o.done();
+             if (err) return callback(err);
+             var access_token = ohauth_1.stringQs(xhr.response);
+             token('oauth_token', access_token.oauth_token);
+             token('oauth_token_secret', access_token.oauth_token_secret);
+             callback(null, oauth);
+           }
 
-           this.id = ++segmentId;
-           this.leftSE = leftSE;
-           leftSE.segment = this;
-           leftSE.otherSE = rightSE;
-           this.rightSE = rightSE;
-           rightSE.segment = this;
-           rightSE.otherSE = leftSE;
-           this.rings = rings;
-           this.windings = windings; // left unset for performance, set later in algorithm
-           // this.ringOut, this.consumedBy, this.prev
-         }
+           get_access_token(oauth_token);
+         }; // # xhr
+         //
+         // A single XMLHttpRequest wrapper that does authenticated calls if the
+         // user has logged in.
 
-         _createClass$1(Segment, [{
-           key: "replaceRightSE",
 
-           /* When a segment is split, the rightSE is replaced with a new sweep event */
-           value: function replaceRightSE(newRightSE) {
-             this.rightSE = newRightSE;
-             this.rightSE.segment = this;
-             this.rightSE.otherSE = this.leftSE;
-             this.leftSE.otherSE = this.rightSE;
-           }
-         }, {
-           key: "bbox",
-           value: function bbox() {
-             var y1 = this.leftSE.point.y;
-             var y2 = this.rightSE.point.y;
-             return {
-               ll: {
-                 x: this.leftSE.point.x,
-                 y: y1 < y2 ? y1 : y2
-               },
-               ur: {
-                 x: this.rightSE.point.x,
-                 y: y1 > y2 ? y1 : y2
-               }
-             };
+         oauth.xhr = function (options, callback) {
+           if (!oauth.authenticated()) {
+             if (o.auto) {
+               return oauth.authenticate(run);
+             } else {
+               callback('not authenticated', null);
+               return;
+             }
+           } else {
+             return run();
            }
-           /* A vector from the left point to the right */
 
-         }, {
-           key: "vector",
-           value: function vector() {
-             return {
-               x: this.rightSE.point.x - this.leftSE.point.x,
-               y: this.rightSE.point.y - this.leftSE.point.y
-             };
+           function run() {
+             var params = timenonce(getAuth(o)),
+                 oauth_token_secret = token('oauth_token_secret'),
+                 url = options.prefix !== false ? o.url + options.path : options.path,
+                 url_parts = url.replace(/#.*$/, '').split('?', 2),
+                 base_url = url_parts[0],
+                 query = url_parts.length === 2 ? url_parts[1] : ''; // https://tools.ietf.org/html/rfc5849#section-3.4.1.3.1
+
+             if ((!options.options || !options.options.header || options.options.header['Content-Type'] === 'application/x-www-form-urlencoded') && options.content) {
+               params = immutable(params, ohauth_1.stringQs(options.content));
+             }
+
+             params.oauth_token = token('oauth_token');
+             params.oauth_signature = ohauth_1.signature(o.oauth_secret, oauth_token_secret, ohauth_1.baseString(options.method, base_url, immutable(params, ohauth_1.stringQs(query))));
+             return ohauth_1.xhr(options.method, url, params, options.content, options.options, done);
            }
-         }, {
-           key: "isAnEndpoint",
-           value: function isAnEndpoint(pt) {
-             return pt.x === this.leftSE.point.x && pt.y === this.leftSE.point.y || pt.x === this.rightSE.point.x && pt.y === this.rightSE.point.y;
+
+           function done(err, xhr) {
+             if (err) return callback(err);else if (xhr.responseXML) return callback(err, xhr.responseXML);else return callback(err, xhr.response);
            }
-           /* Compare this segment with a point.
-            *
-            * A point P is considered to be colinear to a segment if there
-            * exists a distance D such that if we travel along the segment
-            * from one * endpoint towards the other a distance D, we find
-            * ourselves at point P.
-            *
-            * Return value indicates:
-            *
-            *   1: point lies above the segment (to the left of vertical)
-            *   0: point is colinear to segment
-            *  -1: point lies below the segment (to the right of vertical)
-            */
+         }; // pre-authorize this object, if we can just get a token and token_secret
+         // from the start
 
-         }, {
-           key: "comparePoint",
-           value: function comparePoint(point) {
-             if (this.isAnEndpoint(point)) return 0;
-             var lPt = this.leftSE.point;
-             var rPt = this.rightSE.point;
-             var v = this.vector(); // Exactly vertical segments.
 
-             if (lPt.x === rPt.x) {
-               if (point.x === lPt.x) return 0;
-               return point.x < lPt.x ? 1 : -1;
-             } // Nearly vertical segments with an intersection.
-             // Check to see where a point on the line with matching Y coordinate is.
+         oauth.preauth = function (c) {
+           if (!c) return;
+           if (c.oauth_token) token('oauth_token', c.oauth_token);
+           if (c.oauth_token_secret) token('oauth_token_secret', c.oauth_token_secret);
+           return oauth;
+         };
+
+         oauth.options = function (_) {
+           if (!arguments.length) return o;
+           o = _;
+           o.url = o.url || 'https://www.openstreetmap.org';
+           o.landing = o.landing || 'land.html';
+           o.singlepage = o.singlepage || false; // Optional loading and loading-done functions for nice UI feedback.
+           // by default, no-ops
 
+           o.loading = o.loading || function () {};
 
-             var yDist = (point.y - lPt.y) / v.y;
-             var xFromYDist = lPt.x + yDist * v.x;
-             if (point.x === xFromYDist) return 0; // General case.
-             // Check to see where a point on the line with matching X coordinate is.
+           o.done = o.done || function () {};
 
-             var xDist = (point.x - lPt.x) / v.x;
-             var yFromXDist = lPt.y + xDist * v.y;
-             if (point.y === yFromXDist) return 0;
-             return point.y < yFromXDist ? -1 : 1;
-           }
-           /**
-            * Given another segment, returns the first non-trivial intersection
-            * between the two segments (in terms of sweep line ordering), if it exists.
-            *
-            * A 'non-trivial' intersection is one that will cause one or both of the
-            * segments to be split(). As such, 'trivial' vs. 'non-trivial' intersection:
-            *
-            *   * endpoint of segA with endpoint of segB --> trivial
-            *   * endpoint of segA with point along segB --> non-trivial
-            *   * endpoint of segB with point along segA --> non-trivial
-            *   * point along segA with point along segB --> non-trivial
-            *
-            * If no non-trivial intersection exists, return null
-            * Else, return null.
-            */
+           return oauth.preauth(o);
+         }; // 'stamp' an authentication object from `getAuth()`
+         // with a [nonce](http://en.wikipedia.org/wiki/Cryptographic_nonce)
+         // and timestamp
 
-         }, {
-           key: "getIntersection",
-           value: function getIntersection(other) {
-             // If bboxes don't overlap, there can't be any intersections
-             var tBbox = this.bbox();
-             var oBbox = other.bbox();
-             var bboxOverlap = getBboxOverlap(tBbox, oBbox);
-             if (bboxOverlap === null) return null; // We first check to see if the endpoints can be considered intersections.
-             // This will 'snap' intersections to endpoints if possible, and will
-             // handle cases of colinearity.
 
-             var tlp = this.leftSE.point;
-             var trp = this.rightSE.point;
-             var olp = other.leftSE.point;
-             var orp = other.rightSE.point; // does each endpoint touch the other segment?
-             // note that we restrict the 'touching' definition to only allow segments
-             // to touch endpoints that lie forward from where we are in the sweep line pass
+         function timenonce(o) {
+           o.oauth_timestamp = ohauth_1.timestamp();
+           o.oauth_nonce = ohauth_1.nonce();
+           return o;
+         } // get/set tokens. These are prefixed with the base URL so that `osm-auth`
+         // can be used with multiple APIs and the keys in `localStorage`
+         // will not clash
 
-             var touchesOtherLSE = isInBbox(tBbox, olp) && this.comparePoint(olp) === 0;
-             var touchesThisLSE = isInBbox(oBbox, tlp) && other.comparePoint(tlp) === 0;
-             var touchesOtherRSE = isInBbox(tBbox, orp) && this.comparePoint(orp) === 0;
-             var touchesThisRSE = isInBbox(oBbox, trp) && other.comparePoint(trp) === 0; // do left endpoints match?
 
-             if (touchesThisLSE && touchesOtherLSE) {
-               // these two cases are for colinear segments with matching left
-               // endpoints, and one segment being longer than the other
-               if (touchesThisRSE && !touchesOtherRSE) return trp;
-               if (!touchesThisRSE && touchesOtherRSE) return orp; // either the two segments match exactly (two trival intersections)
-               // or just on their left endpoint (one trivial intersection
+         var token;
 
-               return null;
-             } // does this left endpoint matches (other doesn't)
+         if (store_legacy.enabled) {
+           token = function token(x, y) {
+             if (arguments.length === 1) return store_legacy.get(o.url + x);else if (arguments.length === 2) return store_legacy.set(o.url + x, y);
+           };
+         } else {
+           var storage = {};
 
+           token = function token(x, y) {
+             if (arguments.length === 1) return storage[o.url + x];else if (arguments.length === 2) return storage[o.url + x] = y;
+           };
+         } // Get an authentication object. If you just add and remove properties
+         // from a single object, you'll need to use `delete` to make sure that
+         // it doesn't contain undesired properties for authentication
 
-             if (touchesThisLSE) {
-               // check for segments that just intersect on opposing endpoints
-               if (touchesOtherRSE) {
-                 if (tlp.x === orp.x && tlp.y === orp.y) return null;
-               } // t-intersection on left endpoint
 
+         function getAuth(o) {
+           return {
+             oauth_consumer_key: o.oauth_consumer_key,
+             oauth_signature_method: 'HMAC-SHA1'
+           };
+         } // potentially pre-authorize
 
-               return tlp;
-             } // does other left endpoint matches (this doesn't)
 
+         oauth.options(o);
+         return oauth;
+       };
 
-             if (touchesOtherLSE) {
-               // check for segments that just intersect on opposing endpoints
-               if (touchesThisRSE) {
-                 if (trp.x === olp.x && trp.y === olp.y) return null;
-               } // t-intersection on left endpoint
+       var tiler$2 = utilTiler();
+       var dispatch$2 = dispatch$8('apiStatusChange', 'authLoading', 'authDone', 'change', 'loading', 'loaded', 'loadedNotes');
+       var urlroot = 'https://www.openstreetmap.org';
+       var oauth = osmAuth({
+         url: urlroot,
+         oauth_consumer_key: '5A043yRSEugj4DJ5TljuapfnrflWDte8jTOcWLlT',
+         oauth_secret: 'aB3jKq1TRsCOUrfOIZ6oQMEDmv2ptV76PA54NGLL',
+         loading: authLoading,
+         done: authDone
+       }); // 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 olp;
-             } // trivial intersection on right endpoints
+       var _cachedApiStatus;
 
+       var _changeset = {};
 
-             if (touchesThisRSE && touchesOtherRSE) return null; // t-intersections on just one right endpoint
+       var _deferred = new Set();
 
-             if (touchesThisRSE) return trp;
-             if (touchesOtherRSE) return orp; // None of our endpoints intersect. Look for a general intersection between
-             // infinite lines laid over the segments
+       var _connectionID = 1;
+       var _tileZoom = 16;
+       var _noteZoom = 12;
 
-             var pt = intersection$1(tlp, this.vector(), olp, other.vector()); // are the segments parrallel? Note that if they were colinear with overlap,
-             // they would have an endpoint intersection and that case was already handled above
+       var _rateLimitError;
 
-             if (pt === null) return null; // is the intersection found between the lines not on the segments?
+       var _userChangesets;
 
-             if (!isInBbox(bboxOverlap, pt)) return null; // round the the computed point if needed
+       var _userDetails;
 
-             return rounder.round(pt.x, pt.y);
-           }
-           /**
-            * Split the given segment into multiple segments on the given points.
-            *  * Each existing segment will retain its leftSE and a new rightSE will be
-            *    generated for it.
-            *  * A new segment will be generated which will adopt the original segment's
-            *    rightSE, and a new leftSE will be generated for it.
-            *  * If there are more than two points given to split on, new segments
-            *    in the middle will be generated with new leftSE and rightSE's.
-            *  * An array of the newly generated SweepEvents will be returned.
-            *
-            * Warning: input array of points is modified
-            */
+       var _off; // set a default but also load this from the API status
 
-         }, {
-           key: "split",
-           value: function split(point) {
-             var newEvents = [];
-             var alreadyLinked = point.events !== undefined;
-             var newLeftSE = new SweepEvent$1(point, true);
-             var newRightSE = new SweepEvent$1(point, false);
-             var oldRightSE = this.rightSE;
-             this.replaceRightSE(newRightSE);
-             newEvents.push(newRightSE);
-             newEvents.push(newLeftSE);
-             var newSeg = new Segment(newLeftSE, oldRightSE, this.rings.slice(), this.windings.slice()); // when splitting a nearly vertical downward-facing segment,
-             // sometimes one of the resulting new segments is vertical, in which
-             // case its left and right events may need to be swapped
 
-             if (SweepEvent$1.comparePoints(newSeg.leftSE.point, newSeg.rightSE.point) > 0) {
-               newSeg.swapEvents();
-             }
+       var _maxWayNodes = 2000;
 
-             if (SweepEvent$1.comparePoints(this.leftSE.point, this.rightSE.point) > 0) {
-               this.swapEvents();
-             } // in the point we just used to create new sweep events with was already
-             // linked to other events, we need to check if either of the affected
-             // segments should be consumed
+       function authLoading() {
+         dispatch$2.call('authLoading');
+       }
 
+       function authDone() {
+         dispatch$2.call('authDone');
+       }
 
-             if (alreadyLinked) {
-               newLeftSE.checkForConsuming();
-               newRightSE.checkForConsuming();
-             }
+       function abortRequest$2(controllerOrXHR) {
+         if (controllerOrXHR) {
+           controllerOrXHR.abort();
+         }
+       }
 
-             return newEvents;
-           }
-           /* Swap which event is left and right */
+       function hasInflightRequests(cache) {
+         return Object.keys(cache.inflight).length;
+       }
 
-         }, {
-           key: "swapEvents",
-           value: function swapEvents() {
-             var tmpEvt = this.rightSE;
-             this.rightSE = this.leftSE;
-             this.leftSE = tmpEvt;
-             this.leftSE.isLeft = true;
-             this.rightSE.isLeft = false;
+       function abortUnwantedRequests(cache, visibleTiles) {
+         Object.keys(cache.inflight).forEach(function (k) {
+           if (cache.toLoad[k]) return;
+           if (visibleTiles.find(function (tile) {
+             return k === tile.id;
+           })) return;
+           abortRequest$2(cache.inflight[k]);
+           delete cache.inflight[k];
+         });
+       }
 
-             for (var i = 0, iMax = this.windings.length; i < iMax; i++) {
-               this.windings[i] *= -1;
-             }
-           }
-           /* Consume another segment. We take their rings under our wing
-            * and mark them as consumed. Use for perfectly overlapping segments */
+       function getLoc(attrs) {
+         var lon = attrs.lon && attrs.lon.value;
+         var lat = attrs.lat && attrs.lat.value;
+         return [parseFloat(lon), parseFloat(lat)];
+       }
 
-         }, {
-           key: "consume",
-           value: function consume(other) {
-             var consumer = this;
-             var consumee = other;
+       function getNodes(obj) {
+         var elems = obj.getElementsByTagName('nd');
+         var nodes = new Array(elems.length);
 
-             while (consumer.consumedBy) {
-               consumer = consumer.consumedBy;
-             }
+         for (var i = 0, l = elems.length; i < l; i++) {
+           nodes[i] = 'n' + elems[i].attributes.ref.value;
+         }
 
-             while (consumee.consumedBy) {
-               consumee = consumee.consumedBy;
-             }
+         return nodes;
+       }
 
-             var cmp = Segment.compare(consumer, consumee);
-             if (cmp === 0) return; // already consumed
-             // the winner of the consumption is the earlier segment
-             // according to sweep line ordering
+       function getNodesJSON(obj) {
+         var elems = obj.nodes;
+         var nodes = new Array(elems.length);
 
-             if (cmp > 0) {
-               var tmp = consumer;
-               consumer = consumee;
-               consumee = tmp;
-             } // make sure a segment doesn't consume it's prev
+         for (var i = 0, l = elems.length; i < l; i++) {
+           nodes[i] = 'n' + elems[i];
+         }
 
+         return nodes;
+       }
 
-             if (consumer.prev === consumee) {
-               var _tmp = consumer;
-               consumer = consumee;
-               consumee = _tmp;
-             }
+       function getTags(obj) {
+         var elems = obj.getElementsByTagName('tag');
+         var tags = {};
 
-             for (var i = 0, iMax = consumee.rings.length; i < iMax; i++) {
-               var ring = consumee.rings[i];
-               var winding = consumee.windings[i];
-               var index = consumer.rings.indexOf(ring);
+         for (var i = 0, l = elems.length; i < l; i++) {
+           var attrs = elems[i].attributes;
+           tags[attrs.k.value] = attrs.v.value;
+         }
 
-               if (index === -1) {
-                 consumer.rings.push(ring);
-                 consumer.windings.push(winding);
-               } else consumer.windings[index] += winding;
-             }
+         return tags;
+       }
 
-             consumee.rings = null;
-             consumee.windings = null;
-             consumee.consumedBy = consumer; // mark sweep events consumed as to maintain ordering in sweep event queue
+       function getMembers(obj) {
+         var elems = obj.getElementsByTagName('member');
+         var members = new Array(elems.length);
 
-             consumee.leftSE.consumedBy = consumer.leftSE;
-             consumee.rightSE.consumedBy = consumer.rightSE;
-           }
-           /* The first segment previous segment chain that is in the result */
+         for (var i = 0, l = elems.length; i < l; i++) {
+           var attrs = elems[i].attributes;
+           members[i] = {
+             id: attrs.type.value[0] + attrs.ref.value,
+             type: attrs.type.value,
+             role: attrs.role.value
+           };
+         }
 
-         }, {
-           key: "prevInResult",
-           value: function prevInResult() {
-             if (this._prevInResult !== undefined) return this._prevInResult;
-             if (!this.prev) this._prevInResult = null;else if (this.prev.isInResult()) this._prevInResult = this.prev;else this._prevInResult = this.prev.prevInResult();
-             return this._prevInResult;
-           }
-         }, {
-           key: "beforeState",
-           value: function beforeState() {
-             if (this._beforeState !== undefined) return this._beforeState;
-             if (!this.prev) this._beforeState = {
-               rings: [],
-               windings: [],
-               multiPolys: []
-             };else {
-               var seg = this.prev.consumedBy || this.prev;
-               this._beforeState = seg.afterState();
-             }
-             return this._beforeState;
-           }
-         }, {
-           key: "afterState",
-           value: function afterState() {
-             if (this._afterState !== undefined) return this._afterState;
-             var beforeState = this.beforeState();
-             this._afterState = {
-               rings: beforeState.rings.slice(0),
-               windings: beforeState.windings.slice(0),
-               multiPolys: []
-             };
-             var ringsAfter = this._afterState.rings;
-             var windingsAfter = this._afterState.windings;
-             var mpsAfter = this._afterState.multiPolys; // calculate ringsAfter, windingsAfter
+         return members;
+       }
 
-             for (var i = 0, iMax = this.rings.length; i < iMax; i++) {
-               var ring = this.rings[i];
-               var winding = this.windings[i];
-               var index = ringsAfter.indexOf(ring);
+       function getMembersJSON(obj) {
+         var elems = obj.members;
+         var members = new Array(elems.length);
 
-               if (index === -1) {
-                 ringsAfter.push(ring);
-                 windingsAfter.push(winding);
-               } else windingsAfter[index] += winding;
-             } // calcualte polysAfter
+         for (var i = 0, l = elems.length; i < l; i++) {
+           var attrs = elems[i];
+           members[i] = {
+             id: attrs.type[0] + attrs.ref,
+             type: attrs.type,
+             role: attrs.role
+           };
+         }
 
+         return members;
+       }
 
-             var polysAfter = [];
-             var polysExclude = [];
+       function getVisible(attrs) {
+         return !attrs.visible || attrs.visible.value !== 'false';
+       }
 
-             for (var _i = 0, _iMax = ringsAfter.length; _i < _iMax; _i++) {
-               if (windingsAfter[_i] === 0) continue; // non-zero rule
+       function parseComments(comments) {
+         var parsedComments = []; // for each comment
 
-               var _ring = ringsAfter[_i];
-               var poly = _ring.poly;
-               if (polysExclude.indexOf(poly) !== -1) continue;
-               if (_ring.isExterior) polysAfter.push(poly);else {
-                 if (polysExclude.indexOf(poly) === -1) polysExclude.push(poly);
+         for (var i = 0; i < comments.length; i++) {
+           var comment = comments[i];
 
-                 var _index = polysAfter.indexOf(_ring.poly);
+           if (comment.nodeName === 'comment') {
+             var childNodes = comment.childNodes;
+             var parsedComment = {};
 
-                 if (_index !== -1) polysAfter.splice(_index, 1);
-               }
-             } // calculate multiPolysAfter
+             for (var j = 0; j < childNodes.length; j++) {
+               var node = childNodes[j];
+               var nodeName = node.nodeName;
+               if (nodeName === '#text') continue;
+               parsedComment[nodeName] = node.textContent;
 
+               if (nodeName === 'uid') {
+                 var uid = node.textContent;
 
-             for (var _i2 = 0, _iMax2 = polysAfter.length; _i2 < _iMax2; _i2++) {
-               var mp = polysAfter[_i2].multiPoly;
-               if (mpsAfter.indexOf(mp) === -1) mpsAfter.push(mp);
+                 if (uid && !_userCache.user[uid]) {
+                   _userCache.toLoad[uid] = true;
+                 }
+               }
              }
 
-             return this._afterState;
+             if (parsedComment) {
+               parsedComments.push(parsedComment);
+             }
            }
-           /* Is this segment part of the final result? */
-
-         }, {
-           key: "isInResult",
-           value: function isInResult() {
-             // if we've been consumed, we're not in the result
-             if (this.consumedBy) return false;
-             if (this._isInResult !== undefined) return this._isInResult;
-             var mpsBefore = this.beforeState().multiPolys;
-             var mpsAfter = this.afterState().multiPolys;
-
-             switch (operation.type) {
-               case 'union':
-                 {
-                   // UNION - included iff:
-                   //  * On one side of us there is 0 poly interiors AND
-                   //  * On the other side there is 1 or more.
-                   var noBefores = mpsBefore.length === 0;
-                   var noAfters = mpsAfter.length === 0;
-                   this._isInResult = noBefores !== noAfters;
-                   break;
-                 }
+         }
 
-               case 'intersection':
-                 {
-                   // INTERSECTION - included iff:
-                   //  * on one side of us all multipolys are rep. with poly interiors AND
-                   //  * on the other side of us, not all multipolys are repsented
-                   //    with poly interiors
-                   var least;
-                   var most;
+         return parsedComments;
+       }
 
-                   if (mpsBefore.length < mpsAfter.length) {
-                     least = mpsBefore.length;
-                     most = mpsAfter.length;
-                   } else {
-                     least = mpsAfter.length;
-                     most = mpsBefore.length;
-                   }
+       function encodeNoteRtree(note) {
+         return {
+           minX: note.loc[0],
+           minY: note.loc[1],
+           maxX: note.loc[0],
+           maxY: note.loc[1],
+           data: note
+         };
+       }
 
-                   this._isInResult = most === operation.numMultiPolys && least < most;
-                   break;
-                 }
+       var jsonparsers = {
+         node: function nodeData(obj, uid) {
+           return new osmNode({
+             id: uid,
+             visible: typeof obj.visible === 'boolean' ? obj.visible : true,
+             version: obj.version && obj.version.toString(),
+             changeset: obj.changeset && obj.changeset.toString(),
+             timestamp: obj.timestamp,
+             user: obj.user,
+             uid: obj.uid && obj.uid.toString(),
+             loc: [parseFloat(obj.lon), parseFloat(obj.lat)],
+             tags: obj.tags
+           });
+         },
+         way: function wayData(obj, uid) {
+           return new osmWay({
+             id: uid,
+             visible: typeof obj.visible === 'boolean' ? obj.visible : true,
+             version: obj.version && obj.version.toString(),
+             changeset: obj.changeset && obj.changeset.toString(),
+             timestamp: obj.timestamp,
+             user: obj.user,
+             uid: obj.uid && obj.uid.toString(),
+             tags: obj.tags,
+             nodes: getNodesJSON(obj)
+           });
+         },
+         relation: function relationData(obj, uid) {
+           return new osmRelation({
+             id: uid,
+             visible: typeof obj.visible === 'boolean' ? obj.visible : true,
+             version: obj.version && obj.version.toString(),
+             changeset: obj.changeset && obj.changeset.toString(),
+             timestamp: obj.timestamp,
+             user: obj.user,
+             uid: obj.uid && obj.uid.toString(),
+             tags: obj.tags,
+             members: getMembersJSON(obj)
+           });
+         },
+         user: function parseUser(obj, uid) {
+           return {
+             id: uid,
+             display_name: obj.display_name,
+             account_created: obj.account_created,
+             image_url: obj.img && obj.img.href,
+             changesets_count: obj.changesets && obj.changesets.count && obj.changesets.count.toString() || '0',
+             active_blocks: obj.blocks && obj.blocks.received && obj.blocks.received.active && obj.blocks.received.active.toString() || '0'
+           };
+         }
+       };
 
-               case 'xor':
-                 {
-                   // XOR - included iff:
-                   //  * the difference between the number of multipolys represented
-                   //    with poly interiors on our two sides is an odd number
-                   var diff = Math.abs(mpsBefore.length - mpsAfter.length);
-                   this._isInResult = diff % 2 === 1;
-                   break;
-                 }
+       function parseJSON(payload, callback, options) {
+         options = Object.assign({
+           skipSeen: true
+         }, options);
 
-               case 'difference':
-                 {
-                   // DIFFERENCE included iff:
-                   //  * on exactly one side, we have just the subject
-                   var isJustSubject = function isJustSubject(mps) {
-                     return mps.length === 1 && mps[0].isSubject;
-                   };
+         if (!payload) {
+           return callback({
+             message: 'No JSON',
+             status: -1
+           });
+         }
 
-                   this._isInResult = isJustSubject(mpsBefore) !== isJustSubject(mpsAfter);
-                   break;
-                 }
+         var json = payload;
+         if (_typeof(json) !== 'object') json = JSON.parse(payload);
+         if (!json.elements) return callback({
+           message: 'No JSON',
+           status: -1
+         });
+         var children = json.elements;
+         var handle = window.requestIdleCallback(function () {
+           _deferred["delete"](handle);
 
-               default:
-                 throw new Error("Unrecognized operation type found ".concat(operation.type));
-             }
+           var results = [];
+           var result;
 
-             return this._isInResult;
+           for (var i = 0; i < children.length; i++) {
+             result = parseChild(children[i]);
+             if (result) results.push(result);
            }
-         }], [{
-           key: "fromRing",
-           value: function fromRing(pt1, pt2, ring) {
-             var leftPt, rightPt, winding; // ordering the two points according to sweep line ordering
-
-             var cmpPts = SweepEvent$1.comparePoints(pt1, pt2);
 
-             if (cmpPts < 0) {
-               leftPt = pt1;
-               rightPt = pt2;
-               winding = 1;
-             } else if (cmpPts > 0) {
-               leftPt = pt2;
-               rightPt = pt1;
-               winding = -1;
-             } else throw new Error("Tried to create degenerate segment at [".concat(pt1.x, ", ").concat(pt1.y, "]"));
+           callback(null, results);
+         });
 
-             var leftSE = new SweepEvent$1(leftPt, true);
-             var rightSE = new SweepEvent$1(rightPt, false);
-             return new Segment(leftSE, rightSE, [ring], [winding]);
-           }
-         }]);
+         _deferred.add(handle);
 
-         return Segment;
-       }();
+         function parseChild(child) {
+           var parser = jsonparsers[child.type];
+           if (!parser) return null;
+           var uid;
+           uid = osmEntity.id.fromOSM(child.type, child.id);
 
-       var RingIn = /*#__PURE__*/function () {
-         function RingIn(geomRing, poly, isExterior) {
-           _classCallCheck$1(this, RingIn);
+           if (options.skipSeen) {
+             if (_tileCache.seen[uid]) return null; // avoid reparsing a "seen" entity
 
-           if (!Array.isArray(geomRing) || geomRing.length === 0) {
-             throw new Error('Input geometry is not a valid Polygon or MultiPolygon');
+             _tileCache.seen[uid] = true;
            }
 
-           this.poly = poly;
-           this.isExterior = isExterior;
-           this.segments = [];
+           return parser(child, uid);
+         }
+       }
 
-           if (typeof geomRing[0][0] !== 'number' || typeof geomRing[0][1] !== 'number') {
-             throw new Error('Input geometry is not a valid Polygon or MultiPolygon');
-           }
+       function parseUserJSON(payload, callback, options) {
+         options = Object.assign({
+           skipSeen: true
+         }, options);
 
-           var firstPoint = rounder.round(geomRing[0][0], geomRing[0][1]);
-           this.bbox = {
-             ll: {
-               x: firstPoint.x,
-               y: firstPoint.y
-             },
-             ur: {
-               x: firstPoint.x,
-               y: firstPoint.y
-             }
-           };
-           var prevPoint = firstPoint;
+         if (!payload) {
+           return callback({
+             message: 'No JSON',
+             status: -1
+           });
+         }
 
-           for (var i = 1, iMax = geomRing.length; i < iMax; i++) {
-             if (typeof geomRing[i][0] !== 'number' || typeof geomRing[i][1] !== 'number') {
-               throw new Error('Input geometry is not a valid Polygon or MultiPolygon');
-             }
+         var json = payload;
+         if (_typeof(json) !== 'object') json = JSON.parse(payload);
+         if (!json.users && !json.user) return callback({
+           message: 'No JSON',
+           status: -1
+         });
+         var objs = json.users || [json];
+         var handle = window.requestIdleCallback(function () {
+           _deferred["delete"](handle);
 
-             var point = rounder.round(geomRing[i][0], geomRing[i][1]); // skip repeated points
+           var results = [];
+           var result;
 
-             if (point.x === prevPoint.x && point.y === prevPoint.y) continue;
-             this.segments.push(Segment.fromRing(prevPoint, point, this));
-             if (point.x < this.bbox.ll.x) this.bbox.ll.x = point.x;
-             if (point.y < this.bbox.ll.y) this.bbox.ll.y = point.y;
-             if (point.x > this.bbox.ur.x) this.bbox.ur.x = point.x;
-             if (point.y > this.bbox.ur.y) this.bbox.ur.y = point.y;
-             prevPoint = point;
-           } // add segment from last to first if last is not the same as first
+           for (var i = 0; i < objs.length; i++) {
+             result = parseObj(objs[i]);
+             if (result) results.push(result);
+           }
+
+           callback(null, results);
+         });
 
+         _deferred.add(handle);
 
-           if (firstPoint.x !== prevPoint.x || firstPoint.y !== prevPoint.y) {
-             this.segments.push(Segment.fromRing(prevPoint, firstPoint, this));
+         function parseObj(obj) {
+           var uid = obj.user.id && obj.user.id.toString();
+
+           if (options.skipSeen && _userCache.user[uid]) {
+             delete _userCache.toLoad[uid];
+             return null;
            }
+
+           var user = jsonparsers.user(obj.user, uid);
+           _userCache.user[uid] = user;
+           delete _userCache.toLoad[uid];
+           return user;
          }
+       }
 
-         _createClass$1(RingIn, [{
-           key: "getSweepEvents",
-           value: function getSweepEvents() {
-             var sweepEvents = [];
+       var parsers = {
+         node: function nodeData(obj, uid) {
+           var attrs = obj.attributes;
+           return new osmNode({
+             id: uid,
+             visible: getVisible(attrs),
+             version: attrs.version.value,
+             changeset: attrs.changeset && attrs.changeset.value,
+             timestamp: attrs.timestamp && attrs.timestamp.value,
+             user: attrs.user && attrs.user.value,
+             uid: attrs.uid && attrs.uid.value,
+             loc: getLoc(attrs),
+             tags: getTags(obj)
+           });
+         },
+         way: function wayData(obj, uid) {
+           var attrs = obj.attributes;
+           return new osmWay({
+             id: uid,
+             visible: getVisible(attrs),
+             version: attrs.version.value,
+             changeset: attrs.changeset && attrs.changeset.value,
+             timestamp: attrs.timestamp && attrs.timestamp.value,
+             user: attrs.user && attrs.user.value,
+             uid: attrs.uid && attrs.uid.value,
+             tags: getTags(obj),
+             nodes: getNodes(obj)
+           });
+         },
+         relation: function relationData(obj, uid) {
+           var attrs = obj.attributes;
+           return new osmRelation({
+             id: uid,
+             visible: getVisible(attrs),
+             version: attrs.version.value,
+             changeset: attrs.changeset && attrs.changeset.value,
+             timestamp: attrs.timestamp && attrs.timestamp.value,
+             user: attrs.user && attrs.user.value,
+             uid: attrs.uid && attrs.uid.value,
+             tags: getTags(obj),
+             members: getMembers(obj)
+           });
+         },
+         note: function parseNote(obj, uid) {
+           var attrs = obj.attributes;
+           var childNodes = obj.childNodes;
+           var props = {};
+           props.id = uid;
+           props.loc = getLoc(attrs); // if notes are coincident, move them apart slightly
 
-             for (var i = 0, iMax = this.segments.length; i < iMax; i++) {
-               var segment = this.segments[i];
-               sweepEvents.push(segment.leftSE);
-               sweepEvents.push(segment.rightSE);
+           var coincident = false;
+           var epsilon = 0.00001;
+
+           do {
+             if (coincident) {
+               props.loc = geoVecAdd(props.loc, [epsilon, epsilon]);
              }
 
-             return sweepEvents;
-           }
-         }]);
+             var bbox = geoExtent(props.loc).bbox();
+             coincident = _noteCache.rtree.search(bbox).length;
+           } while (coincident); // parse note contents
 
-         return RingIn;
-       }();
 
-       var PolyIn = /*#__PURE__*/function () {
-         function PolyIn(geomPoly, multiPoly) {
-           _classCallCheck$1(this, PolyIn);
+           for (var i = 0; i < childNodes.length; i++) {
+             var node = childNodes[i];
+             var nodeName = node.nodeName;
+             if (nodeName === '#text') continue; // if the element is comments, parse the comments
 
-           if (!Array.isArray(geomPoly)) {
-             throw new Error('Input geometry is not a valid Polygon or MultiPolygon');
+             if (nodeName === 'comments') {
+               props[nodeName] = parseComments(node.childNodes);
+             } else {
+               props[nodeName] = node.textContent;
+             }
            }
 
-           this.exteriorRing = new RingIn(geomPoly[0], this, true); // copy by value
+           var note = new osmNote(props);
+           var item = encodeNoteRtree(note);
+           _noteCache.note[note.id] = note;
 
-           this.bbox = {
-             ll: {
-               x: this.exteriorRing.bbox.ll.x,
-               y: this.exteriorRing.bbox.ll.y
-             },
-             ur: {
-               x: this.exteriorRing.bbox.ur.x,
-               y: this.exteriorRing.bbox.ur.y
-             }
+           _noteCache.rtree.insert(item);
+
+           return note;
+         },
+         user: function parseUser(obj, uid) {
+           var attrs = obj.attributes;
+           var user = {
+             id: uid,
+             display_name: attrs.display_name && attrs.display_name.value,
+             account_created: attrs.account_created && attrs.account_created.value,
+             changesets_count: '0',
+             active_blocks: '0'
            };
-           this.interiorRings = [];
+           var img = obj.getElementsByTagName('img');
 
-           for (var i = 1, iMax = geomPoly.length; i < iMax; i++) {
-             var ring = new RingIn(geomPoly[i], this, false);
-             if (ring.bbox.ll.x < this.bbox.ll.x) this.bbox.ll.x = ring.bbox.ll.x;
-             if (ring.bbox.ll.y < this.bbox.ll.y) this.bbox.ll.y = ring.bbox.ll.y;
-             if (ring.bbox.ur.x > this.bbox.ur.x) this.bbox.ur.x = ring.bbox.ur.x;
-             if (ring.bbox.ur.y > this.bbox.ur.y) this.bbox.ur.y = ring.bbox.ur.y;
-             this.interiorRings.push(ring);
+           if (img && img[0] && img[0].getAttribute('href')) {
+             user.image_url = img[0].getAttribute('href');
            }
 
-           this.multiPoly = multiPoly;
-         }
+           var changesets = obj.getElementsByTagName('changesets');
 
-         _createClass$1(PolyIn, [{
-           key: "getSweepEvents",
-           value: function getSweepEvents() {
-             var sweepEvents = this.exteriorRing.getSweepEvents();
+           if (changesets && changesets[0] && changesets[0].getAttribute('count')) {
+             user.changesets_count = changesets[0].getAttribute('count');
+           }
 
-             for (var i = 0, iMax = this.interiorRings.length; i < iMax; i++) {
-               var ringSweepEvents = this.interiorRings[i].getSweepEvents();
+           var blocks = obj.getElementsByTagName('blocks');
 
-               for (var j = 0, jMax = ringSweepEvents.length; j < jMax; j++) {
-                 sweepEvents.push(ringSweepEvents[j]);
-               }
-             }
+           if (blocks && blocks[0]) {
+             var received = blocks[0].getElementsByTagName('received');
 
-             return sweepEvents;
+             if (received && received[0] && received[0].getAttribute('active')) {
+               user.active_blocks = received[0].getAttribute('active');
+             }
            }
-         }]);
 
-         return PolyIn;
-       }();
+           _userCache.user[uid] = user;
+           delete _userCache.toLoad[uid];
+           return user;
+         }
+       };
 
-       var MultiPolyIn = /*#__PURE__*/function () {
-         function MultiPolyIn(geom, isSubject) {
-           _classCallCheck$1(this, MultiPolyIn);
+       function parseXML(xml, callback, options) {
+         options = Object.assign({
+           skipSeen: true
+         }, options);
 
-           if (!Array.isArray(geom)) {
-             throw new Error('Input geometry is not a valid Polygon or MultiPolygon');
-           }
+         if (!xml || !xml.childNodes) {
+           return callback({
+             message: 'No XML',
+             status: -1
+           });
+         }
 
-           try {
-             // if the input looks like a polygon, convert it to a multipolygon
-             if (typeof geom[0][0][0] === 'number') geom = [geom];
-           } catch (ex) {// The input is either malformed or has empty arrays.
-             // In either case, it will be handled later on.
-           }
+         var root = xml.childNodes[0];
+         var children = root.childNodes;
+         var handle = window.requestIdleCallback(function () {
+           _deferred["delete"](handle);
 
-           this.polys = [];
-           this.bbox = {
-             ll: {
-               x: Number.POSITIVE_INFINITY,
-               y: Number.POSITIVE_INFINITY
-             },
-             ur: {
-               x: Number.NEGATIVE_INFINITY,
-               y: Number.NEGATIVE_INFINITY
-             }
-           };
+           var results = [];
+           var result;
 
-           for (var i = 0, iMax = geom.length; i < iMax; i++) {
-             var poly = new PolyIn(geom[i], this);
-             if (poly.bbox.ll.x < this.bbox.ll.x) this.bbox.ll.x = poly.bbox.ll.x;
-             if (poly.bbox.ll.y < this.bbox.ll.y) this.bbox.ll.y = poly.bbox.ll.y;
-             if (poly.bbox.ur.x > this.bbox.ur.x) this.bbox.ur.x = poly.bbox.ur.x;
-             if (poly.bbox.ur.y > this.bbox.ur.y) this.bbox.ur.y = poly.bbox.ur.y;
-             this.polys.push(poly);
+           for (var i = 0; i < children.length; i++) {
+             result = parseChild(children[i]);
+             if (result) results.push(result);
            }
 
-           this.isSubject = isSubject;
-         }
+           callback(null, results);
+         });
 
-         _createClass$1(MultiPolyIn, [{
-           key: "getSweepEvents",
-           value: function getSweepEvents() {
-             var sweepEvents = [];
+         _deferred.add(handle);
 
-             for (var i = 0, iMax = this.polys.length; i < iMax; i++) {
-               var polySweepEvents = this.polys[i].getSweepEvents();
+         function parseChild(child) {
+           var parser = parsers[child.nodeName];
+           if (!parser) return null;
+           var uid;
 
-               for (var j = 0, jMax = polySweepEvents.length; j < jMax; j++) {
-                 sweepEvents.push(polySweepEvents[j]);
-               }
+           if (child.nodeName === 'user') {
+             uid = child.attributes.id.value;
+
+             if (options.skipSeen && _userCache.user[uid]) {
+               delete _userCache.toLoad[uid];
+               return null;
              }
+           } else if (child.nodeName === 'note') {
+             uid = child.getElementsByTagName('id')[0].textContent;
+           } else {
+             uid = osmEntity.id.fromOSM(child.nodeName, child.attributes.id.value);
 
-             return sweepEvents;
-           }
-         }]);
+             if (options.skipSeen) {
+               if (_tileCache.seen[uid]) return null; // avoid reparsing a "seen" entity
 
-         return MultiPolyIn;
-       }();
+               _tileCache.seen[uid] = true;
+             }
+           }
 
-       var RingOut = /*#__PURE__*/function () {
-         _createClass$1(RingOut, null, [{
-           key: "factory",
+           return parser(child, uid);
+         }
+       } // replace or remove note from rtree
 
-           /* Given the segments from the sweep line pass, compute & return a series
-            * of closed rings from all the segments marked to be part of the result */
-           value: function factory(allSegments) {
-             var ringsOut = [];
 
-             for (var i = 0, iMax = allSegments.length; i < iMax; i++) {
-               var segment = allSegments[i];
-               if (!segment.isInResult() || segment.ringOut) continue;
-               var prevEvent = null;
-               var event = segment.leftSE;
-               var nextEvent = segment.rightSE;
-               var events = [event];
-               var startingPoint = event.point;
-               var intersectionLEs = [];
-               /* Walk the chain of linked events to form a closed ring */
+       function updateRtree(item, replace) {
+         _noteCache.rtree.remove(item, function isEql(a, b) {
+           return a.data.id === b.data.id;
+         });
 
-               while (true) {
-                 prevEvent = event;
-                 event = nextEvent;
-                 events.push(event);
-                 /* Is the ring complete? */
+         if (replace) {
+           _noteCache.rtree.insert(item);
+         }
+       }
 
-                 if (event.point === startingPoint) break;
+       function wrapcb(thisArg, callback, cid) {
+         return function (err, result) {
+           if (err) {
+             // 400 Bad Request, 401 Unauthorized, 403 Forbidden..
+             if (err.status === 400 || err.status === 401 || err.status === 403) {
+               thisArg.logout();
+             }
 
-                 while (true) {
-                   var availableLEs = event.getAvailableLinkedEvents();
-                   /* Did we hit a dead end? This shouldn't happen. Indicates some earlier
-                    * part of the algorithm malfunctioned... please file a bug report. */
+             return callback.call(thisArg, err);
+           } else if (thisArg.getConnectionId() !== cid) {
+             return callback.call(thisArg, {
+               message: 'Connection Switched',
+               status: -1
+             });
+           } else {
+             return callback.call(thisArg, err, result);
+           }
+         };
+       }
 
-                   if (availableLEs.length === 0) {
-                     var firstPt = events[0].point;
-                     var lastPt = events[events.length - 1].point;
-                     throw new Error("Unable to complete output ring starting at [".concat(firstPt.x, ",") + " ".concat(firstPt.y, "]. Last matching segment found ends at") + " [".concat(lastPt.x, ", ").concat(lastPt.y, "]."));
-                   }
-                   /* Only one way to go, so cotinue on the path */
+       var serviceOsm = {
+         init: function init() {
+           utilRebind(this, dispatch$2, 'on');
+         },
+         reset: function reset() {
+           Array.from(_deferred).forEach(function (handle) {
+             window.cancelIdleCallback(handle);
 
+             _deferred["delete"](handle);
+           });
+           _connectionID++;
+           _userChangesets = undefined;
+           _userDetails = undefined;
+           _rateLimitError = undefined;
+           Object.values(_tileCache.inflight).forEach(abortRequest$2);
+           Object.values(_noteCache.inflight).forEach(abortRequest$2);
+           Object.values(_noteCache.inflightPost).forEach(abortRequest$2);
+           if (_changeset.inflight) abortRequest$2(_changeset.inflight);
+           _tileCache = {
+             toLoad: {},
+             loaded: {},
+             inflight: {},
+             seen: {},
+             rtree: new RBush()
+           };
+           _noteCache = {
+             toLoad: {},
+             loaded: {},
+             inflight: {},
+             inflightPost: {},
+             note: {},
+             closed: {},
+             rtree: new RBush()
+           };
+           _userCache = {
+             toLoad: {},
+             user: {}
+           };
+           _cachedApiStatus = undefined;
+           _changeset = {};
+           return this;
+         },
+         getConnectionId: function getConnectionId() {
+           return _connectionID;
+         },
+         changesetURL: function changesetURL(changesetID) {
+           return urlroot + '/changeset/' + changesetID;
+         },
+         changesetsURL: function changesetsURL(center, zoom) {
+           var precision = Math.max(0, Math.ceil(Math.log(zoom) / Math.LN2));
+           return urlroot + '/history#map=' + Math.floor(zoom) + '/' + center[1].toFixed(precision) + '/' + center[0].toFixed(precision);
+         },
+         entityURL: function entityURL(entity) {
+           return urlroot + '/' + entity.type + '/' + entity.osmId();
+         },
+         historyURL: function historyURL(entity) {
+           return urlroot + '/' + entity.type + '/' + entity.osmId() + '/history';
+         },
+         userURL: function userURL(username) {
+           return urlroot + '/user/' + username;
+         },
+         noteURL: function noteURL(note) {
+           return urlroot + '/note/' + note.id;
+         },
+         noteReportURL: function noteReportURL(note) {
+           return urlroot + '/reports/new?reportable_type=Note&reportable_id=' + note.id;
+         },
+         // Generic method to load data from the OSM API
+         // Can handle either auth or unauth calls.
+         loadFromAPI: function loadFromAPI(path, callback, options) {
+           options = Object.assign({
+             skipSeen: true
+           }, options);
+           var that = this;
+           var cid = _connectionID;
 
-                   if (availableLEs.length === 1) {
-                     nextEvent = availableLEs[0].otherSE;
-                     break;
-                   }
-                   /* We must have an intersection. Check for a completed loop */
+           function done(err, payload) {
+             if (that.getConnectionId() !== cid) {
+               if (callback) callback({
+                 message: 'Connection Switched',
+                 status: -1
+               });
+               return;
+             }
 
+             var isAuthenticated = that.authenticated(); // 400 Bad Request, 401 Unauthorized, 403 Forbidden
+             // Logout and retry the request..
 
-                   var indexLE = null;
+             if (isAuthenticated && err && err.status && (err.status === 400 || err.status === 401 || err.status === 403)) {
+               that.logout();
+               that.loadFromAPI(path, callback, options); // else, no retry..
+             } else {
+               // 509 Bandwidth Limit Exceeded, 429 Too Many Requests
+               // Set the rateLimitError flag and trigger a warning..
+               if (!isAuthenticated && !_rateLimitError && err && err.status && (err.status === 509 || err.status === 429)) {
+                 _rateLimitError = err;
+                 dispatch$2.call('change');
+                 that.reloadApiStatus();
+               } else if (err && _cachedApiStatus === 'online' || !err && _cachedApiStatus !== 'online') {
+                 // If the response's error state doesn't match the status,
+                 // it's likely we lost or gained the connection so reload the status
+                 that.reloadApiStatus();
+               }
 
-                   for (var j = 0, jMax = intersectionLEs.length; j < jMax; j++) {
-                     if (intersectionLEs[j].point === event.point) {
-                       indexLE = j;
-                       break;
-                     }
+               if (callback) {
+                 if (err) {
+                   return callback(err);
+                 } else {
+                   if (path.indexOf('.json') !== -1) {
+                     return parseJSON(payload, callback, options);
+                   } else {
+                     return parseXML(payload, callback, options);
                    }
-                   /* Found a completed loop. Cut that off and make a ring */
+                 }
+               }
+             }
+           }
 
+           if (this.authenticated()) {
+             return oauth.xhr({
+               method: 'GET',
+               path: path
+             }, done);
+           } else {
+             var url = urlroot + path;
+             var controller = new AbortController();
+             var fn;
 
-                   if (indexLE !== null) {
-                     var intersectionLE = intersectionLEs.splice(indexLE)[0];
-                     var ringEvents = events.splice(intersectionLE.index);
-                     ringEvents.unshift(ringEvents[0].otherSE);
-                     ringsOut.push(new RingOut(ringEvents.reverse()));
-                     continue;
-                   }
-                   /* register the intersection */
+             if (path.indexOf('.json') !== -1) {
+               fn = d3_json;
+             } else {
+               fn = d3_xml;
+             }
 
+             fn(url, {
+               signal: controller.signal
+             }).then(function (data) {
+               done(null, data);
+             })["catch"](function (err) {
+               if (err.name === 'AbortError') return; // d3-fetch includes status in the error message,
+               // but we can't access the response itself
+               // https://github.com/d3/d3-fetch/issues/27
 
-                   intersectionLEs.push({
-                     index: events.length,
-                     point: event.point
-                   });
-                   /* Choose the left-most option to continue the walk */
+               var match = err.message.match(/^\d{3}/);
 
-                   var comparator = event.getLeftmostComparator(prevEvent);
-                   nextEvent = availableLEs.sort(comparator)[0].otherSE;
-                   break;
-                 }
+               if (match) {
+                 done({
+                   status: +match[0],
+                   statusText: err.message
+                 });
+               } else {
+                 done(err.message);
                }
-
-               ringsOut.push(new RingOut(events));
-             }
-
-             return ringsOut;
+             });
+             return controller;
            }
-         }]);
-
-         function RingOut(events) {
-           _classCallCheck$1(this, RingOut);
+         },
+         // Load a single entity by id (ways and relations use the `/full` call to include
+         // nodes and members). Parent relations are not included, see `loadEntityRelations`.
+         // GET /api/0.6/node/#id
+         // GET /api/0.6/[way|relation]/#id/full
+         loadEntity: function loadEntity(id, callback) {
+           var type = osmEntity.id.type(id);
+           var osmID = osmEntity.id.toOSM(id);
+           var options = {
+             skipSeen: false
+           };
+           this.loadFromAPI('/api/0.6/' + type + '/' + osmID + (type !== 'node' ? '/full' : '') + '.json', function (err, entities) {
+             if (callback) callback(err, {
+               data: entities
+             });
+           }, options);
+         },
+         // Load a single entity with a specific version
+         // GET /api/0.6/[node|way|relation]/#id/#version
+         loadEntityVersion: function loadEntityVersion(id, version, callback) {
+           var type = osmEntity.id.type(id);
+           var osmID = osmEntity.id.toOSM(id);
+           var options = {
+             skipSeen: false
+           };
+           this.loadFromAPI('/api/0.6/' + type + '/' + osmID + '/' + version + '.json', function (err, entities) {
+             if (callback) callback(err, {
+               data: entities
+             });
+           }, options);
+         },
+         // Load the relations of a single entity with the given.
+         // GET /api/0.6/[node|way|relation]/#id/relations
+         loadEntityRelations: function loadEntityRelations(id, callback) {
+           var type = osmEntity.id.type(id);
+           var osmID = osmEntity.id.toOSM(id);
+           var options = {
+             skipSeen: false
+           };
+           this.loadFromAPI('/api/0.6/' + type + '/' + osmID + '/relations.json', function (err, entities) {
+             if (callback) callback(err, {
+               data: entities
+             });
+           }, options);
+         },
+         // Load multiple entities in chunks
+         // (note: callback may be called multiple times)
+         // Unlike `loadEntity`, child nodes and members are not fetched
+         // GET /api/0.6/[nodes|ways|relations]?#parameters
+         loadMultiple: function loadMultiple(ids, callback) {
+           var that = this;
+           var groups = utilArrayGroupBy(utilArrayUniq(ids), osmEntity.id.type);
+           Object.keys(groups).forEach(function (k) {
+             var type = k + 's'; // nodes, ways, relations
 
-           this.events = events;
+             var osmIDs = groups[k].map(function (id) {
+               return osmEntity.id.toOSM(id);
+             });
+             var options = {
+               skipSeen: false
+             };
+             utilArrayChunk(osmIDs, 150).forEach(function (arr) {
+               that.loadFromAPI('/api/0.6/' + type + '.json?' + type + '=' + arr.join(), function (err, entities) {
+                 if (callback) callback(err, {
+                   data: entities
+                 });
+               }, options);
+             });
+           });
+         },
+         // Create, upload, and close a changeset
+         // PUT /api/0.6/changeset/create
+         // POST /api/0.6/changeset/#id/upload
+         // PUT /api/0.6/changeset/#id/close
+         putChangeset: function putChangeset(changeset, changes, callback) {
+           var cid = _connectionID;
 
-           for (var i = 0, iMax = events.length; i < iMax; i++) {
-             events[i].segment.ringOut = this;
+           if (_changeset.inflight) {
+             return callback({
+               message: 'Changeset already inflight',
+               status: -2
+             }, changeset);
+           } else if (_changeset.open) {
+             // reuse existing open changeset..
+             return createdChangeset.call(this, null, _changeset.open);
+           } else {
+             // Open a new changeset..
+             var options = {
+               method: 'PUT',
+               path: '/api/0.6/changeset/create',
+               options: {
+                 header: {
+                   'Content-Type': 'text/xml'
+                 }
+               },
+               content: JXON.stringify(changeset.asJXON())
+             };
+             _changeset.inflight = oauth.xhr(options, wrapcb(this, createdChangeset, cid));
            }
 
-           this.poly = null;
-         }
-
-         _createClass$1(RingOut, [{
-           key: "getGeom",
-           value: function getGeom() {
-             // Remove superfluous points (ie extra points along a straight line),
-             var prevPt = this.events[0].point;
-             var points = [prevPt];
+           function createdChangeset(err, changesetID) {
+             _changeset.inflight = null;
 
-             for (var i = 1, iMax = this.events.length - 1; i < iMax; i++) {
-               var _pt = this.events[i].point;
-               var _nextPt = this.events[i + 1].point;
-               if (compareVectorAngles(_pt, prevPt, _nextPt) === 0) continue;
-               points.push(_pt);
-               prevPt = _pt;
-             } // ring was all (within rounding error of angle calc) colinear points
+             if (err) {
+               return callback(err, changeset);
+             }
 
+             _changeset.open = changesetID;
+             changeset = changeset.update({
+               id: changesetID
+             }); // Upload the changeset..
 
-             if (points.length === 1) return null; // check if the starting point is necessary
+             var options = {
+               method: 'POST',
+               path: '/api/0.6/changeset/' + changesetID + '/upload',
+               options: {
+                 header: {
+                   'Content-Type': 'text/xml'
+                 }
+               },
+               content: JXON.stringify(changeset.osmChangeJXON(changes))
+             };
+             _changeset.inflight = oauth.xhr(options, wrapcb(this, uploadedChangeset, cid));
+           }
 
-             var pt = points[0];
-             var nextPt = points[1];
-             if (compareVectorAngles(pt, prevPt, nextPt) === 0) points.shift();
-             points.push(points[0]);
-             var step = this.isExteriorRing() ? 1 : -1;
-             var iStart = this.isExteriorRing() ? 0 : points.length - 1;
-             var iEnd = this.isExteriorRing() ? points.length : -1;
-             var orderedPoints = [];
+           function uploadedChangeset(err) {
+             _changeset.inflight = null;
+             if (err) return callback(err, changeset); // Upload was successful, safe to call the callback.
+             // Add delay to allow for postgres replication #1646 #2678
 
-             for (var _i = iStart; _i != iEnd; _i += step) {
-               orderedPoints.push([points[_i].x, points[_i].y]);
-             }
+             window.setTimeout(function () {
+               callback(null, changeset);
+             }, 2500);
+             _changeset.open = null; // At this point, we don't really care if the connection was switched..
+             // Only try to close the changeset if we're still talking to the same server.
 
-             return orderedPoints;
-           }
-         }, {
-           key: "isExteriorRing",
-           value: function isExteriorRing() {
-             if (this._isExteriorRing === undefined) {
-               var enclosing = this.enclosingRing();
-               this._isExteriorRing = enclosing ? !enclosing.isExteriorRing() : true;
+             if (this.getConnectionId() === cid) {
+               // Still attempt to close changeset, but ignore response because #2667
+               oauth.xhr({
+                 method: 'PUT',
+                 path: '/api/0.6/changeset/' + changeset.id + '/close',
+                 options: {
+                   header: {
+                     'Content-Type': 'text/xml'
+                   }
+                 }
+               }, function () {
+                 return true;
+               });
              }
-
-             return this._isExteriorRing;
            }
-         }, {
-           key: "enclosingRing",
-           value: function enclosingRing() {
-             if (this._enclosingRing === undefined) {
-               this._enclosingRing = this._calcEnclosingRing();
+         },
+         // Load multiple users in chunks
+         // (note: callback may be called multiple times)
+         // GET /api/0.6/users?users=#id1,#id2,...,#idn
+         loadUsers: function loadUsers(uids, callback) {
+           var toLoad = [];
+           var cached = [];
+           utilArrayUniq(uids).forEach(function (uid) {
+             if (_userCache.user[uid]) {
+               delete _userCache.toLoad[uid];
+               cached.push(_userCache.user[uid]);
+             } else {
+               toLoad.push(uid);
              }
+           });
 
-             return this._enclosingRing;
+           if (cached.length || !this.authenticated()) {
+             callback(undefined, cached);
+             if (!this.authenticated()) return; // require auth
            }
-           /* Returns the ring that encloses this one, if any */
-
-         }, {
-           key: "_calcEnclosingRing",
-           value: function _calcEnclosingRing() {
-             // start with the ealier sweep line event so that the prevSeg
-             // chain doesn't lead us inside of a loop of ours
-             var leftMostEvt = this.events[0];
-
-             for (var i = 1, iMax = this.events.length; i < iMax; i++) {
-               var evt = this.events[i];
-               if (SweepEvent$1.compare(leftMostEvt, evt) > 0) leftMostEvt = evt;
-             }
 
-             var prevSeg = leftMostEvt.segment.prevInResult();
-             var prevPrevSeg = prevSeg ? prevSeg.prevInResult() : null;
+           utilArrayChunk(toLoad, 150).forEach(function (arr) {
+             oauth.xhr({
+               method: 'GET',
+               path: '/api/0.6/users.json?users=' + arr.join()
+             }, wrapcb(this, done, _connectionID));
+           }.bind(this));
 
-             while (true) {
-               // no segment found, thus no ring can enclose us
-               if (!prevSeg) return null; // no segments below prev segment found, thus the ring of the prev
-               // segment must loop back around and enclose us
+           function done(err, payload) {
+             if (err) return callback(err);
+             var options = {
+               skipSeen: true
+             };
+             return parseUserJSON(payload, function (err, results) {
+               if (err) return callback(err);
+               return callback(undefined, results);
+             }, options);
+           }
+         },
+         // Load a given user by id
+         // GET /api/0.6/user/#id
+         loadUser: function loadUser(uid, callback) {
+           if (_userCache.user[uid] || !this.authenticated()) {
+             // require auth
+             delete _userCache.toLoad[uid];
+             return callback(undefined, _userCache.user[uid]);
+           }
 
-               if (!prevPrevSeg) return prevSeg.ringOut; // if the two segments are of different rings, the ring of the prev
-               // segment must either loop around us or the ring of the prev prev
-               // seg, which would make us and the ring of the prev peers
+           oauth.xhr({
+             method: 'GET',
+             path: '/api/0.6/user/' + uid + '.json'
+           }, wrapcb(this, done, _connectionID));
 
-               if (prevPrevSeg.ringOut !== prevSeg.ringOut) {
-                 if (prevPrevSeg.ringOut.enclosingRing() !== prevSeg.ringOut) {
-                   return prevSeg.ringOut;
-                 } else return prevSeg.ringOut.enclosingRing();
-               } // two segments are from the same ring, so this was a penisula
-               // of that ring. iterate downward, keep searching
+           function done(err, payload) {
+             if (err) return callback(err);
+             var options = {
+               skipSeen: true
+             };
+             return parseUserJSON(payload, function (err, results) {
+               if (err) return callback(err);
+               return callback(undefined, results[0]);
+             }, options);
+           }
+         },
+         // Load the details of the logged-in user
+         // GET /api/0.6/user/details
+         userDetails: function userDetails(callback) {
+           if (_userDetails) {
+             // retrieve cached
+             return callback(undefined, _userDetails);
+           }
 
+           oauth.xhr({
+             method: 'GET',
+             path: '/api/0.6/user/details.json'
+           }, wrapcb(this, done, _connectionID));
 
-               prevSeg = prevPrevSeg.prevInResult();
-               prevPrevSeg = prevSeg ? prevSeg.prevInResult() : null;
-             }
+           function done(err, payload) {
+             if (err) return callback(err);
+             var options = {
+               skipSeen: false
+             };
+             return parseUserJSON(payload, function (err, results) {
+               if (err) return callback(err);
+               _userDetails = results[0];
+               return callback(undefined, _userDetails);
+             }, options);
+           }
+         },
+         // Load previous changesets for the logged in user
+         // GET /api/0.6/changesets?user=#id
+         userChangesets: function userChangesets(callback) {
+           if (_userChangesets) {
+             // retrieve cached
+             return callback(undefined, _userChangesets);
            }
-         }]);
-
-         return RingOut;
-       }();
 
-       var PolyOut = /*#__PURE__*/function () {
-         function PolyOut(exteriorRing) {
-           _classCallCheck$1(this, PolyOut);
+           this.userDetails(wrapcb(this, gotDetails, _connectionID));
 
-           this.exteriorRing = exteriorRing;
-           exteriorRing.poly = this;
-           this.interiorRings = [];
-         }
+           function gotDetails(err, user) {
+             if (err) {
+               return callback(err);
+             }
 
-         _createClass$1(PolyOut, [{
-           key: "addInterior",
-           value: function addInterior(ring) {
-             this.interiorRings.push(ring);
-             ring.poly = this;
+             oauth.xhr({
+               method: 'GET',
+               path: '/api/0.6/changesets?user=' + user.id
+             }, wrapcb(this, done, _connectionID));
            }
-         }, {
-           key: "getGeom",
-           value: function getGeom() {
-             var geom = [this.exteriorRing.getGeom()]; // exterior ring was all (within rounding error of angle calc) colinear points
-
-             if (geom[0] === null) return null;
-
-             for (var i = 0, iMax = this.interiorRings.length; i < iMax; i++) {
-               var ringGeom = this.interiorRings[i].getGeom(); // interior ring was all (within rounding error of angle calc) colinear points
 
-               if (ringGeom === null) continue;
-               geom.push(ringGeom);
+           function done(err, xml) {
+             if (err) {
+               return callback(err);
              }
 
-             return geom;
+             _userChangesets = Array.prototype.map.call(xml.getElementsByTagName('changeset'), function (changeset) {
+               return {
+                 tags: getTags(changeset)
+               };
+             }).filter(function (changeset) {
+               var comment = changeset.tags.comment;
+               return comment && comment !== '';
+             });
+             return callback(undefined, _userChangesets);
            }
-         }]);
-
-         return PolyOut;
-       }();
+         },
+         // Fetch the status of the OSM API
+         // GET /api/capabilities
+         status: function status(callback) {
+           var url = urlroot + '/api/capabilities';
+           var errback = wrapcb(this, done, _connectionID);
+           d3_xml(url).then(function (data) {
+             errback(null, data);
+           })["catch"](function (err) {
+             errback(err.message);
+           });
 
-       var MultiPolyOut = /*#__PURE__*/function () {
-         function MultiPolyOut(rings) {
-           _classCallCheck$1(this, MultiPolyOut);
+           function done(err, xml) {
+             if (err) {
+               // the status is null if no response could be retrieved
+               return callback(err, null);
+             } // update blocklists
 
-           this.rings = rings;
-           this.polys = this._composePolys(rings);
-         }
 
-         _createClass$1(MultiPolyOut, [{
-           key: "getGeom",
-           value: function getGeom() {
-             var geom = [];
+             var elements = xml.getElementsByTagName('blacklist');
+             var regexes = [];
 
-             for (var i = 0, iMax = this.polys.length; i < iMax; i++) {
-               var polyGeom = this.polys[i].getGeom(); // exterior ring was all (within rounding error of angle calc) colinear points
+             for (var i = 0; i < elements.length; i++) {
+               var regexString = elements[i].getAttribute('regex'); // needs unencode?
 
-               if (polyGeom === null) continue;
-               geom.push(polyGeom);
+               if (regexString) {
+                 try {
+                   var regex = new RegExp(regexString);
+                   regexes.push(regex);
+                 } catch (e) {
+                   /* noop */
+                 }
+               }
              }
 
-             return geom;
-           }
-         }, {
-           key: "_composePolys",
-           value: function _composePolys(rings) {
-             var polys = [];
-
-             for (var i = 0, iMax = rings.length; i < iMax; i++) {
-               var ring = rings[i];
-               if (ring.poly) continue;
-               if (ring.isExteriorRing()) polys.push(new PolyOut(ring));else {
-                 var enclosingRing = ring.enclosingRing();
-                 if (!enclosingRing.poly) polys.push(new PolyOut(enclosingRing));
-                 enclosingRing.poly.addInterior(ring);
-               }
+             if (regexes.length) {
+               _imageryBlocklists = regexes;
              }
 
-             return polys;
+             if (_rateLimitError) {
+               return callback(_rateLimitError, 'rateLimited');
+             } else {
+               var waynodes = xml.getElementsByTagName('waynodes');
+               var maxWayNodes = waynodes.length && parseInt(waynodes[0].getAttribute('maximum'), 10);
+               if (maxWayNodes && isFinite(maxWayNodes)) _maxWayNodes = maxWayNodes;
+               var apiStatus = xml.getElementsByTagName('status');
+               var val = apiStatus[0].getAttribute('api');
+               return callback(undefined, val);
+             }
            }
-         }]);
+         },
+         // Calls `status` and dispatches an `apiStatusChange` event if the returned
+         // status differs from the cached status.
+         reloadApiStatus: function reloadApiStatus() {
+           // throttle to avoid unnecessary API calls
+           if (!this.throttledReloadApiStatus) {
+             var that = this;
+             this.throttledReloadApiStatus = throttle(function () {
+               that.status(function (err, status) {
+                 if (status !== _cachedApiStatus) {
+                   _cachedApiStatus = status;
+                   dispatch$2.call('apiStatusChange', that, err, status);
+                 }
+               });
+             }, 500);
+           }
+
+           this.throttledReloadApiStatus();
+         },
+         // Returns the maximum number of nodes a single way can have
+         maxWayNodes: function maxWayNodes() {
+           return _maxWayNodes;
+         },
+         // Load data (entities) from the API in tiles
+         // GET /api/0.6/map?bbox=
+         loadTiles: function loadTiles(projection, callback) {
+           if (_off) return; // determine the needed tiles to cover the view
 
-         return MultiPolyOut;
-       }();
-       /**
-        * NOTE:  We must be careful not to change any segments while
-        *        they are in the SplayTree. AFAIK, there's no way to tell
-        *        the tree to rebalance itself - thus before splitting
-        *        a segment that's in the tree, we remove it from the tree,
-        *        do the split, then re-insert it. (Even though splitting a
-        *        segment *shouldn't* change its correct position in the
-        *        sweep line tree, the reality is because of rounding errors,
-        *        it sometimes does.)
-        */
+           var tiles = tiler$2.zoomExtent([_tileZoom, _tileZoom]).getTiles(projection); // abort inflight requests that are no longer needed
 
+           var hadRequests = hasInflightRequests(_tileCache);
+           abortUnwantedRequests(_tileCache, tiles);
 
-       var SweepLine = /*#__PURE__*/function () {
-         function SweepLine(queue) {
-           var comparator = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : Segment.compare;
+           if (hadRequests && !hasInflightRequests(_tileCache)) {
+             dispatch$2.call('loaded'); // stop the spinner
+           } // issue new requests..
 
-           _classCallCheck$1(this, SweepLine);
 
-           this.queue = queue;
-           this.tree = new Tree(comparator);
-           this.segments = [];
-         }
+           tiles.forEach(function (tile) {
+             this.loadTile(tile, callback);
+           }, this);
+         },
+         // Load a single data tile
+         // GET /api/0.6/map?bbox=
+         loadTile: function loadTile(tile, callback) {
+           if (_off) return;
+           if (_tileCache.loaded[tile.id] || _tileCache.inflight[tile.id]) return;
 
-         _createClass$1(SweepLine, [{
-           key: "process",
-           value: function process(event) {
-             var segment = event.segment;
-             var newEvents = []; // if we've already been consumed by another segment,
-             // clean up our body parts and get out
+           if (!hasInflightRequests(_tileCache)) {
+             dispatch$2.call('loading'); // start the spinner
+           }
 
-             if (event.consumedBy) {
-               if (event.isLeft) this.queue.remove(event.otherSE);else this.tree.remove(segment);
-               return newEvents;
-             }
+           var path = '/api/0.6/map.json?bbox=';
+           var options = {
+             skipSeen: true
+           };
+           _tileCache.inflight[tile.id] = this.loadFromAPI(path + tile.extent.toParam(), tileCallback, options);
 
-             var node = event.isLeft ? this.tree.insert(segment) : this.tree.find(segment);
-             if (!node) throw new Error("Unable to find segment #".concat(segment.id, " ") + "[".concat(segment.leftSE.point.x, ", ").concat(segment.leftSE.point.y, "] -> ") + "[".concat(segment.rightSE.point.x, ", ").concat(segment.rightSE.point.y, "] ") + 'in SweepLine tree. Please submit a bug report.');
-             var prevNode = node;
-             var nextNode = node;
-             var prevSeg = undefined;
-             var nextSeg = undefined; // skip consumed segments still in tree
+           function tileCallback(err, parsed) {
+             delete _tileCache.inflight[tile.id];
 
-             while (prevSeg === undefined) {
-               prevNode = this.tree.prev(prevNode);
-               if (prevNode === null) prevSeg = null;else if (prevNode.key.consumedBy === undefined) prevSeg = prevNode.key;
-             } // skip consumed segments still in tree
+             if (!err) {
+               delete _tileCache.toLoad[tile.id];
+               _tileCache.loaded[tile.id] = true;
+               var bbox = tile.extent.bbox();
+               bbox.id = tile.id;
 
+               _tileCache.rtree.insert(bbox);
+             }
 
-             while (nextSeg === undefined) {
-               nextNode = this.tree.next(nextNode);
-               if (nextNode === null) nextSeg = null;else if (nextNode.key.consumedBy === undefined) nextSeg = nextNode.key;
+             if (callback) {
+               callback(err, Object.assign({
+                 data: parsed
+               }, tile));
              }
 
-             if (event.isLeft) {
-               // Check for intersections against the previous segment in the sweep line
-               var prevMySplitter = null;
+             if (!hasInflightRequests(_tileCache)) {
+               dispatch$2.call('loaded'); // stop the spinner
+             }
+           }
+         },
+         isDataLoaded: function isDataLoaded(loc) {
+           var bbox = {
+             minX: loc[0],
+             minY: loc[1],
+             maxX: loc[0],
+             maxY: loc[1]
+           };
+           return _tileCache.rtree.collides(bbox);
+         },
+         // load the tile that covers the given `loc`
+         loadTileAtLoc: function loadTileAtLoc(loc, callback) {
+           // Back off if the toLoad queue is filling up.. re #6417
+           // (Currently `loadTileAtLoc` requests are considered low priority - used by operations to
+           // let users safely edit geometries which extend to unloaded tiles.  We can drop some.)
+           if (Object.keys(_tileCache.toLoad).length > 50) return;
+           var k = geoZoomToScale(_tileZoom + 1);
+           var offset = geoRawMercator().scale(k)(loc);
+           var projection = geoRawMercator().transform({
+             k: k,
+             x: -offset[0],
+             y: -offset[1]
+           });
+           var tiles = tiler$2.zoomExtent([_tileZoom, _tileZoom]).getTiles(projection);
+           tiles.forEach(function (tile) {
+             if (_tileCache.toLoad[tile.id] || _tileCache.loaded[tile.id] || _tileCache.inflight[tile.id]) return;
+             _tileCache.toLoad[tile.id] = true;
+             this.loadTile(tile, callback);
+           }, this);
+         },
+         // Load notes from the API in tiles
+         // GET /api/0.6/notes?bbox=
+         loadNotes: function loadNotes(projection, noteOptions) {
+           noteOptions = Object.assign({
+             limit: 10000,
+             closed: 7
+           }, noteOptions);
+           if (_off) return;
+           var that = this;
+           var path = '/api/0.6/notes?limit=' + noteOptions.limit + '&closed=' + noteOptions.closed + '&bbox=';
 
-               if (prevSeg) {
-                 var prevInter = prevSeg.getIntersection(segment);
+           var throttleLoadUsers = throttle(function () {
+             var uids = Object.keys(_userCache.toLoad);
+             if (!uids.length) return;
+             that.loadUsers(uids, function () {}); // eagerly load user details
+           }, 750); // determine the needed tiles to cover the view
 
-                 if (prevInter !== null) {
-                   if (!segment.isAnEndpoint(prevInter)) prevMySplitter = prevInter;
 
-                   if (!prevSeg.isAnEndpoint(prevInter)) {
-                     var newEventsFromSplit = this._splitSafely(prevSeg, prevInter);
+           var tiles = tiler$2.zoomExtent([_noteZoom, _noteZoom]).getTiles(projection); // abort inflight requests that are no longer needed
 
-                     for (var i = 0, iMax = newEventsFromSplit.length; i < iMax; i++) {
-                       newEvents.push(newEventsFromSplit[i]);
-                     }
-                   }
-                 }
-               } // Check for intersections against the next segment in the sweep line
+           abortUnwantedRequests(_noteCache, tiles); // issue new requests..
 
+           tiles.forEach(function (tile) {
+             if (_noteCache.loaded[tile.id] || _noteCache.inflight[tile.id]) return;
+             var options = {
+               skipSeen: false
+             };
+             _noteCache.inflight[tile.id] = that.loadFromAPI(path + tile.extent.toParam(), function (err) {
+               delete _noteCache.inflight[tile.id];
 
-               var nextMySplitter = null;
+               if (!err) {
+                 _noteCache.loaded[tile.id] = true;
+               }
 
-               if (nextSeg) {
-                 var nextInter = nextSeg.getIntersection(segment);
+               throttleLoadUsers();
+               dispatch$2.call('loadedNotes');
+             }, options);
+           });
+         },
+         // Create a note
+         // POST /api/0.6/notes?params
+         postNoteCreate: function postNoteCreate(note, callback) {
+           if (!this.authenticated()) {
+             return callback({
+               message: 'Not Authenticated',
+               status: -3
+             }, note);
+           }
 
-                 if (nextInter !== null) {
-                   if (!segment.isAnEndpoint(nextInter)) nextMySplitter = nextInter;
+           if (_noteCache.inflightPost[note.id]) {
+             return callback({
+               message: 'Note update already inflight',
+               status: -2
+             }, note);
+           }
 
-                   if (!nextSeg.isAnEndpoint(nextInter)) {
-                     var _newEventsFromSplit = this._splitSafely(nextSeg, nextInter);
+           if (!note.loc[0] || !note.loc[1] || !note.newComment) return; // location & description required
 
-                     for (var _i = 0, _iMax = _newEventsFromSplit.length; _i < _iMax; _i++) {
-                       newEvents.push(_newEventsFromSplit[_i]);
-                     }
-                   }
-                 }
-               } // For simplicity, even if we find more than one intersection we only
-               // spilt on the 'earliest' (sweep-line style) of the intersections.
-               // The other intersection will be handled in a future process().
+           var comment = note.newComment;
 
+           if (note.newCategory && note.newCategory !== 'None') {
+             comment += ' #' + note.newCategory;
+           }
 
-               if (prevMySplitter !== null || nextMySplitter !== null) {
-                 var mySplitter = null;
-                 if (prevMySplitter === null) mySplitter = nextMySplitter;else if (nextMySplitter === null) mySplitter = prevMySplitter;else {
-                   var cmpSplitters = SweepEvent$1.comparePoints(prevMySplitter, nextMySplitter);
-                   mySplitter = cmpSplitters <= 0 ? prevMySplitter : nextMySplitter;
-                 } // Rounding errors can cause changes in ordering,
-                 // so remove afected segments and right sweep events before splitting
+           var path = '/api/0.6/notes?' + utilQsString({
+             lon: note.loc[0],
+             lat: note.loc[1],
+             text: comment
+           });
+           _noteCache.inflightPost[note.id] = oauth.xhr({
+             method: 'POST',
+             path: path
+           }, wrapcb(this, done, _connectionID));
 
-                 this.queue.remove(segment.rightSE);
-                 newEvents.push(segment.rightSE);
+           function done(err, xml) {
+             delete _noteCache.inflightPost[note.id];
 
-                 var _newEventsFromSplit2 = segment.split(mySplitter);
+             if (err) {
+               return callback(err);
+             } // we get the updated note back, remove from caches and reparse..
 
-                 for (var _i2 = 0, _iMax2 = _newEventsFromSplit2.length; _i2 < _iMax2; _i2++) {
-                   newEvents.push(_newEventsFromSplit2[_i2]);
-                 }
-               }
 
-               if (newEvents.length > 0) {
-                 // We found some intersections, so re-do the current event to
-                 // make sure sweep line ordering is totally consistent for later
-                 // use with the segment 'prev' pointers
-                 this.tree.remove(segment);
-                 newEvents.push(event);
+             this.removeNote(note);
+             var options = {
+               skipSeen: false
+             };
+             return parseXML(xml, function (err, results) {
+               if (err) {
+                 return callback(err);
                } else {
-                 // done with left event
-                 this.segments.push(segment);
-                 segment.prev = prevSeg;
+                 return callback(undefined, results[0]);
                }
-             } else {
-               // event.isRight
-               // since we're about to be removed from the sweep line, check for
-               // intersections between our previous and next segments
-               if (prevSeg && nextSeg) {
-                 var inter = prevSeg.getIntersection(nextSeg);
+             }, options);
+           }
+         },
+         // Update a note
+         // POST /api/0.6/notes/#id/comment?text=comment
+         // POST /api/0.6/notes/#id/close?text=comment
+         // POST /api/0.6/notes/#id/reopen?text=comment
+         postNoteUpdate: function postNoteUpdate(note, newStatus, callback) {
+           if (!this.authenticated()) {
+             return callback({
+               message: 'Not Authenticated',
+               status: -3
+             }, note);
+           }
 
-                 if (inter !== null) {
-                   if (!prevSeg.isAnEndpoint(inter)) {
-                     var _newEventsFromSplit3 = this._splitSafely(prevSeg, inter);
+           if (_noteCache.inflightPost[note.id]) {
+             return callback({
+               message: 'Note update already inflight',
+               status: -2
+             }, note);
+           }
 
-                     for (var _i3 = 0, _iMax3 = _newEventsFromSplit3.length; _i3 < _iMax3; _i3++) {
-                       newEvents.push(_newEventsFromSplit3[_i3]);
-                     }
-                   }
+           var action;
 
-                   if (!nextSeg.isAnEndpoint(inter)) {
-                     var _newEventsFromSplit4 = this._splitSafely(nextSeg, inter);
+           if (note.status !== 'closed' && newStatus === 'closed') {
+             action = 'close';
+           } else if (note.status !== 'open' && newStatus === 'open') {
+             action = 'reopen';
+           } else {
+             action = 'comment';
+             if (!note.newComment) return; // when commenting, comment required
+           }
 
-                     for (var _i4 = 0, _iMax4 = _newEventsFromSplit4.length; _i4 < _iMax4; _i4++) {
-                       newEvents.push(_newEventsFromSplit4[_i4]);
-                     }
-                   }
-                 }
-               }
+           var path = '/api/0.6/notes/' + note.id + '/' + action;
 
-               this.tree.remove(segment);
+           if (note.newComment) {
+             path += '?' + utilQsString({
+               text: note.newComment
+             });
+           }
+
+           _noteCache.inflightPost[note.id] = oauth.xhr({
+             method: 'POST',
+             path: path
+           }, wrapcb(this, done, _connectionID));
+
+           function done(err, xml) {
+             delete _noteCache.inflightPost[note.id];
+
+             if (err) {
+               return callback(err);
+             } // we get the updated note back, remove from caches and reparse..
+
+
+             this.removeNote(note); // update closed note cache - used to populate `closed:note` changeset tag
+
+             if (action === 'close') {
+               _noteCache.closed[note.id] = true;
+             } else if (action === 'reopen') {
+               delete _noteCache.closed[note.id];
              }
 
-             return newEvents;
+             var options = {
+               skipSeen: false
+             };
+             return parseXML(xml, function (err, results) {
+               if (err) {
+                 return callback(err);
+               } else {
+                 return callback(undefined, results[0]);
+               }
+             }, options);
            }
-           /* Safely split a segment that is currently in the datastructures
-            * IE - a segment other than the one that is currently being processed. */
-
-         }, {
-           key: "_splitSafely",
-           value: function _splitSafely(seg, pt) {
-             // Rounding errors can cause changes in ordering,
-             // so remove afected segments and right sweep events before splitting
-             // removeNode() doesn't work, so have re-find the seg
-             // https://github.com/w8r/splay-tree/pull/5
-             this.tree.remove(seg);
-             var rightSE = seg.rightSE;
-             this.queue.remove(rightSE);
-             var newEvents = seg.split(pt);
-             newEvents.push(rightSE); // splitting can trigger consumption
+         },
+         "switch": function _switch(options) {
+           urlroot = options.urlroot;
+           oauth.options(Object.assign({
+             url: urlroot,
+             loading: authLoading,
+             done: authDone
+           }, options));
+           this.reset();
+           this.userChangesets(function () {}); // eagerly load user details/changesets
 
-             if (seg.consumedBy === undefined) this.tree.insert(seg);
-             return newEvents;
+           dispatch$2.call('change');
+           return this;
+         },
+         toggle: function toggle(val) {
+           _off = !val;
+           return this;
+         },
+         isChangesetInflight: function isChangesetInflight() {
+           return !!_changeset.inflight;
+         },
+         // get/set cached data
+         // This is used to save/restore the state when entering/exiting the walkthrough
+         // Also used for testing purposes.
+         caches: function caches(obj) {
+           function cloneCache(source) {
+             var target = {};
+             Object.keys(source).forEach(function (k) {
+               if (k === 'rtree') {
+                 target.rtree = new RBush().fromJSON(source.rtree.toJSON()); // clone rbush
+               } else if (k === 'note') {
+                 target.note = {};
+                 Object.keys(source.note).forEach(function (id) {
+                   target.note[id] = osmNote(source.note[id]); // copy notes
+                 });
+               } else {
+                 target[k] = JSON.parse(JSON.stringify(source[k])); // clone deep
+               }
+             });
+             return target;
            }
-         }]);
 
-         return SweepLine;
-       }();
+           if (!arguments.length) {
+             return {
+               tile: cloneCache(_tileCache),
+               note: cloneCache(_noteCache),
+               user: cloneCache(_userCache)
+             };
+           } // access caches directly for testing (e.g., loading notes rtree)
 
-       var POLYGON_CLIPPING_MAX_QUEUE_SIZE = typeof process !== 'undefined' && process.env.POLYGON_CLIPPING_MAX_QUEUE_SIZE || 1000000;
-       var POLYGON_CLIPPING_MAX_SWEEPLINE_SEGMENTS = typeof process !== 'undefined' && process.env.POLYGON_CLIPPING_MAX_SWEEPLINE_SEGMENTS || 1000000;
 
-       var Operation = /*#__PURE__*/function () {
-         function Operation() {
-           _classCallCheck$1(this, Operation);
-         }
+           if (obj === 'get') {
+             return {
+               tile: _tileCache,
+               note: _noteCache,
+               user: _userCache
+             };
+           }
 
-         _createClass$1(Operation, [{
-           key: "run",
-           value: function run(type, geom, moreGeoms) {
-             operation.type = type;
-             rounder.reset();
-             /* Convert inputs to MultiPoly objects */
+           if (obj.tile) {
+             _tileCache = obj.tile;
+             _tileCache.inflight = {};
+           }
 
-             var multipolys = [new MultiPolyIn(geom, true)];
+           if (obj.note) {
+             _noteCache = obj.note;
+             _noteCache.inflight = {};
+             _noteCache.inflightPost = {};
+           }
 
-             for (var i = 0, iMax = moreGeoms.length; i < iMax; i++) {
-               multipolys.push(new MultiPolyIn(moreGeoms[i], false));
-             }
+           if (obj.user) {
+             _userCache = obj.user;
+           }
 
-             operation.numMultiPolys = multipolys.length;
-             /* BBox optimization for difference operation
-              * If the bbox of a multipolygon that's part of the clipping doesn't
-              * intersect the bbox of the subject at all, we can just drop that
-              * multiploygon. */
+           return this;
+         },
+         logout: function logout() {
+           _userChangesets = undefined;
+           _userDetails = undefined;
+           oauth.logout();
+           dispatch$2.call('change');
+           return this;
+         },
+         authenticated: function authenticated() {
+           return oauth.authenticated();
+         },
+         authenticate: function authenticate(callback) {
+           var that = this;
+           var cid = _connectionID;
+           _userChangesets = undefined;
+           _userDetails = undefined;
 
-             if (operation.type === 'difference') {
-               // in place removal
-               var subject = multipolys[0];
-               var _i = 1;
+           function done(err, res) {
+             if (err) {
+               if (callback) callback(err);
+               return;
+             }
 
-               while (_i < multipolys.length) {
-                 if (getBboxOverlap(multipolys[_i].bbox, subject.bbox) !== null) _i++;else multipolys.splice(_i, 1);
-               }
+             if (that.getConnectionId() !== cid) {
+               if (callback) callback({
+                 message: 'Connection Switched',
+                 status: -1
+               });
+               return;
              }
-             /* BBox optimization for intersection operation
-              * If we can find any pair of multipolygons whose bbox does not overlap,
-              * then the result will be empty. */
 
+             _rateLimitError = undefined;
+             dispatch$2.call('change');
+             if (callback) callback(err, res);
+             that.userChangesets(function () {}); // eagerly load user details/changesets
+           }
 
-             if (operation.type === 'intersection') {
-               // TODO: this is O(n^2) in number of polygons. By sorting the bboxes,
-               //       it could be optimized to O(n * ln(n))
-               for (var _i2 = 0, _iMax = multipolys.length; _i2 < _iMax; _i2++) {
-                 var mpA = multipolys[_i2];
+           return oauth.authenticate(done);
+         },
+         imageryBlocklists: function imageryBlocklists() {
+           return _imageryBlocklists;
+         },
+         tileZoom: function tileZoom(val) {
+           if (!arguments.length) return _tileZoom;
+           _tileZoom = val;
+           return this;
+         },
+         // get all cached notes covering the viewport
+         notes: function notes(projection) {
+           var viewport = projection.clipExtent();
+           var min = [viewport[0][0], viewport[1][1]];
+           var max = [viewport[1][0], viewport[0][1]];
+           var bbox = geoExtent(projection.invert(min), projection.invert(max)).bbox();
+           return _noteCache.rtree.search(bbox).map(function (d) {
+             return d.data;
+           });
+         },
+         // get a single note from the cache
+         getNote: function getNote(id) {
+           return _noteCache.note[id];
+         },
+         // remove a single note from the cache
+         removeNote: function removeNote(note) {
+           if (!(note instanceof osmNote) || !note.id) return;
+           delete _noteCache.note[note.id];
+           updateRtree(encodeNoteRtree(note), false); // false = remove
+         },
+         // replace a single note in the cache
+         replaceNote: function replaceNote(note) {
+           if (!(note instanceof osmNote) || !note.id) return;
+           _noteCache.note[note.id] = note;
+           updateRtree(encodeNoteRtree(note), true); // true = replace
 
-                 for (var j = _i2 + 1, jMax = multipolys.length; j < jMax; j++) {
-                   if (getBboxOverlap(mpA.bbox, multipolys[j].bbox) === null) return [];
-                 }
-               }
-             }
-             /* Put segment endpoints in a priority queue */
+           return note;
+         },
+         // Get an array of note IDs closed during this session.
+         // Used to populate `closed:note` changeset tag
+         getClosedIDs: function getClosedIDs() {
+           return Object.keys(_noteCache.closed).sort();
+         }
+       };
 
+       var _apibase$1 = 'https://wiki.openstreetmap.org/w/api.php';
+       var _inflight$1 = {};
+       var _wikibaseCache = {};
+       var _localeIDs = {
+         en: false
+       };
 
-             var queue = new Tree(SweepEvent$1.compare);
+       var debouncedRequest$1 = debounce(request$1, 500, {
+         leading: false
+       });
 
-             for (var _i3 = 0, _iMax2 = multipolys.length; _i3 < _iMax2; _i3++) {
-               var sweepEvents = multipolys[_i3].getSweepEvents();
+       function request$1(url, callback) {
+         if (_inflight$1[url]) return;
+         var controller = new AbortController();
+         _inflight$1[url] = controller;
+         d3_json(url, {
+           signal: controller.signal
+         }).then(function (result) {
+           delete _inflight$1[url];
+           if (callback) callback(null, result);
+         })["catch"](function (err) {
+           delete _inflight$1[url];
+           if (err.name === 'AbortError') return;
+           if (callback) callback(err.message);
+         });
+       }
 
-               for (var _j = 0, _jMax = sweepEvents.length; _j < _jMax; _j++) {
-                 queue.insert(sweepEvents[_j]);
+       var serviceOsmWikibase = {
+         init: function init() {
+           _inflight$1 = {};
+           _wikibaseCache = {};
+           _localeIDs = {};
+         },
+         reset: function reset() {
+           Object.values(_inflight$1).forEach(function (controller) {
+             controller.abort();
+           });
+           _inflight$1 = {};
+         },
 
-                 if (queue.size > POLYGON_CLIPPING_MAX_QUEUE_SIZE) {
-                   // prevents an infinite loop, an otherwise common manifestation of bugs
-                   throw new Error('Infinite loop when putting segment endpoints in a priority queue ' + '(queue size too big). Please file a bug report.');
-                 }
-               }
+         /**
+          * Get the best value for the property, or undefined if not found
+          * @param entity object from wikibase
+          * @param property string e.g. 'P4' for image
+          * @param langCode string e.g. 'fr' for French
+          */
+         claimToValue: function claimToValue(entity, property, langCode) {
+           if (!entity.claims[property]) return undefined;
+           var locale = _localeIDs[langCode];
+           var preferredPick, localePick;
+           entity.claims[property].forEach(function (stmt) {
+             // If exists, use value limited to the needed language (has a qualifier P26 = locale)
+             // Or if not found, use the first value with the "preferred" rank
+             if (!preferredPick && stmt.rank === 'preferred') {
+               preferredPick = stmt;
              }
-             /* Pass the sweep line over those endpoints */
-
-
-             var sweepLine = new SweepLine(queue);
-             var prevQueueSize = queue.size;
-             var node = queue.pop();
-
-             while (node) {
-               var evt = node.key;
-
-               if (queue.size === prevQueueSize) {
-                 // prevents an infinite loop, an otherwise common manifestation of bugs
-                 var seg = evt.segment;
-                 throw new Error("Unable to pop() ".concat(evt.isLeft ? 'left' : 'right', " SweepEvent ") + "[".concat(evt.point.x, ", ").concat(evt.point.y, "] from segment #").concat(seg.id, " ") + "[".concat(seg.leftSE.point.x, ", ").concat(seg.leftSE.point.y, "] -> ") + "[".concat(seg.rightSE.point.x, ", ").concat(seg.rightSE.point.y, "] from queue. ") + 'Please file a bug report.');
-               }
 
-               if (queue.size > POLYGON_CLIPPING_MAX_QUEUE_SIZE) {
-                 // prevents an infinite loop, an otherwise common manifestation of bugs
-                 throw new Error('Infinite loop when passing sweep line over endpoints ' + '(queue size too big). Please file a bug report.');
-               }
+             if (locale && stmt.qualifiers && stmt.qualifiers.P26 && stmt.qualifiers.P26[0].datavalue.value.id === locale) {
+               localePick = stmt;
+             }
+           });
+           var result = localePick || preferredPick;
 
-               if (sweepLine.segments.length > POLYGON_CLIPPING_MAX_SWEEPLINE_SEGMENTS) {
-                 // prevents an infinite loop, an otherwise common manifestation of bugs
-                 throw new Error('Infinite loop when passing sweep line over endpoints ' + '(too many sweep line segments). Please file a bug report.');
-               }
+           if (result) {
+             var datavalue = result.mainsnak.datavalue;
+             return datavalue.type === 'wikibase-entityid' ? datavalue.value.id : datavalue.value;
+           } else {
+             return undefined;
+           }
+         },
 
-               var newEvents = sweepLine.process(evt);
+         /**
+          * Convert monolingual property into a key-value object (language -> value)
+          * @param entity object from wikibase
+          * @param property string e.g. 'P31' for monolingual wiki page title
+          */
+         monolingualClaimToValueObj: function monolingualClaimToValueObj(entity, property) {
+           if (!entity || !entity.claims[property]) return undefined;
+           return entity.claims[property].reduce(function (acc, obj) {
+             var value = obj.mainsnak.datavalue.value;
+             acc[value.language] = value.text;
+             return acc;
+           }, {});
+         },
+         toSitelink: function toSitelink(key, value) {
+           var result = value ? 'Tag:' + key + '=' + value : 'Key:' + key;
+           return result.replace(/_/g, ' ').trim();
+         },
+         //
+         // Pass params object of the form:
+         // {
+         //   key: 'string',
+         //   value: 'string',
+         //   langCode: 'string'
+         // }
+         //
+         getEntity: function getEntity(params, callback) {
+           var doRequest = params.debounce ? debouncedRequest$1 : request$1;
+           var that = this;
+           var titles = [];
+           var result = {};
+           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;
 
-               for (var _i4 = 0, _iMax3 = newEvents.length; _i4 < _iMax3; _i4++) {
-                 var _evt = newEvents[_i4];
-                 if (_evt.consumedBy === undefined) queue.insert(_evt);
+           if (params.langCodes) {
+             params.langCodes.forEach(function (langCode) {
+               if (_localeIDs[langCode] === undefined) {
+                 // If this is the first time we are asking about this locale,
+                 // fetch corresponding entity (if it exists), and cache it.
+                 // If there is no such entry, cache `false` value to avoid re-requesting it.
+                 localeSitelink = ('Locale:' + langCode).replace(/_/g, ' ').trim();
+                 titles.push(localeSitelink);
                }
+             });
+           }
 
-               prevQueueSize = queue.size;
-               node = queue.pop();
-             } // free some memory we don't need anymore
-
-
-             rounder.reset();
-             /* Collect and compile segments we're keeping into a multipolygon */
+           if (rtypeSitelink) {
+             if (_wikibaseCache[rtypeSitelink]) {
+               result.rtype = _wikibaseCache[rtypeSitelink];
+             } else {
+               titles.push(rtypeSitelink);
+             }
+           }
 
-             var ringsOut = RingOut.factory(sweepLine.segments);
-             var result = new MultiPolyOut(ringsOut);
-             return result.getGeom();
+           if (keySitelink) {
+             if (_wikibaseCache[keySitelink]) {
+               result.key = _wikibaseCache[keySitelink];
+             } else {
+               titles.push(keySitelink);
+             }
            }
-         }]);
 
-         return Operation;
-       }(); // singleton available by import
+           if (tagSitelink) {
+             if (_wikibaseCache[tagSitelink]) {
+               result.tag = _wikibaseCache[tagSitelink];
+             } else {
+               titles.push(tagSitelink);
+             }
+           }
 
+           if (!titles.length) {
+             // Nothing to do, we already had everything in the cache
+             return callback(null, result);
+           } // Requesting just the user language code
+           // If backend recognizes the code, it will perform proper fallbacks,
+           // and the result will contain the requested code. If not, all values are returned:
+           // {"zh-tw":{"value":"...","language":"zh-tw","source-language":"zh-hant"}
+           // {"pt-br":{"value":"...","language":"pt","for-language":"pt-br"}}
 
-       var operation = new Operation();
 
-       var union$1 = function union(geom) {
-         for (var _len = arguments.length, moreGeoms = new Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) {
-           moreGeoms[_key - 1] = arguments[_key];
-         }
+           var obj = {
+             action: 'wbgetentities',
+             sites: 'wiki',
+             titles: titles.join('|'),
+             languages: params.langCodes.join('|'),
+             languagefallback: 1,
+             origin: '*',
+             format: 'json' // There is an MW Wikibase API bug https://phabricator.wikimedia.org/T212069
+             // We shouldn't use v1 until it gets fixed, but should switch to it afterwards
+             // formatversion: 2,
 
-         return operation.run('union', geom, moreGeoms);
-       };
+           };
+           var url = _apibase$1 + '?' + utilQsString(obj);
+           doRequest(url, function (err, d) {
+             if (err) {
+               callback(err);
+             } else if (!d.success || d.error) {
+               callback(d.error.messages.map(function (v) {
+                 return v.html['*'];
+               }).join('<br>'));
+             } else {
+               var localeID = false;
+               Object.values(d.entities).forEach(function (res) {
+                 if (res.missing !== '') {
+                   var title = res.sitelinks.wiki.title;
 
-       var intersection$1$1 = function intersection(geom) {
-         for (var _len2 = arguments.length, moreGeoms = new Array(_len2 > 1 ? _len2 - 1 : 0), _key2 = 1; _key2 < _len2; _key2++) {
-           moreGeoms[_key2 - 1] = arguments[_key2];
-         }
+                   if (title === rtypeSitelink) {
+                     _wikibaseCache[rtypeSitelink] = res;
+                     result.rtype = res;
+                   } else if (title === keySitelink) {
+                     _wikibaseCache[keySitelink] = res;
+                     result.key = res;
+                   } else if (title === tagSitelink) {
+                     _wikibaseCache[tagSitelink] = res;
+                     result.tag = res;
+                   } else if (title === localeSitelink) {
+                     localeID = res.id;
+                   } else {
+                     console.log('Unexpected title ' + title); // eslint-disable-line no-console
+                   }
+                 }
+               });
 
-         return operation.run('intersection', geom, moreGeoms);
-       };
+               if (localeSitelink) {
+                 // If locale ID is not found, store false to prevent repeated queries
+                 that.addLocale(params.langCodes[0], localeID);
+               }
 
-       var xor = function xor(geom) {
-         for (var _len3 = arguments.length, moreGeoms = new Array(_len3 > 1 ? _len3 - 1 : 0), _key3 = 1; _key3 < _len3; _key3++) {
-           moreGeoms[_key3 - 1] = arguments[_key3];
-         }
+               callback(null, result);
+             }
+           });
+         },
+         //
+         // Pass params object of the form:
+         // {
+         //   key: 'string',     // required
+         //   value: 'string'    // optional
+         // }
+         //
+         // Get an result object used to display tag documentation
+         // {
+         //   title:        'string',
+         //   description:  'string',
+         //   editURL:      'string',
+         //   imageURL:     'string',
+         //   wiki:         { title: 'string', text: 'string', url: 'string' }
+         // }
+         //
+         getDocs: function getDocs(params, callback) {
+           var that = this;
+           var langCodes = _mainLocalizer.localeCodes().map(function (code) {
+             return code.toLowerCase();
+           });
+           params.langCodes = langCodes;
+           this.getEntity(params, function (err, data) {
+             if (err) {
+               callback(err);
+               return;
+             }
 
-         return operation.run('xor', geom, moreGeoms);
-       };
+             var entity = data.rtype || data.tag || data.key;
 
-       var difference = function difference(subjectGeom) {
-         for (var _len4 = arguments.length, clippingGeoms = new Array(_len4 > 1 ? _len4 - 1 : 0), _key4 = 1; _key4 < _len4; _key4++) {
-           clippingGeoms[_key4 - 1] = arguments[_key4];
-         }
+             if (!entity) {
+               callback('No entity');
+               return;
+             }
 
-         return operation.run('difference', subjectGeom, clippingGeoms);
-       };
+             var i;
+             var description;
 
-       var index$1 = {
-         union: union$1,
-         intersection: intersection$1$1,
-         xor: xor,
-         difference: difference
-       };
+             for (i in langCodes) {
+               var _code = langCodes[i];
 
-       var geojsonPrecision = createCommonjsModule(function (module) {
-         (function () {
-           function parse(t, coordinatePrecision, extrasPrecision) {
-             function point(p) {
-               return p.map(function (e, index) {
-                 if (index < 2) {
-                   return 1 * e.toFixed(coordinatePrecision);
-                 } else {
-                   return 1 * e.toFixed(extrasPrecision);
-                 }
-               });
+               if (entity.descriptions[_code] && entity.descriptions[_code].language === _code) {
+                 description = entity.descriptions[_code];
+                 break;
+               }
              }
 
-             function multi(l) {
-               return l.map(point);
-             }
+             if (!description && Object.values(entity.descriptions).length) description = Object.values(entity.descriptions)[0]; // prepare result
 
-             function poly(p) {
-               return p.map(multi);
-             }
+             var result = {
+               title: entity.title,
+               description: description ? description.value : '',
+               descriptionLocaleCode: description ? description.language : '',
+               editURL: 'https://wiki.openstreetmap.org/wiki/' + entity.title
+             }; // add image
 
-             function multiPoly(m) {
-               return m.map(poly);
-             }
+             if (entity.claims) {
+               var imageroot;
+               var image = that.claimToValue(entity, 'P4', langCodes[0]);
 
-             function geometry(obj) {
-               if (!obj) {
-                 return {};
+               if (image) {
+                 imageroot = 'https://commons.wikimedia.org/w/index.php';
+               } else {
+                 image = that.claimToValue(entity, 'P28', langCodes[0]);
+
+                 if (image) {
+                   imageroot = 'https://wiki.openstreetmap.org/w/index.php';
+                 }
                }
 
-               switch (obj.type) {
-                 case "Point":
-                   obj.coordinates = point(obj.coordinates);
-                   return obj;
+               if (imageroot && image) {
+                 result.imageURL = imageroot + '?' + utilQsString({
+                   title: 'Special:Redirect/file/' + image,
+                   width: 400
+                 });
+               }
+             } // Try to get a wiki page from tag data item first, followed by the corresponding key data item.
+             // If neither tag nor key data item contain a wiki page in the needed language nor English,
+             // get the first found wiki page from either the tag or the key item.
 
-                 case "LineString":
-                 case "MultiPoint":
-                   obj.coordinates = multi(obj.coordinates);
-                   return obj;
 
-                 case "Polygon":
-                 case "MultiLineString":
-                   obj.coordinates = poly(obj.coordinates);
-                   return obj;
+             var rtypeWiki = that.monolingualClaimToValueObj(data.rtype, 'P31');
+             var tagWiki = that.monolingualClaimToValueObj(data.tag, 'P31');
+             var keyWiki = that.monolingualClaimToValueObj(data.key, 'P31');
+             var wikis = [rtypeWiki, tagWiki, keyWiki];
 
-                 case "MultiPolygon":
-                   obj.coordinates = multiPoly(obj.coordinates);
-                   return obj;
+             for (i in wikis) {
+               var wiki = wikis[i];
 
-                 case "GeometryCollection":
-                   obj.geometries = obj.geometries.map(geometry);
-                   return obj;
+               for (var j in langCodes) {
+                 var code = langCodes[j];
+                 var referenceId = langCodes[0].split('-')[0] !== 'en' && code.split('-')[0] === 'en' ? 'inspector.wiki_en_reference' : 'inspector.wiki_reference';
+                 var info = getWikiInfo(wiki, code, referenceId);
 
-                 default:
-                   return {};
+                 if (info) {
+                   result.wiki = info;
+                   break;
+                 }
                }
-             }
-
-             function feature(obj) {
-               obj.geometry = geometry(obj.geometry);
-               return obj;
-             }
 
-             function featureCollection(f) {
-               f.features = f.features.map(feature);
-               return f;
+               if (result.wiki) break;
              }
 
-             function geometryCollection(g) {
-               g.geometries = g.geometries.map(geometry);
-               return g;
-             }
+             callback(null, result); // Helper method to get wiki info if a given language exists
 
-             if (!t) {
-               return t;
+             function getWikiInfo(wiki, langCode, tKey) {
+               if (wiki && wiki[langCode]) {
+                 return {
+                   title: wiki[langCode],
+                   text: tKey,
+                   url: 'https://wiki.openstreetmap.org/wiki/' + wiki[langCode]
+                 };
+               }
              }
+           });
+         },
+         addLocale: function addLocale(langCode, qid) {
+           // Makes it easier to unit test
+           _localeIDs[langCode] = qid;
+         },
+         apibase: function apibase(val) {
+           if (!arguments.length) return _apibase$1;
+           _apibase$1 = val;
+           return this;
+         }
+       };
 
-             switch (t.type) {
-               case "Feature":
-                 return feature(t);
-
-               case "GeometryCollection":
-                 return geometryCollection(t);
-
-               case "FeatureCollection":
-                 return featureCollection(t);
+       var jsonpCache = {};
+       window.jsonpCache = jsonpCache;
+       function jsonpRequest(url, callback) {
+         var request = {
+           abort: function abort() {}
+         };
 
-               case "Point":
-               case "LineString":
-               case "Polygon":
-               case "MultiPoint":
-               case "MultiPolygon":
-               case "MultiLineString":
-                 return geometry(t);
+         if (window.JSONP_FIX) {
+           if (window.JSONP_DELAY === 0) {
+             callback(window.JSONP_FIX);
+           } else {
+             var t = window.setTimeout(function () {
+               callback(window.JSONP_FIX);
+             }, window.JSONP_DELAY || 0);
 
-               default:
-                 return t;
-             }
+             request.abort = function () {
+               window.clearTimeout(t);
+             };
            }
 
-           module.exports = parse;
-           module.exports.parse = parse;
-         })();
-       });
-
-       function isObject$4(obj) {
-         return _typeof(obj) === 'object' && obj !== null;
-       }
-
-       function forEach(obj, cb) {
-         if (Array.isArray(obj)) {
-           obj.forEach(cb);
-         } else if (isObject$4(obj)) {
-           Object.keys(obj).forEach(function (key) {
-             var val = obj[key];
-             cb(val, key);
-           });
+           return request;
          }
-       }
 
-       function getTreeDepth(obj) {
-         var depth = 0;
+         function rand() {
+           var chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz';
+           var c = '';
+           var i = -1;
 
-         if (Array.isArray(obj) || isObject$4(obj)) {
-           forEach(obj, function (val) {
-             if (Array.isArray(val) || isObject$4(val)) {
-               var tmpDepth = getTreeDepth(val);
+           while (++i < 15) {
+             c += chars.charAt(Math.floor(Math.random() * 52));
+           }
 
-               if (tmpDepth > depth) {
-                 depth = tmpDepth;
-               }
-             }
-           });
-           return depth + 1;
+           return c;
          }
 
-         return depth;
-       }
+         function create(url) {
+           var e = url.match(/callback=(\w+)/);
+           var c = e ? e[1] : rand();
 
-       function stringify(obj, options) {
-         options = options || {};
-         var indent = JSON.stringify([1], null, get$5(options, 'indent', 2)).slice(2, -3);
-         var addMargin = get$5(options, 'margins', false);
-         var addArrayMargin = get$5(options, 'arrayMargins', false);
-         var addObjectMargin = get$5(options, 'objectMargins', false);
-         var maxLength = indent === '' ? Infinity : get$5(options, 'maxLength', 80);
-         var maxNesting = get$5(options, 'maxNesting', Infinity);
-         return function _stringify(obj, currentIndent, reserved) {
-           if (obj && typeof obj.toJSON === 'function') {
-             obj = obj.toJSON();
-           }
+           jsonpCache[c] = function (data) {
+             if (jsonpCache[c]) {
+               callback(data);
+             }
 
-           var string = JSON.stringify(obj);
+             finalize();
+           };
 
-           if (string === undefined) {
-             return string;
+           function finalize() {
+             delete jsonpCache[c];
+             script.remove();
            }
 
-           var length = maxLength - currentIndent.length - reserved;
-           var treeDepth = getTreeDepth(obj);
+           request.abort = finalize;
+           return 'jsonpCache.' + c;
+         }
 
-           if (treeDepth <= maxNesting && string.length <= length) {
-             var prettified = prettify(string, {
-               addMargin: addMargin,
-               addArrayMargin: addArrayMargin,
-               addObjectMargin: addObjectMargin
-             });
+         var cb = create(url);
+         var script = select('head').append('script').attr('type', 'text/javascript').attr('src', url.replace(/(\{|%7B)callback(\}|%7D)/, cb));
+         return request;
+       }
 
-             if (prettified.length <= length) {
-               return prettified;
-             }
-           }
+       var bubbleApi = 'https://dev.virtualearth.net/mapcontrol/HumanScaleServices/GetBubbles.ashx?';
+       var streetsideImagesApi = 'https://t.ssl.ak.tiles.virtualearth.net/tiles/';
+       var bubbleAppKey = 'AuftgJsO0Xs8Ts4M1xZUQJQXJNsvmh3IV8DkNieCiy3tCwCUMq76-WpkrBtNAuEm';
+       var pannellumViewerCSS = 'pannellum-streetside/pannellum.css';
+       var pannellumViewerJS = 'pannellum-streetside/pannellum.js';
+       var maxResults = 2000;
+       var tileZoom = 16.5;
+       var tiler$1 = utilTiler().zoomExtent([tileZoom, tileZoom]).skipNullIsland(true);
+       var dispatch$1 = dispatch$8('loadedImages', 'viewerChanged');
+       var minHfov = 10; // zoom in degrees:  20, 10, 5
 
-           if (isObject$4(obj)) {
-             var nextIndent = currentIndent + indent;
-             var items = [];
-             var delimiters;
+       var maxHfov = 90; // zoom out degrees
 
-             var comma = function comma(array, index) {
-               return index === array.length - 1 ? 0 : 1;
-             };
+       var defaultHfov = 45;
+       var _hires = false;
+       var _resolution = 512; // higher numbers are slower - 512, 1024, 2048, 4096
 
-             if (Array.isArray(obj)) {
-               for (var index = 0; index < obj.length; index++) {
-                 items.push(_stringify(obj[index], nextIndent, comma(obj, index)) || 'null');
-               }
+       var _currScene = 0;
 
-               delimiters = '[]';
-             } else {
-               Object.keys(obj).forEach(function (key, index, array) {
-                 var keyPart = JSON.stringify(key) + ': ';
+       var _ssCache;
 
-                 var value = _stringify(obj[key], nextIndent, keyPart.length + comma(array, index));
+       var _pannellumViewer;
 
-                 if (value !== undefined) {
-                   items.push(keyPart + value);
-                 }
-               });
-               delimiters = '{}';
-             }
+       var _sceneOptions = {
+         showFullscreenCtrl: false,
+         autoLoad: true,
+         compass: true,
+         yaw: 0,
+         minHfov: minHfov,
+         maxHfov: maxHfov,
+         hfov: defaultHfov,
+         type: 'cubemap',
+         cubeMap: []
+       };
 
-             if (items.length > 0) {
-               return [delimiters[0], indent + items.join(',\n' + nextIndent), delimiters[1]].join('\n' + currentIndent);
-             }
-           }
+       var _loadViewerPromise;
+       /**
+        * abortRequest().
+        */
 
-           return string;
-         }(obj, '', 0);
-       } // Note: This regex matches even invalid JSON strings, but since we’re
-       // working on the output of `JSON.stringify` we know that only valid strings
-       // are present (unless the user supplied a weird `options.indent` but in
-       // that case we don’t care since the output would be invalid anyway).
 
+       function abortRequest$1(i) {
+         i.abort();
+       }
+       /**
+        * localeTimeStamp().
+        */
 
-       var stringOrChar = /("(?:[^\\"]|\\.)*")|[:,\][}{]/g;
 
-       function prettify(string, options) {
-         options = options || {};
-         var tokens = {
-           '{': '{',
-           '}': '}',
-           '[': '[',
-           ']': ']',
-           ',': ', ',
-           ':': ': '
+       function localeTimestamp(s) {
+         if (!s) return null;
+         var options = {
+           day: 'numeric',
+           month: 'short',
+           year: 'numeric'
          };
+         var d = new Date(s);
+         if (isNaN(d.getTime())) return null;
+         return d.toLocaleString(_mainLocalizer.localeCode(), options);
+       }
+       /**
+        * loadTiles() wraps the process of generating tiles and then fetching image points for each tile.
+        */
 
-         if (options.addMargin || options.addObjectMargin) {
-           tokens['{'] = '{ ';
-           tokens['}'] = ' }';
-         }
 
-         if (options.addMargin || options.addArrayMargin) {
-           tokens['['] = '[ ';
-           tokens[']'] = ' ]';
-         }
+       function loadTiles(which, url, projection, margin) {
+         var tiles = tiler$1.margin(margin).getTiles(projection); // abort inflight requests that are no longer needed
 
-         return string.replace(stringOrChar, function (match, string) {
-           return string ? match : tokens[match];
+         var cache = _ssCache[which];
+         Object.keys(cache.inflight).forEach(function (k) {
+           var wanted = tiles.find(function (tile) {
+             return k.indexOf(tile.id + ',') === 0;
+           });
+
+           if (!wanted) {
+             abortRequest$1(cache.inflight[k]);
+             delete cache.inflight[k];
+           }
+         });
+         tiles.forEach(function (tile) {
+           return loadNextTilePage(which, url, tile);
          });
        }
+       /**
+        * loadNextTilePage() load data for the next tile page in line.
+        */
 
-       function get$5(options, name, defaultValue) {
-         return name in options ? options[name] : defaultValue;
-       }
 
-       var jsonStringifyPrettyCompact = stringify;
+       function loadNextTilePage(which, url, tile) {
+         var cache = _ssCache[which];
+         var nextPage = cache.nextPage[tile.id] || 0;
+         var id = tile.id + ',' + String(nextPage);
+         if (cache.loaded[id] || cache.inflight[id]) return;
+         cache.inflight[id] = getBubbles(url, tile, function (bubbles) {
+           cache.loaded[id] = true;
+           delete cache.inflight[id];
+           if (!bubbles) return; // [].shift() removes the first element, some statistics info, not a bubble point
 
-       var _default$2 = /*#__PURE__*/function () {
-         // constructor
-         //
-         // `fc`  Optional FeatureCollection of known features
-         //
-         // Optionally pass a GeoJSON FeatureCollection of known features which we can refer to later.
-         // Each feature must have a filename-like `id`, for example: `something.geojson`
-         //
-         // {
-         //   "type": "FeatureCollection"
-         //   "features": [
-         //     {
-         //       "type": "Feature",
-         //       "id": "philly_metro.geojson",
-         //       "properties": { … },
-         //       "geometry": { … }
-         //     }
-         //   ]
-         // }
-         function _default(fc) {
-           var _this = this;
+           bubbles.shift();
+           var features = bubbles.map(function (bubble) {
+             if (cache.points[bubble.id]) return null; // skip duplicates
 
-           _classCallCheck(this, _default);
+             var loc = [bubble.lo, bubble.la];
+             var d = {
+               loc: loc,
+               key: bubble.id,
+               ca: bubble.he,
+               captured_at: bubble.cd,
+               captured_by: 'microsoft',
+               // nbn: bubble.nbn,
+               // pbn: bubble.pbn,
+               // ad: bubble.ad,
+               // rn: bubble.rn,
+               pr: bubble.pr,
+               // previous
+               ne: bubble.ne,
+               // next
+               pano: true,
+               sequenceKey: null
+             };
+             cache.points[bubble.id] = d; // a sequence starts here
 
-           // The _cache retains resolved features, so if you ask for the same thing multiple times
-           // we don't repeat the expensive resolving/clipping operations.
-           //
-           // Each feature has a stable identifier that is used as the cache key.
-           // The identifiers look like:
-           // - for point locations, the stringified point:          e.g. '[8.67039,49.41882]'
-           // - for geojson locations, the geojson id:               e.g. 'de-hamburg.geojson'
-           // - for countrycoder locations, feature.id property:     e.g. 'Q2'  (countrycoder uses Wikidata identifiers)
-           // - for aggregated locationSets, +[include]-[exclude]:   e.g '+[Q2]-[Q18,Q27611]'
-           this._cache = {}; // When strict mode = true, throw on invalid locations or locationSets.
-           // When strict mode = false, return `null` for invalid locations or locationSets.
+             if (bubble.pr === undefined) {
+               cache.leaders.push(bubble.id);
+             }
 
-           this._strict = true; // process input FeatureCollection
+             return {
+               minX: loc[0],
+               minY: loc[1],
+               maxX: loc[0],
+               maxY: loc[1],
+               data: d
+             };
+           }).filter(Boolean);
+           cache.rtree.load(features);
+           connectSequences();
 
-           if (fc && fc.type === 'FeatureCollection' && Array.isArray(fc.features)) {
-             fc.features.forEach(function (feature) {
-               feature.properties = feature.properties || {};
-               var props = feature.properties; // get `id` from either `id` or `properties`
+           if (which === 'bubbles') {
+             dispatch$1.call('loadedImages');
+           }
+         });
+       } // call this sometimes to connect the bubbles into sequences
 
-               var id = feature.id || props.id;
-               if (!id || !/^\S+\.geojson$/i.test(id)) return; // ensure `id` exists and is lowercase
 
-               id = id.toLowerCase();
-               feature.id = id;
-               props.id = id; // ensure `area` property exists
+       function connectSequences() {
+         var cache = _ssCache.bubbles;
+         var keepLeaders = [];
 
-               if (!props.area) {
-                 var area = geojsonArea.geometry(feature.geometry) / 1e6; // m² to km²
+         for (var i = 0; i < cache.leaders.length; i++) {
+           var bubble = cache.points[cache.leaders[i]];
+           var seen = {}; // try to make a sequence.. use the key of the leader bubble.
 
-                 props.area = Number(area.toFixed(2));
-               }
+           var sequence = {
+             key: bubble.key,
+             bubbles: []
+           };
+           var complete = false;
 
-               _this._cache[id] = feature;
-             });
-           } // Replace CountryCoder world geometry to be a polygon covering the world.
+           do {
+             sequence.bubbles.push(bubble);
+             seen[bubble.key] = true;
 
+             if (bubble.ne === undefined) {
+               complete = true;
+             } else {
+               bubble = cache.points[bubble.ne]; // advance to next
+             }
+           } while (bubble && !seen[bubble.key] && !complete);
 
-           var world = _cloneDeep(feature$1('Q2'));
+           if (complete) {
+             _ssCache.sequences[sequence.key] = sequence; // assign bubbles to the sequence
 
-           world.geometry = {
-             type: 'Polygon',
-             coordinates: [[[-180, -90], [180, -90], [180, 90], [-180, 90], [-180, -90]]]
-           };
-           world.id = 'Q2';
-           world.properties.id = 'Q2';
-           world.properties.area = geojsonArea.geometry(world.geometry) / 1e6; // m² to km²
+             for (var j = 0; j < sequence.bubbles.length; j++) {
+               sequence.bubbles[j].sequenceKey = sequence.key;
+             } // create a GeoJSON LineString
 
-           this._cache.Q2 = world;
-         } // validateLocation
-         // `location`  The location to validate
-         //
-         // Pass a `location` value to validate
-         //
-         // Returns a result like:
-         //   {
-         //     type:     'point', 'geojson', or 'countrycoder'
-         //     location:  the queried location
-         //     id:        the stable identifier for the feature
-         //   }
-         // or `null` if the location is invalid
-         //
 
+             sequence.geojson = {
+               type: 'LineString',
+               properties: {
+                 captured_at: sequence.bubbles[0] ? sequence.bubbles[0].captured_at : null,
+                 captured_by: sequence.bubbles[0] ? sequence.bubbles[0].captured_by : null,
+                 key: sequence.key
+               },
+               coordinates: sequence.bubbles.map(function (d) {
+                 return d.loc;
+               })
+             };
+           } else {
+             keepLeaders.push(cache.leaders[i]);
+           }
+         } // couldn't complete these, save for later
 
-         _createClass(_default, [{
-           key: "validateLocation",
-           value: function validateLocation(location) {
-             if (Array.isArray(location)) {
-               // a [lon,lat] coordinate pair?
-               if (location.length === 2 && Number.isFinite(location[0]) && Number.isFinite(location[1]) && location[0] >= -180 && location[0] <= 180 && location[1] >= -90 && location[1] <= 90) {
-                 var id = '[' + location.toString() + ']';
-                 return {
-                   type: 'point',
-                   location: location,
-                   id: id
-                 };
-               }
-             } else if (typeof location === 'string' && /^\S+\.geojson$/i.test(location)) {
-               // a .geojson filename?
-               var _id = location.toLowerCase();
 
-               if (this._cache[_id]) {
-                 return {
-                   type: 'geojson',
-                   location: location,
-                   id: _id
-                 };
-               }
-             } else if (typeof location === 'string' || typeof location === 'number') {
-               // a country-coder value?
-               var feature = feature$1(location);
+         cache.leaders = keepLeaders;
+       }
+       /**
+        * getBubbles() handles the request to the server for a tile extent of 'bubbles' (streetside image locations).
+        */
 
-               if (feature) {
-                 // Use wikidata QID as the identifier, since that seems to be the one
-                 // property that everything in CountryCoder is guaranteed to have.
-                 var _id2 = feature.properties.wikidata;
-                 return {
-                   type: 'countrycoder',
-                   location: location,
-                   id: _id2
-                 };
-               }
-             }
 
-             if (this._strict) {
-               throw new Error("validateLocation:  Invalid location: \"".concat(location, "\"."));
-             } else {
-               return null;
-             }
-           } // resolveLocation
-           // `location`  The location to resolve
-           //
-           // Pass a `location` value to resolve
-           //
-           // Returns a result like:
-           //   {
-           //     type:      'point', 'geojson', or 'countrycoder'
-           //     location:  the queried location
-           //     id:        a stable identifier for the feature
-           //     feature:   the resolved GeoJSON feature
-           //   }
-           //  or `null` if the location is invalid
-           //
+       function getBubbles(url, tile, callback) {
+         var rect = tile.extent.rectangle();
+         var urlForRequest = url + utilQsString({
+           n: rect[3],
+           s: rect[1],
+           e: rect[2],
+           w: rect[0],
+           c: maxResults,
+           appkey: bubbleAppKey,
+           jsCallback: '{callback}'
+         });
+         return jsonpRequest(urlForRequest, function (data) {
+           if (!data || data.error) {
+             callback(null);
+           } else {
+             callback(data);
+           }
+         });
+       } // partition viewport into higher zoom tiles
 
-         }, {
-           key: "resolveLocation",
-           value: function resolveLocation(location) {
-             var valid = this.validateLocation(location);
-             if (!valid) return null;
-             var id = valid.id; // return a result from cache if we can
 
-             if (this._cache[id]) {
-               return Object.assign(valid, {
-                 feature: this._cache[id]
-               });
-             } // a [lon,lat] coordinate pair?
+       function partitionViewport(projection) {
+         var z = geoScaleToZoom(projection.scale());
+         var z2 = Math.ceil(z * 2) / 2 + 2.5; // round to next 0.5 and add 2.5
 
+         var tiler = utilTiler().zoomExtent([z2, z2]);
+         return tiler.getTiles(projection).map(function (tile) {
+           return tile.extent;
+         });
+       } // no more than `limit` results per partition.
 
-             if (valid.type === 'point') {
-               var RADIUS = 25000; // meters
 
-               var EDGES = 10;
-               var PRECISION = 3;
-               var area = Math.PI * RADIUS * RADIUS / 1e6; // m² to km²
+       function searchLimited(limit, projection, rtree) {
+         limit = limit || 5;
+         return partitionViewport(projection).reduce(function (result, extent) {
+           var found = rtree.search(extent.bbox()).slice(0, limit).map(function (d) {
+             return d.data;
+           });
+           return found.length ? result.concat(found) : result;
+         }, []);
+       }
+       /**
+        * loadImage()
+        */
 
-               var feature = this._cache[id] = geojsonPrecision({
-                 type: 'Feature',
-                 id: id,
-                 properties: {
-                   id: id,
-                   area: Number(area.toFixed(2))
-                 },
-                 geometry: circleToPolygon(location, RADIUS, EDGES)
-               }, PRECISION);
-               return Object.assign(valid, {
-                 feature: feature
-               }); // a .geojson filename?
-             } else if (valid.type === 'geojson') ; else if (valid.type === 'countrycoder') {
-               var _feature = _cloneDeep(feature$1(id));
 
-               var props = _feature.properties; // -> This block of code is weird and requires some explanation. <-
-               // CountryCoder includes higher level features which are made up of members.
-               // These features don't have their own geometry, but CountryCoder provides an
-               //   `aggregateFeature` method to combine these members into a MultiPolygon.
-               // BUT, when we try to actually work with these aggregated MultiPolygons,
-               //   Turf/JSTS gets crashy because of topography bugs.
-               // SO, we'll aggregate the features ourselves by unioning them together.
-               // This approach also has the benefit of removing all the internal boaders and
-               //   simplifying the regional polygons a lot.
+       function loadImage(imgInfo) {
+         return new Promise(function (resolve) {
+           var img = new Image();
 
-               if (Array.isArray(props.members)) {
-                 var seed = _feature.geometry ? _feature : null;
-                 var aggregate = props.members.reduce(_locationReducer.bind(this), seed);
-                 _feature.geometry = aggregate.geometry;
-               } // ensure `area` property exists
+           img.onload = function () {
+             var canvas = document.getElementById('ideditor-canvas' + imgInfo.face);
+             var ctx = canvas.getContext('2d');
+             ctx.drawImage(img, imgInfo.x, imgInfo.y);
+             resolve({
+               imgInfo: imgInfo,
+               status: 'ok'
+             });
+           };
 
+           img.onerror = function () {
+             resolve({
+               data: imgInfo,
+               status: 'error'
+             });
+           };
 
-               if (!props.area) {
-                 var _area = geojsonArea.geometry(_feature.geometry) / 1e6; // m² to km²
+           img.setAttribute('crossorigin', '');
+           img.src = imgInfo.url;
+         });
+       }
+       /**
+        * loadCanvas()
+        */
 
 
-                 props.area = Number(_area.toFixed(2));
-               } // ensure `id` property exists
+       function loadCanvas(imageGroup) {
+         return Promise.all(imageGroup.map(loadImage)).then(function (data) {
+           var canvas = document.getElementById('ideditor-canvas' + data[0].imgInfo.face);
+           var which = {
+             '01': 0,
+             '02': 1,
+             '03': 2,
+             '10': 3,
+             '11': 4,
+             '12': 5
+           };
+           var face = data[0].imgInfo.face;
+           _sceneOptions.cubeMap[which[face]] = canvas.toDataURL('image/jpeg', 1.0);
+           return {
+             status: 'loadCanvas for face ' + data[0].imgInfo.face + 'ok'
+           };
+         });
+       }
+       /**
+        * loadFaces()
+        */
 
 
-               _feature.id = id;
-               props.id = id;
-               this._cache[id] = _feature;
-               return Object.assign(valid, {
-                 feature: _feature
-               });
-             }
+       function loadFaces(faceGroup) {
+         return Promise.all(faceGroup.map(loadCanvas)).then(function () {
+           return {
+             status: 'loadFaces done'
+           };
+         });
+       }
 
-             if (this._strict) {
-               throw new Error("resolveLocation:  Couldn't resolve location \"".concat(location, "\"."));
-             } else {
-               return null;
-             }
-           } // validateLocationSet
-           // `locationSet`  the locationSet to validate
-           //
-           // Pass a locationSet Object to validate like:
-           //   {
-           //     include: [ Array of locations ],
-           //     exclude: [ Array of locations ]
-           //   }
-           //
-           // Returns a result like:
-           //   {
-           //     type:         'locationset'
-           //     locationSet:  the queried locationSet
-           //     id:           the stable identifier for the feature
-           //   }
-           // or `null` if the locationSet is invalid
-           //
+       function setupCanvas(selection, reset) {
+         if (reset) {
+           selection.selectAll('#ideditor-stitcher-canvases').remove();
+         } // Add the Streetside working canvases. These are used for 'stitching', or combining,
+         // multiple images for each of the six faces, before passing to the Pannellum control as DataUrls
+
+
+         selection.selectAll('#ideditor-stitcher-canvases').data([0]).enter().append('div').attr('id', 'ideditor-stitcher-canvases').attr('display', 'none').selectAll('canvas').data(['canvas01', 'canvas02', 'canvas03', 'canvas10', 'canvas11', 'canvas12']).enter().append('canvas').attr('id', function (d) {
+           return 'ideditor-' + d;
+         }).attr('width', _resolution).attr('height', _resolution);
+       }
+
+       function qkToXY(qk) {
+         var x = 0;
+         var y = 0;
+         var scale = 256;
+
+         for (var i = qk.length; i > 0; i--) {
+           var key = qk[i - 1];
+           x += +(key === '1' || key === '3') * scale;
+           y += +(key === '2' || key === '3') * scale;
+           scale *= 2;
+         }
 
-         }, {
-           key: "validateLocationSet",
-           value: function validateLocationSet(locationSet) {
-             locationSet = locationSet || {};
-             var validator = this.validateLocation.bind(this);
-             var include = (locationSet.include || []).map(validator).filter(Boolean);
-             var exclude = (locationSet.exclude || []).map(validator).filter(Boolean);
+         return [x, y];
+       }
 
-             if (!include.length) {
-               if (this._strict) {
-                 throw new Error("validateLocationSet:  LocationSet includes nothing.");
-               } else {
-                 // non-strict mode, replace an empty locationSet with one that includes "the world"
-                 locationSet.include = ['Q2'];
-                 include = [{
-                   type: 'countrycoder',
-                   location: 'Q2',
-                   id: 'Q2'
-                 }];
-               }
-             } // generate stable identifier
+       function getQuadKeys() {
+         var dim = _resolution / 256;
+         var quadKeys;
 
+         if (dim === 16) {
+           quadKeys = ['0000', '0001', '0010', '0011', '0100', '0101', '0110', '0111', '1000', '1001', '1010', '1011', '1100', '1101', '1110', '1111', '0002', '0003', '0012', '0013', '0102', '0103', '0112', '0113', '1002', '1003', '1012', '1013', '1102', '1103', '1112', '1113', '0020', '0021', '0030', '0031', '0120', '0121', '0130', '0131', '1020', '1021', '1030', '1031', '1120', '1121', '1130', '1131', '0022', '0023', '0032', '0033', '0122', '0123', '0132', '0133', '1022', '1023', '1032', '1033', '1122', '1123', '1132', '1133', '0200', '0201', '0210', '0211', '0300', '0301', '0310', '0311', '1200', '1201', '1210', '1211', '1300', '1301', '1310', '1311', '0202', '0203', '0212', '0213', '0302', '0303', '0312', '0313', '1202', '1203', '1212', '1213', '1302', '1303', '1312', '1313', '0220', '0221', '0230', '0231', '0320', '0321', '0330', '0331', '1220', '1221', '1230', '1231', '1320', '1321', '1330', '1331', '0222', '0223', '0232', '0233', '0322', '0323', '0332', '0333', '1222', '1223', '1232', '1233', '1322', '1323', '1332', '1333', '2000', '2001', '2010', '2011', '2100', '2101', '2110', '2111', '3000', '3001', '3010', '3011', '3100', '3101', '3110', '3111', '2002', '2003', '2012', '2013', '2102', '2103', '2112', '2113', '3002', '3003', '3012', '3013', '3102', '3103', '3112', '3113', '2020', '2021', '2030', '2031', '2120', '2121', '2130', '2131', '3020', '3021', '3030', '3031', '3120', '3121', '3130', '3131', '2022', '2023', '2032', '2033', '2122', '2123', '2132', '2133', '3022', '3023', '3032', '3033', '3122', '3123', '3132', '3133', '2200', '2201', '2210', '2211', '2300', '2301', '2310', '2311', '3200', '3201', '3210', '3211', '3300', '3301', '3310', '3311', '2202', '2203', '2212', '2213', '2302', '2303', '2312', '2313', '3202', '3203', '3212', '3213', '3302', '3303', '3312', '3313', '2220', '2221', '2230', '2231', '2320', '2321', '2330', '2331', '3220', '3221', '3230', '3231', '3320', '3321', '3330', '3331', '2222', '2223', '2232', '2233', '2322', '2323', '2332', '2333', '3222', '3223', '3232', '3233', '3322', '3323', '3332', '3333'];
+         } else if (dim === 8) {
+           quadKeys = ['000', '001', '010', '011', '100', '101', '110', '111', '002', '003', '012', '013', '102', '103', '112', '113', '020', '021', '030', '031', '120', '121', '130', '131', '022', '023', '032', '033', '122', '123', '132', '133', '200', '201', '210', '211', '300', '301', '310', '311', '202', '203', '212', '213', '302', '303', '312', '313', '220', '221', '230', '231', '320', '321', '330', '331', '222', '223', '232', '233', '322', '323', '332', '333'];
+         } else if (dim === 4) {
+           quadKeys = ['00', '01', '10', '11', '02', '03', '12', '13', '20', '21', '30', '31', '22', '23', '32', '33'];
+         } else {
+           // dim === 2
+           quadKeys = ['0', '1', '2', '3'];
+         }
 
-             include.sort(_sortLocations);
-             var id = '+[' + include.map(function (d) {
-               return d.id;
-             }).join(',') + ']';
+         return quadKeys;
+       }
 
-             if (exclude.length) {
-               exclude.sort(_sortLocations);
-               id += '-[' + exclude.map(function (d) {
-                 return d.id;
-               }).join(',') + ']';
-             }
+       var serviceStreetside = {
+         /**
+          * init() initialize streetside.
+          */
+         init: function init() {
+           if (!_ssCache) {
+             this.reset();
+           }
 
-             return {
-               type: 'locationset',
-               locationSet: locationSet,
-               id: id
-             };
-           } // resolveLocationSet
-           // `locationSet`  the locationSet to resolve
-           //
-           // Pass a locationSet Object to validate like:
-           //   {
-           //     include: [ Array of locations ],
-           //     exclude: [ Array of locations ]
-           //   }
-           //
-           // Returns a result like:
-           //   {
-           //     type:         'locationset'
-           //     locationSet:  the queried locationSet
-           //     id:           the stable identifier for the feature
-           //     feature:      the resolved GeoJSON feature
-           //   }
-           // or `null` if the locationSet is invalid
-           //
+           this.event = utilRebind(this, dispatch$1, 'on');
+         },
 
-         }, {
-           key: "resolveLocationSet",
-           value: function resolveLocationSet(locationSet) {
-             locationSet = locationSet || {};
-             var valid = this.validateLocationSet(locationSet);
-             if (!valid) return null;
-             var id = valid.id; // return a result from cache if we can
+         /**
+          * reset() reset the cache.
+          */
+         reset: function reset() {
+           if (_ssCache) {
+             Object.values(_ssCache.bubbles.inflight).forEach(abortRequest$1);
+           }
 
-             if (this._cache[id]) {
-               return Object.assign(valid, {
-                 feature: this._cache[id]
-               });
-             }
+           _ssCache = {
+             bubbles: {
+               inflight: {},
+               loaded: {},
+               nextPage: {},
+               rtree: new RBush(),
+               points: {},
+               leaders: []
+             },
+             sequences: {}
+           };
+         },
 
-             var resolver = this.resolveLocation.bind(this);
-             var include = (locationSet.include || []).map(resolver).filter(Boolean);
-             var exclude = (locationSet.exclude || []).map(resolver).filter(Boolean); // return quickly if it's a single included location..
+         /**
+          * bubbles()
+          */
+         bubbles: function bubbles(projection) {
+           var limit = 5;
+           return searchLimited(limit, projection, _ssCache.bubbles.rtree);
+         },
+         cachedImage: function cachedImage(imageKey) {
+           return _ssCache.bubbles.points[imageKey];
+         },
+         sequences: function sequences(projection) {
+           var viewport = projection.clipExtent();
+           var min = [viewport[0][0], viewport[1][1]];
+           var max = [viewport[1][0], viewport[0][1]];
+           var bbox = geoExtent(projection.invert(min), projection.invert(max)).bbox();
+           var seen = {};
+           var results = []; // all sequences for bubbles in viewport
 
-             if (include.length === 1 && exclude.length === 0) {
-               return Object.assign(valid, {
-                 feature: include[0].feature
-               });
-             } // calculate unions
+           _ssCache.bubbles.rtree.search(bbox).forEach(function (d) {
+             var key = d.data.sequenceKey;
 
+             if (key && !seen[key]) {
+               seen[key] = true;
+               results.push(_ssCache.sequences[key].geojson);
+             }
+           });
 
-             var includeGeoJSON = include.map(function (d) {
-               return d.location;
-             }).reduce(_locationReducer.bind(this), null);
-             var excludeGeoJSON = exclude.map(function (d) {
-               return d.location;
-             }).reduce(_locationReducer.bind(this), null); // calculate difference, update `area` and return result
+           return results;
+         },
 
-             var resultGeoJSON = excludeGeoJSON ? _clip(includeGeoJSON, excludeGeoJSON, 'DIFFERENCE') : includeGeoJSON;
-             var area = geojsonArea.geometry(resultGeoJSON.geometry) / 1e6; // m² to km²
+         /**
+          * loadBubbles()
+          */
+         loadBubbles: function loadBubbles(projection, margin) {
+           // by default: request 2 nearby tiles so we can connect sequences.
+           if (margin === undefined) margin = 2;
+           loadTiles('bubbles', bubbleApi, projection, margin);
+         },
+         viewer: function viewer() {
+           return _pannellumViewer;
+         },
+         initViewer: function initViewer() {
+           if (!window.pannellum) return;
+           if (_pannellumViewer) return;
+           _currScene += 1;
 
-             resultGeoJSON.id = id;
-             resultGeoJSON.properties = {
-               id: id,
-               area: Number(area.toFixed(2))
-             };
-             this._cache[id] = resultGeoJSON;
-             return Object.assign(valid, {
-               feature: resultGeoJSON
-             });
-           } // strict
-           //
+           var sceneID = _currScene.toString();
 
-         }, {
-           key: "strict",
-           value: function strict(val) {
-             if (val === undefined) {
-               // get
-               return this._strict;
-             } else {
-               // set
-               this._strict = val;
-               return this;
-             }
-           } // cache
-           // convenience method to access the internal cache
+           var options = {
+             'default': {
+               firstScene: sceneID
+             },
+             scenes: {}
+           };
+           options.scenes[sceneID] = _sceneOptions;
+           _pannellumViewer = window.pannellum.viewer('ideditor-viewer-streetside', options);
+         },
+         ensureViewerLoaded: function ensureViewerLoaded(context) {
+           if (_loadViewerPromise) return _loadViewerPromise; // create ms-wrapper, a photo wrapper class
 
-         }, {
-           key: "cache",
-           value: function cache() {
-             return this._cache;
-           } // stringify
-           // convenience method to prettyStringify the given object
+           var wrap = context.container().select('.photoviewer').selectAll('.ms-wrapper').data([0]); // inject ms-wrapper into the photoviewer div
+           // (used by all to house each custom photo viewer)
 
-         }, {
-           key: "stringify",
-           value: function stringify(obj, options) {
-             return jsonStringifyPrettyCompact(obj, options);
-           }
-         }]);
+           var wrapEnter = wrap.enter().append('div').attr('class', 'photo-wrapper ms-wrapper').classed('hide', true);
+           var that = this;
+           var pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse'; // inject div to support streetside viewer (pannellum) and attribution line
 
-         return _default;
-       }(); // Wrap the mfogel/polygon-clipping library and return a GeoJSON feature.
+           wrapEnter.append('div').attr('id', 'ideditor-viewer-streetside').on(pointerPrefix + 'down.streetside', function () {
+             select(window).on(pointerPrefix + 'move.streetside', function () {
+               dispatch$1.call('viewerChanged');
+             }, true);
+           }).on(pointerPrefix + 'up.streetside pointercancel.streetside', function () {
+             select(window).on(pointerPrefix + 'move.streetside', null); // continue dispatching events for a few seconds, in case viewer has inertia.
 
-       function _clip(a, b, which) {
-         var fn = {
-           UNION: index$1.union,
-           DIFFERENCE: index$1.difference
-         }[which];
-         var coords = fn(a.geometry.coordinates, b.geometry.coordinates);
-         return {
-           type: 'Feature',
-           properties: {},
-           geometry: {
-             type: whichType(coords),
-             coordinates: coords
-           }
-         }; // is this a Polygon or a MultiPolygon?
+             var t = timer(function (elapsed) {
+               dispatch$1.call('viewerChanged');
 
-         function whichType(coords) {
-           var a = Array.isArray(coords);
-           var b = a && Array.isArray(coords[0]);
-           var c = b && Array.isArray(coords[0][0]);
-           var d = c && Array.isArray(coords[0][0][0]);
-           return d ? 'MultiPolygon' : 'Polygon';
-         }
-       } // Reduce an array of locations into a single GeoJSON feature
+               if (elapsed > 2000) {
+                 t.stop();
+               }
+             });
+           }).append('div').attr('class', 'photo-attribution fillD');
+           var controlsEnter = wrapEnter.append('div').attr('class', 'photo-controls-wrap').append('div').attr('class', 'photo-controls');
+           controlsEnter.append('button').on('click.back', step(-1)).html('◄');
+           controlsEnter.append('button').on('click.forward', step(1)).html('►'); // create working canvas for stitching together images
 
+           wrap = wrap.merge(wrapEnter).call(setupCanvas, true); // Register viewer resize handler
 
-       function _locationReducer(accumulator, location) {
-         /* eslint-disable no-console, no-invalid-this */
-         var result;
+           context.ui().photoviewer.on('resize.streetside', function () {
+             if (_pannellumViewer) {
+               _pannellumViewer.resize();
+             }
+           });
+           _loadViewerPromise = new Promise(function (resolve, reject) {
+             var loadedCount = 0;
 
-         try {
-           var resolved = this.resolveLocation(location);
+             function loaded() {
+               loadedCount += 1; // wait until both files are loaded
 
-           if (!resolved || !resolved.feature) {
-             console.warn("Warning:  Couldn't resolve location \"".concat(location, "\""));
-             return accumulator;
-           }
+               if (loadedCount === 2) resolve();
+             }
 
-           result = !accumulator ? resolved.feature : _clip(accumulator, resolved.feature, 'UNION');
-         } catch (e) {
-           console.warn("Warning:  Error resolving location \"".concat(location, "\""));
-           console.warn(e);
-           result = accumulator;
-         }
+             var head = select('head'); // load streetside pannellum viewer css
 
-         return result;
-         /* eslint-enable no-console, no-invalid-this */
-       }
+             head.selectAll('#ideditor-streetside-viewercss').data([0]).enter().append('link').attr('id', 'ideditor-streetside-viewercss').attr('rel', 'stylesheet').attr('crossorigin', 'anonymous').attr('href', context.asset(pannellumViewerCSS)).on('load.serviceStreetside', loaded).on('error.serviceStreetside', function () {
+               reject();
+             }); // load streetside pannellum viewer js
 
-       function _cloneDeep(obj) {
-         return JSON.parse(JSON.stringify(obj));
-       } // Sorting the location lists is ok because they end up unioned together.
-       // This sorting makes it possible to generate a deterministic id.
+             head.selectAll('#ideditor-streetside-viewerjs').data([0]).enter().append('script').attr('id', 'ideditor-streetside-viewerjs').attr('crossorigin', 'anonymous').attr('src', context.asset(pannellumViewerJS)).on('load.serviceStreetside', loaded).on('error.serviceStreetside', function () {
+               reject();
+             });
+           })["catch"](function () {
+             _loadViewerPromise = null;
+           });
+           return _loadViewerPromise;
 
+           function step(stepBy) {
+             return function () {
+               var viewer = context.container().select('.photoviewer');
+               var selected = viewer.empty() ? undefined : viewer.datum();
+               if (!selected) return;
+               var nextID = stepBy === 1 ? selected.ne : selected.pr;
 
-       function _sortLocations(a, b) {
-         var rank = {
-           countrycoder: 1,
-           geojson: 2,
-           point: 3
-         };
-         var aRank = rank[a.type];
-         var bRank = rank[b.type];
-         return aRank > bRank ? 1 : aRank < bRank ? -1 : a.id.localeCompare(b.id);
-       }
+               var yaw = _pannellumViewer.getYaw();
 
-       var _oci = null;
-       function uiSuccess(context) {
-         var MAXEVENTS = 2;
-         var dispatch$1 = dispatch('cancel');
+               var ca = selected.ca + yaw;
+               var origin = selected.loc; // construct a search trapezoid pointing out from current bubble
 
-         var _changeset;
+               var meters = 35;
+               var p1 = [origin[0] + geoMetersToLon(meters / 5, origin[1]), origin[1]];
+               var p2 = [origin[0] + geoMetersToLon(meters / 2, origin[1]), origin[1] + geoMetersToLat(meters)];
+               var p3 = [origin[0] - geoMetersToLon(meters / 2, origin[1]), origin[1] + geoMetersToLat(meters)];
+               var p4 = [origin[0] - geoMetersToLon(meters / 5, origin[1]), origin[1]];
+               var poly = [p1, p2, p3, p4, p1]; // rotate it to face forward/backward
 
-         var _location;
+               var angle = (stepBy === 1 ? ca : ca + 180) * (Math.PI / 180);
+               poly = geoRotate(poly, -angle, origin);
+               var extent = poly.reduce(function (extent, point) {
+                 return extent.extend(geoExtent(point));
+               }, geoExtent()); // find nearest other bubble in the search polygon
 
-         ensureOSMCommunityIndex(); // start fetching the data
+               var minDist = Infinity;
 
-         function ensureOSMCommunityIndex() {
-           var data = _mainFileFetcher;
-           return Promise.all([data.get('oci_resources'), data.get('oci_features')]).then(function (vals) {
-             if (_oci) return _oci;
-             var ociResources = vals[0].resources;
-             var loco = new _default$2(vals[1]);
-             var ociFeatures = {};
-             Object.values(ociResources).forEach(function (resource) {
-               var feature = loco.resolveLocationSet(resource.locationSet).feature;
-               var ociFeature = ociFeatures[feature.id];
+               _ssCache.bubbles.rtree.search(extent.bbox()).forEach(function (d) {
+                 if (d.data.key === selected.key) return;
+                 if (!geoPointInPolygon(d.data.loc, poly)) return;
+                 var dist = geoVecLength(d.data.loc, selected.loc);
+                 var theta = selected.ca - d.data.ca;
+                 var minTheta = Math.min(Math.abs(theta), 360 - Math.abs(theta));
 
-               if (!ociFeature) {
-                 ociFeature = JSON.parse(JSON.stringify(feature)); // deep clone
+                 if (minTheta > 20) {
+                   dist += 5; // penalize distance if camera angles don't match
+                 }
 
-                 ociFeature.properties.resourceIDs = new Set();
-                 ociFeatures[feature.id] = ociFeature;
-               }
+                 if (dist < minDist) {
+                   nextID = d.data.key;
+                   minDist = dist;
+                 }
+               });
 
-               ociFeature.properties.resourceIDs.add(resource.id);
-             });
-             return _oci = {
-               features: ociFeatures,
-               resources: ociResources,
-               query: whichPolygon_1({
-                 type: 'FeatureCollection',
-                 features: Object.values(ociFeatures)
-               })
+               var nextBubble = nextID && that.cachedImage(nextID);
+               if (!nextBubble) return;
+               context.map().centerEase(nextBubble.loc);
+               that.selectImage(context, nextBubble.key).yaw(yaw).showViewer(context);
              };
-           });
-         } // string-to-date parsing in JavaScript is weird
-
+           }
+         },
+         yaw: function yaw(_yaw) {
+           if (typeof _yaw !== 'number') return _yaw;
+           _sceneOptions.yaw = _yaw;
+           return this;
+         },
 
-         function parseEventDate(when) {
-           if (!when) return;
-           var raw = when.trim();
-           if (!raw) return;
+         /**
+          * showViewer()
+          */
+         showViewer: function showViewer(context) {
+           var wrap = context.container().select('.photoviewer').classed('hide', false);
+           var isHidden = wrap.selectAll('.photo-wrapper.ms-wrapper.hide').size();
 
-           if (!/Z$/.test(raw)) {
-             // if no trailing 'Z', add one
-             raw += 'Z'; // this forces date to be parsed as a UTC date
+           if (isHidden) {
+             wrap.selectAll('.photo-wrapper:not(.ms-wrapper)').classed('hide', true);
+             wrap.selectAll('.photo-wrapper.ms-wrapper').classed('hide', false);
            }
 
-           var parsed = new Date(raw);
-           return new Date(parsed.toUTCString().substr(0, 25)); // convert to local timezone
-         }
+           return this;
+         },
 
-         function success(selection) {
-           var header = selection.append('div').attr('class', 'header fillL');
-           header.append('h3').html(_t.html('success.just_edited'));
-           header.append('button').attr('class', 'close').on('click', function () {
-             return dispatch$1.call('cancel');
-           }).call(svgIcon('#iD-icon-close'));
-           var body = selection.append('div').attr('class', 'body save-success fillL');
-           var summary = body.append('div').attr('class', 'save-summary');
-           summary.append('h3').html(_t.html('success.thank_you' + (_location ? '_location' : ''), {
-             where: _location
-           }));
-           summary.append('p').html(_t.html('success.help_html')).append('a').attr('class', 'link-out').attr('target', '_blank').attr('href', _t('success.help_link_url')).call(svgIcon('#iD-icon-out-link', 'inline')).append('span').html(_t.html('success.help_link_text'));
-           var osm = context.connection();
-           if (!osm) return;
-           var changesetURL = osm.changesetURL(_changeset.id);
-           var table = summary.append('table').attr('class', 'summary-table');
-           var row = table.append('tr').attr('class', 'summary-row');
-           row.append('td').attr('class', 'cell-icon summary-icon').append('a').attr('target', '_blank').attr('href', changesetURL).append('svg').attr('class', 'logo-small').append('use').attr('xlink:href', '#iD-logo-osm');
-           var summaryDetail = row.append('td').attr('class', 'cell-detail summary-detail');
-           summaryDetail.append('a').attr('class', 'cell-detail summary-view-on-osm').attr('target', '_blank').attr('href', changesetURL).html(_t.html('success.view_on_osm'));
-           summaryDetail.append('div').html(_t.html('success.changeset_id', {
-             changeset_id: "<a href=\"".concat(changesetURL, "\" target=\"_blank\">").concat(_changeset.id, "</a>")
-           })); // Get OSM community index features intersecting the map..
+         /**
+          * hideViewer()
+          */
+         hideViewer: function hideViewer(context) {
+           var viewer = context.container().select('.photoviewer');
+           if (!viewer.empty()) viewer.datum(null);
+           viewer.classed('hide', true).selectAll('.photo-wrapper').classed('hide', true);
+           context.container().selectAll('.viewfield-group, .sequence, .icon-sign').classed('currentView', false);
+           this.updateUrlImage(null);
+           return this.setStyles(context, null, true);
+         },
 
-           ensureOSMCommunityIndex().then(function (oci) {
-             var communities = [];
-             var properties = oci.query(context.map().center(), true) || []; // Gather the communities from the result
-
-             properties.forEach(function (props) {
-               var resourceIDs = Array.from(props.resourceIDs);
-               resourceIDs.forEach(function (resourceID) {
-                 var resource = oci.resources[resourceID];
-                 communities.push({
-                   area: props.area || Infinity,
-                   order: resource.order || 0,
-                   resource: resource
-                 });
-               });
-             }); // sort communities by feature area ascending, community order descending
+         /**
+          * selectImage().
+          */
+         selectImage: function selectImage(context, key) {
+           var that = this;
+           var d = this.cachedImage(key);
+           var viewer = context.container().select('.photoviewer');
+           if (!viewer.empty()) viewer.datum(d);
+           this.setStyles(context, null, true);
+           var wrap = context.container().select('.photoviewer .ms-wrapper');
+           var attribution = wrap.selectAll('.photo-attribution').html('');
+           wrap.selectAll('.pnlm-load-box') // display "loading.."
+           .style('display', 'block');
+           if (!d) return this;
+           this.updateUrlImage(key);
+           _sceneOptions.northOffset = d.ca;
+           var line1 = attribution.append('div').attr('class', 'attribution-row');
+           var hiresDomId = utilUniqueDomId('streetside-hires'); // Add hires checkbox
 
-             communities.sort(function (a, b) {
-               return a.area - b.area || b.order - a.order;
-             });
-             body.call(showCommunityLinks, communities.map(function (c) {
-               return c.resource;
-             }));
+           var label = line1.append('label').attr('for', hiresDomId).attr('class', 'streetside-hires');
+           label.append('input').attr('type', 'checkbox').attr('id', hiresDomId).property('checked', _hires).on('click', function (d3_event) {
+             d3_event.stopPropagation();
+             _hires = !_hires;
+             _resolution = _hires ? 1024 : 512;
+             wrap.call(setupCanvas, true);
+             var viewstate = {
+               yaw: _pannellumViewer.getYaw(),
+               pitch: _pannellumViewer.getPitch(),
+               hfov: _pannellumViewer.getHfov()
+             };
+             _sceneOptions = Object.assign(_sceneOptions, viewstate);
+             that.selectImage(context, d.key).showViewer(context);
            });
-         }
+           label.append('span').html(_t.html('streetside.hires'));
+           var captureInfo = line1.append('div').attr('class', 'attribution-capture-info'); // Add capture date
 
-         function showCommunityLinks(selection, resources) {
-           var communityLinks = selection.append('div').attr('class', 'save-communityLinks');
-           communityLinks.append('h3').html(_t.html('success.like_osm'));
-           var table = communityLinks.append('table').attr('class', 'community-table');
-           var row = table.selectAll('.community-row').data(resources);
-           var rowEnter = row.enter().append('tr').attr('class', 'community-row');
-           rowEnter.append('td').attr('class', 'cell-icon community-icon').append('a').attr('target', '_blank').attr('href', function (d) {
-             return d.url;
-           }).append('svg').attr('class', 'logo-small').append('use').attr('xlink:href', function (d) {
-             return "#community-".concat(d.type);
-           });
-           var communityDetail = rowEnter.append('td').attr('class', 'cell-detail community-detail');
-           communityDetail.each(showCommunityDetails);
-           communityLinks.append('div').attr('class', 'community-missing').html(_t.html('success.missing')).append('a').attr('class', 'link-out').attr('target', '_blank').call(svgIcon('#iD-icon-out-link', 'inline')).attr('href', 'https://github.com/osmlab/osm-community-index/issues').append('span').html(_t.html('success.tell_us'));
-         }
+           if (d.captured_by) {
+             var yyyy = new Date().getFullYear();
+             captureInfo.append('a').attr('class', 'captured_by').attr('target', '_blank').attr('href', 'https://www.microsoft.com/en-us/maps/streetside').html('©' + yyyy + ' Microsoft');
+             captureInfo.append('span').html('|');
+           }
 
-         function showCommunityDetails(d) {
-           var selection = select(this);
-           var communityID = d.id;
-           var replacements = {
-             url: linkify(d.url),
-             signupUrl: linkify(d.signupUrl || d.url)
-           };
-           selection.append('div').attr('class', 'community-name').append('a').attr('target', '_blank').attr('href', d.url).html(_t.html("community.".concat(d.id, ".name")));
-           var descriptionHTML = _t.html("community.".concat(d.id, ".description"), replacements);
+           if (d.captured_at) {
+             captureInfo.append('span').attr('class', 'captured_at').html(localeTimestamp(d.captured_at));
+           } // Add image links
 
-           if (d.type === 'reddit') {
-             // linkify subreddits  #4997
-             descriptionHTML = descriptionHTML.replace(/(\/r\/\w*\/*)/i, function (match) {
-               return linkify(d.url, match);
-             });
-           }
 
-           selection.append('div').attr('class', 'community-description').html(descriptionHTML);
+           var line2 = attribution.append('div').attr('class', 'attribution-row');
+           line2.append('a').attr('class', 'image-view-link').attr('target', '_blank').attr('href', 'https://www.bing.com/maps?cp=' + d.loc[1] + '~' + d.loc[0] + '&lvl=17&dir=' + d.ca + '&style=x&v=2&sV=1').html(_t.html('streetside.view_on_bing'));
+           line2.append('a').attr('class', 'image-report-link').attr('target', '_blank').attr('href', 'https://www.bing.com/maps/privacyreport/streetsideprivacyreport?bubbleid=' + encodeURIComponent(d.key) + '&focus=photo&lat=' + d.loc[1] + '&lng=' + d.loc[0] + '&z=17').html(_t.html('streetside.report'));
+           var bubbleIdQuadKey = d.key.toString(4);
+           var paddingNeeded = 16 - bubbleIdQuadKey.length;
 
-           if (d.extendedDescription || d.languageCodes && d.languageCodes.length) {
-             selection.append('div').call(uiDisclosure(context, "community-more-".concat(d.id), false).expanded(false).updatePreference(false).label(_t.html('success.more')).content(showMore));
+           for (var i = 0; i < paddingNeeded; i++) {
+             bubbleIdQuadKey = '0' + bubbleIdQuadKey;
            }
 
-           var nextEvents = (d.events || []).map(function (event) {
-             event.date = parseEventDate(event.when);
-             return event;
-           }).filter(function (event) {
-             // date is valid and future (or today)
-             var t = event.date.getTime();
-             var now = new Date().setHours(0, 0, 0, 0);
-             return !isNaN(t) && t >= now;
-           }).sort(function (a, b) {
-             // sort by date ascending
-             return a.date < b.date ? -1 : a.date > b.date ? 1 : 0;
-           }).slice(0, MAXEVENTS); // limit number of events shown
+           var imgUrlPrefix = streetsideImagesApi + 'hs' + bubbleIdQuadKey;
+           var imgUrlSuffix = '.jpg?g=6338&n=z'; // Cubemap face code order matters here: front=01, right=02, back=03, left=10, up=11, down=12
+
+           var faceKeys = ['01', '02', '03', '10', '11', '12']; // Map images to cube faces
+
+           var quadKeys = getQuadKeys();
+           var faces = faceKeys.map(function (faceKey) {
+             return quadKeys.map(function (quadKey) {
+               var xy = qkToXY(quadKey);
+               return {
+                 face: faceKey,
+                 url: imgUrlPrefix + faceKey + quadKey + imgUrlSuffix,
+                 x: xy[0],
+                 y: xy[1]
+               };
+             });
+           });
+           loadFaces(faces).then(function () {
+             if (!_pannellumViewer) {
+               that.initViewer();
+             } else {
+               // make a new scene
+               _currScene += 1;
 
-           if (nextEvents.length) {
-             selection.append('div').call(uiDisclosure(context, "community-events-".concat(d.id), false).expanded(false).updatePreference(false).label(_t.html('success.events')).content(showNextEvents)).select('.hide-toggle').append('span').attr('class', 'badge-text').html(nextEvents.length);
-           }
+               var sceneID = _currScene.toString();
 
-           function showMore(selection) {
-             var more = selection.selectAll('.community-more').data([0]);
-             var moreEnter = more.enter().append('div').attr('class', 'community-more');
+               _pannellumViewer.addScene(sceneID, _sceneOptions).loadScene(sceneID); // remove previous scene
 
-             if (d.extendedDescription) {
-               moreEnter.append('div').attr('class', 'community-extended-description').html(_t.html("community.".concat(d.id, ".extendedDescription"), replacements));
-             }
 
-             if (d.languageCodes && d.languageCodes.length) {
-               var languageList = d.languageCodes.map(function (code) {
-                 return _mainLocalizer.languageName(code);
-               }).join(', ');
-               moreEnter.append('div').attr('class', 'community-languages').html(_t.html('success.languages', {
-                 languages: languageList
-               }));
+               if (_currScene > 2) {
+                 sceneID = (_currScene - 1).toString();
+
+                 _pannellumViewer.removeScene(sceneID);
+               }
              }
+           });
+           return this;
+         },
+         getSequenceKeyForBubble: function getSequenceKeyForBubble(d) {
+           return d && d.sequenceKey;
+         },
+         // Updates the currently highlighted sequence and selected bubble.
+         // Reset is only necessary when interacting with the viewport because
+         // this implicitly changes the currently selected bubble/sequence
+         setStyles: function setStyles(context, hovered, reset) {
+           if (reset) {
+             // reset all layers
+             context.container().selectAll('.viewfield-group').classed('highlighted', false).classed('hovered', false).classed('currentView', false);
+             context.container().selectAll('.sequence').classed('highlighted', false).classed('currentView', false);
            }
 
-           function showNextEvents(selection) {
-             var events = selection.append('div').attr('class', 'community-events');
-             var item = events.selectAll('.community-event').data(nextEvents);
-             var itemEnter = item.enter().append('div').attr('class', 'community-event');
-             itemEnter.append('div').attr('class', 'community-event-name').append('a').attr('target', '_blank').attr('href', function (d) {
-               return d.url;
-             }).html(function (d) {
-               var name = d.name;
-
-               if (d.i18n && d.id) {
-                 name = _t("community.".concat(communityID, ".events.").concat(d.id, ".name"), {
-                   "default": name
-                 });
-               }
+           var hoveredBubbleKey = hovered && hovered.key;
+           var hoveredSequenceKey = this.getSequenceKeyForBubble(hovered);
+           var hoveredSequence = hoveredSequenceKey && _ssCache.sequences[hoveredSequenceKey];
+           var hoveredBubbleKeys = hoveredSequence && hoveredSequence.bubbles.map(function (d) {
+             return d.key;
+           }) || [];
+           var viewer = context.container().select('.photoviewer');
+           var selected = viewer.empty() ? undefined : viewer.datum();
+           var selectedBubbleKey = selected && selected.key;
+           var selectedSequenceKey = this.getSequenceKeyForBubble(selected);
+           var selectedSequence = selectedSequenceKey && _ssCache.sequences[selectedSequenceKey];
+           var selectedBubbleKeys = selectedSequence && selectedSequence.bubbles.map(function (d) {
+             return d.key;
+           }) || []; // highlight sibling viewfields on either the selected or the hovered sequences
 
-               return name;
-             });
-             itemEnter.append('div').attr('class', 'community-event-when').html(function (d) {
-               var options = {
-                 weekday: 'short',
-                 day: 'numeric',
-                 month: 'short',
-                 year: 'numeric'
-               };
+           var highlightedBubbleKeys = utilArrayUnion(hoveredBubbleKeys, selectedBubbleKeys);
+           context.container().selectAll('.layer-streetside-images .viewfield-group').classed('highlighted', function (d) {
+             return highlightedBubbleKeys.indexOf(d.key) !== -1;
+           }).classed('hovered', function (d) {
+             return d.key === hoveredBubbleKey;
+           }).classed('currentView', function (d) {
+             return d.key === selectedBubbleKey;
+           });
+           context.container().selectAll('.layer-streetside-images .sequence').classed('highlighted', function (d) {
+             return d.properties.key === hoveredSequenceKey;
+           }).classed('currentView', function (d) {
+             return d.properties.key === selectedSequenceKey;
+           }); // update viewfields if needed
 
-               if (d.date.getHours() || d.date.getMinutes()) {
-                 // include time if it has one
-                 options.hour = 'numeric';
-                 options.minute = 'numeric';
-               }
+           context.container().selectAll('.layer-streetside-images .viewfield-group .viewfield').attr('d', viewfieldPath);
 
-               return d.date.toLocaleString(_mainLocalizer.localeCode(), options);
-             });
-             itemEnter.append('div').attr('class', 'community-event-where').html(function (d) {
-               var where = d.where;
+           function viewfieldPath() {
+             var d = this.parentNode.__data__;
 
-               if (d.i18n && d.id) {
-                 where = _t("community.".concat(communityID, ".events.").concat(d.id, ".where"), {
-                   "default": where
-                 });
-               }
+             if (d.pano && d.key !== selectedBubbleKey) {
+               return 'M 8,13 m -10,0 a 10,10 0 1,0 20,0 a 10,10 0 1,0 -20,0';
+             } else {
+               return 'M 6,9 C 8,8.4 8,8.4 10,9 L 16,-2 C 12,-5 4,-5 0,-2 z';
+             }
+           }
 
-               return where;
-             });
-             itemEnter.append('div').attr('class', 'community-event-description').html(function (d) {
-               var description = d.description;
+           return this;
+         },
+         updateUrlImage: function updateUrlImage(imageKey) {
+           if (!window.mocha) {
+             var hash = utilStringQs(window.location.hash);
 
-               if (d.i18n && d.id) {
-                 description = _t("community.".concat(communityID, ".events.").concat(d.id, ".description"), {
-                   "default": description
-                 });
-               }
+             if (imageKey) {
+               hash.photo = 'streetside/' + imageKey;
+             } else {
+               delete hash.photo;
+             }
 
-               return description;
-             });
+             window.location.replace('#' + utilQsString(hash, true));
            }
+         },
 
-           function linkify(url, text) {
-             text = text || url;
-             return "<a target=\"_blank\" href=\"".concat(url, "\">").concat(text, "</a>");
-           }
+         /**
+          * cache().
+          */
+         cache: function cache() {
+           return _ssCache;
          }
+       };
 
-         success.changeset = function (val) {
-           if (!arguments.length) return _changeset;
-           _changeset = val;
-           return success;
-         };
+       var _apibase = 'https://taginfo.openstreetmap.org/api/4/';
+       var _inflight = {};
+       var _popularKeys = {};
+       var _taginfoCache = {};
+       var tag_sorts = {
+         point: 'count_nodes',
+         vertex: 'count_nodes',
+         area: 'count_ways',
+         line: 'count_ways'
+       };
+       var tag_sort_members = {
+         point: 'count_node_members',
+         vertex: 'count_node_members',
+         area: 'count_way_members',
+         line: 'count_way_members',
+         relation: 'count_relation_members'
+       };
+       var tag_filters = {
+         point: 'nodes',
+         vertex: 'nodes',
+         area: 'ways',
+         line: 'ways'
+       };
+       var tag_members_fractions = {
+         point: 'count_node_members_fraction',
+         vertex: 'count_node_members_fraction',
+         area: 'count_way_members_fraction',
+         line: 'count_way_members_fraction',
+         relation: 'count_relation_members_fraction'
+       };
 
-         success.location = function (val) {
-           if (!arguments.length) return _location;
-           _location = val;
-           return success;
-         };
+       function sets(params, n, o) {
+         if (params.geometry && o[params.geometry]) {
+           params[n] = o[params.geometry];
+         }
 
-         return utilRebind(success, dispatch$1, 'on');
+         return params;
        }
 
-       function modeSave(context) {
-         var mode = {
-           id: 'save'
-         };
-         var keybinding = utilKeybinding('modeSave');
-         var commit = uiCommit(context).on('cancel', cancel);
-
-         var _conflictsUi; // uiConflicts
+       function setFilter(params) {
+         return sets(params, 'filter', tag_filters);
+       }
 
+       function setSort(params) {
+         return sets(params, 'sortname', tag_sorts);
+       }
 
-         var _location;
+       function setSortMembers(params) {
+         return sets(params, 'sortname', tag_sort_members);
+       }
 
-         var _success;
+       function clean(params) {
+         return utilObjectOmit(params, ['geometry', 'debounce']);
+       }
 
-         var uploader = context.uploader().on('saveStarted.modeSave', function () {
-           keybindingOff();
-         }) // fire off some async work that we want to be ready later
-         .on('willAttemptUpload.modeSave', prepareForSuccess).on('progressChanged.modeSave', showProgress).on('resultNoChanges.modeSave', function () {
-           cancel();
-         }).on('resultErrors.modeSave', showErrors).on('resultConflicts.modeSave', showConflicts).on('resultSuccess.modeSave', showSuccess);
+       function filterKeys(type) {
+         var count_type = type ? 'count_' + type : 'count_all';
+         return function (d) {
+           return parseFloat(d[count_type]) > 2500 || d.in_wiki;
+         };
+       }
 
-         function cancel() {
-           context.enter(modeBrowse(context));
-         }
+       function filterMultikeys(prefix) {
+         return function (d) {
+           // d.key begins with prefix, and d.key contains no additional ':'s
+           var re = new RegExp('^' + prefix + '(.*)$');
+           var matches = d.key.match(re) || [];
+           return matches.length === 2 && matches[1].indexOf(':') === -1;
+         };
+       }
 
-         function showProgress(num, total) {
-           var modal = context.container().select('.loading-modal .modal-section');
-           var progress = modal.selectAll('.progress').data([0]); // enter/update
+       function filterValues(allowUpperCase) {
+         return function (d) {
+           if (d.value.match(/[;,]/) !== null) return false; // exclude some punctuation
 
-           progress.enter().append('div').attr('class', 'progress').merge(progress).text(_t('save.conflict_progress', {
-             num: num,
-             total: total
-           }));
-         }
+           if (!allowUpperCase && d.value.match(/[A-Z*]/) !== null) return false; // exclude uppercase letters
 
-         function showConflicts(changeset, conflicts, origChanges) {
-           var selection = context.container().select('.sidebar').append('div').attr('class', 'sidebar-component');
-           context.container().selectAll('.main-content').classed('active', true).classed('inactive', false);
-           _conflictsUi = uiConflicts(context).conflictList(conflicts).origChanges(origChanges).on('cancel', function () {
-             context.container().selectAll('.main-content').classed('active', false).classed('inactive', true);
-             selection.remove();
-             keybindingOn();
-             uploader.cancelConflictResolution();
-           }).on('save', function () {
-             context.container().selectAll('.main-content').classed('active', false).classed('inactive', true);
-             selection.remove();
-             uploader.processResolvedConflicts(changeset);
-           });
-           selection.call(_conflictsUi);
-         }
+           return parseFloat(d.fraction) > 0.0;
+         };
+       }
 
-         function showErrors(errors) {
-           keybindingOn();
-           var selection = uiConfirm(context.container());
-           selection.select('.modal-section.header').append('h3').text(_t('save.error'));
-           addErrors(selection, errors);
-           selection.okButton();
-         }
+       function filterRoles(geometry) {
+         return function (d) {
+           if (d.role === '') return false; // exclude empty role
 
-         function addErrors(selection, data) {
-           var message = selection.select('.modal-section.message-text');
-           var items = message.selectAll('.error-container').data(data);
-           var enter = items.enter().append('div').attr('class', 'error-container');
-           enter.append('a').attr('class', 'error-description').attr('href', '#').classed('hide-toggle', true).text(function (d) {
-             return d.msg || _t('save.unknown_error_details');
-           }).on('click', function (d3_event) {
-             d3_event.preventDefault();
-             var error = select(this);
-             var detail = select(this.nextElementSibling);
-             var exp = error.classed('expanded');
-             detail.style('display', exp ? 'none' : 'block');
-             error.classed('expanded', !exp);
-           });
-           var details = enter.append('div').attr('class', 'error-detail-container').style('display', 'none');
-           details.append('ul').attr('class', 'error-detail-list').selectAll('li').data(function (d) {
-             return d.details || [];
-           }).enter().append('li').attr('class', 'error-detail-item').text(function (d) {
-             return d;
-           });
-           items.exit().remove();
-         }
+           if (d.role.match(/[A-Z*;,]/) !== null) return false; // exclude uppercase letters and some punctuation
 
-         function showSuccess(changeset) {
-           commit.reset();
+           return parseFloat(d[tag_members_fractions[geometry]]) > 0.0;
+         };
+       }
 
-           var ui = _success.changeset(changeset).location(_location).on('cancel', function () {
-             context.ui().sidebar.hide();
-           });
+       function valKey(d) {
+         return {
+           value: d.key,
+           title: d.key
+         };
+       }
 
-           context.enter(modeBrowse(context).sidebar(ui));
-         }
+       function valKeyDescription(d) {
+         var obj = {
+           value: d.value,
+           title: d.description || d.value
+         };
 
-         function keybindingOn() {
-           select(document).call(keybinding.on('⎋', cancel, true));
+         if (d.count) {
+           obj.count = d.count;
          }
 
-         function keybindingOff() {
-           select(document).call(keybinding.unbind);
-         } // Reverse geocode current map location so we can display a message on
-         // the success screen like "Thank you for editing around place, region."
+         return obj;
+       }
 
+       function roleKey(d) {
+         return {
+           value: d.role,
+           title: d.role
+         };
+       } // sort keys with ':' lower than keys without ':'
 
-         function prepareForSuccess() {
-           _success = uiSuccess(context);
-           _location = null;
-           if (!services.geocoder) return;
-           services.geocoder.reverse(context.map().center(), function (err, result) {
-             if (err || !result || !result.address) return;
-             var addr = result.address;
-             var place = addr && (addr.town || addr.city || addr.county) || '';
-             var region = addr && (addr.state || addr.country) || '';
-             var separator = place && region ? _t('success.thank_you_where.separator') : '';
-             _location = _t('success.thank_you_where.format', {
-               place: place,
-               separator: separator,
-               region: region
-             });
-           });
-         }
 
-         mode.selectedIDs = function () {
-           return _conflictsUi ? _conflictsUi.shownEntityIds() : [];
-         };
+       function sortKeys(a, b) {
+         return a.key.indexOf(':') === -1 && b.key.indexOf(':') !== -1 ? -1 : a.key.indexOf(':') !== -1 && b.key.indexOf(':') === -1 ? 1 : 0;
+       }
 
-         mode.enter = function () {
-           // Show sidebar
-           context.ui().sidebar.expand();
+       var debouncedRequest = debounce(request, 300, {
+         leading: false
+       });
 
-           function done() {
-             context.ui().sidebar.show(commit);
-           }
+       function request(url, params, exactMatch, callback, loaded) {
+         if (_inflight[url]) return;
+         if (checkCache(url, params, exactMatch, callback)) return;
+         var controller = new AbortController();
+         _inflight[url] = controller;
+         d3_json(url, {
+           signal: controller.signal
+         }).then(function (result) {
+           delete _inflight[url];
+           if (loaded) loaded(null, result);
+         })["catch"](function (err) {
+           delete _inflight[url];
+           if (err.name === 'AbortError') return;
+           if (loaded) loaded(err.message);
+         });
+       }
 
-           keybindingOn();
-           context.container().selectAll('.main-content').classed('active', false).classed('inactive', true);
-           var osm = context.connection();
+       function checkCache(url, params, exactMatch, callback) {
+         var rp = params.rp || 25;
+         var testQuery = params.query || '';
+         var testUrl = url;
 
-           if (!osm) {
-             cancel();
-             return;
-           }
+         do {
+           var hit = _taginfoCache[testUrl]; // exact match, or shorter match yielding fewer than max results (rp)
 
-           if (osm.authenticated()) {
-             done();
-           } else {
-             osm.authenticate(function (err) {
-               if (err) {
-                 cancel();
-               } else {
-                 done();
-               }
-             });
-           }
-         };
+           if (hit && (url === testUrl || hit.length < rp)) {
+             callback(null, hit);
+             return true;
+           } // don't try to shorten the query
 
-         mode.exit = function () {
-           keybindingOff();
-           context.container().selectAll('.main-content').classed('active', true).classed('inactive', false);
-           context.ui().sidebar.hide();
-         };
 
-         return mode;
-       }
+           if (exactMatch || !testQuery.length) return false; // do shorten the query to see if we already have a cached result
+           // that has returned fewer than max results (rp)
 
-       function uiToolOldDrawModes(context) {
-         var tool = {
-           id: 'old_modes',
-           label: _t.html('toolbar.add_feature')
-         };
-         var modes = [modeAddPoint(context, {
-           title: _t.html('modes.add_point.title'),
-           button: 'point',
-           description: _t.html('modes.add_point.description'),
-           preset: _mainPresetIndex.item('point'),
-           key: '1'
-         }), modeAddLine(context, {
-           title: _t.html('modes.add_line.title'),
-           button: 'line',
-           description: _t.html('modes.add_line.description'),
-           preset: _mainPresetIndex.item('line'),
-           key: '2'
-         }), modeAddArea(context, {
-           title: _t.html('modes.add_area.title'),
-           button: 'area',
-           description: _t.html('modes.add_area.description'),
-           preset: _mainPresetIndex.item('area'),
-           key: '3'
-         })];
+           testQuery = testQuery.slice(0, -1);
+           testUrl = url.replace(/&query=(.*?)&/, '&query=' + testQuery + '&');
+         } while (testQuery.length >= 0);
 
-         function enabled() {
-           return osmEditable();
-         }
+         return false;
+       }
 
-         function osmEditable() {
-           return context.editable();
-         }
+       var serviceTaginfo = {
+         init: function init() {
+           _inflight = {};
+           _taginfoCache = {};
+           _popularKeys = {
+             // manually exclude some keys – #5377, #7485
+             postal_code: true,
+             full_name: true,
+             loc_name: true,
+             reg_name: true,
+             short_name: true,
+             sorting_name: true,
+             artist_name: true,
+             nat_name: true,
+             long_name: true,
+             'bridge:name': true
+           }; // Fetch popular keys.  We'll exclude these from `values`
+           // lookups because they stress taginfo, and they aren't likely
+           // to yield meaningful autocomplete results.. see #3955
 
-         modes.forEach(function (mode) {
-           context.keybinding().on(mode.key, function () {
-             if (!enabled()) return;
+           var params = {
+             rp: 100,
+             sortname: 'values_all',
+             sortorder: 'desc',
+             page: 1,
+             debounce: false,
+             lang: _mainLocalizer.languageCode()
+           };
+           this.keys(params, function (err, data) {
+             if (err) return;
+             data.forEach(function (d) {
+               if (d.value === 'opening_hours') return; // exception
 
-             if (mode.id === context.mode().id) {
-               context.enter(modeBrowse(context));
+               _popularKeys[d.value] = true;
+             });
+           });
+         },
+         reset: function reset() {
+           Object.values(_inflight).forEach(function (controller) {
+             controller.abort();
+           });
+           _inflight = {};
+         },
+         keys: function keys(params, callback) {
+           var doRequest = params.debounce ? debouncedRequest : request;
+           params = clean(setSort(params));
+           params = Object.assign({
+             rp: 10,
+             sortname: 'count_all',
+             sortorder: 'desc',
+             page: 1,
+             lang: _mainLocalizer.languageCode()
+           }, params);
+           var url = _apibase + 'keys/all?' + utilQsString(params);
+           doRequest(url, params, false, callback, function (err, d) {
+             if (err) {
+               callback(err);
+             } else {
+               var f = filterKeys(params.filter);
+               var result = d.data.filter(f).sort(sortKeys).map(valKey);
+               _taginfoCache[url] = result;
+               callback(null, result);
+             }
+           });
+         },
+         multikeys: function multikeys(params, callback) {
+           var doRequest = params.debounce ? debouncedRequest : request;
+           params = clean(setSort(params));
+           params = Object.assign({
+             rp: 25,
+             sortname: 'count_all',
+             sortorder: 'desc',
+             page: 1,
+             lang: _mainLocalizer.languageCode()
+           }, params);
+           var prefix = params.query;
+           var url = _apibase + 'keys/all?' + utilQsString(params);
+           doRequest(url, params, true, callback, function (err, d) {
+             if (err) {
+               callback(err);
              } else {
-               context.enter(mode);
+               var f = filterMultikeys(prefix);
+               var result = d.data.filter(f).map(valKey);
+               _taginfoCache[url] = result;
+               callback(null, result);
              }
            });
-         });
+         },
+         values: function values(params, callback) {
+           // Exclude popular keys from values lookups.. see #3955
+           var key = params.key;
 
-         tool.render = function (selection) {
-           var wrap = selection.append('div').attr('class', 'joined').style('display', 'flex');
+           if (key && _popularKeys[key]) {
+             callback(null, []);
+             return;
+           }
 
-           var debouncedUpdate = debounce(update, 500, {
-             leading: true,
-             trailing: true
+           var doRequest = params.debounce ? debouncedRequest : request;
+           params = clean(setSort(setFilter(params)));
+           params = Object.assign({
+             rp: 25,
+             sortname: 'count_all',
+             sortorder: 'desc',
+             page: 1,
+             lang: _mainLocalizer.languageCode()
+           }, params);
+           var url = _apibase + 'key/values?' + utilQsString(params);
+           doRequest(url, params, false, callback, function (err, d) {
+             if (err) {
+               callback(err);
+             } else {
+               // In most cases we prefer taginfo value results with lowercase letters.
+               // A few OSM keys expect values to contain uppercase values (see #3377).
+               // This is not an exhaustive list (e.g. `name` also has uppercase values)
+               // but these are the fields where taginfo value lookup is most useful.
+               var re = /network|taxon|genus|species|brand|grape_variety|royal_cypher|listed_status|booth|rating|stars|:output|_hours|_times|_ref|manufacturer|country|target|brewery/;
+               var allowUpperCase = re.test(params.key);
+               var f = filterValues(allowUpperCase);
+               var result = d.data.filter(f).map(valKeyDescription);
+               _taginfoCache[url] = result;
+               callback(null, result);
+             }
            });
+         },
+         roles: function roles(params, callback) {
+           var doRequest = params.debounce ? debouncedRequest : request;
+           var geometry = params.geometry;
+           params = clean(setSortMembers(params));
+           params = Object.assign({
+             rp: 25,
+             sortname: 'count_all_members',
+             sortorder: 'desc',
+             page: 1,
+             lang: _mainLocalizer.languageCode()
+           }, params);
+           var url = _apibase + 'relation/roles?' + utilQsString(params);
+           doRequest(url, params, true, callback, function (err, d) {
+             if (err) {
+               callback(err);
+             } else {
+               var f = filterRoles(geometry);
+               var result = d.data.filter(f).map(roleKey);
+               _taginfoCache[url] = result;
+               callback(null, result);
+             }
+           });
+         },
+         docs: function docs(params, callback) {
+           var doRequest = params.debounce ? debouncedRequest : request;
+           params = clean(setSort(params));
+           var path = 'key/wiki_pages?';
 
-           context.map().on('move.modes', debouncedUpdate).on('drawn.modes', debouncedUpdate);
-           context.on('enter.modes', update);
-           update();
-
-           function update() {
-             var buttons = wrap.selectAll('button.add-button').data(modes, function (d) {
-               return d.id;
-             }); // exit
-
-             buttons.exit().remove(); // enter
-
-             var buttonsEnter = buttons.enter().append('button').attr('class', function (d) {
-               return d.id + ' add-button bar-button';
-             }).on('click.mode-buttons', function (d3_event, d) {
-               if (!enabled()) return; // When drawing, ignore accidental clicks on mode buttons - #4042
-
-               var currMode = context.mode().id;
-               if (/^draw/.test(currMode)) return;
-
-               if (d.id === currMode) {
-                 context.enter(modeBrowse(context));
-               } else {
-                 context.enter(d);
-               }
-             }).call(uiTooltip().placement('bottom').title(function (d) {
-               return d.description;
-             }).keys(function (d) {
-               return [d.key];
-             }).scrollContainer(context.container().select('.top-toolbar')));
-             buttonsEnter.each(function (d) {
-               select(this).call(svgIcon('#iD-icon-' + d.button));
-             });
-             buttonsEnter.append('span').attr('class', 'label').html(function (mode) {
-               return mode.title;
-             }); // if we are adding/removing the buttons, check if toolbar has overflowed
-
-             if (buttons.enter().size() || buttons.exit().size()) {
-               context.ui().checkOverflow('.top-toolbar', true);
-             } // update
+           if (params.value) {
+             path = 'tag/wiki_pages?';
+           } else if (params.rtype) {
+             path = 'relation/wiki_pages?';
+           }
 
+           var url = _apibase + path + utilQsString(params);
+           doRequest(url, params, true, callback, function (err, d) {
+             if (err) {
+               callback(err);
+             } else {
+               _taginfoCache[url] = d.data;
+               callback(null, d.data);
+             }
+           });
+         },
+         apibase: function apibase(_) {
+           if (!arguments.length) return _apibase;
+           _apibase = _;
+           return this;
+         }
+       };
 
-             buttons = buttons.merge(buttonsEnter).classed('disabled', function (d) {
-               return !enabled();
-             }).classed('active', function (d) {
-               return context.mode() && context.mode().button === d.button;
-             });
-           }
-         };
+       /**
+        * Wraps a GeoJSON {@link Geometry} in a GeoJSON {@link Feature}.
+        *
+        * @name feature
+        * @param {Geometry} geometry input geometry
+        * @param {Object} [properties={}] an Object of key-value pairs to add as properties
+        * @param {Object} [options={}] Optional Parameters
+        * @param {Array<number>} [options.bbox] Bounding Box Array [west, south, east, north] associated with the Feature
+        * @param {string|number} [options.id] Identifier associated with the Feature
+        * @returns {Feature} a GeoJSON Feature
+        * @example
+        * var geometry = {
+        *   "type": "Point",
+        *   "coordinates": [110, 50]
+        * };
+        *
+        * var feature = turf.feature(geometry);
+        *
+        * //=feature
+        */
 
-         return tool;
-       }
+       function feature(geom, properties, options) {
+         if (options === void 0) {
+           options = {};
+         }
 
-       function uiToolNotes(context) {
-         var tool = {
-           id: 'notes',
-           label: _t.html('modes.add_note.label')
+         var feat = {
+           type: "Feature"
          };
-         var mode = modeAddNote(context);
 
-         function enabled() {
-           return notesEnabled() && notesEditable();
+         if (options.id === 0 || options.id) {
+           feat.id = options.id;
          }
 
-         function notesEnabled() {
-           var noteLayer = context.layers().layer('notes');
-           return noteLayer && noteLayer.enabled();
+         if (options.bbox) {
+           feat.bbox = options.bbox;
          }
 
-         function notesEditable() {
-           var mode = context.mode();
-           return context.map().notesEditable() && mode && mode.id !== 'save';
+         feat.properties = properties || {};
+         feat.geometry = geom;
+         return feat;
+       }
+       /**
+        * Creates a {@link Polygon} {@link Feature} from an Array of LinearRings.
+        *
+        * @name polygon
+        * @param {Array<Array<Array<number>>>} coordinates an array of LinearRings
+        * @param {Object} [properties={}] an Object of key-value pairs to add as properties
+        * @param {Object} [options={}] Optional Parameters
+        * @param {Array<number>} [options.bbox] Bounding Box Array [west, south, east, north] associated with the Feature
+        * @param {string|number} [options.id] Identifier associated with the Feature
+        * @returns {Feature<Polygon>} Polygon Feature
+        * @example
+        * var polygon = turf.polygon([[[-5, 52], [-4, 56], [-2, 51], [-7, 54], [-5, 52]]], { name: 'poly1' });
+        *
+        * //=polygon
+        */
+
+       function polygon(coordinates, properties, options) {
+         if (options === void 0) {
+           options = {};
          }
 
-         context.keybinding().on(mode.key, function () {
-           if (!enabled()) return;
+         for (var _i = 0, coordinates_1 = coordinates; _i < coordinates_1.length; _i++) {
+           var ring = coordinates_1[_i];
 
-           if (mode.id === context.mode().id) {
-             context.enter(modeBrowse(context));
-           } else {
-             context.enter(mode);
+           if (ring.length < 4) {
+             throw new Error("Each LinearRing of a Polygon must have 4 or more Positions.");
            }
-         });
-
-         tool.render = function (selection) {
-           var debouncedUpdate = debounce(update, 500, {
-             leading: true,
-             trailing: true
-           });
-
-           context.map().on('move.notes', debouncedUpdate).on('drawn.notes', debouncedUpdate);
-           context.on('enter.notes', update);
-           update();
-
-           function update() {
-             var showNotes = notesEnabled();
-             var data = showNotes ? [mode] : [];
-             var buttons = selection.selectAll('button.add-button').data(data, function (d) {
-               return d.id;
-             }); // exit
-
-             buttons.exit().remove(); // enter
-
-             var buttonsEnter = buttons.enter().append('button').attr('class', function (d) {
-               return d.id + ' add-button bar-button';
-             }).on('click.notes', function (d3_event, d) {
-               if (!enabled()) return; // When drawing, ignore accidental clicks on mode buttons - #4042
-
-               var currMode = context.mode().id;
-               if (/^draw/.test(currMode)) return;
-
-               if (d.id === currMode) {
-                 context.enter(modeBrowse(context));
-               } else {
-                 context.enter(d);
-               }
-             }).call(uiTooltip().placement('bottom').title(function (d) {
-               return d.description;
-             }).keys(function (d) {
-               return [d.key];
-             }).scrollContainer(context.container().select('.top-toolbar')));
-             buttonsEnter.each(function (d) {
-               select(this).call(svgIcon(d.icon || '#iD-icon-' + d.button));
-             }); // if we are adding/removing the buttons, check if toolbar has overflowed
-
-             if (buttons.enter().size() || buttons.exit().size()) {
-               context.ui().checkOverflow('.top-toolbar', true);
-             } // update
-
 
-             buttons = buttons.merge(buttonsEnter).classed('disabled', function (d) {
-               return !enabled();
-             }).classed('active', function (d) {
-               return context.mode() && context.mode().button === d.button;
-             });
+           for (var j = 0; j < ring[ring.length - 1].length; j++) {
+             // Check if first point of Polygon contains two numbers
+             if (ring[ring.length - 1][j] !== ring[0][j]) {
+               throw new Error("First and last Position are not equivalent.");
+             }
            }
-         };
+         }
 
-         tool.uninstall = function () {
-           context.on('enter.editor.notes', null).on('exit.editor.notes', null).on('enter.notes', null);
-           context.map().on('move.notes', null).on('drawn.notes', null);
+         var geom = {
+           type: "Polygon",
+           coordinates: coordinates
          };
-
-         return tool;
+         return feature(geom, properties, options);
        }
+       /**
+        * Creates a {@link LineString} {@link Feature} from an Array of Positions.
+        *
+        * @name lineString
+        * @param {Array<Array<number>>} coordinates an array of Positions
+        * @param {Object} [properties={}] an Object of key-value pairs to add as properties
+        * @param {Object} [options={}] Optional Parameters
+        * @param {Array<number>} [options.bbox] Bounding Box Array [west, south, east, north] associated with the Feature
+        * @param {string|number} [options.id] Identifier associated with the Feature
+        * @returns {Feature<LineString>} LineString Feature
+        * @example
+        * var linestring1 = turf.lineString([[-24, 63], [-23, 60], [-25, 65], [-20, 69]], {name: 'line 1'});
+        * var linestring2 = turf.lineString([[-14, 43], [-13, 40], [-15, 45], [-10, 49]], {name: 'line 2'});
+        *
+        * //=linestring1
+        * //=linestring2
+        */
 
-       function uiToolSave(context) {
-         var tool = {
-           id: 'save',
-           label: _t.html('save.title')
-         };
-         var button = null;
-         var tooltipBehavior = null;
-         var history = context.history();
-         var key = uiCmd('⌘S');
-         var _numChanges = 0;
-
-         function isSaving() {
-           var mode = context.mode();
-           return mode && mode.id === 'save';
+       function lineString(coordinates, properties, options) {
+         if (options === void 0) {
+           options = {};
          }
 
-         function isDisabled() {
-           return _numChanges === 0 || isSaving();
+         if (coordinates.length < 2) {
+           throw new Error("coordinates must be an array of two or more positions");
          }
 
-         function save(d3_event) {
-           d3_event.preventDefault();
+         var geom = {
+           type: "LineString",
+           coordinates: coordinates
+         };
+         return feature(geom, properties, options);
+       }
+       /**
+        * Creates a {@link Feature<MultiLineString>} based on a
+        * coordinate array. Properties can be added optionally.
+        *
+        * @name multiLineString
+        * @param {Array<Array<Array<number>>>} coordinates an array of LineStrings
+        * @param {Object} [properties={}] an Object of key-value pairs to add as properties
+        * @param {Object} [options={}] Optional Parameters
+        * @param {Array<number>} [options.bbox] Bounding Box Array [west, south, east, north] associated with the Feature
+        * @param {string|number} [options.id] Identifier associated with the Feature
+        * @returns {Feature<MultiLineString>} a MultiLineString feature
+        * @throws {Error} if no coordinates are passed
+        * @example
+        * var multiLine = turf.multiLineString([[[0,0],[10,10]]]);
+        *
+        * //=multiLine
+        */
 
-           if (!context.inIntro() && !isSaving() && history.hasChanges()) {
-             context.enter(modeSave(context));
-           }
+       function multiLineString(coordinates, properties, options) {
+         if (options === void 0) {
+           options = {};
          }
 
-         function bgColor() {
-           var step;
+         var geom = {
+           type: "MultiLineString",
+           coordinates: coordinates
+         };
+         return feature(geom, properties, options);
+       }
+       /**
+        * Creates a {@link Feature<MultiPolygon>} based on a
+        * coordinate array. Properties can be added optionally.
+        *
+        * @name multiPolygon
+        * @param {Array<Array<Array<Array<number>>>>} coordinates an array of Polygons
+        * @param {Object} [properties={}] an Object of key-value pairs to add as properties
+        * @param {Object} [options={}] Optional Parameters
+        * @param {Array<number>} [options.bbox] Bounding Box Array [west, south, east, north] associated with the Feature
+        * @param {string|number} [options.id] Identifier associated with the Feature
+        * @returns {Feature<MultiPolygon>} a multipolygon feature
+        * @throws {Error} if no coordinates are passed
+        * @example
+        * var multiPoly = turf.multiPolygon([[[[0,0],[0,10],[10,10],[10,0],[0,0]]]]);
+        *
+        * //=multiPoly
+        *
+        */
 
-           if (_numChanges === 0) {
-             return null;
-           } else if (_numChanges <= 50) {
-             step = _numChanges / 50;
-             return d3_interpolateRgb('#fff', '#ff8')(step); // white -> yellow
-           } else {
-             step = Math.min((_numChanges - 50) / 50, 1.0);
-             return d3_interpolateRgb('#ff8', '#f88')(step); // yellow -> red
-           }
+       function multiPolygon(coordinates, properties, options) {
+         if (options === void 0) {
+           options = {};
          }
 
-         function updateCount() {
-           var val = history.difference().summary().length;
-           if (val === _numChanges) return;
-           _numChanges = val;
+         var geom = {
+           type: "MultiPolygon",
+           coordinates: coordinates
+         };
+         return feature(geom, properties, options);
+       }
 
-           if (tooltipBehavior) {
-             tooltipBehavior.title(_t.html(_numChanges > 0 ? 'save.help' : 'save.no_changes')).keys([key]);
-           }
+       /**
+        * Get Geometry from Feature or Geometry Object
+        *
+        * @param {Feature|Geometry} geojson GeoJSON Feature or Geometry Object
+        * @returns {Geometry|null} GeoJSON Geometry Object
+        * @throws {Error} if geojson is not a Feature or Geometry Object
+        * @example
+        * var point = {
+        *   "type": "Feature",
+        *   "properties": {},
+        *   "geometry": {
+        *     "type": "Point",
+        *     "coordinates": [110, 40]
+        *   }
+        * }
+        * var geom = turf.getGeom(point)
+        * //={"type": "Point", "coordinates": [110, 40]}
+        */
 
-           if (button) {
-             button.classed('disabled', isDisabled()).style('background', bgColor());
-             button.select('span.count').html(_numChanges);
-           }
+       function getGeom(geojson) {
+         if (geojson.type === "Feature") {
+           return geojson.geometry;
          }
 
-         tool.render = function (selection) {
-           tooltipBehavior = uiTooltip().placement('bottom').title(_t.html('save.no_changes')).keys([key]).scrollContainer(context.container().select('.top-toolbar'));
-           var lastPointerUpType;
-           button = selection.append('button').attr('class', 'save disabled bar-button').on('pointerup', function (d3_event) {
-             lastPointerUpType = d3_event.pointerType;
-           }).on('click', function (d3_event) {
-             save(d3_event);
+         return geojson;
+       }
 
-             if (_numChanges === 0 && (lastPointerUpType === 'touch' || lastPointerUpType === 'pen')) {
-               // there are no tooltips for touch interactions so flash feedback instead
-               context.ui().flash.duration(2000).iconName('#iD-icon-save').iconClass('disabled').label(_t.html('save.no_changes'))();
-             }
+       // Cohen-Sutherland line clipping algorithm, adapted to efficiently
+       // handle polylines rather than just segments
+       function lineclip(points, bbox, result) {
+         var len = points.length,
+             codeA = bitCode(points[0], bbox),
+             part = [],
+             i,
+             codeB,
+             lastCode;
+         var a;
+         var b;
+         if (!result) result = [];
 
-             lastPointerUpType = null;
-           }).call(tooltipBehavior);
-           button.call(svgIcon('#iD-icon-save'));
-           button.append('span').attr('class', 'count').attr('aria-hidden', 'true').html('0');
-           updateCount();
-           context.keybinding().on(key, save, true);
-           context.history().on('change.save', updateCount);
-           context.on('enter.save', function () {
-             if (button) {
-               button.classed('disabled', isDisabled());
+         for (i = 1; i < len; i++) {
+           a = points[i - 1];
+           b = points[i];
+           codeB = lastCode = bitCode(b, bbox);
 
-               if (isSaving()) {
-                 button.call(tooltipBehavior.hide);
+           while (true) {
+             if (!(codeA | codeB)) {
+               // accept
+               part.push(a);
+
+               if (codeB !== lastCode) {
+                 // segment went outside
+                 part.push(b);
+
+                 if (i < len - 1) {
+                   // start a new line
+                   result.push(part);
+                   part = [];
+                 }
+               } else if (i === len - 1) {
+                 part.push(b);
                }
+
+               break;
+             } else if (codeA & codeB) {
+               // trivial reject
+               break;
+             } else if (codeA) {
+               // a outside, intersect with clip edge
+               a = intersect(a, b, codeA, bbox);
+               codeA = bitCode(a, bbox);
+             } else {
+               // b outside
+               b = intersect(a, b, codeB, bbox);
+               codeB = bitCode(b, bbox);
              }
-           });
-         };
+           }
 
-         tool.uninstall = function () {
-           context.keybinding().off(key, true);
-           context.history().on('change.save', null);
-           context.on('enter.save', null);
-           button = null;
-           tooltipBehavior = null;
-         };
+           codeA = lastCode;
+         }
 
-         return tool;
-       }
+         if (part.length) result.push(part);
+         return result;
+       } // Sutherland-Hodgeman polygon clipping algorithm
 
-       function uiToolSidebarToggle(context) {
-         var tool = {
-           id: 'sidebar_toggle',
-           label: _t.html('toolbar.inspect')
-         };
+       function polygonclip(points, bbox) {
+         var result, edge, prev, prevInside, i, p, inside; // clip against each side of the clip rectangle
 
-         tool.render = function (selection) {
-           selection.append('button').attr('class', 'bar-button').on('click', function () {
-             context.ui().sidebar.toggle();
-           }).call(uiTooltip().placement('bottom').title(_t.html('sidebar.tooltip')).keys([_t('sidebar.key')]).scrollContainer(context.container().select('.top-toolbar'))).call(svgIcon('#iD-icon-sidebar-' + (_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left')));
-         };
+         for (edge = 1; edge <= 8; edge *= 2) {
+           result = [];
+           prev = points[points.length - 1];
+           prevInside = !(bitCode(prev, bbox) & edge);
 
-         return tool;
-       }
+           for (i = 0; i < points.length; i++) {
+             p = points[i];
+             inside = !(bitCode(p, bbox) & edge); // if segment goes through the clip window, add an intersection
 
-       function uiToolUndoRedo(context) {
-         var tool = {
-           id: 'undo_redo',
-           label: _t.html('toolbar.undo_redo')
-         };
-         var commands = [{
-           id: 'undo',
-           cmd: uiCmd('⌘Z'),
-           action: function action() {
-             context.undo();
-           },
-           annotation: function annotation() {
-             return context.history().undoAnnotation();
-           },
-           icon: 'iD-icon-' + (_mainLocalizer.textDirection() === 'rtl' ? 'redo' : 'undo')
-         }, {
-           id: 'redo',
-           cmd: uiCmd('⌘⇧Z'),
-           action: function action() {
-             context.redo();
-           },
-           annotation: function annotation() {
-             return context.history().redoAnnotation();
-           },
-           icon: 'iD-icon-' + (_mainLocalizer.textDirection() === 'rtl' ? 'undo' : 'redo')
-         }];
+             if (inside !== prevInside) result.push(intersect(prev, p, edge, bbox));
+             if (inside) result.push(p); // add a point if it's inside
 
-         function editable() {
-           return context.mode() && context.mode().id !== 'save' && context.map().editableDataEnabled(true
-           /* ignore min zoom */
-           );
-         }
+             prev = p;
+             prevInside = inside;
+           }
 
-         tool.render = function (selection) {
-           var tooltipBehavior = uiTooltip().placement('bottom').title(function (d) {
-             return d.annotation() ? _t.html(d.id + '.tooltip', {
-               action: d.annotation()
-             }) : _t.html(d.id + '.nothing');
-           }).keys(function (d) {
-             return [d.cmd];
-           }).scrollContainer(context.container().select('.top-toolbar'));
-           var lastPointerUpType;
-           var buttons = selection.selectAll('button').data(commands).enter().append('button').attr('class', function (d) {
-             return 'disabled ' + d.id + '-button bar-button';
-           }).on('pointerup', function (d3_event) {
-             // `pointerup` is always called before `click`
-             lastPointerUpType = d3_event.pointerType;
-           }).on('click', function (d3_event, d) {
-             d3_event.preventDefault();
-             var annotation = d.annotation();
+           points = result;
+           if (!points.length) break;
+         }
 
-             if (editable() && annotation) {
-               d.action();
-             }
+         return result;
+       } // intersect a segment against one of the 4 lines that make up the bbox
 
-             if (editable() && (lastPointerUpType === 'touch' || lastPointerUpType === 'pen')) {
-               // there are no tooltips for touch interactions so flash feedback instead
-               var text = annotation ? _t(d.id + '.tooltip', {
-                 action: annotation
-               }) : _t(d.id + '.nothing');
-               context.ui().flash.duration(2000).iconName('#' + d.icon).iconClass(annotation ? '' : 'disabled').label(text)();
-             }
+       function intersect(a, b, edge, bbox) {
+         return edge & 8 ? [a[0] + (b[0] - a[0]) * (bbox[3] - a[1]) / (b[1] - a[1]), bbox[3]] // top
+         : edge & 4 ? [a[0] + (b[0] - a[0]) * (bbox[1] - a[1]) / (b[1] - a[1]), bbox[1]] // bottom
+         : edge & 2 ? [bbox[2], a[1] + (b[1] - a[1]) * (bbox[2] - a[0]) / (b[0] - a[0])] // right
+         : edge & 1 ? [bbox[0], a[1] + (b[1] - a[1]) * (bbox[0] - a[0]) / (b[0] - a[0])] // left
+         : null;
+       } // bit code reflects the point position relative to the bbox:
+       //         left  mid  right
+       //    top  1001  1000  1010
+       //    mid  0001  0000  0010
+       // bottom  0101  0100  0110
 
-             lastPointerUpType = null;
-           }).call(tooltipBehavior);
-           buttons.each(function (d) {
-             select(this).call(svgIcon('#' + d.icon));
-           });
-           context.keybinding().on(commands[0].cmd, function (d3_event) {
-             d3_event.preventDefault();
-             if (editable()) commands[0].action();
-           }).on(commands[1].cmd, function (d3_event) {
-             d3_event.preventDefault();
-             if (editable()) commands[1].action();
-           });
 
-           var debouncedUpdate = debounce(update, 500, {
-             leading: true,
-             trailing: true
-           });
+       function bitCode(p, bbox) {
+         var code = 0;
+         if (p[0] < bbox[0]) code |= 1; // left
+         else if (p[0] > bbox[2]) code |= 2; // right
 
-           context.map().on('move.undo_redo', debouncedUpdate).on('drawn.undo_redo', debouncedUpdate);
-           context.history().on('change.undo_redo', function (difference) {
-             if (difference) update();
-           });
-           context.on('enter.undo_redo', update);
+         if (p[1] < bbox[1]) code |= 4; // bottom
+         else if (p[1] > bbox[3]) code |= 8; // top
 
-           function update() {
-             buttons.classed('disabled', function (d) {
-               return !editable() || !d.annotation();
-             }).each(function () {
-               var selection = select(this);
+         return code;
+       }
 
-               if (!selection.select('.tooltip.in').empty()) {
-                 selection.call(tooltipBehavior.updateContent);
-               }
-             });
-           }
-         };
+       /**
+        * Takes a {@link Feature} and a bbox and clips the feature to the bbox using
+        * [lineclip](https://github.com/mapbox/lineclip).
+        * May result in degenerate edges when clipping Polygons.
+        *
+        * @name bboxClip
+        * @param {Feature<LineString|MultiLineString|Polygon|MultiPolygon>} feature feature to clip to the bbox
+        * @param {BBox} bbox extent in [minX, minY, maxX, maxY] order
+        * @returns {Feature<LineString|MultiLineString|Polygon|MultiPolygon>} clipped Feature
+        * @example
+        * var bbox = [0, 0, 10, 10];
+        * var poly = turf.polygon([[[2, 2], [8, 4], [12, 8], [3, 7], [2, 2]]]);
+        *
+        * var clipped = turf.bboxClip(poly, bbox);
+        *
+        * //addToMap
+        * var addToMap = [bbox, poly, clipped]
+        */
 
-         tool.uninstall = function () {
-           context.keybinding().off(commands[0].cmd).off(commands[1].cmd);
-           context.map().on('move.undo_redo', null).on('drawn.undo_redo', null);
-           context.history().on('change.undo_redo', null);
-           context.on('enter.undo_redo', null);
-         };
+       function bboxClip(feature, bbox) {
+         var geom = getGeom(feature);
+         var type = geom.type;
+         var properties = feature.type === "Feature" ? feature.properties : {};
+         var coords = geom.coordinates;
 
-         return tool;
-       }
+         switch (type) {
+           case "LineString":
+           case "MultiLineString":
+             var lines_1 = [];
 
-       function uiTopToolbar(context) {
-         var sidebarToggle = uiToolSidebarToggle(context),
-             modes = uiToolOldDrawModes(context),
-             notes = uiToolNotes(context),
-             undoRedo = uiToolUndoRedo(context),
-             save = uiToolSave(context);
+             if (type === "LineString") {
+               coords = [coords];
+             }
 
-         function notesEnabled() {
-           var noteLayer = context.layers().layer('notes');
-           return noteLayer && noteLayer.enabled();
-         }
+             coords.forEach(function (line) {
+               lineclip(line, bbox, lines_1);
+             });
 
-         function topToolbar(bar) {
-           bar.on('wheel.topToolbar', function (d3_event) {
-             if (!d3_event.deltaX) {
-               // translate vertical scrolling into horizontal scrolling in case
-               // the user doesn't have an input device that can scroll horizontally
-               bar.node().scrollLeft += d3_event.deltaY;
+             if (lines_1.length === 1) {
+               return lineString(lines_1[0], properties);
              }
-           });
 
-           var debouncedUpdate = debounce(update, 500, {
-             leading: true,
-             trailing: true
-           });
+             return multiLineString(lines_1, properties);
 
-           context.layers().on('change.topToolbar', debouncedUpdate);
-           update();
+           case "Polygon":
+             return polygon(clipPolygon(coords, bbox), properties);
 
-           function update() {
-             var tools = [sidebarToggle, 'spacer', modes];
-             tools.push('spacer');
+           case "MultiPolygon":
+             return multiPolygon(coords.map(function (poly) {
+               return clipPolygon(poly, bbox);
+             }), properties);
 
-             if (notesEnabled()) {
-               tools = tools.concat([notes, 'spacer']);
+           default:
+             throw new Error("geometry " + type + " not supported");
+         }
+       }
+
+       function clipPolygon(rings, bbox) {
+         var outRings = [];
+
+         for (var _i = 0, rings_1 = rings; _i < rings_1.length; _i++) {
+           var ring = rings_1[_i];
+           var clipped = polygonclip(ring, bbox);
+
+           if (clipped.length > 0) {
+             if (clipped[0][0] !== clipped[clipped.length - 1][0] || clipped[0][1] !== clipped[clipped.length - 1][1]) {
+               clipped.push(clipped[0]);
              }
 
-             tools = tools.concat([undoRedo, save]);
-             var toolbarItems = bar.selectAll('.toolbar-item').data(tools, function (d) {
-               return d.id || d;
-             });
-             toolbarItems.exit().each(function (d) {
-               if (d.uninstall) {
-                 d.uninstall();
-               }
-             }).remove();
-             var itemsEnter = toolbarItems.enter().append('div').attr('class', function (d) {
-               var classes = 'toolbar-item ' + (d.id || d).replace('_', '-');
-               if (d.klass) classes += ' ' + d.klass;
-               return classes;
-             });
-             var actionableItems = itemsEnter.filter(function (d) {
-               return d !== 'spacer';
-             });
-             actionableItems.append('div').attr('class', 'item-content').each(function (d) {
-               select(this).call(d.render, bar);
-             });
-             actionableItems.append('div').attr('class', 'item-label').html(function (d) {
-               return d.label;
-             });
+             if (clipped.length >= 4) {
+               outRings.push(clipped);
+             }
            }
          }
 
-         return topToolbar;
+         return outRings;
        }
 
-       var sawVersion = null;
-       var isNewVersion = false;
-       var isNewUser = false;
-       function uiVersion(context) {
-         var currVersion = context.version;
-         var matchedVersion = currVersion.match(/\d+\.\d+\.\d+.*/);
-
-         if (sawVersion === null && matchedVersion !== null) {
-           if (corePreferences('sawVersion')) {
-             isNewUser = false;
-             isNewVersion = corePreferences('sawVersion') !== currVersion && currVersion.indexOf('-') === -1;
-           } else {
-             isNewUser = true;
-             isNewVersion = true;
-           }
-
-           corePreferences('sawVersion', currVersion);
-           sawVersion = currVersion;
-         }
+       var tiler = utilTiler().tileSize(512).margin(1);
+       var dispatch = dispatch$8('loadedData');
 
-         return function (selection) {
-           selection.append('a').attr('target', '_blank').attr('href', 'https://github.com/openstreetmap/iD').html(currVersion); // only show new version indicator to users that have used iD before
+       var _vtCache;
 
-           if (isNewVersion && !isNewUser) {
-             selection.append('a').attr('class', 'badge').attr('target', '_blank').attr('href', 'https://github.com/openstreetmap/iD/blob/release/CHANGELOG.md#whats-new').call(svgIcon('#maki-gift-11')).call(uiTooltip().title(_t.html('version.whats_new', {
-               version: currVersion
-             })).placement('top').scrollContainer(context.container().select('.main-footer-wrap')));
-           }
-         };
+       function abortRequest(controller) {
+         controller.abort();
        }
 
-       function uiZoom(context) {
-         var zooms = [{
-           id: 'zoom-in',
-           icon: 'iD-icon-plus',
-           title: _t.html('zoom.in'),
-           action: zoomIn,
-           disabled: function disabled() {
-             return !context.map().canZoomIn();
-           },
-           disabledTitle: _t.html('zoom.disabled.in'),
-           key: '+'
-         }, {
-           id: 'zoom-out',
-           icon: 'iD-icon-minus',
-           title: _t.html('zoom.out'),
-           action: zoomOut,
-           disabled: function disabled() {
-             return !context.map().canZoomOut();
-           },
-           disabledTitle: _t.html('zoom.disabled.out'),
-           key: '-'
-         }];
+       function vtToGeoJSON(data, tile, mergeCache) {
+         var vectorTile$1 = new vectorTile.VectorTile(new pbf(data));
+         var layers = Object.keys(vectorTile$1.layers);
 
-         function zoomIn(d3_event) {
-           if (d3_event.shiftKey) return;
-           d3_event.preventDefault();
-           context.map().zoomIn();
+         if (!Array.isArray(layers)) {
+           layers = [layers];
          }
 
-         function zoomOut(d3_event) {
-           if (d3_event.shiftKey) return;
-           d3_event.preventDefault();
-           context.map().zoomOut();
-         }
+         var features = [];
+         layers.forEach(function (layerID) {
+           var layer = vectorTile$1.layers[layerID];
 
-         function zoomInFurther(d3_event) {
-           if (d3_event.shiftKey) return;
-           d3_event.preventDefault();
-           context.map().zoomInFurther();
-         }
+           if (layer) {
+             for (var i = 0; i < layer.length; i++) {
+               var feature = layer.feature(i).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
+               var geometry = feature.geometry; // Treat all Polygons as MultiPolygons
 
-         function zoomOutFurther(d3_event) {
-           if (d3_event.shiftKey) return;
-           d3_event.preventDefault();
-           context.map().zoomOutFurther();
-         }
+               if (geometry.type === 'Polygon') {
+                 geometry.type = 'MultiPolygon';
+                 geometry.coordinates = [geometry.coordinates];
+               }
 
-         return function (selection) {
-           var tooltipBehavior = uiTooltip().placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left').title(function (d) {
-             if (d.disabled()) {
-               return d.disabledTitle;
-             }
+               var isClipped = false; // Clip to tile bounds
 
-             return d.title;
-           }).keys(function (d) {
-             return [d.key];
-           });
-           var lastPointerUpType;
-           var buttons = selection.selectAll('button').data(zooms).enter().append('button').attr('class', function (d) {
-             return d.id;
-           }).on('pointerup.editor', function (d3_event) {
-             lastPointerUpType = d3_event.pointerType;
-           }).on('click.editor', function (d3_event, d) {
-             if (!d.disabled()) {
-               d.action(d3_event);
-             } else if (lastPointerUpType === 'touch' || lastPointerUpType === 'pen') {
-               context.ui().flash.duration(2000).iconName('#' + d.icon).iconClass('disabled').label(d.disabledTitle)();
-             }
+               if (geometry.type === 'MultiPolygon') {
+                 var featureClip = bboxClip(feature, tile.extent.rectangle());
 
-             lastPointerUpType = null;
-           }).call(tooltipBehavior);
-           buttons.each(function (d) {
-             select(this).call(svgIcon('#' + d.icon, 'light'));
-           });
-           utilKeybinding.plusKeys.forEach(function (key) {
-             context.keybinding().on([key], zoomIn);
-             context.keybinding().on([uiCmd('⌥' + key)], zoomInFurther);
-           });
-           utilKeybinding.minusKeys.forEach(function (key) {
-             context.keybinding().on([key], zoomOut);
-             context.keybinding().on([uiCmd('⌥' + key)], zoomOutFurther);
-           });
+                 if (!fastDeepEqual(feature.geometry, featureClip.geometry)) {
+                   // feature = featureClip;
+                   isClipped = true;
+                 }
 
-           function updateButtonStates() {
-             buttons.classed('disabled', function (d) {
-               return d.disabled();
-             }).each(function () {
-               var selection = select(this);
+                 if (!feature.geometry.coordinates.length) continue; // not actually on this tile
 
-               if (!selection.select('.tooltip.in').empty()) {
-                 selection.call(tooltipBehavior.updateContent);
-               }
-             });
-           }
+                 if (!feature.geometry.coordinates[0].length) continue; // not actually on this tile
+               } // Generate some unique IDs and add some metadata
 
-           updateButtonStates();
-           context.map().on('move.uiZoom', updateButtonStates);
-         };
-       }
 
-       function uiZoomToSelection(context) {
-         function isDisabled() {
-           var mode = context.mode();
-           return !mode || !mode.zoomToSelected;
-         }
+               var featurehash = utilHashcode(fastJsonStableStringify(feature));
+               var propertyhash = utilHashcode(fastJsonStableStringify(feature.properties || {}));
+               feature.__layerID__ = layerID.replace(/[^_a-zA-Z0-9\-]/g, '_');
+               feature.__featurehash__ = featurehash;
+               feature.__propertyhash__ = propertyhash;
+               features.push(feature); // Clipped Polygons at same zoom with identical properties can get merged
 
-         var _lastPointerUpType;
+               if (isClipped && geometry.type === 'MultiPolygon') {
+                 var merged = mergeCache[propertyhash];
 
-         function pointerup(d3_event) {
-           _lastPointerUpType = d3_event.pointerType;
-         }
+                 if (merged && merged.length) {
+                   var other = merged[0];
+                   var coords = index.union(feature.geometry.coordinates, other.geometry.coordinates);
 
-         function click(d3_event) {
-           d3_event.preventDefault();
+                   if (!coords || !coords.length) {
+                     continue; // something failed in polygon union
+                   }
 
-           if (isDisabled()) {
-             if (_lastPointerUpType === 'touch' || _lastPointerUpType === 'pen') {
-               context.ui().flash.duration(2000).iconName('#iD-icon-framed-dot').iconClass('disabled').label(_t.html('inspector.zoom_to.no_selection'))();
-             }
-           } else {
-             var mode = context.mode();
+                   merged.push(feature);
 
-             if (mode && mode.zoomToSelected) {
-               mode.zoomToSelected();
+                   for (var j = 0; j < merged.length; j++) {
+                     // all these features get...
+                     merged[j].geometry.coordinates = coords; // same coords
+
+                     merged[j].__featurehash__ = featurehash; // same hash, so deduplication works
+                   }
+                 } else {
+                   mergeCache[propertyhash] = [feature];
+                 }
+               }
              }
            }
+         });
+         return features;
+       }
 
-           _lastPointerUpType = null;
-         }
-
-         return function (selection) {
-           var tooltipBehavior = uiTooltip().placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left').title(function () {
-             if (isDisabled()) {
-               return _t.html('inspector.zoom_to.no_selection');
-             }
+       function loadTile(source, tile) {
+         if (source.loaded[tile.id] || source.inflight[tile.id]) return;
+         var url = source.template.replace('{x}', tile.xyz[0]).replace('{y}', tile.xyz[1]) // TMS-flipped y coordinate
+         .replace(/\{[t-]y\}/, Math.pow(2, tile.xyz[2]) - tile.xyz[1] - 1).replace(/\{z(oom)?\}/, tile.xyz[2]).replace(/\{switch:([^}]+)\}/, function (s, r) {
+           var subdomains = r.split(',');
+           return subdomains[(tile.xyz[0] + tile.xyz[1]) % subdomains.length];
+         });
+         var controller = new AbortController();
+         source.inflight[tile.id] = controller;
+         fetch(url, {
+           signal: controller.signal
+         }).then(function (response) {
+           if (!response.ok) {
+             throw new Error(response.status + ' ' + response.statusText);
+           }
 
-             return _t.html('inspector.zoom_to.title');
-           }).keys([_t('inspector.zoom_to.key')]);
-           var button = selection.append('button').on('pointerup', pointerup).on('click', click).call(svgIcon('#iD-icon-framed-dot', 'light')).call(tooltipBehavior);
+           source.loaded[tile.id] = [];
+           delete source.inflight[tile.id];
+           return response.arrayBuffer();
+         }).then(function (data) {
+           if (!data) {
+             throw new Error('No Data');
+           }
 
-           function setEnabledState() {
-             button.classed('disabled', isDisabled());
+           var z = tile.xyz[2];
 
-             if (!button.select('.tooltip.in').empty()) {
-               button.call(tooltipBehavior.updateContent);
-             }
+           if (!source.canMerge[z]) {
+             source.canMerge[z] = {}; // initialize mergeCache
            }
 
-           context.on('enter.uiZoomToSelection', setEnabledState);
-           setEnabledState();
-         };
+           source.loaded[tile.id] = vtToGeoJSON(data, tile, source.canMerge[z]);
+           dispatch.call('loadedData');
+         })["catch"](function () {
+           source.loaded[tile.id] = [];
+           delete source.inflight[tile.id];
+         });
        }
 
-       function uiPane(id, context) {
-         var _key;
-
-         var _label = '';
-         var _description = '';
-         var _iconName = '';
-
-         var _sections; // array of uiSection objects
+       var serviceVectorTile = {
+         init: function init() {
+           if (!_vtCache) {
+             this.reset();
+           }
 
+           this.event = utilRebind(this, dispatch, 'on');
+         },
+         reset: function reset() {
+           for (var sourceID in _vtCache) {
+             var source = _vtCache[sourceID];
 
-         var _paneSelection = select(null);
+             if (source && source.inflight) {
+               Object.values(source.inflight).forEach(abortRequest);
+             }
+           }
 
-         var _paneTooltip;
+           _vtCache = {};
+         },
+         addSource: function addSource(sourceID, template) {
+           _vtCache[sourceID] = {
+             template: template,
+             inflight: {},
+             loaded: {},
+             canMerge: {}
+           };
+           return _vtCache[sourceID];
+         },
+         data: function data(sourceID, projection) {
+           var source = _vtCache[sourceID];
+           if (!source) return [];
+           var tiles = tiler.getTiles(projection);
+           var seen = {};
+           var results = [];
 
-         var pane = {
-           id: id
-         };
+           for (var i = 0; i < tiles.length; i++) {
+             var features = source.loaded[tiles[i].id];
+             if (!features || !features.length) continue;
 
-         pane.label = function (val) {
-           if (!arguments.length) return _label;
-           _label = val;
-           return pane;
-         };
+             for (var j = 0; j < features.length; j++) {
+               var feature = features[j];
+               var hash = feature.__featurehash__;
+               if (seen[hash]) continue;
+               seen[hash] = true; // return a shallow copy, because the hash may change
+               // later if this feature gets merged with another
 
-         pane.key = function (val) {
-           if (!arguments.length) return _key;
-           _key = val;
-           return pane;
-         };
+               results.push(Object.assign({}, feature)); // shallow copy
+             }
+           }
 
-         pane.description = function (val) {
-           if (!arguments.length) return _description;
-           _description = val;
-           return pane;
-         };
+           return results;
+         },
+         loadTiles: function loadTiles(sourceID, template, projection) {
+           var source = _vtCache[sourceID];
 
-         pane.iconName = function (val) {
-           if (!arguments.length) return _iconName;
-           _iconName = val;
-           return pane;
-         };
+           if (!source) {
+             source = this.addSource(sourceID, template);
+           }
 
-         pane.sections = function (val) {
-           if (!arguments.length) return _sections;
-           _sections = val;
-           return pane;
-         };
+           var tiles = tiler.getTiles(projection); // abort inflight requests that are no longer needed
 
-         pane.selection = function () {
-           return _paneSelection;
-         };
+           Object.keys(source.inflight).forEach(function (k) {
+             var wanted = tiles.find(function (tile) {
+               return k === tile.id;
+             });
 
-         function hidePane() {
-           context.ui().togglePanes();
+             if (!wanted) {
+               abortRequest(source.inflight[k]);
+               delete source.inflight[k];
+             }
+           });
+           tiles.forEach(function (tile) {
+             loadTile(source, tile);
+           });
+         },
+         cache: function cache() {
+           return _vtCache;
          }
+       };
 
-         pane.togglePane = function (d3_event) {
-           if (d3_event) d3_event.preventDefault();
-
-           _paneTooltip.hide();
+       var apibase = 'https://www.wikidata.org/w/api.php?';
+       var _wikidataCache = {};
+       var serviceWikidata = {
+         init: function init() {},
+         reset: function reset() {
+           _wikidataCache = {};
+         },
+         // Search for Wikidata items matching the query
+         itemsForSearchQuery: function itemsForSearchQuery(query, callback) {
+           if (!query) {
+             if (callback) callback('No query', {});
+             return;
+           }
 
-           context.ui().togglePanes(!_paneSelection.classed('shown') ? _paneSelection : undefined);
-         };
+           var lang = this.languagesToQuery()[0];
+           var url = apibase + utilQsString({
+             action: 'wbsearchentities',
+             format: 'json',
+             formatversion: 2,
+             search: query,
+             type: 'item',
+             // the language to search
+             language: lang,
+             // the language for the label and description in the result
+             uselang: lang,
+             limit: 10,
+             origin: '*'
+           });
+           d3_json(url).then(function (result) {
+             if (result && result.error) {
+               throw new Error(result.error);
+             }
 
-         pane.renderToggleButton = function (selection) {
-           if (!_paneTooltip) {
-             _paneTooltip = uiTooltip().placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left').title(_description).keys([_key]);
+             if (callback) callback(null, result.search || {});
+           })["catch"](function (err) {
+             if (callback) callback(err.message, {});
+           });
+         },
+         // Given a Wikipedia language and article title,
+         // return an array of corresponding Wikidata entities.
+         itemsByTitle: function itemsByTitle(lang, title, callback) {
+           if (!title) {
+             if (callback) callback('No title', {});
+             return;
            }
 
-           selection.append('button').on('click', pane.togglePane).call(svgIcon('#' + _iconName, 'light')).call(_paneTooltip);
-         };
+           lang = lang || 'en';
+           var url = apibase + utilQsString({
+             action: 'wbgetentities',
+             format: 'json',
+             formatversion: 2,
+             sites: lang.replace(/-/g, '_') + 'wiki',
+             titles: title,
+             languages: 'en',
+             // shrink response by filtering to one language
+             origin: '*'
+           });
+           d3_json(url).then(function (result) {
+             if (result && result.error) {
+               throw new Error(result.error);
+             }
 
-         pane.renderContent = function (selection) {
-           // override to fully customize content
-           if (_sections) {
-             _sections.forEach(function (section) {
-               selection.call(section.render);
-             });
+             if (callback) callback(null, result.entities || {});
+           })["catch"](function (err) {
+             if (callback) callback(err.message, {});
+           });
+         },
+         languagesToQuery: function languagesToQuery() {
+           return _mainLocalizer.localeCodes().map(function (code) {
+             return code.toLowerCase();
+           }).filter(function (code) {
+             // HACK: en-us isn't a wikidata language. We should really be filtering by
+             // the languages known to be supported by wikidata.
+             return code !== 'en-us';
+           });
+         },
+         entityByQID: function entityByQID(qid, callback) {
+           if (!qid) {
+             callback('No qid', {});
+             return;
            }
-         };
 
-         pane.renderPane = function (selection) {
-           _paneSelection = selection.append('div').attr('class', 'fillL map-pane hide ' + id + '-pane').attr('pane', id);
+           if (_wikidataCache[qid]) {
+             if (callback) callback(null, _wikidataCache[qid]);
+             return;
+           }
 
-           var heading = _paneSelection.append('div').attr('class', 'pane-heading');
+           var langs = this.languagesToQuery();
+           var url = apibase + utilQsString({
+             action: 'wbgetentities',
+             format: 'json',
+             formatversion: 2,
+             ids: qid,
+             props: 'labels|descriptions|claims|sitelinks',
+             sitefilter: langs.map(function (d) {
+               return d + 'wiki';
+             }).join('|'),
+             languages: langs.join('|'),
+             languagefallback: 1,
+             origin: '*'
+           });
+           d3_json(url).then(function (result) {
+             if (result && result.error) {
+               throw new Error(result.error);
+             }
 
-           heading.append('h2').html(_label);
-           heading.append('button').on('click', hidePane).call(svgIcon('#iD-icon-close'));
+             if (callback) callback(null, result.entities[qid] || {});
+           })["catch"](function (err) {
+             if (callback) callback(err.message, {});
+           });
+         },
+         // Pass `params` object of the form:
+         // {
+         //   qid: 'string'      // brand wikidata  (e.g. 'Q37158')
+         // }
+         //
+         // Get an result object used to display tag documentation
+         // {
+         //   title:        'string',
+         //   description:  'string',
+         //   editURL:      'string',
+         //   imageURL:     'string',
+         //   wiki:         { title: 'string', text: 'string', url: 'string' }
+         // }
+         //
+         getDocs: function getDocs(params, callback) {
+           var langs = this.languagesToQuery();
+           this.entityByQID(params.qid, function (err, entity) {
+             if (err || !entity) {
+               callback(err || 'No entity');
+               return;
+             }
 
-           _paneSelection.append('div').attr('class', 'pane-content').call(pane.renderContent);
+             var i;
+             var description;
 
-           if (_key) {
-             context.keybinding().on(_key, pane.togglePane);
-           }
-         };
+             for (i in langs) {
+               var code = langs[i];
 
-         return pane;
-       }
+               if (entity.descriptions[code] && entity.descriptions[code].language === code) {
+                 description = entity.descriptions[code];
+                 break;
+               }
+             }
 
-       function uiSectionBackgroundDisplayOptions(context) {
-         var section = uiSection('background-display-options', context).label(_t.html('background.display_options')).disclosureContent(renderDisclosureContent);
+             if (!description && Object.values(entity.descriptions).length) description = Object.values(entity.descriptions)[0]; // prepare result
 
-         var _detected = utilDetect();
+             var result = {
+               title: entity.id,
+               description: description ? description.value : '',
+               descriptionLocaleCode: description ? description.language : '',
+               editURL: 'https://www.wikidata.org/wiki/' + entity.id
+             }; // add image
 
-         var _storedOpacity = corePreferences('background-opacity');
+             if (entity.claims) {
+               var imageroot = 'https://commons.wikimedia.org/w/index.php';
+               var props = ['P154', 'P18']; // logo image, image
 
-         var _minVal = 0;
+               var prop, image;
 
-         var _maxVal = _detected.cssfilters ? 3 : 1;
+               for (i = 0; i < props.length; i++) {
+                 prop = entity.claims[props[i]];
 
-         var _sliders = _detected.cssfilters ? ['brightness', 'contrast', 'saturation', 'sharpness'] : ['brightness'];
+                 if (prop && Object.keys(prop).length > 0) {
+                   image = prop[Object.keys(prop)[0]].mainsnak.datavalue.value;
 
-         var _options = {
-           brightness: _storedOpacity !== null ? +_storedOpacity : 1,
-           contrast: 1,
-           saturation: 1,
-           sharpness: 1
-         };
+                   if (image) {
+                     result.imageURL = imageroot + '?' + utilQsString({
+                       title: 'Special:Redirect/file/' + image,
+                       width: 400
+                     });
+                     break;
+                   }
+                 }
+               }
+             }
 
-         function clamp(x, min, max) {
-           return Math.max(min, Math.min(x, max));
-         }
+             if (entity.sitelinks) {
+               var englishLocale = _mainLocalizer.languageCode().toLowerCase() === 'en'; // must be one of these that we requested..
 
-         function updateValue(d, val) {
-           val = clamp(val, _minVal, _maxVal);
-           _options[d] = val;
-           context.background()[d](val);
+               for (i = 0; i < langs.length; i++) {
+                 // check each, in order of preference
+                 var w = langs[i] + 'wiki';
 
-           if (d === 'brightness') {
-             corePreferences('background-opacity', val);
-           }
+                 if (entity.sitelinks[w]) {
+                   var title = entity.sitelinks[w].title;
+                   var tKey = 'inspector.wiki_reference';
 
-           section.reRender();
-         }
+                   if (!englishLocale && langs[i] === 'en') {
+                     // user's locale isn't English but
+                     tKey = 'inspector.wiki_en_reference'; // we are sending them to enwiki anyway..
+                   }
 
-         function renderDisclosureContent(selection) {
-           var container = selection.selectAll('.display-options-container').data([0]);
-           var containerEnter = container.enter().append('div').attr('class', 'display-options-container controls-list'); // add slider controls
+                   result.wiki = {
+                     title: title,
+                     text: tKey,
+                     url: 'https://' + langs[i] + '.wikipedia.org/wiki/' + title.replace(/ /g, '_')
+                   };
+                   break;
+                 }
+               }
+             }
 
-           var slidersEnter = containerEnter.selectAll('.display-control').data(_sliders).enter().append('div').attr('class', function (d) {
-             return 'display-control display-control-' + d;
-           });
-           slidersEnter.append('h5').html(function (d) {
-             return _t.html('background.' + d);
-           }).append('span').attr('class', function (d) {
-             return 'display-option-value display-option-value-' + d;
+             callback(null, result);
            });
-           var sildersControlEnter = slidersEnter.append('div').attr('class', 'control-wrap');
-           sildersControlEnter.append('input').attr('class', function (d) {
-             return 'display-option-input display-option-input-' + d;
-           }).attr('type', 'range').attr('min', _minVal).attr('max', _maxVal).attr('step', '0.05').on('input', function (d3_event, d) {
-             var val = select(this).property('value');
+         }
+       };
 
-             if (!val && d3_event && d3_event.target) {
-               val = d3_event.target.value;
-             }
+       var endpoint = 'https://en.wikipedia.org/w/api.php?';
+       var serviceWikipedia = {
+         init: function init() {},
+         reset: function reset() {},
+         search: function search(lang, query, callback) {
+           if (!query) {
+             if (callback) callback('No Query', []);
+             return;
+           }
 
-             updateValue(d, val);
+           lang = lang || 'en';
+           var url = endpoint.replace('en', lang) + utilQsString({
+             action: 'query',
+             list: 'search',
+             srlimit: '10',
+             srinfo: 'suggestion',
+             format: 'json',
+             origin: '*',
+             srsearch: query
            });
-           sildersControlEnter.append('button').attr('title', _t('background.reset')).attr('class', function (d) {
-             return 'display-option-reset display-option-reset-' + d;
-           }).on('click', function (d3_event, d) {
-             if (d3_event.button !== 0) return;
-             updateValue(d, 1);
-           }).call(svgIcon('#iD-icon-' + (_mainLocalizer.textDirection() === 'rtl' ? 'redo' : 'undo'))); // reset all button
+           d3_json(url).then(function (result) {
+             if (result && result.error) {
+               throw new Error(result.error);
+             } else if (!result || !result.query || !result.query.search) {
+               throw new Error('No Results');
+             }
 
-           containerEnter.append('a').attr('class', 'display-option-resetlink').attr('href', '#').html(_t.html('background.reset_all')).on('click', function (d3_event) {
-             d3_event.preventDefault();
+             if (callback) {
+               var titles = result.query.search.map(function (d) {
+                 return d.title;
+               });
+               callback(null, titles);
+             }
+           })["catch"](function (err) {
+             if (callback) callback(err, []);
+           });
+         },
+         suggestions: function suggestions(lang, query, callback) {
+           if (!query) {
+             if (callback) callback('', []);
+             return;
+           }
 
-             for (var i = 0; i < _sliders.length; i++) {
-               updateValue(_sliders[i], 1);
+           lang = lang || 'en';
+           var url = endpoint.replace('en', lang) + utilQsString({
+             action: 'opensearch',
+             namespace: 0,
+             suggest: '',
+             format: 'json',
+             origin: '*',
+             search: query
+           });
+           d3_json(url).then(function (result) {
+             if (result && result.error) {
+               throw new Error(result.error);
+             } else if (!result || result.length < 2) {
+               throw new Error('No Results');
              }
-           }); // update
 
-           container = containerEnter.merge(container);
-           container.selectAll('.display-option-input').property('value', function (d) {
-             return _options[d];
+             if (callback) callback(null, result[1] || []);
+           })["catch"](function (err) {
+             if (callback) callback(err.message, []);
            });
-           container.selectAll('.display-option-value').html(function (d) {
-             return Math.floor(_options[d] * 100) + '%';
+         },
+         translations: function translations(lang, title, callback) {
+           if (!title) {
+             if (callback) callback('No Title');
+             return;
+           }
+
+           var url = endpoint.replace('en', lang) + utilQsString({
+             action: 'query',
+             prop: 'langlinks',
+             format: 'json',
+             origin: '*',
+             lllimit: 500,
+             titles: title
            });
-           container.selectAll('.display-option-reset').classed('disabled', function (d) {
-             return _options[d] === 1;
-           }); // first time only, set brightness if needed
+           d3_json(url).then(function (result) {
+             if (result && result.error) {
+               throw new Error(result.error);
+             } else if (!result || !result.query || !result.query.pages) {
+               throw new Error('No Results');
+             }
 
-           if (containerEnter.size() && _options.brightness !== 1) {
-             context.background().brightness(_options.brightness);
-           }
-         }
+             if (callback) {
+               var list = result.query.pages[Object.keys(result.query.pages)[0]];
+               var translations = {};
 
-         return section;
-       }
+               if (list && list.langlinks) {
+                 list.langlinks.forEach(function (d) {
+                   translations[d.lang] = d['*'];
+                 });
+               }
 
-       function uiSettingsCustomBackground() {
-         var dispatch$1 = dispatch('change');
+               callback(null, translations);
+             }
+           })["catch"](function (err) {
+             if (callback) callback(err.message);
+           });
+         }
+       };
 
-         function render(selection) {
-           // keep separate copies of original and current settings
-           var _origSettings = {
-             template: corePreferences('background-custom-template')
-           };
-           var _currSettings = {
-             template: corePreferences('background-custom-template')
-           };
-           var example = 'https://{switch:a,b,c}.tile.openstreetmap.org/{zoom}/{x}/{y}.png';
-           var modal = uiConfirm(selection).okButton();
-           modal.classed('settings-modal settings-custom-background', true);
-           modal.select('.modal-section.header').append('h3').html(_t.html('settings.custom_background.header'));
-           var textSection = modal.select('.modal-section.message-text');
-           var instructions = "".concat(_t.html('settings.custom_background.instructions.info'), "\n") + '\n' + "#### ".concat(_t.html('settings.custom_background.instructions.wms.tokens_label'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.wms.tokens.proj'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.wms.tokens.wkid'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.wms.tokens.dimensions'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.wms.tokens.bbox'), "\n") + '\n' + "#### ".concat(_t.html('settings.custom_background.instructions.tms.tokens_label'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.tms.tokens.xyz'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.tms.tokens.flipped_y'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.tms.tokens.switch'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.tms.tokens.quadtile'), "\n") + "* ".concat(_t.html('settings.custom_background.instructions.tms.tokens.scale_factor'), "\n") + '\n' + "#### ".concat(_t.html('settings.custom_background.instructions.example'), "\n") + "`".concat(example, "`");
-           textSection.append('div').attr('class', 'instructions-template').html(marked_1(instructions));
-           textSection.append('textarea').attr('class', 'field-template').attr('placeholder', _t('settings.custom_background.template.placeholder')).call(utilNoAuto).property('value', _currSettings.template); // insert a cancel button
+       var services = {
+         geocoder: serviceNominatim,
+         keepRight: serviceKeepRight,
+         improveOSM: serviceImproveOSM,
+         osmose: serviceOsmose,
+         mapillary: serviceMapillary,
+         nsi: serviceNsi,
+         openstreetcam: serviceOpenstreetcam,
+         osm: serviceOsm,
+         osmWikibase: serviceOsmWikibase,
+         maprules: serviceMapRules,
+         streetside: serviceStreetside,
+         taginfo: serviceTaginfo,
+         vectorTile: serviceVectorTile,
+         wikidata: serviceWikidata,
+         wikipedia: serviceWikipedia
+       };
+
+       function modeDragNote(context) {
+         var mode = {
+           id: 'drag-note',
+           button: 'browse'
+         };
+         var edit = behaviorEdit(context);
 
-           var buttonSection = modal.select('.modal-section.buttons');
-           buttonSection.insert('button', '.ok-button').attr('class', 'button cancel-button secondary-action').html(_t.html('confirm.cancel'));
-           buttonSection.select('.cancel-button').on('click.cancel', clickCancel);
-           buttonSection.select('.ok-button').attr('disabled', isSaveDisabled).on('click.save', clickSave);
+         var _nudgeInterval;
 
-           function isSaveDisabled() {
-             return null;
-           } // restore the original template
+         var _lastLoc;
 
+         var _note; // most current note.. dragged note may have stale datum.
 
-           function clickCancel() {
-             textSection.select('.field-template').property('value', _origSettings.template);
-             corePreferences('background-custom-template', _origSettings.template);
-             this.blur();
-             modal.close();
-           } // accept the current template
 
+         function startNudge(d3_event, nudge) {
+           if (_nudgeInterval) window.clearInterval(_nudgeInterval);
+           _nudgeInterval = window.setInterval(function () {
+             context.map().pan(nudge);
+             doMove(d3_event, nudge);
+           }, 50);
+         }
 
-           function clickSave() {
-             _currSettings.template = textSection.select('.field-template').property('value');
-             corePreferences('background-custom-template', _currSettings.template);
-             this.blur();
-             modal.close();
-             dispatch$1.call('change', this, _currSettings);
+         function stopNudge() {
+           if (_nudgeInterval) {
+             window.clearInterval(_nudgeInterval);
+             _nudgeInterval = null;
            }
          }
 
-         return utilRebind(render, dispatch$1, 'on');
-       }
+         function origin(note) {
+           return context.projection(note.loc);
+         }
 
-       function uiSectionBackgroundList(context) {
-         var _backgroundList = select(null);
+         function start(d3_event, note) {
+           _note = note;
+           var osm = services.osm;
 
-         var _customSource = context.background().findSource('custom');
+           if (osm) {
+             // Get latest note from cache.. The marker may have a stale datum bound to it
+             // and dragging it around can sometimes delete the users note comment.
+             _note = osm.getNote(_note.id);
+           }
 
-         var _settingsCustomBackground = uiSettingsCustomBackground().on('change', customChanged);
+           context.surface().selectAll('.note-' + _note.id).classed('active', true);
+           context.perform(actionNoop());
+           context.enter(mode);
+           context.selectedNoteID(_note.id);
+         }
 
-         var section = uiSection('background-list', context).label(_t.html('background.backgrounds')).disclosureContent(renderDisclosureContent);
+         function move(d3_event, entity, point) {
+           d3_event.stopPropagation();
+           _lastLoc = context.projection.invert(point);
+           doMove(d3_event);
+           var nudge = geoViewportEdge(point, context.map().dimensions());
 
-         function previousBackgroundID() {
-           return corePreferences('background-last-used-toggle');
+           if (nudge) {
+             startNudge(d3_event, nudge);
+           } else {
+             stopNudge();
+           }
          }
 
-         function renderDisclosureContent(selection) {
-           // the background list
-           var container = selection.selectAll('.layer-background-list').data([0]);
-           _backgroundList = container.enter().append('ul').attr('class', 'layer-list layer-background-list').attr('dir', 'auto').merge(container); // add minimap toggle below list
-
-           var bgExtrasListEnter = selection.selectAll('.bg-extras-list').data([0]).enter().append('ul').attr('class', 'layer-list bg-extras-list');
-           var minimapLabelEnter = bgExtrasListEnter.append('li').attr('class', 'minimap-toggle-item').append('label').call(uiTooltip().title(_t.html('background.minimap.tooltip')).keys([_t('background.minimap.key')]).placement('top'));
-           minimapLabelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event) {
-             d3_event.preventDefault();
-             uiMapInMap.toggle();
-           });
-           minimapLabelEnter.append('span').html(_t.html('background.minimap.description'));
-           var panelLabelEnter = bgExtrasListEnter.append('li').attr('class', 'background-panel-toggle-item').append('label').call(uiTooltip().title(_t.html('background.panel.tooltip')).keys([uiCmd('⌘⇧' + _t('info_panels.background.key'))]).placement('top'));
-           panelLabelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event) {
-             d3_event.preventDefault();
-             context.ui().info.toggle('background');
-           });
-           panelLabelEnter.append('span').html(_t.html('background.panel.description'));
-           var locPanelLabelEnter = bgExtrasListEnter.append('li').attr('class', 'location-panel-toggle-item').append('label').call(uiTooltip().title(_t.html('background.location_panel.tooltip')).keys([uiCmd('⌘⇧' + _t('info_panels.location.key'))]).placement('top'));
-           locPanelLabelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event) {
-             d3_event.preventDefault();
-             context.ui().info.toggle('location');
-           });
-           locPanelLabelEnter.append('span').html(_t.html('background.location_panel.description')); // "Info / Report a Problem" link
+         function doMove(d3_event, nudge) {
+           nudge = nudge || [0, 0];
+           var currPoint = d3_event && d3_event.point || context.projection(_lastLoc);
+           var currMouse = geoVecSubtract(currPoint, nudge);
+           var loc = context.projection.invert(currMouse);
+           _note = _note.move(loc);
+           var osm = services.osm;
 
-           selection.selectAll('.imagery-faq').data([0]).enter().append('div').attr('class', 'imagery-faq').append('a').attr('target', '_blank').call(svgIcon('#iD-icon-out-link', 'inline')).attr('href', 'https://github.com/openstreetmap/iD/blob/develop/FAQ.md#how-can-i-report-an-issue-with-background-imagery').append('span').html(_t.html('background.imagery_problem_faq'));
+           if (osm) {
+             osm.replaceNote(_note); // update note cache
+           }
 
-           _backgroundList.call(drawListItems, 'radio', function (d3_event, d) {
-             chooseBackground(d);
-           }, function (d) {
-             return !d.isHidden() && !d.overlay;
-           });
+           context.replace(actionNoop()); // trigger redraw
          }
 
-         function setTooltips(selection) {
-           selection.each(function (d, i, nodes) {
-             var item = select(this).select('label');
-             var span = item.select('span');
-             var placement = i < nodes.length / 2 ? 'bottom' : 'top';
-             var description = d.description();
-             var isOverflowing = span.property('clientWidth') !== span.property('scrollWidth');
-             item.call(uiTooltip().destroyAny);
+         function end() {
+           context.replace(actionNoop()); // trigger redraw
 
-             if (d.id === previousBackgroundID()) {
-               item.call(uiTooltip().placement(placement).title('<div>' + _t.html('background.switch') + '</div>').keys([uiCmd('⌘' + _t('background.key'))]));
-             } else if (description || isOverflowing) {
-               item.call(uiTooltip().placement(placement).title(description || d.label()));
-             }
-           });
+           context.selectedNoteID(_note.id).enter(modeSelectNote(context, _note.id));
          }
 
-         function drawListItems(layerList, type, change, filter) {
-           var sources = context.background().sources(context.map().extent(), context.map().zoom(), true).filter(filter).sort(function (a, b) {
-             return a.best() && !b.best() ? -1 : b.best() && !a.best() ? 1 : d3_descending(a.area(), b.area()) || d3_ascending(a.name(), b.name()) || 0;
-           });
-           var layerLinks = layerList.selectAll('li') // We have to be a bit inefficient about reordering the list since
-           // arrow key navigation of radio values likes to work in the order
-           // they were added, not the display document order.
-           .data(sources, function (d, i) {
-             return d.id + '---' + i;
-           });
-           layerLinks.exit().remove();
-           var enter = layerLinks.enter().append('li').classed('layer-custom', function (d) {
-             return d.id === 'custom';
-           }).classed('best', function (d) {
-             return d.best();
-           });
-           var label = enter.append('label');
-           label.append('input').attr('type', type).attr('name', 'background-layer').attr('value', function (d) {
-             return d.id;
-           }).on('change', change);
-           label.append('span').html(function (d) {
-             return d.label();
-           });
-           enter.filter(function (d) {
-             return d.id === 'custom';
-           }).append('button').attr('class', 'layer-browse').call(uiTooltip().title(_t.html('settings.custom_background.tooltip')).placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left')).on('click', function (d3_event) {
-             d3_event.preventDefault();
-             editCustom();
-           }).call(svgIcon('#iD-icon-more'));
-           enter.filter(function (d) {
-             return d.best();
-           }).append('div').attr('class', 'best').call(uiTooltip().title(_t.html('background.best_imagery')).placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left')).append('span').html('&#9733;');
-           layerList.call(updateLayerSelections);
-         }
+         var drag = behaviorDrag().selector('.layer-touch.markers .target.note.new').surface(context.container().select('.main-map').node()).origin(origin).on('start', start).on('move', move).on('end', end);
 
-         function updateLayerSelections(selection) {
-           function active(d) {
-             return context.background().showsLayer(d);
-           }
+         mode.enter = function () {
+           context.install(edit);
+         };
 
-           selection.selectAll('li').classed('active', active).classed('switch', function (d) {
-             return d.id === previousBackgroundID();
-           }).call(setTooltips).selectAll('input').property('checked', active);
-         }
+         mode.exit = function () {
+           context.ui().sidebar.hover.cancel();
+           context.uninstall(edit);
+           context.surface().selectAll('.active').classed('active', false);
+           stopNudge();
+         };
 
-         function chooseBackground(d) {
-           if (d.id === 'custom' && !d.template()) {
-             return editCustom();
-           }
+         mode.behavior = drag;
+         return mode;
+       }
 
-           var previousBackground = context.background().baseLayerSource();
-           corePreferences('background-last-used-toggle', previousBackground.id);
-           corePreferences('background-last-used', d.id);
-           context.background().baseLayerSource(d);
-         }
+       function modeSelectData(context, selectedDatum) {
+         var mode = {
+           id: 'select-data',
+           button: 'browse'
+         };
+         var keybinding = utilKeybinding('select-data');
+         var dataEditor = uiDataEditor(context);
+         var behaviors = [behaviorBreathe(), behaviorHover(context), behaviorSelect(context), behaviorLasso(context), modeDragNode(context).behavior, modeDragNote(context).behavior]; // class the data as selected, or return to browse mode if the data is gone
 
-         function customChanged(d) {
-           if (d && d.template) {
-             _customSource.template(d.template);
+         function selectData(d3_event, drawn) {
+           var selection = context.surface().selectAll('.layer-mapdata .data' + selectedDatum.__featurehash__);
 
-             chooseBackground(_customSource);
-           } else {
-             _customSource.template('');
+           if (selection.empty()) {
+             // Return to browse mode if selected DOM elements have
+             // disappeared because the user moved them out of view..
+             var source = d3_event && d3_event.type === 'zoom' && d3_event.sourceEvent;
 
-             chooseBackground(context.background().findSource('none'));
+             if (drawn && source && (source.type === 'pointermove' || source.type === 'mousemove' || source.type === 'touchmove')) {
+               context.enter(modeBrowse(context));
+             }
+           } else {
+             selection.classed('selected', true);
            }
          }
 
-         function editCustom() {
-           context.container().call(_settingsCustomBackground);
+         function esc() {
+           if (context.container().select('.combobox').size()) return;
+           context.enter(modeBrowse(context));
          }
 
-         context.background().on('change.background_list', function () {
-           _backgroundList.call(updateLayerSelections);
-         });
-         context.map().on('move.background_list', debounce(function () {
-           // layers in-view may have changed due to map move
-           window.requestIdleCallback(section.reRender);
-         }, 1000));
-         return section;
+         mode.zoomToSelected = function () {
+           var extent = geoExtent(d3_geoBounds(selectedDatum));
+           context.map().centerZoomEase(extent.center(), context.map().trimmedExtentZoom(extent));
+         };
+
+         mode.enter = function () {
+           behaviors.forEach(context.install);
+           keybinding.on(_t('inspector.zoom_to.key'), mode.zoomToSelected).on('⎋', esc, true);
+           select(document).call(keybinding);
+           selectData();
+           var sidebar = context.ui().sidebar;
+           sidebar.show(dataEditor.datum(selectedDatum)); // expand the sidebar, avoid obscuring the data if needed
+
+           var extent = geoExtent(d3_geoBounds(selectedDatum));
+           sidebar.expand(sidebar.intersects(extent));
+           context.map().on('drawn.select-data', selectData);
+         };
+
+         mode.exit = function () {
+           behaviors.forEach(context.uninstall);
+           select(document).call(keybinding.unbind);
+           context.surface().selectAll('.layer-mapdata .selected').classed('selected hover', false);
+           context.map().on('drawn.select-data', null);
+           context.ui().sidebar.hide();
+         };
+
+         return mode;
        }
 
-       function uiSectionBackgroundOffset(context) {
-         var section = uiSection('background-offset', context).label(_t.html('background.fix_misalignment')).disclosureContent(renderDisclosureContent).expandedByDefault(false);
+       function behaviorSelect(context) {
+         var _tolerancePx = 4; // see also behaviorDrag
 
-         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
+         var _lastMouseEvent = null;
+         var _showMenu = false;
+         var _downPointers = {};
+         var _longPressTimeout = null;
+         var _lastInteractionType = null; // the id of the down pointer that's enabling multiselection while down
 
-         var _directions = [['top', [0, -0.5]], ['left', [-0.5, 0]], ['right', [0.5, 0]], ['bottom', [0, 0.5]]];
+         var _multiselectionPointerId = null; // use pointer events on supported platforms; fallback to mouse events
 
-         function updateValue() {
-           var meters = geoOffsetToMeters(context.background().offset());
-           var x = +meters[0].toFixed(2);
-           var y = +meters[1].toFixed(2);
-           context.container().selectAll('.nudge-inner-rect').select('input').classed('error', false).property('value', x + ', ' + y);
-           context.container().selectAll('.nudge-reset').classed('disabled', function () {
-             return x === 0 && y === 0;
-           });
-         }
+         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
 
-         function resetOffset() {
-           context.background().offset([0, 0]);
-           updateValue();
-         }
+         function keydown(d3_event) {
+           if (d3_event.keyCode === 32) {
+             // don't react to spacebar events during text input
+             var activeNode = document.activeElement;
+             if (activeNode && new Set(['INPUT', 'TEXTAREA']).has(activeNode.nodeName)) return;
+           }
 
-         function nudge(d) {
-           context.background().nudge(d, context.map().zoom());
-           updateValue();
-         }
+           if (d3_event.keyCode === 93 || // context menu key
+           d3_event.keyCode === 32) {
+             // spacebar
+             d3_event.preventDefault();
+           }
 
-         function inputOffset() {
-           var input = select(this);
-           var d = input.node().value;
-           if (d === '') return resetOffset();
-           d = d.replace(/;/g, ',').split(',').map(function (n) {
-             // if n is NaN, it will always get mapped to false.
-             return !isNaN(n) && n;
-           });
+           if (d3_event.repeat) return; // ignore repeated events for held keys
+           // if any key is pressed the user is probably doing something other than long-pressing
 
-           if (d.length !== 2 || !d[0] || !d[1]) {
-             input.classed('error', true);
-             return;
+           cancelLongPress();
+
+           if (d3_event.shiftKey) {
+             context.surface().classed('behavior-multiselect', true);
            }
 
-           context.background().offset(geoMetersToOffset(d));
-           updateValue();
+           if (d3_event.keyCode === 32) {
+             // spacebar
+             if (!_downPointers.spacebar && _lastMouseEvent) {
+               cancelLongPress();
+               _longPressTimeout = window.setTimeout(didLongPress, 500, 'spacebar', 'spacebar');
+               _downPointers.spacebar = {
+                 firstEvent: _lastMouseEvent,
+                 lastEvent: _lastMouseEvent
+               };
+             }
+           }
          }
 
-         function dragOffset(d3_event) {
-           if (d3_event.button !== 0) return;
-           var origin = [d3_event.clientX, d3_event.clientY];
-           var pointerId = d3_event.pointerId || 'mouse';
-           context.container().append('div').attr('class', 'nudge-surface');
-           select(window).on(_pointerPrefix + 'move.drag-bg-offset', pointermove).on(_pointerPrefix + 'up.drag-bg-offset', pointerup);
+         function keyup(d3_event) {
+           cancelLongPress();
 
-           if (_pointerPrefix === 'pointer') {
-             select(window).on('pointercancel.drag-bg-offset', pointerup);
+           if (!d3_event.shiftKey) {
+             context.surface().classed('behavior-multiselect', false);
            }
 
-           function pointermove(d3_event) {
-             if (pointerId !== (d3_event.pointerId || 'mouse')) return;
-             var latest = [d3_event.clientX, d3_event.clientY];
-             var d = [-(origin[0] - latest[0]) / 4, -(origin[1] - latest[1]) / 4];
-             origin = latest;
-             nudge(d);
-           }
+           if (d3_event.keyCode === 93) {
+             // context menu key
+             d3_event.preventDefault();
+             _lastInteractionType = 'menukey';
+             contextmenu(d3_event);
+           } else if (d3_event.keyCode === 32) {
+             // spacebar
+             var pointer = _downPointers.spacebar;
 
-           function pointerup(d3_event) {
-             if (pointerId !== (d3_event.pointerId || 'mouse')) return;
-             if (d3_event.button !== 0) return;
-             context.container().selectAll('.nudge-surface').remove();
-             select(window).on('.drag-bg-offset', null);
+             if (pointer) {
+               delete _downPointers.spacebar;
+               if (pointer.done) return;
+               d3_event.preventDefault();
+               _lastInteractionType = 'spacebar';
+               click(pointer.firstEvent, pointer.lastEvent, 'spacebar');
+             }
            }
          }
 
-         function renderDisclosureContent(selection) {
-           var container = selection.selectAll('.nudge-container').data([0]);
-           var containerEnter = container.enter().append('div').attr('class', 'nudge-container');
-           containerEnter.append('div').attr('class', 'nudge-instructions').html(_t.html('background.offset'));
-           var nudgeWrapEnter = containerEnter.append('div').attr('class', 'nudge-controls-wrap');
-           var nudgeEnter = nudgeWrapEnter.append('div').attr('class', 'nudge-outer-rect').on(_pointerPrefix + 'down', dragOffset);
-           nudgeEnter.append('div').attr('class', 'nudge-inner-rect').append('input').attr('type', 'text').on('change', inputOffset);
-           nudgeWrapEnter.append('div').selectAll('button').data(_directions).enter().append('button').attr('class', function (d) {
-             return d[0] + ' nudge';
-           }).on('click', function (d3_event, d) {
-             nudge(d[1]);
-           });
-           nudgeWrapEnter.append('button').attr('title', _t('background.reset')).attr('class', 'nudge-reset disabled').on('click', function (d3_event) {
-             d3_event.preventDefault();
-             resetOffset();
-           }).call(svgIcon('#iD-icon-' + (_mainLocalizer.textDirection() === 'rtl' ? 'redo' : 'undo')));
-           updateValue();
+         function pointerdown(d3_event) {
+           var id = (d3_event.pointerId || 'mouse').toString();
+           cancelLongPress();
+           if (d3_event.buttons && d3_event.buttons !== 1) return;
+           context.ui().closeEditMenu();
+           _longPressTimeout = window.setTimeout(didLongPress, 500, id, 'longdown-' + (d3_event.pointerType || 'mouse'));
+           _downPointers[id] = {
+             firstEvent: d3_event,
+             lastEvent: d3_event
+           };
          }
 
-         context.background().on('change.backgroundOffset-update', updateValue);
-         return section;
-       }
+         function didLongPress(id, interactionType) {
+           var pointer = _downPointers[id];
+           if (!pointer) return;
 
-       function uiSectionOverlayList(context) {
-         var section = uiSection('overlay-list', context).label(_t.html('background.overlays')).disclosureContent(renderDisclosureContent);
+           for (var i in _downPointers) {
+             // don't allow this or any currently down pointer to trigger another click
+             _downPointers[i].done = true;
+           } // treat long presses like right-clicks
 
-         var _overlayList = select(null);
 
-         function setTooltips(selection) {
-           selection.each(function (d, i, nodes) {
-             var item = select(this).select('label');
-             var span = item.select('span');
-             var placement = i < nodes.length / 2 ? 'bottom' : 'top';
-             var description = d.description();
-             var isOverflowing = span.property('clientWidth') !== span.property('scrollWidth');
-             item.call(uiTooltip().destroyAny);
+           _longPressTimeout = null;
+           _lastInteractionType = interactionType;
+           _showMenu = true;
+           click(pointer.firstEvent, pointer.lastEvent, id);
+         }
 
-             if (description || isOverflowing) {
-               item.call(uiTooltip().placement(placement).title(description || d.name()));
+         function pointermove(d3_event) {
+           var id = (d3_event.pointerId || 'mouse').toString();
+
+           if (_downPointers[id]) {
+             _downPointers[id].lastEvent = d3_event;
+           }
+
+           if (!d3_event.pointerType || d3_event.pointerType === 'mouse') {
+             _lastMouseEvent = d3_event;
+
+             if (_downPointers.spacebar) {
+               _downPointers.spacebar.lastEvent = d3_event;
              }
-           });
+           }
          }
 
-         function updateLayerSelections(selection) {
-           function active(d) {
-             return context.background().showsLayer(d);
+         function pointerup(d3_event) {
+           var id = (d3_event.pointerId || 'mouse').toString();
+           var pointer = _downPointers[id];
+           if (!pointer) return;
+           delete _downPointers[id];
+
+           if (_multiselectionPointerId === id) {
+             _multiselectionPointerId = null;
            }
 
-           selection.selectAll('li').classed('active', active).call(setTooltips).selectAll('input').property('checked', active);
+           if (pointer.done) return;
+           click(pointer.firstEvent, d3_event, id);
          }
 
-         function chooseOverlay(d3_event, d) {
+         function pointercancel(d3_event) {
+           var id = (d3_event.pointerId || 'mouse').toString();
+           if (!_downPointers[id]) return;
+           delete _downPointers[id];
+
+           if (_multiselectionPointerId === id) {
+             _multiselectionPointerId = null;
+           }
+         }
+
+         function contextmenu(d3_event) {
            d3_event.preventDefault();
-           context.background().toggleOverlayLayer(d);
 
-           _overlayList.call(updateLayerSelections);
+           if (!+d3_event.clientX && !+d3_event.clientY) {
+             if (_lastMouseEvent) {
+               d3_event.sourceEvent = _lastMouseEvent;
+             } else {
+               return;
+             }
+           } else {
+             _lastMouseEvent = d3_event;
+             _lastInteractionType = 'rightclick';
+           }
 
-           document.activeElement.blur();
+           _showMenu = true;
+           click(d3_event, d3_event);
          }
 
-         function drawListItems(layerList, type, change, filter) {
-           var sources = context.background().sources(context.map().extent(), context.map().zoom(), true).filter(filter);
-           var layerLinks = layerList.selectAll('li').data(sources, function (d) {
-             return d.name();
-           });
-           layerLinks.exit().remove();
-           var enter = layerLinks.enter().append('li');
-           var label = enter.append('label');
-           label.append('input').attr('type', type).attr('name', 'layers').on('change', change);
-           label.append('span').html(function (d) {
-             return d.label();
-           });
-           layerList.selectAll('li').sort(sortSources);
-           layerList.call(updateLayerSelections);
+         function click(firstEvent, lastEvent, pointerId) {
+           cancelLongPress();
+           var mapNode = context.container().select('.main-map').node(); // Use the `main-map` coordinate system since the surface and supersurface
+           // are transformed when drag-panning.
+
+           var pointGetter = utilFastMouse(mapNode);
+           var p1 = pointGetter(firstEvent);
+           var p2 = pointGetter(lastEvent);
+           var dist = geoVecLength(p1, p2);
+
+           if (dist > _tolerancePx || !mapContains(lastEvent)) {
+             resetProperties();
+             return;
+           }
+
+           var targetDatum = lastEvent.target.__data__;
+           var multiselectEntityId;
+
+           if (!_multiselectionPointerId) {
+             // If a different pointer than the one triggering this click is down on a
+             // feature, treat this and all future clicks as multiselection until that
+             // pointer is raised.
+             var selectPointerInfo = pointerDownOnSelection(pointerId);
+
+             if (selectPointerInfo) {
+               _multiselectionPointerId = selectPointerInfo.pointerId; // if the other feature isn't selected yet, make sure we select it
+
+               multiselectEntityId = !selectPointerInfo.selected && selectPointerInfo.entityId;
+               _downPointers[selectPointerInfo.pointerId].done = true;
+             }
+           } // support multiselect if data is already selected
+
+
+           var isMultiselect = context.mode().id === 'select' && ( // and shift key is down
+           lastEvent && lastEvent.shiftKey || // or we're lasso-selecting
+           context.surface().select('.lasso').node() || // or a pointer is down over a selected feature
+           _multiselectionPointerId && !multiselectEntityId);
+
+           processClick(targetDatum, isMultiselect, p2, multiselectEntityId);
 
-           function sortSources(a, b) {
-             return a.best() && !b.best() ? -1 : b.best() && !a.best() ? 1 : d3_descending(a.area(), b.area()) || d3_ascending(a.name(), b.name()) || 0;
+           function mapContains(event) {
+             var rect = mapNode.getBoundingClientRect();
+             return event.clientX >= rect.left && event.clientX <= rect.right && event.clientY >= rect.top && event.clientY <= rect.bottom;
            }
-         }
 
-         function renderDisclosureContent(selection) {
-           var container = selection.selectAll('.layer-overlay-list').data([0]);
-           _overlayList = container.enter().append('ul').attr('class', 'layer-list layer-overlay-list').attr('dir', 'auto').merge(container);
+           function pointerDownOnSelection(skipPointerId) {
+             var mode = context.mode();
+             var selectedIDs = mode.id === 'select' ? mode.selectedIDs() : [];
 
-           _overlayList.call(drawListItems, 'checkbox', chooseOverlay, function (d) {
-             return !d.isHidden() && d.overlay;
-           });
-         }
+             for (var pointerId in _downPointers) {
+               if (pointerId === 'spacebar' || pointerId === skipPointerId) continue;
+               var pointerInfo = _downPointers[pointerId];
+               var p1 = pointGetter(pointerInfo.firstEvent);
+               var p2 = pointGetter(pointerInfo.lastEvent);
+               if (geoVecLength(p1, p2) > _tolerancePx) continue;
+               var datum = pointerInfo.firstEvent.target.__data__;
+               var entity = datum && datum.properties && datum.properties.entity || datum;
 
-         context.map().on('move.overlay_list', debounce(function () {
-           // layers in-view may have changed due to map move
-           window.requestIdleCallback(section.reRender);
-         }, 1000));
-         return section;
-       }
+               if (context.graph().hasEntity(entity.id)) {
+                 return {
+                   pointerId: pointerId,
+                   entityId: entity.id,
+                   selected: selectedIDs.indexOf(entity.id) !== -1
+                 };
+               }
+             }
 
-       function uiPaneBackground(context) {
-         var backgroundPane = uiPane('background', context).key(_t('background.key')).label(_t.html('background.title')).description(_t.html('background.description')).iconName('iD-icon-layers').sections([uiSectionBackgroundList(context), uiSectionOverlayList(context), uiSectionBackgroundDisplayOptions(context), uiSectionBackgroundOffset(context)]);
-         return backgroundPane;
-       }
+             return null;
+           }
+         }
 
-       function uiPaneHelp(context) {
-         var docKeys = [['help', ['welcome', 'open_data_h', 'open_data', 'before_start_h', 'before_start', 'open_source_h', 'open_source', 'open_source_help']], ['overview', ['navigation_h', 'navigation_drag', 'navigation_zoom', 'features_h', 'features', 'nodes_ways']], ['editing', ['select_h', 'select_left_click', 'select_right_click', 'select_space', 'multiselect_h', 'multiselect', 'multiselect_shift_click', 'multiselect_lasso', 'undo_redo_h', 'undo_redo', 'save_h', 'save', 'save_validation', 'upload_h', 'upload', 'backups_h', 'backups', 'keyboard_h', 'keyboard']], ['feature_editor', ['intro', 'definitions', 'type_h', 'type', 'type_picker', 'fields_h', 'fields_all_fields', 'fields_example', 'fields_add_field', 'tags_h', 'tags_all_tags', 'tags_resources']], ['points', ['intro', 'add_point_h', 'add_point', 'add_point_finish', 'move_point_h', 'move_point', 'delete_point_h', 'delete_point', 'delete_point_command']], ['lines', ['intro', 'add_line_h', 'add_line', 'add_line_draw', 'add_line_continue', 'add_line_finish', 'modify_line_h', 'modify_line_dragnode', 'modify_line_addnode', 'connect_line_h', 'connect_line', 'connect_line_display', 'connect_line_drag', 'connect_line_tag', 'disconnect_line_h', 'disconnect_line_command', 'move_line_h', 'move_line_command', 'move_line_connected', 'delete_line_h', 'delete_line', 'delete_line_command']], ['areas', ['intro', 'point_or_area_h', 'point_or_area', 'add_area_h', 'add_area_command', 'add_area_draw', 'add_area_continue', 'add_area_finish', 'square_area_h', 'square_area_command', 'modify_area_h', 'modify_area_dragnode', 'modify_area_addnode', 'delete_area_h', 'delete_area', 'delete_area_command']], ['relations', ['intro', 'edit_relation_h', 'edit_relation', 'edit_relation_add', 'edit_relation_delete', 'maintain_relation_h', 'maintain_relation', 'relation_types_h', 'multipolygon_h', 'multipolygon', 'multipolygon_create', 'multipolygon_merge', 'turn_restriction_h', 'turn_restriction', 'turn_restriction_field', 'turn_restriction_editing', 'route_h', 'route', 'route_add', 'boundary_h', 'boundary', 'boundary_add']], ['operations', ['intro', 'intro_2', 'straighten', 'orthogonalize', 'circularize', 'move', 'rotate', 'reflect', 'continue', 'reverse', 'disconnect', 'split', 'extract', 'merge', 'delete', 'downgrade', 'copy_paste']], ['notes', ['intro', 'add_note_h', 'add_note', 'place_note', 'move_note', 'update_note_h', 'update_note', 'save_note_h', 'save_note']], ['imagery', ['intro', 'sources_h', 'choosing', 'sources', 'offsets_h', 'offset', 'offset_change']], ['streetlevel', ['intro', 'using_h', 'using', 'photos', 'viewer']], ['gps', ['intro', 'survey', 'using_h', 'using', 'tracing', 'upload']], ['qa', ['intro', 'tools_h', 'tools', 'issues_h', 'issues']]];
-         var headings = {
-           'help.help.open_data_h': 3,
-           'help.help.before_start_h': 3,
-           'help.help.open_source_h': 3,
-           'help.overview.navigation_h': 3,
-           'help.overview.features_h': 3,
-           'help.editing.select_h': 3,
-           'help.editing.multiselect_h': 3,
-           'help.editing.undo_redo_h': 3,
-           'help.editing.save_h': 3,
-           'help.editing.upload_h': 3,
-           'help.editing.backups_h': 3,
-           'help.editing.keyboard_h': 3,
-           'help.feature_editor.type_h': 3,
-           'help.feature_editor.fields_h': 3,
-           'help.feature_editor.tags_h': 3,
-           'help.points.add_point_h': 3,
-           'help.points.move_point_h': 3,
-           'help.points.delete_point_h': 3,
-           'help.lines.add_line_h': 3,
-           'help.lines.modify_line_h': 3,
-           'help.lines.connect_line_h': 3,
-           'help.lines.disconnect_line_h': 3,
-           'help.lines.move_line_h': 3,
-           'help.lines.delete_line_h': 3,
-           'help.areas.point_or_area_h': 3,
-           'help.areas.add_area_h': 3,
-           'help.areas.square_area_h': 3,
-           'help.areas.modify_area_h': 3,
-           'help.areas.delete_area_h': 3,
-           'help.relations.edit_relation_h': 3,
-           'help.relations.maintain_relation_h': 3,
-           'help.relations.relation_types_h': 2,
-           'help.relations.multipolygon_h': 3,
-           'help.relations.turn_restriction_h': 3,
-           'help.relations.route_h': 3,
-           'help.relations.boundary_h': 3,
-           'help.notes.add_note_h': 3,
-           'help.notes.update_note_h': 3,
-           'help.notes.save_note_h': 3,
-           'help.imagery.sources_h': 3,
-           'help.imagery.offsets_h': 3,
-           'help.streetlevel.using_h': 3,
-           'help.gps.using_h': 3,
-           'help.qa.tools_h': 3,
-           'help.qa.issues_h': 3
-         }; // For each section, squash all the texts into a single markdown document
+         function processClick(datum, isMultiselect, point, alsoSelectId) {
+           var mode = context.mode();
+           var showMenu = _showMenu;
+           var interactionType = _lastInteractionType;
+           var entity = datum && datum.properties && datum.properties.entity;
+           if (entity) datum = entity;
 
-         var docs = docKeys.map(function (key) {
-           var helpkey = 'help.' + key[0];
-           var helpPaneReplacements = {
-             version: context.version
-           };
-           var text = key[1].reduce(function (all, part) {
-             var subkey = helpkey + '.' + part;
-             var depth = headings[subkey]; // is this subkey a heading?
+           if (datum && datum.type === 'midpoint') {
+             // treat targeting midpoints as if targeting the parent way
+             datum = datum.parents[0];
+           }
 
-             var hhh = depth ? Array(depth + 1).join('#') + ' ' : ''; // if so, prepend with some ##'s
+           var newMode;
 
-             return all + hhh + helpHtml(subkey, helpPaneReplacements) + '\n\n';
-           }, '');
-           return {
-             title: _t.html(helpkey + '.title'),
-             content: marked_1(text.trim()) // use keyboard key styling for shortcuts
-             .replace(/<code>/g, '<kbd>').replace(/<\/code>/g, '<\/kbd>')
-           };
-         });
-         var helpPane = uiPane('help', context).key(_t('help.key')).label(_t.html('help.title')).description(_t.html('help.title')).iconName('iD-icon-help');
+           if (datum instanceof osmEntity) {
+             // targeting an entity
+             var selectedIDs = context.selectedIDs();
+             context.selectedNoteID(null);
+             context.selectedErrorID(null);
 
-         helpPane.renderContent = function (content) {
-           function clickHelp(d, i) {
-             var rtl = _mainLocalizer.textDirection() === 'rtl';
-             content.property('scrollTop', 0);
-             helpPane.selection().select('.pane-heading h2').html(d.title);
-             body.html(d.content);
-             body.selectAll('a').attr('target', '_blank');
-             menuItems.classed('selected', function (m) {
-               return m.title === d.title;
-             });
-             nav.html('');
+             if (!isMultiselect) {
+               // don't change the selection if we're toggling the menu atop a multiselection
+               if (!showMenu || selectedIDs.length <= 1 || selectedIDs.indexOf(datum.id) === -1) {
+                 if (alsoSelectId === datum.id) alsoSelectId = null;
+                 selectedIDs = (alsoSelectId ? [alsoSelectId] : []).concat([datum.id]); // always enter modeSelect even if the entity is already
+                 // selected since listeners may expect `context.enter` events,
+                 // e.g. in the walkthrough
 
-             if (rtl) {
-               nav.call(drawNext).call(drawPrevious);
+                 newMode = mode.id === 'select' ? mode.selectedIDs(selectedIDs) : modeSelect(context, selectedIDs).selectBehavior(behavior);
+                 context.enter(newMode);
+               }
              } else {
-               nav.call(drawPrevious).call(drawNext);
-             }
-
-             function drawNext(selection) {
-               if (i < docs.length - 1) {
-                 var nextLink = selection.append('a').attr('href', '#').attr('class', 'next').on('click', function (d3_event) {
-                   d3_event.preventDefault();
-                   clickHelp(docs[i + 1], i + 1);
-                 });
-                 nextLink.append('span').html(docs[i + 1].title).call(svgIcon(rtl ? '#iD-icon-backward' : '#iD-icon-forward', 'inline'));
+               if (selectedIDs.indexOf(datum.id) !== -1) {
+                 // clicked entity is already in the selectedIDs list..
+                 if (!showMenu) {
+                   // deselect clicked entity, then reenter select mode or return to browse mode..
+                   selectedIDs = selectedIDs.filter(function (id) {
+                     return id !== datum.id;
+                   });
+                   newMode = selectedIDs.length ? mode.selectedIDs(selectedIDs) : modeBrowse(context).selectBehavior(behavior);
+                   context.enter(newMode);
+                 }
+               } else {
+                 // clicked entity is not in the selected list, add it..
+                 selectedIDs = selectedIDs.concat([datum.id]);
+                 newMode = mode.selectedIDs(selectedIDs);
+                 context.enter(newMode);
                }
              }
+           } else if (datum && datum.__featurehash__ && !isMultiselect) {
+             // targeting custom data
+             context.selectedNoteID(null).enter(modeSelectData(context, datum));
+           } else if (datum instanceof osmNote && !isMultiselect) {
+             // targeting a note
+             context.selectedNoteID(datum.id).enter(modeSelectNote(context, datum.id));
+           } else if (datum instanceof QAItem & !isMultiselect) {
+             // targeting an external QA issue
+             context.selectedErrorID(datum.id).enter(modeSelectError(context, datum.id, datum.service));
+           } else {
+             // targeting nothing
+             context.selectedNoteID(null);
+             context.selectedErrorID(null);
 
-             function drawPrevious(selection) {
-               if (i > 0) {
-                 var prevLink = selection.append('a').attr('href', '#').attr('class', 'previous').on('click', function (d3_event) {
-                   d3_event.preventDefault();
-                   clickHelp(docs[i - 1], i - 1);
-                 });
-                 prevLink.call(svgIcon(rtl ? '#iD-icon-forward' : '#iD-icon-backward', 'inline')).append('span').html(docs[i - 1].title);
-               }
+             if (!isMultiselect && mode.id !== 'browse') {
+               context.enter(modeBrowse(context));
              }
            }
 
-           function clickWalkthrough(d3_event) {
-             d3_event.preventDefault();
-             if (context.inIntro()) return;
-             context.container().call(uiIntro(context));
-             context.ui().togglePanes();
-           }
+           context.ui().closeEditMenu(); // always request to show the edit menu in case the mode needs it
 
-           function clickShortcuts(d3_event) {
-             d3_event.preventDefault();
-             context.container().call(context.ui().shortcuts, true);
-           }
+           if (showMenu) context.ui().showEditMenu(point, interactionType);
+           resetProperties();
+         }
 
-           var toc = content.append('ul').attr('class', 'toc');
-           var menuItems = toc.selectAll('li').data(docs).enter().append('li').append('a').attr('href', '#').html(function (d) {
-             return d.title;
-           }).on('click', function (d3_event, d) {
-             d3_event.preventDefault();
-             clickHelp(d, docs.indexOf(d));
+         function cancelLongPress() {
+           if (_longPressTimeout) window.clearTimeout(_longPressTimeout);
+           _longPressTimeout = null;
+         }
+
+         function resetProperties() {
+           cancelLongPress();
+           _showMenu = false;
+           _lastInteractionType = null; // don't reset _lastMouseEvent since it might still be useful
+         }
+
+         function behavior(selection) {
+           resetProperties();
+           _lastMouseEvent = context.map().lastPointerEvent();
+           select(window).on('keydown.select', keydown).on('keyup.select', keyup).on(_pointerPrefix + 'move.select', pointermove, true).on(_pointerPrefix + 'up.select', pointerup, true).on('pointercancel.select', pointercancel, true).on('contextmenu.select-window', function (d3_event) {
+             // Edge and IE really like to show the contextmenu on the
+             // menubar when user presses a keyboard menu button
+             // even after we've already preventdefaulted the key event.
+             var e = d3_event;
+
+             if (+e.clientX === 0 && +e.clientY === 0) {
+               d3_event.preventDefault();
+             }
            });
-           var shortcuts = toc.append('li').attr('class', 'shortcuts').call(uiTooltip().title(_t.html('shortcuts.tooltip')).keys(['?']).placement('top')).append('a').attr('href', '#').on('click', clickShortcuts);
-           shortcuts.append('div').html(_t.html('shortcuts.title'));
-           var walkthrough = toc.append('li').attr('class', 'walkthrough').append('a').attr('href', '#').on('click', clickWalkthrough);
-           walkthrough.append('svg').attr('class', 'logo logo-walkthrough').append('use').attr('xlink:href', '#iD-logo-walkthrough');
-           walkthrough.append('div').html(_t.html('splash.walkthrough'));
-           var helpContent = content.append('div').attr('class', 'left-content');
-           var body = helpContent.append('div').attr('class', 'body');
-           var nav = helpContent.append('div').attr('class', 'nav');
-           clickHelp(docs[0], 0);
+           selection.on(_pointerPrefix + 'down.select', pointerdown).on('contextmenu.select', contextmenu);
+           /*if (d3_event && d3_event.shiftKey) {
+               context.surface()
+                   .classed('behavior-multiselect', true);
+           }*/
+         }
+
+         behavior.off = function (selection) {
+           cancelLongPress();
+           select(window).on('keydown.select', null).on('keyup.select', null).on('contextmenu.select-window', null).on(_pointerPrefix + 'move.select', null, true).on(_pointerPrefix + 'up.select', null, true).on('pointercancel.select', null, true);
+           selection.on(_pointerPrefix + 'down.select', null).on('contextmenu.select', null);
+           context.surface().classed('behavior-multiselect', false);
          };
 
-         return helpPane;
+         return behavior;
        }
 
-       function uiSectionValidationIssues(id, severity, context) {
-         var _issues = [];
-         var section = uiSection(id, context).label(function () {
-           if (!_issues) return '';
-           var issueCountText = _issues.length > 1000 ? '1000+' : String(_issues.length);
-           return _t('inspector.title_count', {
-             title: _t.html('issues.' + severity + 's.list_title'),
-             count: issueCountText
-           });
-         }).disclosureContent(renderDisclosureContent).shouldDisplay(function () {
-           return _issues && _issues.length;
+       function operationContinue(context, selectedIDs) {
+         var _entities = selectedIDs.map(function (id) {
+           return context.graph().entity(id);
          });
 
-         function getOptions() {
-           return {
-             what: corePreferences('validate-what') || 'edited',
-             where: corePreferences('validate-where') || 'all'
-           };
-         } // get and cache the issues to display, unordered
+         var _geometries = Object.assign({
+           line: [],
+           vertex: []
+         }, utilArrayGroupBy(_entities, function (entity) {
+           return entity.geometry(context.graph());
+         }));
 
+         var _vertex = _geometries.vertex.length && _geometries.vertex[0];
 
-         function reloadIssues() {
-           _issues = context.validator().getIssuesBySeverity(getOptions())[severity];
+         function candidateWays() {
+           return _vertex ? context.graph().parentWays(_vertex).filter(function (parent) {
+             return parent.geometry(context.graph()) === 'line' && !parent.isClosed() && parent.affix(_vertex.id) && (_geometries.line.length === 0 || _geometries.line[0] === parent);
+           }) : [];
          }
 
-         function renderDisclosureContent(selection) {
-           var center = context.map().center();
-           var graph = context.graph(); // sort issues by distance away from the center of the map
-
-           var issues = _issues.map(function withDistance(issue) {
-             var extent = issue.extent(graph);
-             var dist = extent ? geoSphericalDistance(center, extent.center()) : 0;
-             return Object.assign(issue, {
-               dist: dist
-             });
-           }).sort(function byDistance(a, b) {
-             return a.dist - b.dist;
-           }); // cut off at 1000
-
-
-           issues = issues.slice(0, 1000); //renderIgnoredIssuesReset(_warningsSelection);
-
-           selection.call(drawIssuesList, issues);
-         }
+         var _candidates = candidateWays();
 
-         function drawIssuesList(selection, issues) {
-           var list = selection.selectAll('.issues-list').data([0]);
-           list = list.enter().append('ul').attr('class', 'layer-list issues-list ' + severity + 's-list').merge(list);
-           var items = list.selectAll('li').data(issues, function (d) {
-             return d.id;
-           }); // Exit
+         var operation = function operation() {
+           var candidate = _candidates[0];
+           context.enter(modeDrawLine(context, candidate.id, context.graph(), 'line', candidate.affix(_vertex.id), true));
+         };
 
-           items.exit().remove(); // Enter
+         operation.relatedEntityIds = function () {
+           return _candidates.length ? [_candidates[0].id] : [];
+         };
 
-           var itemsEnter = items.enter().append('li').attr('class', function (d) {
-             return 'issue severity-' + d.severity;
-           });
-           var labelsEnter = itemsEnter.append('button').attr('class', 'issue-label').on('click', function (d3_event, d) {
-             context.validator().focusIssue(d);
-           }).on('mouseover', function (d3_event, d) {
-             utilHighlightEntities(d.entityIds, true, context);
-           }).on('mouseout', function (d3_event, d) {
-             utilHighlightEntities(d.entityIds, false, context);
-           });
-           var textEnter = labelsEnter.append('span').attr('class', 'issue-text');
-           textEnter.append('span').attr('class', 'issue-icon').each(function (d) {
-             var iconName = '#iD-icon-' + (d.severity === 'warning' ? 'alert' : 'error');
-             select(this).call(svgIcon(iconName));
-           });
-           textEnter.append('span').attr('class', 'issue-message');
-           /*
-           labelsEnter
-               .append('span')
-               .attr('class', 'issue-autofix')
-               .each(function(d) {
-                   if (!d.autoFix) return;
-                    d3_select(this)
-                       .append('button')
-                       .attr('title', t('issues.fix_one.title'))
-                       .datum(d.autoFix)  // set button datum to the autofix
-                       .attr('class', 'autofix action')
-                       .on('click', function(d3_event, d) {
-                           d3_event.preventDefault();
-                           d3_event.stopPropagation();
-                            var issuesEntityIDs = d.issue.entityIds;
-                           utilHighlightEntities(issuesEntityIDs.concat(d.entityIds), false, context);
-                            context.perform.apply(context, d.autoArgs);
-                           context.validator().validate();
-                       })
-                       .call(svgIcon('#iD-icon-wrench'));
-               });
-           */
-           // Update
+         operation.available = function () {
+           return _geometries.vertex.length === 1 && _geometries.line.length <= 1 && !context.features().hasHiddenConnections(_vertex, context.graph());
+         };
 
-           items = items.merge(itemsEnter).order();
-           items.selectAll('.issue-message').html(function (d) {
-             return d.message(context);
-           });
-           /*
-           // autofix
-           var canAutoFix = issues.filter(function(issue) { return issue.autoFix; });
-            var autoFixAll = selection.selectAll('.autofix-all')
-               .data(canAutoFix.length ? [0] : []);
-            // exit
-           autoFixAll.exit()
-               .remove();
-            // enter
-           var autoFixAllEnter = autoFixAll.enter()
-               .insert('div', '.issues-list')
-               .attr('class', 'autofix-all');
-            var linkEnter = autoFixAllEnter
-               .append('a')
-               .attr('class', 'autofix-all-link')
-               .attr('href', '#');
-            linkEnter
-               .append('span')
-               .attr('class', 'autofix-all-link-text')
-               .html(t.html('issues.fix_all.title'));
-            linkEnter
-               .append('span')
-               .attr('class', 'autofix-all-link-icon')
-               .call(svgIcon('#iD-icon-wrench'));
-            if (severity === 'warning') {
-               renderIgnoredIssuesReset(selection);
+         operation.disabled = function () {
+           if (_candidates.length === 0) {
+             return 'not_eligible';
+           } else if (_candidates.length > 1) {
+             return 'multiple';
            }
-            // update
-           autoFixAll = autoFixAll
-               .merge(autoFixAllEnter);
-            autoFixAll.selectAll('.autofix-all-link')
-               .on('click', function() {
-                   context.pauseChangeDispatch();
-                   context.perform(actionNoop());
-                   canAutoFix.forEach(function(issue) {
-                       var args = issue.autoFix.autoArgs.slice();  // copy
-                       if (typeof args[args.length - 1] !== 'function') {
-                           args.pop();
-                       }
-                       args.push(t('issues.fix_all.annotation'));
-                       context.replace.apply(context, args);
-                   });
-                   context.resumeChangeDispatch();
-                   context.validator().validate();
-               });
-           */
-         }
-
-         context.validator().on('validated.uiSectionValidationIssues' + id, function () {
-           window.requestIdleCallback(function () {
-             reloadIssues();
-             section.reRender();
-           });
-         });
-         context.map().on('move.uiSectionValidationIssues' + id, debounce(function () {
-           window.requestIdleCallback(function () {
-             if (getOptions().where === 'visible') {
-               // must refetch issues if they are viewport-dependent
-               reloadIssues();
-             } // always reload list to re-sort-by-distance
 
+           return false;
+         };
 
-             section.reRender();
-           });
-         }, 1000));
-         return section;
-       }
+         operation.tooltip = function () {
+           var disable = operation.disabled();
+           return disable ? _t('operations.continue.' + disable) : _t('operations.continue.description');
+         };
 
-       function uiSectionValidationOptions(context) {
-         var section = uiSection('issues-options', context).content(renderContent);
+         operation.annotation = function () {
+           return _t('operations.continue.annotation.line');
+         };
 
-         function renderContent(selection) {
-           var container = selection.selectAll('.issues-options-container').data([0]);
-           container = container.enter().append('div').attr('class', 'issues-options-container').merge(container);
-           var data = [{
-             key: 'what',
-             values: ['edited', 'all']
-           }, {
-             key: 'where',
-             values: ['visible', 'all']
-           }];
-           var options = container.selectAll('.issues-option').data(data, function (d) {
-             return d.key;
-           });
-           var optionsEnter = options.enter().append('div').attr('class', function (d) {
-             return 'issues-option issues-option-' + d.key;
-           });
-           optionsEnter.append('div').attr('class', 'issues-option-title').html(function (d) {
-             return _t.html('issues.options.' + d.key + '.title');
-           });
-           var valuesEnter = optionsEnter.selectAll('label').data(function (d) {
-             return d.values.map(function (val) {
-               return {
-                 value: val,
-                 key: d.key
-               };
-             });
-           }).enter().append('label');
-           valuesEnter.append('input').attr('type', 'radio').attr('name', function (d) {
-             return 'issues-option-' + d.key;
-           }).attr('value', function (d) {
-             return d.value;
-           }).property('checked', function (d) {
-             return getOptions()[d.key] === d.value;
-           }).on('change', function (d3_event, d) {
-             updateOptionValue(d3_event, d.key, d.value);
-           });
-           valuesEnter.append('span').html(function (d) {
-             return _t.html('issues.options.' + d.key + '.' + d.value);
+         operation.id = 'continue';
+         operation.keys = [_t('operations.continue.key')];
+         operation.title = _t('operations.continue.title');
+         operation.behavior = behaviorOperation(context).which(operation);
+         return operation;
+       }
+
+       function operationCopy(context, selectedIDs) {
+         function getFilteredIdsToCopy() {
+           return selectedIDs.filter(function (selectedID) {
+             var entity = context.graph().hasEntity(selectedID); // don't copy untagged vertices separately from ways
+
+             return entity.hasInterestingTags() || entity.geometry(context.graph()) !== 'vertex';
            });
          }
 
-         function getOptions() {
-           return {
-             what: corePreferences('validate-what') || 'edited',
-             // 'all', 'edited'
-             where: corePreferences('validate-where') || 'all' // 'all', 'visible'
+         var operation = function operation() {
+           var graph = context.graph();
+           var selected = groupEntities(getFilteredIdsToCopy(), graph);
+           var canCopy = [];
+           var skip = {};
+           var entity;
+           var i;
 
-           };
-         }
+           for (i = 0; i < selected.relation.length; i++) {
+             entity = selected.relation[i];
 
-         function updateOptionValue(d3_event, d, val) {
-           if (!val && d3_event && d3_event.target) {
-             val = d3_event.target.value;
+             if (!skip[entity.id] && entity.isComplete(graph)) {
+               canCopy.push(entity.id);
+               skip = getDescendants(entity.id, graph, skip);
+             }
            }
 
-           corePreferences('validate-' + d, val);
-           context.validator().validate();
-         }
+           for (i = 0; i < selected.way.length; i++) {
+             entity = selected.way[i];
 
-         return section;
-       }
+             if (!skip[entity.id]) {
+               canCopy.push(entity.id);
+               skip = getDescendants(entity.id, graph, skip);
+             }
+           }
 
-       function uiSectionValidationRules(context) {
-         var MINSQUARE = 0;
-         var MAXSQUARE = 20;
-         var DEFAULTSQUARE = 5; // see also unsquare_way.js
+           for (i = 0; i < selected.node.length; i++) {
+             entity = selected.node[i];
 
-         var section = uiSection('issues-rules', context).disclosureContent(renderDisclosureContent).label(_t.html('issues.rules.title'));
+             if (!skip[entity.id]) {
+               canCopy.push(entity.id);
+             }
+           }
 
-         var _ruleKeys = context.validator().getRuleKeys().filter(function (key) {
-           return key !== 'maprules';
-         }).sort(function (key1, key2) {
-           // alphabetize by localized title
-           return _t('issues.' + key1 + '.title') < _t('issues.' + key2 + '.title') ? -1 : 1;
-         });
+           context.copyIDs(canCopy);
 
-         function renderDisclosureContent(selection) {
-           var container = selection.selectAll('.issues-rulelist-container').data([0]);
-           var containerEnter = container.enter().append('div').attr('class', 'issues-rulelist-container');
-           containerEnter.append('ul').attr('class', 'layer-list issue-rules-list');
-           var ruleLinks = containerEnter.append('div').attr('class', 'issue-rules-links section-footer');
-           ruleLinks.append('a').attr('class', 'issue-rules-link').attr('href', '#').html(_t.html('issues.disable_all')).on('click', function (d3_event) {
-             d3_event.preventDefault();
-             context.validator().disableRules(_ruleKeys);
-           });
-           ruleLinks.append('a').attr('class', 'issue-rules-link').attr('href', '#').html(_t.html('issues.enable_all')).on('click', function (d3_event) {
-             d3_event.preventDefault();
-             context.validator().disableRules([]);
-           }); // Update
+           if (_point && (canCopy.length !== 1 || graph.entity(canCopy[0]).type !== 'node')) {
+             // store the anchor coordinates if copying more than a single node
+             context.copyLonLat(context.projection.invert(_point));
+           } else {
+             context.copyLonLat(null);
+           }
+         };
 
-           container = container.merge(containerEnter);
-           container.selectAll('.issue-rules-list').call(drawListItems, _ruleKeys, 'checkbox', 'rule', toggleRule, isRuleEnabled);
+         function groupEntities(ids, graph) {
+           var entities = ids.map(function (id) {
+             return graph.entity(id);
+           });
+           return Object.assign({
+             relation: [],
+             way: [],
+             node: []
+           }, utilArrayGroupBy(entities, 'type'));
          }
 
-         function drawListItems(selection, data, type, name, change, active) {
-           var items = selection.selectAll('li').data(data); // Exit
-
-           items.exit().remove(); // Enter
-
-           var enter = items.enter().append('li');
+         function getDescendants(id, graph, descendants) {
+           var entity = graph.entity(id);
+           var children;
+           descendants = descendants || {};
 
-           if (name === 'rule') {
-             enter.call(uiTooltip().title(function (d) {
-               return _t.html('issues.' + d + '.tip');
-             }).placement('top'));
+           if (entity.type === 'relation') {
+             children = entity.members.map(function (m) {
+               return m.id;
+             });
+           } else if (entity.type === 'way') {
+             children = entity.nodes;
+           } else {
+             children = [];
            }
 
-           var label = enter.append('label');
-           label.append('input').attr('type', type).attr('name', name).on('change', change);
-           label.append('span').html(function (d) {
-             var params = {};
-
-             if (d === 'unsquare_way') {
-               params.val = '<span class="square-degrees"></span>';
+           for (var i = 0; i < children.length; i++) {
+             if (!descendants[children[i]]) {
+               descendants[children[i]] = true;
+               descendants = getDescendants(children[i], graph, descendants);
              }
+           }
 
-             return _t.html('issues.' + d + '.title', params);
-           }); // Update
+           return descendants;
+         }
 
-           items = items.merge(enter);
-           items.classed('active', active).selectAll('input').property('checked', active).property('indeterminate', false); // user-configurable square threshold
+         operation.available = function () {
+           return getFilteredIdsToCopy().length > 0;
+         };
 
-           var degStr = corePreferences('validate-square-degrees');
+         operation.disabled = function () {
+           var extent = utilTotalExtent(getFilteredIdsToCopy(), context.graph());
 
-           if (degStr === null) {
-             degStr = DEFAULTSQUARE.toString();
+           if (extent.percentContainedIn(context.map().extent()) < 0.8) {
+             return 'too_large';
            }
 
-           var span = items.selectAll('.square-degrees');
-           var input = span.selectAll('.square-degrees-input').data([0]); // enter / update
-
-           input.enter().append('input').attr('type', 'number').attr('min', MINSQUARE.toString()).attr('max', MAXSQUARE.toString()).attr('step', '0.5').attr('class', 'square-degrees-input').call(utilNoAuto).on('click', function (d3_event) {
-             d3_event.preventDefault();
-             d3_event.stopPropagation();
-             this.select();
-           }).on('keyup', function (d3_event) {
-             if (d3_event.keyCode === 13) {
-               // ↩ Return
-               this.blur();
-               this.select();
-             }
-           }).on('blur', changeSquare).merge(input).property('value', degStr);
-         }
+           return false;
+         };
 
-         function changeSquare() {
-           var input = select(this);
-           var degStr = utilGetSetValue(input).trim();
-           var degNum = parseFloat(degStr, 10);
+         operation.availableForKeypress = function () {
+           var selection = window.getSelection && window.getSelection(); // if the user has text selected then let them copy that, not the selected feature
 
-           if (!isFinite(degNum)) {
-             degNum = DEFAULTSQUARE;
-           } else if (degNum > MAXSQUARE) {
-             degNum = MAXSQUARE;
-           } else if (degNum < MINSQUARE) {
-             degNum = MINSQUARE;
-           }
+           return !selection || !selection.toString();
+         };
 
-           degNum = Math.round(degNum * 10) / 10; // round to 1 decimal
+         operation.tooltip = function () {
+           var disable = operation.disabled();
+           return disable ? _t('operations.copy.' + disable, {
+             n: selectedIDs.length
+           }) : _t('operations.copy.description', {
+             n: selectedIDs.length
+           });
+         };
 
-           degStr = degNum.toString();
-           input.property('value', degStr);
-           corePreferences('validate-square-degrees', degStr);
-           context.validator().reloadUnsquareIssues();
-         }
+         operation.annotation = function () {
+           return _t('operations.copy.annotation', {
+             n: selectedIDs.length
+           });
+         };
 
-         function isRuleEnabled(d) {
-           return context.validator().isRuleEnabled(d);
-         }
+         var _point;
 
-         function toggleRule(d3_event, d) {
-           context.validator().toggleRule(d);
-         }
+         operation.point = function (val) {
+           _point = val;
+           return operation;
+         };
 
-         context.validator().on('validated.uiSectionValidationRules', function () {
-           window.requestIdleCallback(section.reRender);
-         });
-         return section;
+         operation.id = 'copy';
+         operation.keys = [uiCmd('⌘C')];
+         operation.title = _t('operations.copy.title');
+         operation.behavior = behaviorOperation(context).which(operation);
+         return operation;
        }
 
-       function uiSectionValidationStatus(context) {
-         var section = uiSection('issues-status', context).content(renderContent).shouldDisplay(function () {
-           var issues = context.validator().getIssues(getOptions());
-           return issues.length === 0;
+       function operationDisconnect(context, selectedIDs) {
+         var _vertexIDs = [];
+         var _wayIDs = [];
+         var _otherIDs = [];
+         var _actions = [];
+         selectedIDs.forEach(function (id) {
+           var entity = context.entity(id);
+
+           if (entity.type === 'way') {
+             _wayIDs.push(id);
+           } else if (entity.geometry(context.graph()) === 'vertex') {
+             _vertexIDs.push(id);
+           } else {
+             _otherIDs.push(id);
+           }
          });
 
-         function getOptions() {
-           return {
-             what: corePreferences('validate-what') || 'edited',
-             where: corePreferences('validate-where') || 'all'
-           };
-         }
+         var _coords,
+             _descriptionID = '',
+             _annotationID = 'features';
 
-         function renderContent(selection) {
-           var box = selection.selectAll('.box').data([0]);
-           var boxEnter = box.enter().append('div').attr('class', 'box');
-           boxEnter.append('div').call(svgIcon('#iD-icon-apply', 'pre-text'));
-           var noIssuesMessage = boxEnter.append('span');
-           noIssuesMessage.append('strong').attr('class', 'message');
-           noIssuesMessage.append('br');
-           noIssuesMessage.append('span').attr('class', 'details');
-           renderIgnoredIssuesReset(selection);
-           setNoIssuesText(selection);
-         }
+         var _disconnectingVertexIds = [];
+         var _disconnectingWayIds = [];
 
-         function renderIgnoredIssuesReset(selection) {
-           var ignoredIssues = context.validator().getIssues({
-             what: 'all',
-             where: 'all',
-             includeDisabledRules: true,
-             includeIgnored: 'only'
-           });
-           var resetIgnored = selection.selectAll('.reset-ignored').data(ignoredIssues.length ? [0] : []); // exit
+         if (_vertexIDs.length > 0) {
+           // At the selected vertices, disconnect the selected ways, if any, else
+           // disconnect all connected ways
+           _disconnectingVertexIds = _vertexIDs;
 
-           resetIgnored.exit().remove(); // enter
+           _vertexIDs.forEach(function (vertexID) {
+             var action = actionDisconnect(vertexID);
 
-           var resetIgnoredEnter = resetIgnored.enter().append('div').attr('class', 'reset-ignored section-footer');
-           resetIgnoredEnter.append('a').attr('href', '#'); // update
+             if (_wayIDs.length > 0) {
+               var waysIDsForVertex = _wayIDs.filter(function (wayID) {
+                 var way = context.entity(wayID);
+                 return way.nodes.indexOf(vertexID) !== -1;
+               });
 
-           resetIgnored = resetIgnored.merge(resetIgnoredEnter);
-           resetIgnored.select('a').html(_t('inspector.title_count', {
-             title: _t.html('issues.reset_ignored'),
-             count: ignoredIssues.length
-           }));
-           resetIgnored.on('click', function (d3_event) {
-             d3_event.preventDefault();
-             context.validator().resetIgnoredIssues();
-           });
-         }
+               action.limitWays(waysIDsForVertex);
+             }
 
-         function setNoIssuesText(selection) {
-           var opts = getOptions();
+             _actions.push(action);
 
-           function checkForHiddenIssues(cases) {
-             for (var type in cases) {
-               var hiddenOpts = cases[type];
-               var hiddenIssues = context.validator().getIssues(hiddenOpts);
+             _disconnectingWayIds = _disconnectingWayIds.concat(context.graph().parentWays(context.graph().entity(vertexID)).map(function (d) {
+               return d.id;
+             }));
+           });
 
-               if (hiddenIssues.length) {
-                 selection.select('.box .details').html(_t.html('issues.no_issues.hidden_issues.' + type, {
-                   count: hiddenIssues.length.toString()
-                 }));
-                 return;
-               }
-             }
+           _disconnectingWayIds = utilArrayUniq(_disconnectingWayIds).filter(function (id) {
+             return _wayIDs.indexOf(id) === -1;
+           });
+           _descriptionID += _actions.length === 1 ? 'single_point.' : 'multiple_points.';
 
-             selection.select('.box .details').html(_t.html('issues.no_issues.hidden_issues.none'));
+           if (_wayIDs.length === 1) {
+             _descriptionID += 'single_way.' + context.graph().geometry(_wayIDs[0]);
+           } else {
+             _descriptionID += _wayIDs.length === 0 ? 'no_ways' : 'multiple_ways';
            }
+         } else if (_wayIDs.length > 0) {
+           // Disconnect the selected ways from each other, if they're connected,
+           // else disconnect them from all connected ways
+           var ways = _wayIDs.map(function (id) {
+             return context.entity(id);
+           });
 
-           var messageType;
+           var nodes = utilGetAllNodes(_wayIDs, context.graph());
+           _coords = nodes.map(function (n) {
+             return n.loc;
+           }); // actions for connected nodes shared by at least two selected ways
 
-           if (opts.what === 'edited' && opts.where === 'visible') {
-             messageType = 'edits_in_view';
-             checkForHiddenIssues({
-               elsewhere: {
-                 what: 'edited',
-                 where: 'all'
-               },
-               everything_else: {
-                 what: 'all',
-                 where: 'visible'
-               },
-               disabled_rules: {
-                 what: 'edited',
-                 where: 'visible',
-                 includeDisabledRules: 'only'
-               },
-               everything_else_elsewhere: {
-                 what: 'all',
-                 where: 'all'
-               },
-               disabled_rules_elsewhere: {
-                 what: 'edited',
-                 where: 'all',
-                 includeDisabledRules: 'only'
-               },
-               ignored_issues: {
-                 what: 'edited',
-                 where: 'visible',
-                 includeIgnored: 'only'
-               },
-               ignored_issues_elsewhere: {
-                 what: 'edited',
-                 where: 'all',
-                 includeIgnored: 'only'
+           var sharedActions = [];
+           var sharedNodes = []; // actions for connected nodes
+
+           var unsharedActions = [];
+           var unsharedNodes = [];
+           nodes.forEach(function (node) {
+             var action = actionDisconnect(node.id).limitWays(_wayIDs);
+
+             if (action.disabled(context.graph()) !== 'not_connected') {
+               var count = 0;
+
+               for (var i in ways) {
+                 var way = ways[i];
+
+                 if (way.nodes.indexOf(node.id) !== -1) {
+                   count += 1;
+                 }
+
+                 if (count > 1) break;
                }
-             });
-           } else if (opts.what === 'edited' && opts.where === 'all') {
-             messageType = 'edits';
-             checkForHiddenIssues({
-               everything_else: {
-                 what: 'all',
-                 where: 'all'
-               },
-               disabled_rules: {
-                 what: 'edited',
-                 where: 'all',
-                 includeDisabledRules: 'only'
-               },
-               ignored_issues: {
-                 what: 'edited',
-                 where: 'all',
-                 includeIgnored: 'only'
+
+               if (count > 1) {
+                 sharedActions.push(action);
+                 sharedNodes.push(node);
+               } else {
+                 unsharedActions.push(action);
+                 unsharedNodes.push(node);
                }
+             }
+           });
+           _descriptionID += 'no_points.';
+           _descriptionID += _wayIDs.length === 1 ? 'single_way.' : 'multiple_ways.';
+
+           if (sharedActions.length) {
+             // if any nodes are shared, only disconnect the selected ways from each other
+             _actions = sharedActions;
+             _disconnectingVertexIds = sharedNodes.map(function (node) {
+               return node.id;
              });
-           } else if (opts.what === 'all' && opts.where === 'visible') {
-             messageType = 'everything_in_view';
-             checkForHiddenIssues({
-               elsewhere: {
-                 what: 'all',
-                 where: 'all'
-               },
-               disabled_rules: {
-                 what: 'all',
-                 where: 'visible',
-                 includeDisabledRules: 'only'
-               },
-               disabled_rules_elsewhere: {
-                 what: 'all',
-                 where: 'all',
-                 includeDisabledRules: 'only'
-               },
-               ignored_issues: {
-                 what: 'all',
-                 where: 'visible',
-                 includeIgnored: 'only'
-               },
-               ignored_issues_elsewhere: {
-                 what: 'all',
-                 where: 'all',
-                 includeIgnored: 'only'
-               }
+             _descriptionID += 'conjoined';
+             _annotationID = 'from_each_other';
+           } else {
+             // if no nodes are shared, disconnect the selected ways from all connected ways
+             _actions = unsharedActions;
+             _disconnectingVertexIds = unsharedNodes.map(function (node) {
+               return node.id;
              });
-           } else if (opts.what === 'all' && opts.where === 'all') {
-             messageType = 'everything';
-             checkForHiddenIssues({
-               disabled_rules: {
-                 what: 'all',
-                 where: 'all',
-                 includeDisabledRules: 'only'
-               },
-               ignored_issues: {
-                 what: 'all',
-                 where: 'all',
-                 includeIgnored: 'only'
-               }
+
+             if (_wayIDs.length === 1) {
+               _descriptionID += context.graph().geometry(_wayIDs[0]);
+             } else {
+               _descriptionID += 'separate';
+             }
+           }
+         }
+
+         var _extent = utilTotalExtent(_disconnectingVertexIds, context.graph());
+
+         var operation = function operation() {
+           context.perform(function (graph) {
+             return _actions.reduce(function (graph, action) {
+               return action(graph);
+             }, graph);
+           }, operation.annotation());
+           context.validator().validate();
+         };
+
+         operation.relatedEntityIds = function () {
+           if (_vertexIDs.length) {
+             return _disconnectingWayIds;
+           }
+
+           return _disconnectingVertexIds;
+         };
+
+         operation.available = function () {
+           if (_actions.length === 0) return false;
+           if (_otherIDs.length !== 0) return false;
+           if (_vertexIDs.length !== 0 && _wayIDs.length !== 0 && !_wayIDs.every(function (wayID) {
+             return _vertexIDs.some(function (vertexID) {
+               var way = context.entity(wayID);
+               return way.nodes.indexOf(vertexID) !== -1;
              });
+           })) return false;
+           return true;
+         };
+
+         operation.disabled = function () {
+           var reason;
+
+           for (var actionIndex in _actions) {
+             reason = _actions[actionIndex].disabled(context.graph());
+             if (reason) return reason;
            }
 
-           if (opts.what === 'edited' && context.history().difference().summary().length === 0) {
-             messageType = 'no_edits';
+           if (_extent && _extent.percentContainedIn(context.map().extent()) < 0.8) {
+             return 'too_large.' + ((_vertexIDs.length ? _vertexIDs : _wayIDs).length === 1 ? 'single' : 'multiple');
+           } else if (_coords && someMissing()) {
+             return 'not_downloaded';
+           } else if (selectedIDs.some(context.hasHiddenConnections)) {
+             return 'connected_to_hidden';
            }
 
-           selection.select('.box .message').html(_t.html('issues.no_issues.message.' + messageType));
-         }
+           return false;
 
-         context.validator().on('validated.uiSectionValidationStatus', function () {
-           window.requestIdleCallback(section.reRender);
-         });
-         context.map().on('move.uiSectionValidationStatus', debounce(function () {
-           window.requestIdleCallback(section.reRender);
-         }, 1000));
-         return section;
-       }
+           function someMissing() {
+             if (context.inIntro()) return false;
+             var osm = context.connection();
 
-       function uiPaneIssues(context) {
-         var issuesPane = uiPane('issues', context).key(_t('issues.key')).label(_t.html('issues.title')).description(_t.html('issues.title')).iconName('iD-icon-alert').sections([uiSectionValidationOptions(context), uiSectionValidationStatus(context), uiSectionValidationIssues('issues-errors', 'error', context), uiSectionValidationIssues('issues-warnings', 'warning', context), uiSectionValidationRules(context)]);
-         return issuesPane;
-       }
+             if (osm) {
+               var missing = _coords.filter(function (loc) {
+                 return !osm.isDataLoaded(loc);
+               });
 
-       function uiSettingsCustomData(context) {
-         var dispatch$1 = dispatch('change');
+               if (missing.length) {
+                 missing.forEach(function (loc) {
+                   context.loadTileAtLoc(loc);
+                 });
+                 return true;
+               }
+             }
 
-         function render(selection) {
-           var dataLayer = context.layers().layer('data'); // keep separate copies of original and current settings
+             return false;
+           }
+         };
 
-           var _origSettings = {
-             fileList: dataLayer && dataLayer.fileList() || null,
-             url: corePreferences('settings-custom-data-url')
-           };
-           var _currSettings = {
-             fileList: dataLayer && dataLayer.fileList() || null,
-             url: corePreferences('settings-custom-data-url')
-           }; // var example = 'https://{switch:a,b,c}.tile.openstreetmap.org/{zoom}/{x}/{y}.png';
+         operation.tooltip = function () {
+           var disable = operation.disabled();
 
-           var modal = uiConfirm(selection).okButton();
-           modal.classed('settings-modal settings-custom-data', true);
-           modal.select('.modal-section.header').append('h3').html(_t.html('settings.custom_data.header'));
-           var textSection = modal.select('.modal-section.message-text');
-           textSection.append('pre').attr('class', 'instructions-file').html(_t.html('settings.custom_data.file.instructions'));
-           textSection.append('input').attr('class', 'field-file').attr('type', 'file').property('files', _currSettings.fileList) // works for all except IE11
-           .on('change', function (d3_event) {
-             var files = d3_event.target.files;
+           if (disable) {
+             return _t('operations.disconnect.' + disable);
+           }
 
-             if (files && files.length) {
-               _currSettings.url = '';
-               textSection.select('.field-url').property('value', '');
-               _currSettings.fileList = files;
-             } else {
-               _currSettings.fileList = null;
-             }
-           });
-           textSection.append('h4').html(_t.html('settings.custom_data.or'));
-           textSection.append('pre').attr('class', 'instructions-url').html(_t.html('settings.custom_data.url.instructions'));
-           textSection.append('textarea').attr('class', 'field-url').attr('placeholder', _t('settings.custom_data.url.placeholder')).call(utilNoAuto).property('value', _currSettings.url); // insert a cancel button
+           return _t('operations.disconnect.description.' + _descriptionID);
+         };
 
-           var buttonSection = modal.select('.modal-section.buttons');
-           buttonSection.insert('button', '.ok-button').attr('class', 'button cancel-button secondary-action').html(_t.html('confirm.cancel'));
-           buttonSection.select('.cancel-button').on('click.cancel', clickCancel);
-           buttonSection.select('.ok-button').attr('disabled', isSaveDisabled).on('click.save', clickSave);
+         operation.annotation = function () {
+           return _t('operations.disconnect.annotation.' + _annotationID);
+         };
 
-           function isSaveDisabled() {
-             return null;
-           } // restore the original url
+         operation.id = 'disconnect';
+         operation.keys = [_t('operations.disconnect.key')];
+         operation.title = _t('operations.disconnect.title');
+         operation.behavior = behaviorOperation(context).which(operation);
+         return operation;
+       }
+
+       function operationDowngrade(context, selectedIDs) {
+         var _affectedFeatureCount = 0;
 
+         var _downgradeType = downgradeTypeForEntityIDs(selectedIDs);
 
-           function clickCancel() {
-             textSection.select('.field-url').property('value', _origSettings.url);
-             corePreferences('settings-custom-data-url', _origSettings.url);
-             this.blur();
-             modal.close();
-           } // accept the current url
+         var _multi = _affectedFeatureCount === 1 ? 'single' : 'multiple';
 
+         function downgradeTypeForEntityIDs(entityIds) {
+           var downgradeType;
+           _affectedFeatureCount = 0;
 
-           function clickSave() {
-             _currSettings.url = textSection.select('.field-url').property('value').trim(); // one or the other but not both
+           for (var i in entityIds) {
+             var entityID = entityIds[i];
+             var type = downgradeTypeForEntityID(entityID);
 
-             if (_currSettings.url) {
-               _currSettings.fileList = null;
-             }
+             if (type) {
+               _affectedFeatureCount += 1;
 
-             if (_currSettings.fileList) {
-               _currSettings.url = '';
+               if (downgradeType && type !== downgradeType) {
+                 if (downgradeType !== 'generic' && type !== 'generic') {
+                   downgradeType = 'building_address';
+                 } else {
+                   downgradeType = 'generic';
+                 }
+               } else {
+                 downgradeType = type;
+               }
              }
-
-             corePreferences('settings-custom-data-url', _currSettings.url);
-             this.blur();
-             modal.close();
-             dispatch$1.call('change', this, _currSettings);
            }
+
+           return downgradeType;
          }
 
-         return utilRebind(render, dispatch$1, 'on');
-       }
+         function downgradeTypeForEntityID(entityID) {
+           var graph = context.graph();
+           var entity = graph.entity(entityID);
+           var preset = _mainPresetIndex.match(entity, graph);
+           if (!preset || preset.isFallback()) return null;
 
-       function uiSectionDataLayers(context) {
-         var settingsCustomData = uiSettingsCustomData(context).on('change', customChanged);
-         var layers = context.layers();
-         var section = uiSection('data-layers', context).label(_t.html('map_data.data_layers')).disclosureContent(renderDisclosureContent);
+           if (entity.type === 'node' && preset.id !== 'address' && Object.keys(entity.tags).some(function (key) {
+             return key.match(/^addr:.{1,}/);
+           })) {
+             return 'address';
+           }
 
-         function renderDisclosureContent(selection) {
-           var container = selection.selectAll('.data-layer-container').data([0]);
-           container.enter().append('div').attr('class', 'data-layer-container').merge(container).call(drawOsmItems).call(drawQAItems).call(drawCustomDataItems).call(drawVectorItems) // Beta - Detroit mapping challenge
-           .call(drawPanelItems);
-         }
+           var geometry = entity.geometry(graph);
 
-         function showsLayer(which) {
-           var layer = layers.layer(which);
+           if (geometry === 'area' && entity.tags.building && !preset.tags.building) {
+             return 'building';
+           }
 
-           if (layer) {
-             return layer.enabled();
+           if (geometry === 'vertex' && Object.keys(entity.tags).length) {
+             return 'generic';
            }
 
-           return false;
+           return null;
          }
 
-         function setLayer(which, enabled) {
-           // Don't allow layer changes while drawing - #6584
-           var mode = context.mode();
-           if (mode && /^draw/.test(mode.id)) return;
-           var layer = layers.layer(which);
+         var buildingKeysToKeep = ['architect', 'building', 'height', 'layer', 'source', 'type', 'wheelchair'];
+         var addressKeysToKeep = ['source'];
 
-           if (layer) {
-             layer.enabled(enabled);
+         var operation = function operation() {
+           context.perform(function (graph) {
+             for (var i in selectedIDs) {
+               var entityID = selectedIDs[i];
+               var type = downgradeTypeForEntityID(entityID);
+               if (!type) continue;
+               var tags = Object.assign({}, graph.entity(entityID).tags); // shallow copy
 
-             if (!enabled && (which === 'osm' || which === 'notes')) {
-               context.enter(modeBrowse(context));
-             }
-           }
-         }
+               for (var key in tags) {
+                 if (type === 'address' && addressKeysToKeep.indexOf(key) !== -1) continue;
 
-         function toggleLayer(which) {
-           setLayer(which, !showsLayer(which));
-         }
+                 if (type === 'building') {
+                   if (buildingKeysToKeep.indexOf(key) !== -1 || key.match(/^building:.{1,}/) || key.match(/^roof:.{1,}/)) continue;
+                 }
 
-         function drawOsmItems(selection) {
-           var osmKeys = ['osm', 'notes'];
-           var osmLayers = layers.all().filter(function (obj) {
-             return osmKeys.indexOf(obj.id) !== -1;
-           });
-           var ul = selection.selectAll('.layer-list-osm').data([0]);
-           ul = ul.enter().append('ul').attr('class', 'layer-list layer-list-osm').merge(ul);
-           var li = ul.selectAll('.list-item').data(osmLayers);
-           li.exit().remove();
-           var liEnter = li.enter().append('li').attr('class', function (d) {
-             return 'list-item list-item-' + d.id;
-           });
-           var labelEnter = liEnter.append('label').each(function (d) {
-             if (d.id === 'osm') {
-               select(this).call(uiTooltip().title(_t.html('map_data.layers.' + d.id + '.tooltip')).keys([uiCmd('⌥' + _t('area_fill.wireframe.key'))]).placement('bottom'));
-             } else {
-               select(this).call(uiTooltip().title(_t.html('map_data.layers.' + d.id + '.tooltip')).placement('bottom'));
-             }
-           });
-           labelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event, d) {
-             toggleLayer(d.id);
-           });
-           labelEnter.append('span').html(function (d) {
-             return _t.html('map_data.layers.' + d.id + '.title');
-           }); // Update
+                 if (type !== 'generic') {
+                   if (key.match(/^addr:.{1,}/) || key.match(/^source:.{1,}/)) continue;
+                 }
 
-           li.merge(liEnter).classed('active', function (d) {
-             return d.layer.enabled();
-           }).selectAll('input').property('checked', function (d) {
-             return d.layer.enabled();
-           });
-         }
+                 delete tags[key];
+               }
 
-         function drawQAItems(selection) {
-           var qaKeys = ['keepRight', 'improveOSM', 'osmose'];
-           var qaLayers = layers.all().filter(function (obj) {
-             return qaKeys.indexOf(obj.id) !== -1;
-           });
-           var ul = selection.selectAll('.layer-list-qa').data([0]);
-           ul = ul.enter().append('ul').attr('class', 'layer-list layer-list-qa').merge(ul);
-           var li = ul.selectAll('.list-item').data(qaLayers);
-           li.exit().remove();
-           var liEnter = li.enter().append('li').attr('class', function (d) {
-             return 'list-item list-item-' + d.id;
-           });
-           var labelEnter = liEnter.append('label').each(function (d) {
-             select(this).call(uiTooltip().title(_t.html('map_data.layers.' + d.id + '.tooltip')).placement('bottom'));
-           });
-           labelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event, d) {
-             toggleLayer(d.id);
-           });
-           labelEnter.append('span').html(function (d) {
-             return _t.html('map_data.layers.' + d.id + '.title');
-           }); // Update
+               graph = actionChangeTags(entityID, tags)(graph);
+             }
 
-           li.merge(liEnter).classed('active', function (d) {
-             return d.layer.enabled();
-           }).selectAll('input').property('checked', function (d) {
-             return d.layer.enabled();
-           });
-         } // Beta feature - sample vector layers to support Detroit Mapping Challenge
-         // https://github.com/osmus/detroit-mapping-challenge
+             return graph;
+           }, operation.annotation());
+           context.validator().validate(); // refresh the select mode to enable the delete operation
 
+           context.enter(modeSelect(context, selectedIDs));
+         };
 
-         function drawVectorItems(selection) {
-           var dataLayer = layers.layer('data');
-           var vtData = [{
-             name: 'Detroit Neighborhoods/Parks',
-             src: 'neighborhoods-parks',
-             tooltip: 'Neighborhood boundaries and parks as compiled by City of Detroit in concert with community groups.',
-             template: 'https://{switch:a,b,c,d}.tiles.mapbox.com/v4/jonahadkins.cjksmur6x34562qp9iv1u3ksf-54hev,jonahadkins.cjksmqxdx33jj2wp90xd9x2md-4e5y2/{z}/{x}/{y}.vector.pbf?access_token=pk.eyJ1Ijoiam9uYWhhZGtpbnMiLCJhIjoiRlVVVkx3VSJ9.9sdVEK_B_VkEXPjssU5MqA'
-           }, {
-             name: 'Detroit Composite POIs',
-             src: 'composite-poi',
-             tooltip: 'Fire Inspections, Business Licenses, and other public location data collated from the City of Detroit.',
-             template: 'https://{switch:a,b,c,d}.tiles.mapbox.com/v4/jonahadkins.cjksmm6a02sli31myxhsr7zf3-2sw8h/{z}/{x}/{y}.vector.pbf?access_token=pk.eyJ1Ijoiam9uYWhhZGtpbnMiLCJhIjoiRlVVVkx3VSJ9.9sdVEK_B_VkEXPjssU5MqA'
-           }, {
-             name: 'Detroit All-The-Places POIs',
-             src: 'alltheplaces-poi',
-             tooltip: 'Public domain business location data created by web scrapers.',
-             template: 'https://{switch:a,b,c,d}.tiles.mapbox.com/v4/jonahadkins.cjksmswgk340g2vo06p1w9w0j-8fjjc/{z}/{x}/{y}.vector.pbf?access_token=pk.eyJ1Ijoiam9uYWhhZGtpbnMiLCJhIjoiRlVVVkx3VSJ9.9sdVEK_B_VkEXPjssU5MqA'
-           }]; // Only show this if the map is around Detroit..
+         operation.available = function () {
+           return _downgradeType;
+         };
 
-           var detroit = geoExtent([-83.5, 42.1], [-82.8, 42.5]);
-           var showVectorItems = context.map().zoom() > 9 && detroit.contains(context.map().center());
-           var container = selection.selectAll('.vectortile-container').data(showVectorItems ? [0] : []);
-           container.exit().remove();
-           var containerEnter = container.enter().append('div').attr('class', 'vectortile-container');
-           containerEnter.append('h4').attr('class', 'vectortile-header').html('Detroit Vector Tiles (Beta)');
-           containerEnter.append('ul').attr('class', 'layer-list layer-list-vectortile');
-           containerEnter.append('div').attr('class', 'vectortile-footer').append('a').attr('target', '_blank').call(svgIcon('#iD-icon-out-link', 'inline')).attr('href', 'https://github.com/osmus/detroit-mapping-challenge').append('span').html('About these layers');
-           container = container.merge(containerEnter);
-           var ul = container.selectAll('.layer-list-vectortile');
-           var li = ul.selectAll('.list-item').data(vtData);
-           li.exit().remove();
-           var liEnter = li.enter().append('li').attr('class', function (d) {
-             return 'list-item list-item-' + d.src;
-           });
-           var labelEnter = liEnter.append('label').each(function (d) {
-             select(this).call(uiTooltip().title(d.tooltip).placement('top'));
-           });
-           labelEnter.append('input').attr('type', 'radio').attr('name', 'vectortile').on('change', selectVTLayer);
-           labelEnter.append('span').html(function (d) {
-             return d.name;
-           }); // Update
+         operation.disabled = function () {
+           if (selectedIDs.some(hasWikidataTag)) {
+             return 'has_wikidata_tag';
+           }
 
-           li.merge(liEnter).classed('active', isVTLayerSelected).selectAll('input').property('checked', isVTLayerSelected);
+           return false;
 
-           function isVTLayerSelected(d) {
-             return dataLayer && dataLayer.template() === d.template;
+           function hasWikidataTag(id) {
+             var entity = context.entity(id);
+             return entity.tags.wikidata && entity.tags.wikidata.trim().length > 0;
            }
+         };
 
-           function selectVTLayer(d3_event, d) {
-             corePreferences('settings-custom-data-url', d.template);
+         operation.tooltip = function () {
+           var disable = operation.disabled();
+           return disable ? _t('operations.downgrade.' + disable + '.' + _multi) : _t('operations.downgrade.description.' + _downgradeType);
+         };
 
-             if (dataLayer) {
-               dataLayer.template(d.template, d.src);
-               dataLayer.enabled(true);
-             }
+         operation.annotation = function () {
+           var suffix;
+
+           if (_downgradeType === 'building_address') {
+             suffix = 'generic';
+           } else {
+             suffix = _downgradeType;
            }
-         }
 
-         function drawCustomDataItems(selection) {
-           var dataLayer = layers.layer('data');
-           var hasData = dataLayer && dataLayer.hasData();
-           var showsData = hasData && dataLayer.enabled();
-           var ul = selection.selectAll('.layer-list-data').data(dataLayer ? [0] : []); // Exit
+           return _t('operations.downgrade.annotation.' + suffix, {
+             n: _affectedFeatureCount
+           });
+         };
 
-           ul.exit().remove(); // Enter
+         operation.id = 'downgrade';
+         operation.keys = [uiCmd('⌫')];
+         operation.title = _t('operations.downgrade.title');
+         operation.behavior = behaviorOperation(context).which(operation);
+         return operation;
+       }
 
-           var ulEnter = ul.enter().append('ul').attr('class', 'layer-list layer-list-data');
-           var liEnter = ulEnter.append('li').attr('class', 'list-item-data');
-           var labelEnter = liEnter.append('label').call(uiTooltip().title(_t.html('map_data.layers.custom.tooltip')).placement('top'));
-           labelEnter.append('input').attr('type', 'checkbox').on('change', function () {
-             toggleLayer('data');
-           });
-           labelEnter.append('span').html(_t.html('map_data.layers.custom.title'));
-           liEnter.append('button').attr('class', 'open-data-options').call(uiTooltip().title(_t.html('settings.custom_data.tooltip')).placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left')).on('click', function (d3_event) {
-             d3_event.preventDefault();
-             editCustom();
-           }).call(svgIcon('#iD-icon-more'));
-           liEnter.append('button').attr('class', 'zoom-to-data').call(uiTooltip().title(_t.html('map_data.layers.custom.zoom')).placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left')).on('click', function (d3_event) {
-             if (select(this).classed('disabled')) return;
-             d3_event.preventDefault();
-             d3_event.stopPropagation();
-             dataLayer.fitZoom();
-           }).call(svgIcon('#iD-icon-framed-dot', 'monochrome')); // Update
+       function operationExtract(context, selectedIDs) {
+         var _amount = selectedIDs.length === 1 ? 'single' : 'multiple';
 
-           ul = ul.merge(ulEnter);
-           ul.selectAll('.list-item-data').classed('active', showsData).selectAll('label').classed('deemphasize', !hasData).selectAll('input').property('disabled', !hasData).property('checked', showsData);
-           ul.selectAll('button.zoom-to-data').classed('disabled', !hasData);
-         }
+         var _geometries = utilArrayUniq(selectedIDs.map(function (entityID) {
+           return context.graph().hasEntity(entityID) && context.graph().geometry(entityID);
+         }).filter(Boolean));
 
-         function editCustom() {
-           context.container().call(settingsCustomData);
-         }
+         var _geometryID = _geometries.length === 1 ? _geometries[0] : 'feature';
 
-         function customChanged(d) {
-           var dataLayer = layers.layer('data');
+         var _extent;
 
-           if (d && d.url) {
-             dataLayer.url(d.url);
-           } else if (d && d.fileList) {
-             dataLayer.fileList(d.fileList);
+         var _actions = selectedIDs.map(function (entityID) {
+           var graph = context.graph();
+           var entity = graph.hasEntity(entityID);
+           if (!entity || !entity.hasInterestingTags()) return null;
+           if (entity.type === 'node' && graph.parentWays(entity).length === 0) return null;
+
+           if (entity.type !== 'node') {
+             var preset = _mainPresetIndex.match(entity, graph); // only allow extraction from ways/relations if the preset supports points
+
+             if (preset.geometry.indexOf('point') === -1) return null;
            }
-         }
 
-         function drawPanelItems(selection) {
-           var panelsListEnter = selection.selectAll('.md-extras-list').data([0]).enter().append('ul').attr('class', 'layer-list md-extras-list');
-           var historyPanelLabelEnter = panelsListEnter.append('li').attr('class', 'history-panel-toggle-item').append('label').call(uiTooltip().title(_t.html('map_data.history_panel.tooltip')).keys([uiCmd('⌘⇧' + _t('info_panels.history.key'))]).placement('top'));
-           historyPanelLabelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event) {
-             d3_event.preventDefault();
-             context.ui().info.toggle('history');
-           });
-           historyPanelLabelEnter.append('span').html(_t.html('map_data.history_panel.title'));
-           var measurementPanelLabelEnter = panelsListEnter.append('li').attr('class', 'measurement-panel-toggle-item').append('label').call(uiTooltip().title(_t.html('map_data.measurement_panel.tooltip')).keys([uiCmd('⌘⇧' + _t('info_panels.measurement.key'))]).placement('top'));
-           measurementPanelLabelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event) {
-             d3_event.preventDefault();
-             context.ui().info.toggle('measurement');
-           });
-           measurementPanelLabelEnter.append('span').html(_t.html('map_data.measurement_panel.title'));
-         }
+           _extent = _extent ? _extent.extend(entity.extent(graph)) : entity.extent(graph);
+           return actionExtract(entityID, context.projection);
+         }).filter(Boolean);
 
-         context.layers().on('change.uiSectionDataLayers', section.reRender);
-         context.map().on('move.uiSectionDataLayers', debounce(function () {
-           // Detroit layers may have moved in or out of view
-           window.requestIdleCallback(section.reRender);
-         }, 1000));
-         return section;
-       }
+         var operation = function operation() {
+           var combinedAction = function combinedAction(graph) {
+             _actions.forEach(function (action) {
+               graph = action(graph);
+             });
 
-       function uiSectionMapFeatures(context) {
-         var _features = context.features().keys();
+             return graph;
+           };
 
-         var section = uiSection('map-features', context).label(_t.html('map_data.map_features')).disclosureContent(renderDisclosureContent).expandedByDefault(false);
+           context.perform(combinedAction, operation.annotation()); // do the extract
 
-         function renderDisclosureContent(selection) {
-           var container = selection.selectAll('.layer-feature-list-container').data([0]);
-           var containerEnter = container.enter().append('div').attr('class', 'layer-feature-list-container');
-           containerEnter.append('ul').attr('class', 'layer-list layer-feature-list');
-           var footer = containerEnter.append('div').attr('class', 'feature-list-links section-footer');
-           footer.append('a').attr('class', 'feature-list-link').attr('href', '#').html(_t.html('issues.disable_all')).on('click', function (d3_event) {
-             d3_event.preventDefault();
-             context.features().disableAll();
+           var extractedNodeIDs = _actions.map(function (action) {
+             return action.getExtractedNodeID();
            });
-           footer.append('a').attr('class', 'feature-list-link').attr('href', '#').html(_t.html('issues.enable_all')).on('click', function (d3_event) {
-             d3_event.preventDefault();
-             context.features().enableAll();
-           }); // Update
 
-           container = container.merge(containerEnter);
-           container.selectAll('.layer-feature-list').call(drawListItems, _features, 'checkbox', 'feature', clickFeature, showsFeature);
-         }
+           context.enter(modeSelect(context, extractedNodeIDs));
+         };
 
-         function drawListItems(selection, data, type, name, change, active) {
-           var items = selection.selectAll('li').data(data); // Exit
+         operation.available = function () {
+           return _actions.length && selectedIDs.length === _actions.length;
+         };
 
-           items.exit().remove(); // Enter
+         operation.disabled = function () {
+           if (_extent && _extent.percentContainedIn(context.map().extent()) < 0.8) {
+             return 'too_large';
+           } else if (selectedIDs.some(function (entityID) {
+             return context.graph().geometry(entityID) === 'vertex' && context.hasHiddenConnections(entityID);
+           })) {
+             return 'connected_to_hidden';
+           }
 
-           var enter = items.enter().append('li').call(uiTooltip().title(function (d) {
-             var tip = _t.html(name + '.' + d + '.tooltip');
+           return false;
+         };
 
-             if (autoHiddenFeature(d)) {
-               var msg = showsLayer('osm') ? _t.html('map_data.autohidden') : _t.html('map_data.osmhidden');
-               tip += '<div>' + msg + '</div>';
-             }
+         operation.tooltip = function () {
+           var disableReason = operation.disabled();
 
-             return tip;
-           }).placement('top'));
-           var label = enter.append('label');
-           label.append('input').attr('type', type).attr('name', name).on('change', change);
-           label.append('span').html(function (d) {
-             return _t.html(name + '.' + d + '.description');
-           }); // Update
+           if (disableReason) {
+             return _t('operations.extract.' + disableReason + '.' + _amount);
+           } else {
+             return _t('operations.extract.description.' + _geometryID + '.' + _amount);
+           }
+         };
 
-           items = items.merge(enter);
-           items.classed('active', active).selectAll('input').property('checked', active).property('indeterminate', autoHiddenFeature);
-         }
+         operation.annotation = function () {
+           return _t('operations.extract.annotation', {
+             n: selectedIDs.length
+           });
+         };
 
-         function autoHiddenFeature(d) {
-           return context.features().autoHidden(d);
-         }
+         operation.id = 'extract';
+         operation.keys = [_t('operations.extract.key')];
+         operation.title = _t('operations.extract.title');
+         operation.behavior = behaviorOperation(context).which(operation);
+         return operation;
+       }
+
+       function operationMerge(context, selectedIDs) {
+         var _action = getAction();
+
+         function getAction() {
+           // prefer a non-disabled action first
+           var join = actionJoin(selectedIDs);
+           if (!join.disabled(context.graph())) return join;
+           var merge = actionMerge(selectedIDs);
+           if (!merge.disabled(context.graph())) return merge;
+           var mergePolygon = actionMergePolygon(selectedIDs);
+           if (!mergePolygon.disabled(context.graph())) return mergePolygon;
+           var mergeNodes = actionMergeNodes(selectedIDs);
+           if (!mergeNodes.disabled(context.graph())) return mergeNodes; // otherwise prefer an action with an interesting disabled reason
 
-         function showsFeature(d) {
-           return context.features().enabled(d);
+           if (join.disabled(context.graph()) !== 'not_eligible') return join;
+           if (merge.disabled(context.graph()) !== 'not_eligible') return merge;
+           if (mergePolygon.disabled(context.graph()) !== 'not_eligible') return mergePolygon;
+           return mergeNodes;
          }
 
-         function clickFeature(d3_event, d) {
-           context.features().toggle(d);
-         }
+         var operation = function operation() {
+           if (operation.disabled()) return;
+           context.perform(_action, operation.annotation());
+           context.validator().validate();
+           var resultIDs = selectedIDs.filter(context.hasEntity);
 
-         function showsLayer(id) {
-           var layer = context.layers().layer(id);
-           return layer && layer.enabled();
-         } // add listeners
+           if (resultIDs.length > 1) {
+             var interestingIDs = resultIDs.filter(function (id) {
+               return context.entity(id).hasInterestingTags();
+             });
+             if (interestingIDs.length) resultIDs = interestingIDs;
+           }
 
+           context.enter(modeSelect(context, resultIDs));
+         };
 
-         context.features().on('change.map_features', section.reRender);
-         return section;
-       }
+         operation.available = function () {
+           return selectedIDs.length >= 2;
+         };
 
-       function uiSectionMapStyleOptions(context) {
-         var section = uiSection('fill-area', context).label(_t.html('map_data.style_options')).disclosureContent(renderDisclosureContent).expandedByDefault(false);
+         operation.disabled = function () {
+           var actionDisabled = _action.disabled(context.graph());
 
-         function renderDisclosureContent(selection) {
-           var container = selection.selectAll('.layer-fill-list').data([0]);
-           container.enter().append('ul').attr('class', 'layer-list layer-fill-list').merge(container).call(drawListItems, context.map().areaFillOptions, 'radio', 'area_fill', setFill, isActiveFill);
-           var container2 = selection.selectAll('.layer-visual-diff-list').data([0]);
-           container2.enter().append('ul').attr('class', 'layer-list layer-visual-diff-list').merge(container2).call(drawListItems, ['highlight_edits'], 'checkbox', 'visual_diff', toggleHighlightEdited, function () {
-             return context.surface().classed('highlight-edited');
-           });
-         }
+           if (actionDisabled) return actionDisabled;
+           var osm = context.connection();
 
-         function drawListItems(selection, data, type, name, change, active) {
-           var items = selection.selectAll('li').data(data); // Exit
+           if (osm && _action.resultingWayNodesLength && _action.resultingWayNodesLength(context.graph()) > osm.maxWayNodes()) {
+             return 'too_many_vertices';
+           }
 
-           items.exit().remove(); // Enter
+           return false;
+         };
 
-           var enter = items.enter().append('li').call(uiTooltip().title(function (d) {
-             return _t.html(name + '.' + d + '.tooltip');
-           }).keys(function (d) {
-             var key = d === 'wireframe' ? _t('area_fill.wireframe.key') : null;
-             if (d === 'highlight_edits') key = _t('map_data.highlight_edits.key');
-             return key ? [key] : null;
-           }).placement('top'));
-           var label = enter.append('label');
-           label.append('input').attr('type', type).attr('name', name).on('change', change);
-           label.append('span').html(function (d) {
-             return _t.html(name + '.' + d + '.description');
-           }); // Update
+         operation.tooltip = function () {
+           var disabled = operation.disabled();
 
-           items = items.merge(enter);
-           items.classed('active', active).selectAll('input').property('checked', active).property('indeterminate', false);
-         }
+           if (disabled) {
+             if (disabled === 'restriction') {
+               return _t('operations.merge.restriction', {
+                 relation: _mainPresetIndex.item('type/restriction').name()
+               });
+             }
 
-         function isActiveFill(d) {
-           return context.map().activeAreaFill() === d;
-         }
+             return _t('operations.merge.' + disabled);
+           }
 
-         function toggleHighlightEdited(d3_event) {
-           d3_event.preventDefault();
-           context.map().toggleHighlightEdited();
-         }
+           return _t('operations.merge.description');
+         };
 
-         function setFill(d3_event, d) {
-           context.map().activeAreaFill(d);
-         }
+         operation.annotation = function () {
+           return _t('operations.merge.annotation', {
+             n: selectedIDs.length
+           });
+         };
 
-         context.map().on('changeHighlighting.ui_style, changeAreaFill.ui_style', section.reRender);
-         return section;
+         operation.id = 'merge';
+         operation.keys = [_t('operations.merge.key')];
+         operation.title = _t('operations.merge.title');
+         operation.behavior = behaviorOperation(context).which(operation);
+         return operation;
        }
 
-       function uiSectionPhotoOverlays(context) {
-         var layers = context.layers();
-         var section = uiSection('photo-overlays', context).label(_t.html('photo_overlays.title')).disclosureContent(renderDisclosureContent).expandedByDefault(false);
-
-         function renderDisclosureContent(selection) {
-           var container = selection.selectAll('.photo-overlay-container').data([0]);
-           container.enter().append('div').attr('class', 'photo-overlay-container').merge(container).call(drawPhotoItems).call(drawPhotoTypeItems).call(drawDateFilter).call(drawUsernameFilter);
-         }
+       function operationPaste(context) {
+         var _pastePoint;
 
-         function drawPhotoItems(selection) {
-           var photoKeys = context.photos().overlayLayerIDs();
-           var photoLayers = layers.all().filter(function (obj) {
-             return photoKeys.indexOf(obj.id) !== -1;
-           });
-           var data = photoLayers.filter(function (obj) {
-             return obj.layer.supported();
+         var operation = function operation() {
+           if (!_pastePoint) return;
+           var oldIDs = context.copyIDs();
+           if (!oldIDs.length) return;
+           var projection = context.projection;
+           var extent = geoExtent();
+           var oldGraph = context.copyGraph();
+           var newIDs = [];
+           var action = actionCopyEntities(oldIDs, oldGraph);
+           context.perform(action);
+           var copies = action.copies();
+           var originals = new Set();
+           Object.values(copies).forEach(function (entity) {
+             originals.add(entity.id);
            });
 
-           function layerSupported(d) {
-             return d.layer && d.layer.supported();
-           }
+           for (var id in copies) {
+             var oldEntity = oldGraph.entity(id);
+             var newEntity = copies[id];
 
-           function layerEnabled(d) {
-             return layerSupported(d) && d.layer.enabled();
-           }
+             extent._extend(oldEntity.extent(oldGraph)); // Exclude child nodes from newIDs if their parent way was also copied.
 
-           var ul = selection.selectAll('.layer-list-photos').data([0]);
-           ul = ul.enter().append('ul').attr('class', 'layer-list layer-list-photos').merge(ul);
-           var li = ul.selectAll('.list-item-photos').data(data);
-           li.exit().remove();
-           var liEnter = li.enter().append('li').attr('class', function (d) {
-             var classes = 'list-item-photos list-item-' + d.id;
 
-             if (d.id === 'mapillary-signs' || d.id === 'mapillary-map-features') {
-               classes += ' indented';
+             var parents = context.graph().parentWays(newEntity);
+             var parentCopied = parents.some(function (parent) {
+               return originals.has(parent.id);
+             });
+
+             if (!parentCopied) {
+               newIDs.push(newEntity.id);
              }
+           } // Use the location of the copy operation to offset the paste location,
+           // or else use the center of the pasted extent
 
-             return classes;
-           });
-           var labelEnter = liEnter.append('label').each(function (d) {
-             var titleID;
-             if (d.id === 'mapillary-signs') titleID = 'mapillary.signs.tooltip';else if (d.id === 'mapillary') titleID = 'mapillary_images.tooltip';else if (d.id === 'openstreetcam') titleID = 'openstreetcam_images.tooltip';else titleID = d.id.replace(/-/g, '_') + '.tooltip';
-             select(this).call(uiTooltip().title(_t.html(titleID)).placement('top'));
-           });
-           labelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event, d) {
-             toggleLayer(d.id);
-           });
-           labelEnter.append('span').html(function (d) {
-             var id = d.id;
-             if (id === 'mapillary-signs') id = 'photo_overlays.traffic_signs';
-             return _t.html(id.replace(/-/g, '_') + '.title');
-           }); // Update
 
-           li.merge(liEnter).classed('active', layerEnabled).selectAll('input').property('checked', layerEnabled);
-         }
+           var copyPoint = context.copyLonLat() && projection(context.copyLonLat()) || projection(extent.center());
+           var delta = geoVecSubtract(_pastePoint, copyPoint); // Move the pasted objects to be anchored at the paste location
 
-         function drawPhotoTypeItems(selection) {
-           var data = context.photos().allPhotoTypes();
+           context.replace(actionMove(newIDs, delta, projection), operation.annotation());
+           context.enter(modeSelect(context, newIDs));
+         };
 
-           function typeEnabled(d) {
-             return context.photos().showsPhotoType(d);
-           }
+         operation.point = function (val) {
+           _pastePoint = val;
+           return operation;
+         };
 
-           var ul = selection.selectAll('.layer-list-photo-types').data([0]);
-           ul.exit().remove();
-           ul = ul.enter().append('ul').attr('class', 'layer-list layer-list-photo-types').merge(ul);
-           var li = ul.selectAll('.list-item-photo-types').data(context.photos().shouldFilterByPhotoType() ? data : []);
-           li.exit().remove();
-           var liEnter = li.enter().append('li').attr('class', function (d) {
-             return 'list-item-photo-types list-item-' + d;
-           });
-           var labelEnter = liEnter.append('label').each(function (d) {
-             select(this).call(uiTooltip().title(_t.html('photo_overlays.photo_type.' + d + '.tooltip')).placement('top'));
-           });
-           labelEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event, d) {
-             context.photos().togglePhotoType(d);
-           });
-           labelEnter.append('span').html(function (d) {
-             return _t.html('photo_overlays.photo_type.' + d + '.title');
-           }); // Update
+         operation.available = function () {
+           return context.mode().id === 'browse';
+         };
 
-           li.merge(liEnter).classed('active', typeEnabled).selectAll('input').property('checked', typeEnabled);
-         }
+         operation.disabled = function () {
+           return !context.copyIDs().length;
+         };
 
-         function drawDateFilter(selection) {
-           var data = context.photos().dateFilters();
+         operation.tooltip = function () {
+           var oldGraph = context.copyGraph();
+           var ids = context.copyIDs();
 
-           function filterEnabled(d) {
-             return context.photos().dateFilterValue(d);
+           if (!ids.length) {
+             return _t('operations.paste.nothing_copied');
            }
 
-           var ul = selection.selectAll('.layer-list-date-filter').data([0]);
-           ul.exit().remove();
-           ul = ul.enter().append('ul').attr('class', 'layer-list layer-list-date-filter').merge(ul);
-           var li = ul.selectAll('.list-item-date-filter').data(context.photos().shouldFilterByDate() ? data : []);
-           li.exit().remove();
-           var liEnter = li.enter().append('li').attr('class', 'list-item-date-filter');
-           var labelEnter = liEnter.append('label').each(function (d) {
-             select(this).call(uiTooltip().title(_t.html('photo_overlays.date_filter.' + d + '.tooltip')).placement('top'));
-           });
-           labelEnter.append('span').html(function (d) {
-             return _t.html('photo_overlays.date_filter.' + d + '.title');
-           });
-           labelEnter.append('input').attr('type', 'date').attr('class', 'list-item-input').attr('placeholder', _t('units.year_month_day')).call(utilNoAuto).each(function (d) {
-             utilGetSetValue(select(this), context.photos().dateFilterValue(d) || '');
-           }).on('change', function (d3_event, d) {
-             var value = utilGetSetValue(select(this)).trim();
-             context.photos().setDateFilter(d, value, true); // reload the displayed dates
-
-             li.selectAll('input').each(function (d) {
-               utilGetSetValue(select(this), context.photos().dateFilterValue(d) || '');
-             });
+           return _t('operations.paste.description', {
+             feature: utilDisplayLabel(oldGraph.entity(ids[0]), oldGraph),
+             n: ids.length
            });
-           li = li.merge(liEnter).classed('active', filterEnabled);
-         }
-
-         function drawUsernameFilter(selection) {
-           function filterEnabled() {
-             return context.photos().usernames();
-           }
+         };
 
-           var ul = selection.selectAll('.layer-list-username-filter').data([0]);
-           ul.exit().remove();
-           ul = ul.enter().append('ul').attr('class', 'layer-list layer-list-username-filter').merge(ul);
-           var li = ul.selectAll('.list-item-username-filter').data(context.photos().shouldFilterByUsername() ? ['username-filter'] : []);
-           li.exit().remove();
-           var liEnter = li.enter().append('li').attr('class', 'list-item-username-filter');
-           var labelEnter = liEnter.append('label').each(function () {
-             select(this).call(uiTooltip().title(_t.html('photo_overlays.username_filter.tooltip')).placement('top'));
-           });
-           labelEnter.append('span').html(_t.html('photo_overlays.username_filter.title'));
-           labelEnter.append('input').attr('type', 'text').attr('class', 'list-item-input').call(utilNoAuto).property('value', usernameValue).on('change', function () {
-             var value = select(this).property('value');
-             context.photos().setUsernameFilter(value, true);
-             select(this).property('value', usernameValue);
+         operation.annotation = function () {
+           var ids = context.copyIDs();
+           return _t('operations.paste.annotation', {
+             n: ids.length
            });
-           li.merge(liEnter).classed('active', filterEnabled);
+         };
 
-           function usernameValue() {
-             var usernames = context.photos().usernames();
-             if (usernames) return usernames.join('; ');
-             return usernames;
-           }
-         }
+         operation.id = 'paste';
+         operation.keys = [uiCmd('⌘V')];
+         operation.title = _t('operations.paste.title');
+         return operation;
+       }
 
-         function toggleLayer(which) {
-           setLayer(which, !showsLayer(which));
-         }
+       function operationReverse(context, selectedIDs) {
+         var operation = function operation() {
+           context.perform(function combinedReverseAction(graph) {
+             actions().forEach(function (action) {
+               graph = action(graph);
+             });
+             return graph;
+           }, operation.annotation());
+           context.validator().validate();
+         };
 
-         function showsLayer(which) {
-           var layer = layers.layer(which);
+         function actions(situation) {
+           return selectedIDs.map(function (entityID) {
+             var entity = context.hasEntity(entityID);
+             if (!entity) return null;
 
-           if (layer) {
-             return layer.enabled();
-           }
+             if (situation === 'toolbar') {
+               if (entity.type === 'way' && !entity.isOneWay() && !entity.isSided()) return null;
+             }
 
-           return false;
+             var geometry = entity.geometry(context.graph());
+             if (entity.type !== 'node' && geometry !== 'line') return null;
+             var action = actionReverse(entityID);
+             if (action.disabled(context.graph())) return null;
+             return action;
+           }).filter(Boolean);
          }
 
-         function setLayer(which, enabled) {
-           var layer = layers.layer(which);
-
-           if (layer) {
-             layer.enabled(enabled);
-           }
+         function reverseTypeID() {
+           var acts = actions();
+           var nodeActionCount = acts.filter(function (act) {
+             var entity = context.hasEntity(act.entityID());
+             return entity && entity.type === 'node';
+           }).length;
+           if (nodeActionCount === 0) return 'line';
+           if (nodeActionCount === acts.length) return 'point';
+           return 'feature';
          }
 
-         context.layers().on('change.uiSectionPhotoOverlays', section.reRender);
-         context.photos().on('change.uiSectionPhotoOverlays', section.reRender);
-         return section;
-       }
-
-       function uiPaneMapData(context) {
-         var mapDataPane = uiPane('map-data', context).key(_t('map_data.key')).label(_t.html('map_data.title')).description(_t.html('map_data.description')).iconName('iD-icon-data').sections([uiSectionDataLayers(context), uiSectionPhotoOverlays(context), uiSectionMapStyleOptions(context), uiSectionMapFeatures(context)]);
-         return mapDataPane;
-       }
+         operation.available = function (situation) {
+           return actions(situation).length > 0;
+         };
 
-       function uiSectionPrivacy(context) {
-         var section = uiSection('preferences-third-party', context).label(_t.html('preferences.privacy.title')).disclosureContent(renderDisclosureContent);
+         operation.disabled = function () {
+           return false;
+         };
 
-         var _showThirdPartyIcons = corePreferences('preferences.privacy.thirdpartyicons') || 'true';
+         operation.tooltip = function () {
+           return _t('operations.reverse.description.' + reverseTypeID());
+         };
 
-         function renderDisclosureContent(selection) {
-           // enter
-           var privacyOptionsListEnter = selection.selectAll('.privacy-options-list').data([0]).enter().append('ul').attr('class', 'layer-list privacy-options-list');
-           var thirdPartyIconsEnter = privacyOptionsListEnter.append('li').attr('class', 'privacy-third-party-icons-item').append('label').call(uiTooltip().title(_t.html('preferences.privacy.third_party_icons.tooltip')).placement('bottom'));
-           thirdPartyIconsEnter.append('input').attr('type', 'checkbox').on('change', function (d3_event) {
-             d3_event.preventDefault();
-             _showThirdPartyIcons = _showThirdPartyIcons === 'true' ? 'false' : 'true';
-             corePreferences('preferences.privacy.thirdpartyicons', _showThirdPartyIcons);
-             update();
+         operation.annotation = function () {
+           var acts = actions();
+           return _t('operations.reverse.annotation.' + reverseTypeID(), {
+             n: acts.length
            });
-           thirdPartyIconsEnter.append('span').html(_t.html('preferences.privacy.third_party_icons.description')); // Privacy Policy link
+         };
 
-           selection.selectAll('.privacy-link').data([0]).enter().append('div').attr('class', 'privacy-link').append('a').attr('target', '_blank').call(svgIcon('#iD-icon-out-link', 'inline')).attr('href', 'https://github.com/openstreetmap/iD/blob/release/PRIVACY.md').append('span').html(_t.html('preferences.privacy.privacy_link'));
-           update();
+         operation.id = 'reverse';
+         operation.keys = [_t('operations.reverse.key')];
+         operation.title = _t('operations.reverse.title');
+         operation.behavior = behaviorOperation(context).which(operation);
+         return operation;
+       }
 
-           function update() {
-             selection.selectAll('.privacy-third-party-icons-item').classed('active', _showThirdPartyIcons === 'true').select('input').property('checked', _showThirdPartyIcons === 'true');
-           }
-         }
+       function operationSplit(context, selectedIDs) {
+         var _vertexIds = selectedIDs.filter(function (id) {
+           return context.graph().geometry(id) === 'vertex';
+         });
 
-         return section;
-       }
+         var _selectedWayIds = selectedIDs.filter(function (id) {
+           var entity = context.graph().hasEntity(id);
+           return entity && entity.type === 'way';
+         });
 
-       function uiPanePreferences(context) {
-         var preferencesPane = uiPane('preferences', context).key(_t('preferences.key')).label(_t.html('preferences.title')).description(_t.html('preferences.description')).iconName('fas-user-cog').sections([uiSectionPrivacy(context)]);
-         return preferencesPane;
-       }
+         var _isAvailable = _vertexIds.length > 0 && _vertexIds.length + _selectedWayIds.length === selectedIDs.length;
 
-       function uiInit(context) {
-         var _initCounter = 0;
-         var _needWidth = {};
+         var _action = actionSplit(_vertexIds);
 
-         var _lastPointerType;
+         var _ways = [];
+         var _geometry = 'feature';
+         var _waysAmount = 'single';
 
-         function render(container) {
-           container.on('click.ui', function (d3_event) {
-             // we're only concerned with the primary mouse button
-             if (d3_event.button !== 0) return;
-             if (!d3_event.composedPath) return; // some targets have default click events we don't want to override
+         var _nodesAmount = _vertexIds.length === 1 ? 'single' : 'multiple';
 
-             var isOkayTarget = d3_event.composedPath().some(function (node) {
-               // we only care about element nodes
-               return node.nodeType === 1 && ( // clicking <input> focuses it and/or changes a value
-               node.nodeName === 'INPUT' || // clicking <label> affects its <input> by default
-               node.nodeName === 'LABEL' || // clicking <a> opens a hyperlink by default
-               node.nodeName === 'A');
-             });
-             if (isOkayTarget) return; // disable double-tap-to-zoom on touchscreens
+         if (_isAvailable) {
+           if (_selectedWayIds.length) _action.limitWays(_selectedWayIds);
+           _ways = _action.ways(context.graph());
+           var geometries = {};
 
-             d3_event.preventDefault();
+           _ways.forEach(function (way) {
+             geometries[way.geometry(context.graph())] = true;
            });
-           var detected = utilDetect(); // only WebKit supports gesture events
 
-           if ('GestureEvent' in window && // Listening for gesture events on iOS 13.4+ breaks double-tapping,
-           // but we only need to do this on desktop Safari anyway. – #7694
-           !detected.isMobileWebKit) {
-             // On iOS we disable pinch-to-zoom of the UI via the `touch-action`
-             // CSS property, but on desktop Safari we need to manually cancel the
-             // default gesture events.
-             container.on('gesturestart.ui gesturechange.ui gestureend.ui', function (d3_event) {
-               // disable pinch-to-zoom of the UI via multitouch trackpads on macOS Safari
-               d3_event.preventDefault();
-             });
+           if (Object.keys(geometries).length === 1) {
+             _geometry = Object.keys(geometries)[0];
            }
 
-           if ('PointerEvent' in window) {
-             select(window).on('pointerdown.ui pointerup.ui', function (d3_event) {
-               var pointerType = d3_event.pointerType || 'mouse';
+           _waysAmount = _ways.length === 1 ? 'single' : 'multiple';
+         }
 
-               if (_lastPointerType !== pointerType) {
-                 _lastPointerType = pointerType;
-                 container.attr('pointer', pointerType);
-               }
-             }, true);
-           } else {
-             _lastPointerType = 'mouse';
-             container.attr('pointer', 'mouse');
-           }
+         var operation = function operation() {
+           var difference = context.perform(_action, operation.annotation()); // select both the nodes and the ways so the mapper can immediately disconnect them if desired
 
-           container.attr('lang', _mainLocalizer.localeCode()).attr('dir', _mainLocalizer.textDirection()); // setup fullscreen keybindings (no button shown at this time)
+           var idsToSelect = _vertexIds.concat(difference.extantIDs().filter(function (id) {
+             // filter out relations that may have had member additions
+             return context.entity(id).type === 'way';
+           }));
 
-           container.call(uiFullScreen(context));
-           var map = context.map();
-           map.redrawEnable(false); // don't draw until we've set zoom/lat/long
+           context.enter(modeSelect(context, idsToSelect));
+         };
 
-           map.on('hitMinZoom.ui', function () {
-             ui.flash.iconName('#iD-icon-no').label(_t.html('cannot_zoom'))();
+         operation.relatedEntityIds = function () {
+           return _selectedWayIds.length ? [] : _ways.map(function (way) {
+             return way.id;
            });
-           container.append('svg').attr('id', 'ideditor-defs').call(ui.svgDefs);
-           container.append('div').attr('class', 'sidebar').call(ui.sidebar);
-           var content = container.append('div').attr('class', 'main-content active'); // Top toolbar
-
-           content.append('div').attr('class', 'top-toolbar-wrap').append('div').attr('class', 'top-toolbar fillD').call(uiTopToolbar(context));
-           content.append('div').attr('class', 'main-map').attr('dir', 'ltr').call(map);
-           var overMap = content.append('div').attr('class', 'over-map'); // HACK: Mobile Safari 14 likes to select anything selectable when long-
-           // pressing, even if it's not targeted. This conflicts with long-pressing
-           // to show the edit menu. We add a selectable offscreen element as the first
-           // child to trick Safari into not showing the selection UI.
-
-           overMap.append('div').attr('class', 'select-trap').text('t');
-           overMap.call(uiMapInMap(context)).call(uiNotice(context));
-           overMap.append('div').attr('class', 'spinner').call(uiSpinner(context)); // Map controls
-
-           var controls = overMap.append('div').attr('class', 'map-controls');
-           controls.append('div').attr('class', 'map-control zoombuttons').call(uiZoom(context));
-           controls.append('div').attr('class', 'map-control zoom-to-selection-control').call(uiZoomToSelection(context));
-           controls.append('div').attr('class', 'map-control geolocate-control').call(uiGeolocate(context)); // Add panes
-           // This should happen after map is initialized, as some require surface()
+         };
 
-           var panes = overMap.append('div').attr('class', 'map-panes');
-           var uiPanes = [uiPaneBackground(context), uiPaneMapData(context), uiPaneIssues(context), uiPanePreferences(context), uiPaneHelp(context)];
-           uiPanes.forEach(function (pane) {
-             controls.append('div').attr('class', 'map-control map-pane-control ' + pane.id + '-control').call(pane.renderToggleButton);
-             panes.call(pane.renderPane);
-           });
-           ui.info = uiInfo(context);
-           overMap.call(ui.info);
-           overMap.append('div').attr('class', 'photoviewer').classed('al', true) // 'al'=left,  'ar'=right
-           .classed('hide', true).call(ui.photoviewer);
-           overMap.append('div').attr('class', 'attribution-wrap').attr('dir', 'ltr').call(uiAttribution(context)); // Add footer
+         operation.available = function () {
+           return _isAvailable;
+         };
 
-           var about = content.append('div').attr('class', 'map-footer');
-           about.append('div').attr('class', 'api-status').call(uiStatus(context));
-           var footer = about.append('div').attr('class', 'map-footer-bar fillD');
-           footer.append('div').attr('class', 'flash-wrap footer-hide');
-           var footerWrap = footer.append('div').attr('class', 'main-footer-wrap footer-show');
-           footerWrap.append('div').attr('class', 'scale-block').call(uiScale(context));
-           var aboutList = footerWrap.append('div').attr('class', 'info-block').append('ul').attr('class', 'map-footer-list');
-           aboutList.append('li').attr('class', 'user-list').call(uiContributors(context));
-           var apiConnections = context.apiConnections();
+         operation.disabled = function () {
+           var reason = _action.disabled(context.graph());
 
-           if (apiConnections && apiConnections.length > 1) {
-             aboutList.append('li').attr('class', 'source-switch').call(uiSourceSwitch(context).keys(apiConnections));
+           if (reason) {
+             return reason;
+           } else if (selectedIDs.some(context.hasHiddenConnections)) {
+             return 'connected_to_hidden';
            }
 
-           aboutList.append('li').attr('class', 'issues-info').call(uiIssuesInfo(context));
-           aboutList.append('li').attr('class', 'feature-warning').call(uiFeatureInfo(context));
-           var issueLinks = aboutList.append('li');
-           issueLinks.append('a').attr('target', '_blank').attr('href', 'https://github.com/openstreetmap/iD/issues').call(svgIcon('#iD-icon-bug', 'light')).call(uiTooltip().title(_t.html('report_a_bug')).placement('top'));
-           issueLinks.append('a').attr('target', '_blank').attr('href', 'https://github.com/openstreetmap/iD/blob/develop/CONTRIBUTING.md#translating').call(svgIcon('#iD-icon-translate', 'light')).call(uiTooltip().title(_t.html('help_translate')).placement('top'));
-           aboutList.append('li').attr('class', 'version').call(uiVersion(context));
+           return false;
+         };
 
-           if (!context.embed()) {
-             aboutList.call(uiAccount(context));
-           } // Setup map dimensions and move map to initial center/zoom.
-           // This should happen after .main-content and toolbars exist.
+         operation.tooltip = function () {
+           var disable = operation.disabled();
+           if (disable) return _t('operations.split.' + disable);
+           return _t('operations.split.description.' + _geometry + '.' + _waysAmount + '.' + _nodesAmount + '_node');
+         };
 
+         operation.annotation = function () {
+           return _t('operations.split.annotation.' + _geometry, {
+             n: _ways.length
+           });
+         };
 
-           ui.onResize();
-           map.redrawEnable(true);
-           ui.hash = behaviorHash(context);
-           ui.hash();
+         operation.id = 'split';
+         operation.keys = [_t('operations.split.key')];
+         operation.title = _t('operations.split.title');
+         operation.behavior = behaviorOperation(context).which(operation);
+         return operation;
+       }
 
-           if (!ui.hash.hadHash) {
-             map.centerZoom([0, 0], 2);
-           } // Bind events
+       function operationStraighten(context, selectedIDs) {
+         var _wayIDs = selectedIDs.filter(function (id) {
+           return id.charAt(0) === 'w';
+         });
 
+         var _nodeIDs = selectedIDs.filter(function (id) {
+           return id.charAt(0) === 'n';
+         });
 
-           window.onbeforeunload = function () {
-             return context.save();
-           };
+         var _amount = (_wayIDs.length ? _wayIDs : _nodeIDs).length === 1 ? 'single' : 'multiple';
 
-           window.onunload = function () {
-             context.history().unlock();
-           };
+         var _nodes = utilGetAllNodes(selectedIDs, context.graph());
 
-           select(window).on('resize.editor', function () {
-             ui.onResize();
-           });
-           var panPixels = 80;
-           context.keybinding().on('⌫', function (d3_event) {
-             d3_event.preventDefault();
-           }).on([_t('sidebar.key'), '`', '²', '@'], ui.sidebar.toggle) // #5663, #6864 - common QWERTY, AZERTY
-           .on('←', pan([panPixels, 0])).on('↑', pan([0, panPixels])).on('→', pan([-panPixels, 0])).on('↓', pan([0, -panPixels])).on(uiCmd('⌥←'), pan([map.dimensions()[0], 0])).on(uiCmd('⌥↑'), pan([0, map.dimensions()[1]])).on(uiCmd('⌥→'), pan([-map.dimensions()[0], 0])).on(uiCmd('⌥↓'), pan([0, -map.dimensions()[1]])).on(uiCmd('⌘' + _t('background.key')), function quickSwitch(d3_event) {
-             if (d3_event) {
-               d3_event.stopImmediatePropagation();
-               d3_event.preventDefault();
-             }
+         var _coords = _nodes.map(function (n) {
+           return n.loc;
+         });
 
-             var previousBackground = context.background().findSource(corePreferences('background-last-used-toggle'));
+         var _extent = utilTotalExtent(selectedIDs, context.graph());
 
-             if (previousBackground) {
-               var currentBackground = context.background().baseLayerSource();
-               corePreferences('background-last-used-toggle', currentBackground.id);
-               corePreferences('background-last-used', previousBackground.id);
-               context.background().baseLayerSource(previousBackground);
-             }
-           }).on(_t('area_fill.wireframe.key'), function toggleWireframe(d3_event) {
-             d3_event.preventDefault();
-             d3_event.stopPropagation();
-             context.map().toggleWireframe();
-           }).on(uiCmd('⌥' + _t('area_fill.wireframe.key')), function toggleOsmData(d3_event) {
-             d3_event.preventDefault();
-             d3_event.stopPropagation(); // Don't allow layer changes while drawing - #6584
+         var _action = chooseAction();
 
-             var mode = context.mode();
-             if (mode && /^draw/.test(mode.id)) return;
-             var layer = context.layers().layer('osm');
+         var _geometry;
 
-             if (layer) {
-               layer.enabled(!layer.enabled());
+         function chooseAction() {
+           // straighten selected nodes
+           if (_wayIDs.length === 0 && _nodeIDs.length > 2) {
+             _geometry = 'point';
+             return actionStraightenNodes(_nodeIDs, context.projection); // straighten selected ways (possibly between range of 2 selected nodes)
+           } else if (_wayIDs.length > 0 && (_nodeIDs.length === 0 || _nodeIDs.length === 2)) {
+             var startNodeIDs = [];
+             var endNodeIDs = [];
 
-               if (!layer.enabled()) {
-                 context.enter(modeBrowse(context));
+             for (var i = 0; i < selectedIDs.length; i++) {
+               var entity = context.entity(selectedIDs[i]);
+
+               if (entity.type === 'node') {
+                 continue;
+               } else if (entity.type !== 'way' || entity.isClosed()) {
+                 return null; // exit early, can't straighten these
                }
-             }
-           }).on(_t('map_data.highlight_edits.key'), function toggleHighlightEdited(d3_event) {
-             d3_event.preventDefault();
-             context.map().toggleHighlightEdited();
-           });
-           context.on('enter.editor', function (entered) {
-             container.classed('mode-' + entered.id, true);
-           }).on('exit.editor', function (exited) {
-             container.classed('mode-' + exited.id, false);
-           });
-           context.enter(modeBrowse(context));
 
-           if (!_initCounter++) {
-             if (!ui.hash.startWalkthrough) {
-               context.container().call(uiSplash(context)).call(uiRestore(context));
-             }
+               startNodeIDs.push(entity.first());
+               endNodeIDs.push(entity.last());
+             } // Remove duplicate end/startNodeIDs (duplicate nodes cannot be at the line end)
 
-             context.container().call(ui.shortcuts);
-           }
 
-           var osm = context.connection();
-           var auth = uiLoading(context).message(_t.html('loading_auth')).blocking(true);
+             startNodeIDs = startNodeIDs.filter(function (n) {
+               return startNodeIDs.indexOf(n) === startNodeIDs.lastIndexOf(n);
+             });
+             endNodeIDs = endNodeIDs.filter(function (n) {
+               return endNodeIDs.indexOf(n) === endNodeIDs.lastIndexOf(n);
+             }); // Ensure all ways are connected (i.e. only 2 unique endpoints/startpoints)
 
-           if (osm && auth) {
-             osm.on('authLoading.ui', function () {
-               context.container().call(auth);
-             }).on('authDone.ui', function () {
-               auth.close();
+             if (utilArrayDifference(startNodeIDs, endNodeIDs).length + utilArrayDifference(endNodeIDs, startNodeIDs).length !== 2) return null; // Ensure path contains at least 3 unique nodes
+
+             var wayNodeIDs = utilGetAllNodes(_wayIDs, context.graph()).map(function (node) {
+               return node.id;
              });
-           }
+             if (wayNodeIDs.length <= 2) return null; // If range of 2 selected nodes is supplied, ensure nodes lie on the selected path
 
-           _initCounter++;
+             if (_nodeIDs.length === 2 && (wayNodeIDs.indexOf(_nodeIDs[0]) === -1 || wayNodeIDs.indexOf(_nodeIDs[1]) === -1)) return null;
 
-           if (ui.hash.startWalkthrough) {
-             ui.hash.startWalkthrough = false;
-             context.container().call(uiIntro(context));
-           }
+             if (_nodeIDs.length) {
+               // If we're only straightenting between two points, we only need that extent visible
+               _extent = utilTotalExtent(_nodeIDs, context.graph());
+             }
 
-           function pan(d) {
-             return function (d3_event) {
-               if (d3_event.shiftKey) return;
-               if (context.container().select('.combobox').size()) return;
-               d3_event.preventDefault();
-               context.map().pan(d, 100);
-             };
+             _geometry = 'line';
+             return actionStraightenWay(selectedIDs, context.projection);
            }
+
+           return null;
          }
 
-         var ui = {};
+         function operation() {
+           if (!_action) return;
+           context.perform(_action, operation.annotation());
+           window.setTimeout(function () {
+             context.validator().validate();
+           }, 300); // after any transition
+         }
 
-         var _loadPromise; // renders the iD interface into the container node
+         operation.available = function () {
+           return Boolean(_action);
+         };
 
+         operation.disabled = function () {
+           var reason = _action.disabled(context.graph());
 
-         ui.ensureLoaded = function () {
-           if (_loadPromise) return _loadPromise;
-           return _loadPromise = Promise.all([// must have strings and presets before loading the UI
-           _mainLocalizer.ensureLoaded(), _mainPresetIndex.ensureLoaded()]).then(function () {
-             if (!context.container().empty()) render(context.container());
-           })["catch"](function (err) {
-             return console.error(err);
-           }); // eslint-disable-line
-         }; // `ui.restart()` will destroy and rebuild the entire iD interface,
-         // for example to switch the locale while iD is running.
+           if (reason) {
+             return reason;
+           } else if (_extent.percentContainedIn(context.map().extent()) < 0.8) {
+             return 'too_large';
+           } else if (someMissing()) {
+             return 'not_downloaded';
+           } else if (selectedIDs.some(context.hasHiddenConnections)) {
+             return 'connected_to_hidden';
+           }
 
+           return false;
 
-         ui.restart = function () {
-           context.keybinding().clear();
-           _loadPromise = null;
-           context.container().selectAll('*').remove();
-           ui.ensureLoaded();
-         };
+           function someMissing() {
+             if (context.inIntro()) return false;
+             var osm = context.connection();
 
-         ui.lastPointerType = function () {
-           return _lastPointerType;
+             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;
+           }
          };
 
-         ui.svgDefs = svgDefs(context);
-         ui.flash = uiFlash(context);
-         ui.sidebar = uiSidebar(context);
-         ui.photoviewer = uiPhotoviewer(context);
-         ui.shortcuts = uiShortcuts(context);
+         operation.tooltip = function () {
+           var disable = operation.disabled();
+           return disable ? _t('operations.straighten.' + disable + '.' + _amount) : _t('operations.straighten.description.' + _geometry + (_wayIDs.length === 1 ? '' : 's'));
+         };
 
-         ui.onResize = function (withPan) {
-           var map = context.map(); // Recalc dimensions of map and sidebar.. (`true` = force recalc)
-           // This will call `getBoundingClientRect` and trigger reflow,
-           //  but the values will be cached for later use.
+         operation.annotation = function () {
+           return _t('operations.straighten.annotation.' + _geometry, {
+             n: _wayIDs.length ? _wayIDs.length : _nodeIDs.length
+           });
+         };
 
-           var mapDimensions = utilGetDimensions(context.container().select('.main-content'), true);
-           utilGetDimensions(context.container().select('.sidebar'), true);
+         operation.id = 'straighten';
+         operation.keys = [_t('operations.straighten.key')];
+         operation.title = _t('operations.straighten.title');
+         operation.behavior = behaviorOperation(context).which(operation);
+         return operation;
+       }
 
-           if (withPan !== undefined) {
-             map.redrawEnable(false);
-             map.pan(withPan);
-             map.redrawEnable(true);
-           }
+       var Operations = /*#__PURE__*/Object.freeze({
+               __proto__: null,
+               operationCircularize: operationCircularize,
+               operationContinue: operationContinue,
+               operationCopy: operationCopy,
+               operationDelete: operationDelete,
+               operationDisconnect: operationDisconnect,
+               operationDowngrade: operationDowngrade,
+               operationExtract: operationExtract,
+               operationMerge: operationMerge,
+               operationMove: operationMove,
+               operationOrthogonalize: operationOrthogonalize,
+               operationPaste: operationPaste,
+               operationReflectShort: operationReflectShort,
+               operationReflectLong: operationReflectLong,
+               operationReverse: operationReverse,
+               operationRotate: operationRotate,
+               operationSplit: operationSplit,
+               operationStraighten: operationStraighten
+       });
 
-           map.dimensions(mapDimensions);
-           ui.photoviewer.onMapResize(); // check if header or footer have overflowed
+       function modeSelect(context, selectedIDs) {
+         var mode = {
+           id: 'select',
+           button: 'browse'
+         };
+         var keybinding = utilKeybinding('select');
 
-           ui.checkOverflow('.top-toolbar');
-           ui.checkOverflow('.map-footer-bar'); // Use outdated code so it works on Explorer
+         var _breatheBehavior = behaviorBreathe();
 
-           var resizeWindowEvent = document.createEvent('Event');
-           resizeWindowEvent.initEvent('resizeWindow', true, true);
-           document.dispatchEvent(resizeWindowEvent);
-         }; // Call checkOverflow when resizing or whenever the contents change.
+         var _modeDragNode = modeDragNode(context);
 
+         var _selectBehavior;
 
-         ui.checkOverflow = function (selector, reset) {
-           if (reset) {
-             delete _needWidth[selector];
-           }
+         var _behaviors = [];
+         var _operations = [];
+         var _newFeature = false;
+         var _follow = false; // `_focusedParentWayId` is used when we visit a vertex with multiple
+         // parents, and we want to remember which parent line we started on.
 
-           var selection = context.container().select(selector);
-           if (selection.empty()) return;
-           var scrollWidth = selection.property('scrollWidth');
-           var clientWidth = selection.property('clientWidth');
-           var needed = _needWidth[selector] || scrollWidth;
+         var _focusedParentWayId;
 
-           if (scrollWidth > clientWidth) {
-             // overflow happening
-             selection.classed('narrow', true);
+         var _focusedVertexIds;
 
-             if (!_needWidth[selector]) {
-               _needWidth[selector] = scrollWidth;
-             }
-           } else if (scrollWidth >= needed) {
-             selection.classed('narrow', false);
+         function singular() {
+           if (selectedIDs && selectedIDs.length === 1) {
+             return context.hasEntity(selectedIDs[0]);
            }
-         };
+         }
 
-         ui.togglePanes = function (showPane) {
-           var hidePanes = context.container().selectAll('.map-pane.shown');
-           var side = _mainLocalizer.textDirection() === 'ltr' ? 'right' : 'left';
-           hidePanes.classed('shown', false).classed('hide', true);
-           context.container().selectAll('.map-pane-control button').classed('active', false);
+         function selectedEntities() {
+           return selectedIDs.map(function (id) {
+             return context.hasEntity(id);
+           }).filter(Boolean);
+         }
 
-           if (showPane) {
-             hidePanes.classed('shown', false).classed('hide', true).style(side, '-500px');
-             context.container().selectAll('.' + showPane.attr('pane') + '-control button').classed('active', true);
-             showPane.classed('shown', true).classed('hide', false);
+         function checkSelectedIDs() {
+           var ids = [];
 
-             if (hidePanes.empty()) {
-               showPane.style(side, '-500px').transition().duration(200).style(side, '0px');
-             } else {
-               showPane.style(side, '0px');
-             }
-           } else {
-             hidePanes.classed('shown', true).classed('hide', false).style(side, '0px').transition().duration(200).style(side, '-500px').on('end', function () {
-               select(this).classed('shown', false).classed('hide', true);
+           if (Array.isArray(selectedIDs)) {
+             ids = selectedIDs.filter(function (id) {
+               return context.hasEntity(id);
              });
            }
-         };
-
-         var _editMenu = uiEditMenu(context);
-
-         ui.editMenu = function () {
-           return _editMenu;
-         };
-
-         ui.showEditMenu = function (anchorPoint, triggerType, operations) {
-           // remove any displayed menu
-           ui.closeEditMenu();
-           if (!operations && context.mode().operations) operations = context.mode().operations();
-           if (!operations || !operations.length) return; // disable menu if in wide selection, for example
-
-           if (!context.map().editableDataEnabled()) return;
-           var surfaceNode = context.surface().node();
 
-           if (surfaceNode.focus) {
-             // FF doesn't support it
-             // focus the surface or else clicking off the menu may not trigger modeBrowse
-             surfaceNode.focus();
+           if (!ids.length) {
+             context.enter(modeBrowse(context));
+             return false;
+           } else if (selectedIDs.length > 1 && ids.length === 1 || selectedIDs.length === 1 && ids.length > 1) {
+             // switch between single- and multi-select UI
+             context.enter(modeSelect(context, ids));
+             return false;
            }
 
-           operations.forEach(function (operation) {
-             if (operation.point) operation.point(anchorPoint);
-           });
+           selectedIDs = ids;
+           return true;
+         } // find the parent ways for nextVertex, previousVertex, and selectParent
 
-           _editMenu.anchorLoc(anchorPoint).triggerType(triggerType).operations(operations); // render the menu
 
+         function parentWaysIdsOfSelection(onlyCommonParents) {
+           var graph = context.graph();
+           var parents = [];
 
-           context.map().supersurface.call(_editMenu);
-         };
+           for (var i = 0; i < selectedIDs.length; i++) {
+             var entity = context.hasEntity(selectedIDs[i]);
 
-         ui.closeEditMenu = function () {
-           // remove any existing menu no matter how it was added
-           context.map().supersurface.select('.edit-menu').remove();
-         };
+             if (!entity || entity.geometry(graph) !== 'vertex') {
+               return []; // selection includes some non-vertices
+             }
 
-         var _saveLoading = select(null);
+             var currParents = graph.parentWays(entity).map(function (w) {
+               return w.id;
+             });
 
-         context.uploader().on('saveStarted.ui', function () {
-           _saveLoading = uiLoading(context).message(_t.html('save.uploading')).blocking(true);
-           context.container().call(_saveLoading); // block input during upload
-         }).on('saveEnded.ui', function () {
-           _saveLoading.close();
+             if (!parents.length) {
+               parents = currParents;
+               continue;
+             }
 
-           _saveLoading = select(null);
-         });
-         return ui;
-       }
+             parents = (onlyCommonParents ? utilArrayIntersection : utilArrayUnion)(parents, currParents);
 
-       function coreContext() {
-         var _this = this;
+             if (!parents.length) {
+               return [];
+             }
+           }
 
-         var dispatch$1 = dispatch('enter', 'exit', 'change');
-         var context = utilRebind({}, dispatch$1, 'on');
+           return parents;
+         } // find the child nodes for selected ways
 
-         var _deferred = new Set();
 
-         context.version = '2.19.6';
-         context.privacyVersion = '20200407'; // iD will alter the hash so cache the parameters intended to setup the session
+         function childNodeIdsOfSelection(onlyCommon) {
+           var graph = context.graph();
+           var childs = [];
 
-         context.initialHashParams = window.location.hash ? utilStringQs(window.location.hash) : {};
-         context.isFirstSession = !corePreferences('sawSplash') && !corePreferences('sawPrivacyVersion');
-         /* Changeset */
-         // An osmChangeset object. Not loaded until needed.
+           for (var i = 0; i < selectedIDs.length; i++) {
+             var entity = context.hasEntity(selectedIDs[i]);
 
-         context.changeset = null;
-         var _defaultChangesetComment = context.initialHashParams.comment;
-         var _defaultChangesetSource = context.initialHashParams.source;
-         var _defaultChangesetHashtags = context.initialHashParams.hashtags;
+             if (!entity || !['area', 'line'].includes(entity.geometry(graph))) {
+               return []; // selection includes non-area/non-line
+             }
 
-         context.defaultChangesetComment = function (val) {
-           if (!arguments.length) return _defaultChangesetComment;
-           _defaultChangesetComment = val;
-           return context;
-         };
+             var currChilds = graph.childNodes(entity).map(function (node) {
+               return node.id;
+             });
 
-         context.defaultChangesetSource = function (val) {
-           if (!arguments.length) return _defaultChangesetSource;
-           _defaultChangesetSource = val;
-           return context;
-         };
+             if (!childs.length) {
+               childs = currChilds;
+               continue;
+             }
 
-         context.defaultChangesetHashtags = function (val) {
-           if (!arguments.length) return _defaultChangesetHashtags;
-           _defaultChangesetHashtags = val;
-           return context;
-         };
-         /* Document title */
+             childs = (onlyCommon ? utilArrayIntersection : utilArrayUnion)(childs, currChilds);
 
-         /* (typically shown as the label for the browser window/tab) */
-         // If true, iD will update the title based on what the user is doing
+             if (!childs.length) {
+               return [];
+             }
+           }
 
+           return childs;
+         }
 
-         var _setsDocumentTitle = true;
+         function checkFocusedParent() {
+           if (_focusedParentWayId) {
+             var parents = parentWaysIdsOfSelection(true);
+             if (parents.indexOf(_focusedParentWayId) === -1) _focusedParentWayId = null;
+           }
+         }
 
-         context.setsDocumentTitle = function (val) {
-           if (!arguments.length) return _setsDocumentTitle;
-           _setsDocumentTitle = val;
-           return context;
-         }; // The part of the title that is always the same
+         function parentWayIdForVertexNavigation() {
+           var parentIds = parentWaysIdsOfSelection(true);
 
+           if (_focusedParentWayId && parentIds.indexOf(_focusedParentWayId) !== -1) {
+             // prefer the previously seen parent
+             return _focusedParentWayId;
+           }
 
-         var _documentTitleBase = document.title;
+           return parentIds.length ? parentIds[0] : null;
+         }
 
-         context.documentTitleBase = function (val) {
-           if (!arguments.length) return _documentTitleBase;
-           _documentTitleBase = val;
-           return context;
+         mode.selectedIDs = function (val) {
+           if (!arguments.length) return selectedIDs;
+           selectedIDs = val;
+           return mode;
          };
-         /* User interface and keybinding */
-
 
-         var _ui;
-
-         context.ui = function () {
-           return _ui;
+         mode.zoomToSelected = function () {
+           context.map().zoomToEase(selectedEntities());
          };
 
-         context.lastPointerType = function () {
-           return _ui.lastPointerType();
+         mode.newFeature = function (val) {
+           if (!arguments.length) return _newFeature;
+           _newFeature = val;
+           return mode;
          };
 
-         var _keybinding = utilKeybinding('context');
+         mode.selectBehavior = function (val) {
+           if (!arguments.length) return _selectBehavior;
+           _selectBehavior = val;
+           return mode;
+         };
 
-         context.keybinding = function () {
-           return _keybinding;
+         mode.follow = function (val) {
+           if (!arguments.length) return _follow;
+           _follow = val;
+           return mode;
          };
 
-         select(document).call(_keybinding);
-         /* Straight accessors. Avoid using these if you can. */
-         // Instantiate the connection here because it doesn't require passing in
-         // `context` and it's needed for pre-init calls like `preauth`
-
-         var _connection = services.osm;
+         function loadOperations() {
+           _operations.forEach(function (operation) {
+             if (operation.behavior) {
+               context.uninstall(operation.behavior);
+             }
+           });
 
-         var _history;
+           _operations = Object.values(Operations).map(function (o) {
+             return o(context, selectedIDs);
+           }).filter(function (o) {
+             return o.id !== 'delete' && o.id !== 'downgrade' && o.id !== 'copy';
+           }).concat([// group copy/downgrade/delete operation together at the end of the list
+           operationCopy(context, selectedIDs), operationDowngrade(context, selectedIDs), operationDelete(context, selectedIDs)]).filter(function (operation) {
+             return operation.available();
+           });
 
-         var _validator;
+           _operations.forEach(function (operation) {
+             if (operation.behavior) {
+               context.install(operation.behavior);
+             }
+           }); // remove any displayed menu
 
-         var _uploader;
 
-         context.connection = function () {
-           return _connection;
-         };
+           context.ui().closeEditMenu();
+         }
 
-         context.history = function () {
-           return _history;
+         mode.operations = function () {
+           return _operations;
          };
 
-         context.validator = function () {
-           return _validator;
-         };
+         mode.enter = function () {
+           if (!checkSelectedIDs()) return;
+           context.features().forceVisible(selectedIDs);
 
-         context.uploader = function () {
-           return _uploader;
-         };
-         /* Connection */
+           _modeDragNode.restoreSelectedIDs(selectedIDs);
 
+           loadOperations();
 
-         context.preauth = function (options) {
-           if (_connection) {
-             _connection["switch"](options);
+           if (!_behaviors.length) {
+             if (!_selectBehavior) _selectBehavior = behaviorSelect(context);
+             _behaviors = [behaviorPaste(context), _breatheBehavior, behaviorHover(context).on('hover', context.ui().sidebar.hoverModeSelect), _selectBehavior, behaviorLasso(context), _modeDragNode.behavior, modeDragNote(context).behavior];
            }
 
-           return context;
-         };
-         /* connection options for source switcher (optional) */
+           _behaviors.forEach(context.install);
 
+           keybinding.on(_t('inspector.zoom_to.key'), mode.zoomToSelected).on(['[', 'pgup'], previousVertex).on([']', 'pgdown'], nextVertex).on(['{', uiCmd('⌘['), 'home'], firstVertex).on(['}', uiCmd('⌘]'), 'end'], lastVertex).on(uiCmd('⇧←'), nudgeSelection([-10, 0])).on(uiCmd('⇧↑'), nudgeSelection([0, -10])).on(uiCmd('⇧→'), nudgeSelection([10, 0])).on(uiCmd('⇧↓'), nudgeSelection([0, 10])).on(uiCmd('⇧⌥←'), nudgeSelection([-100, 0])).on(uiCmd('⇧⌥↑'), nudgeSelection([0, -100])).on(uiCmd('⇧⌥→'), nudgeSelection([100, 0])).on(uiCmd('⇧⌥↓'), nudgeSelection([0, 100])).on(utilKeybinding.plusKeys.map(function (key) {
+             return uiCmd('⇧' + key);
+           }), scaleSelection(1.05)).on(utilKeybinding.plusKeys.map(function (key) {
+             return uiCmd('⇧⌥' + key);
+           }), scaleSelection(Math.pow(1.05, 5))).on(utilKeybinding.minusKeys.map(function (key) {
+             return uiCmd('⇧' + key);
+           }), scaleSelection(1 / 1.05)).on(utilKeybinding.minusKeys.map(function (key) {
+             return uiCmd('⇧⌥' + key);
+           }), scaleSelection(1 / Math.pow(1.05, 5))).on(['\\', 'pause'], focusNextParent).on(uiCmd('⌘↑'), selectParent).on(uiCmd('⌘↓'), selectChild).on('⎋', esc, true);
+           select(document).call(keybinding);
+           context.ui().sidebar.select(selectedIDs, _newFeature);
+           context.history().on('change.select', function () {
+             loadOperations(); // reselect after change in case relation members were removed or added
 
-         var _apiConnections;
+             selectElements();
+           }).on('undone.select', checkSelectedIDs).on('redone.select', checkSelectedIDs);
+           context.map().on('drawn.select', selectElements).on('crossEditableZoom.select', function () {
+             selectElements();
 
-         context.apiConnections = function (val) {
-           if (!arguments.length) return _apiConnections;
-           _apiConnections = val;
-           return context;
-         }; // A string or array or locale codes to prefer over the browser's settings
+             _breatheBehavior.restartIfNeeded(context.surface());
+           });
+           context.map().doubleUpHandler().on('doubleUp.modeSelect', didDoubleUp);
+           selectElements();
 
+           if (_follow) {
+             var extent = geoExtent();
+             var graph = context.graph();
+             selectedIDs.forEach(function (id) {
+               var entity = context.entity(id);
 
-         context.locale = function (locale) {
-           if (!arguments.length) return _mainLocalizer.localeCode();
-           _mainLocalizer.preferredLocaleCodes(locale);
-           return context;
-         };
+               extent._extend(entity.extent(graph));
+             });
+             var loc = extent.center();
+             context.map().centerEase(loc); // we could enter the mode multiple times, so reset follow for next time
 
-         function afterLoad(cid, callback) {
-           return function (err, result) {
-             if (err) {
-               // 400 Bad Request, 401 Unauthorized, 403 Forbidden..
-               if (err.status === 400 || err.status === 401 || err.status === 403) {
-                 if (_connection) {
-                   _connection.logout();
-                 }
-               }
+             _follow = false;
+           }
 
-               if (typeof callback === 'function') {
-                 callback(err);
-               }
+           function nudgeSelection(delta) {
+             return function () {
+               // prevent nudging during low zoom selection
+               if (!context.map().withinEditableZoom()) return;
+               var moveOp = operationMove(context, selectedIDs);
 
-               return;
-             } else if (_connection && _connection.getConnectionId() !== cid) {
-               if (typeof callback === 'function') {
-                 callback({
-                   message: 'Connection Switched',
-                   status: -1
-                 });
+               if (moveOp.disabled()) {
+                 context.ui().flash.duration(4000).iconName('#iD-operation-' + moveOp.id).iconClass('operation disabled').label(moveOp.tooltip)();
+               } else {
+                 context.perform(actionMove(selectedIDs, delta, context.projection), moveOp.annotation());
+                 context.validator().validate();
                }
+             };
+           }
 
-               return;
-             } else {
-               _history.merge(result.data, result.extent);
-
-               if (typeof callback === 'function') {
-                 callback(err, result);
-               }
+           function scaleSelection(factor) {
+             return function () {
+               // prevent scaling during low zoom selection
+               if (!context.map().withinEditableZoom()) return;
+               var nodes = utilGetAllNodes(selectedIDs, context.graph());
+               var isUp = factor > 1; // can only scale if multiple nodes are selected
 
-               return;
-             }
-           };
-         }
+               if (nodes.length <= 1) return;
+               var extent = utilTotalExtent(selectedIDs, context.graph()); // These disabled checks would normally be handled by an operation
+               // object, but we don't want an actual scale operation at this point.
 
-         context.loadTiles = function (projection, callback) {
-           var handle = window.requestIdleCallback(function () {
-             _deferred["delete"](handle);
+               function scalingDisabled() {
+                 if (tooSmall()) {
+                   return 'too_small';
+                 } else if (extent.percentContainedIn(context.map().extent()) < 0.8) {
+                   return 'too_large';
+                 } else if (someMissing() || selectedIDs.some(incompleteRelation)) {
+                   return 'not_downloaded';
+                 } else if (selectedIDs.some(context.hasHiddenConnections)) {
+                   return 'connected_to_hidden';
+                 }
 
-             if (_connection && context.editableDataEnabled()) {
-               var cid = _connection.getConnectionId();
+                 return false;
 
-               _connection.loadTiles(projection, afterLoad(cid, callback));
-             }
-           });
+                 function tooSmall() {
+                   if (isUp) return false;
+                   var dLon = Math.abs(extent[1][0] - extent[0][0]);
+                   var dLat = Math.abs(extent[1][1] - extent[0][1]);
+                   return dLon < geoMetersToLon(1, extent[1][1]) && dLat < geoMetersToLat(1);
+                 }
 
-           _deferred.add(handle);
-         };
+                 function someMissing() {
+                   if (context.inIntro()) return false;
+                   var osm = context.connection();
 
-         context.loadTileAtLoc = function (loc, callback) {
-           var handle = window.requestIdleCallback(function () {
-             _deferred["delete"](handle);
+                   if (osm) {
+                     var missing = nodes.filter(function (n) {
+                       return !osm.isDataLoaded(n.loc);
+                     });
 
-             if (_connection && context.editableDataEnabled()) {
-               var cid = _connection.getConnectionId();
+                     if (missing.length) {
+                       missing.forEach(function (loc) {
+                         context.loadTileAtLoc(loc);
+                       });
+                       return true;
+                     }
+                   }
 
-               _connection.loadTileAtLoc(loc, afterLoad(cid, callback));
-             }
-           });
+                   return false;
+                 }
 
-           _deferred.add(handle);
-         };
+                 function incompleteRelation(id) {
+                   var entity = context.entity(id);
+                   return entity.type === 'relation' && !entity.isComplete(context.graph());
+                 }
+               }
 
-         context.loadEntity = function (entityID, callback) {
-           if (_connection) {
-             var cid = _connection.getConnectionId();
+               var disabled = scalingDisabled();
 
-             _connection.loadEntity(entityID, afterLoad(cid, callback));
+               if (disabled) {
+                 var multi = selectedIDs.length === 1 ? 'single' : 'multiple';
+                 context.ui().flash.duration(4000).iconName('#iD-icon-no').iconClass('operation disabled').label(_t('operations.scale.' + disabled + '.' + multi))();
+               } else {
+                 var pivot = context.projection(extent.center());
+                 var annotation = _t('operations.scale.annotation.' + (isUp ? 'up' : 'down') + '.feature', {
+                   n: selectedIDs.length
+                 });
+                 context.perform(actionScale(selectedIDs, pivot, factor, context.projection), annotation);
+                 context.validator().validate();
+               }
+             };
            }
-         };
-
-         context.zoomToEntity = function (entityID, zoomTo) {
-           // be sure to load the entity even if we're not going to zoom to it
-           context.loadEntity(entityID, function (err, result) {
-             if (err) return;
 
-             if (zoomTo !== false) {
-               var entity = result.data.find(function (e) {
-                 return e.id === entityID;
-               });
+           function didDoubleUp(d3_event, loc) {
+             if (!context.map().withinEditableZoom()) return;
+             var target = select(d3_event.target);
+             var datum = target.datum();
+             var entity = datum && datum.properties && datum.properties.entity;
+             if (!entity) return;
 
-               if (entity) {
-                 _map.zoomTo(entity);
-               }
+             if (entity instanceof osmWay && target.classed('target')) {
+               var choice = geoChooseEdge(context.graph().childNodes(entity), loc, context.projection);
+               var prev = entity.nodes[choice.index - 1];
+               var next = entity.nodes[choice.index];
+               context.perform(actionAddMidpoint({
+                 loc: choice.loc,
+                 edge: [prev, next]
+               }, osmNode()), _t('operations.add.annotation.vertex'));
+             } else if (entity.type === 'midpoint') {
+               context.perform(actionAddMidpoint({
+                 loc: entity.loc,
+                 edge: entity.edge
+               }, osmNode()), _t('operations.add.annotation.vertex'));
              }
-           });
-
-           _map.on('drawn.zoomToEntity', function () {
-             if (!context.hasEntity(entityID)) return;
-
-             _map.on('drawn.zoomToEntity', null);
+           }
 
-             context.on('enter.zoomToEntity', null);
-             context.enter(modeSelect(context, [entityID]));
-           });
+           function selectElements() {
+             if (!checkSelectedIDs()) return;
+             var surface = context.surface();
+             surface.selectAll('.selected-member').classed('selected-member', false);
+             surface.selectAll('.selected').classed('selected', false);
+             surface.selectAll('.related').classed('related', false); // reload `_focusedParentWayId` based on the current selection
 
-           context.on('enter.zoomToEntity', function () {
-             if (_mode.id !== 'browse') {
-               _map.on('drawn.zoomToEntity', null);
+             checkFocusedParent();
 
-               context.on('enter.zoomToEntity', null);
+             if (_focusedParentWayId) {
+               surface.selectAll(utilEntitySelector([_focusedParentWayId])).classed('related', true);
              }
-           });
-         };
-
-         var _minEditableZoom = 16;
-
-         context.minEditableZoom = function (val) {
-           if (!arguments.length) return _minEditableZoom;
-           _minEditableZoom = val;
 
-           if (_connection) {
-             _connection.tileZoom(val);
+             if (context.map().withinEditableZoom()) {
+               // Apply selection styling if not in wide selection
+               surface.selectAll(utilDeepMemberSelector(selectedIDs, context.graph(), true
+               /* skipMultipolgonMembers */
+               )).classed('selected-member', true);
+               surface.selectAll(utilEntityOrDeepMemberSelector(selectedIDs, context.graph())).classed('selected', true);
+             }
            }
 
-           return context;
-         }; // String length limits in Unicode characters, not JavaScript UTF-16 code units
-
-
-         context.maxCharsForTagKey = function () {
-           return 255;
-         };
+           function esc() {
+             if (context.container().select('.combobox').size()) return;
+             context.enter(modeBrowse(context));
+           }
 
-         context.maxCharsForTagValue = function () {
-           return 255;
-         };
+           function firstVertex(d3_event) {
+             d3_event.preventDefault();
+             var entity = singular();
+             var parentId = parentWayIdForVertexNavigation();
+             var way;
 
-         context.maxCharsForRelationRole = function () {
-           return 255;
-         };
+             if (entity && entity.type === 'way') {
+               way = entity;
+             } else if (parentId) {
+               way = context.entity(parentId);
+             }
 
-         function cleanOsmString(val, maxChars) {
-           // be lenient with input
-           if (val === undefined || val === null) {
-             val = '';
-           } else {
-             val = val.toString();
-           } // remove whitespace
+             _focusedParentWayId = way && way.id;
 
+             if (way) {
+               context.enter(mode.selectedIDs([way.first()]).follow(true));
+             }
+           }
 
-           val = val.trim(); // use the canonical form of the string
+           function lastVertex(d3_event) {
+             d3_event.preventDefault();
+             var entity = singular();
+             var parentId = parentWayIdForVertexNavigation();
+             var way;
 
-           if (val.normalize) val = val.normalize('NFC'); // trim to the number of allowed characters
+             if (entity && entity.type === 'way') {
+               way = entity;
+             } else if (parentId) {
+               way = context.entity(parentId);
+             }
 
-           return utilUnicodeCharsTruncated(val, maxChars);
-         }
+             _focusedParentWayId = way && way.id;
 
-         context.cleanTagKey = function (val) {
-           return cleanOsmString(val, context.maxCharsForTagKey());
-         };
+             if (way) {
+               context.enter(mode.selectedIDs([way.last()]).follow(true));
+             }
+           }
 
-         context.cleanTagValue = function (val) {
-           return cleanOsmString(val, context.maxCharsForTagValue());
-         };
+           function previousVertex(d3_event) {
+             d3_event.preventDefault();
+             var parentId = parentWayIdForVertexNavigation();
+             _focusedParentWayId = parentId;
+             if (!parentId) return;
+             var way = context.entity(parentId);
+             var length = way.nodes.length;
+             var curr = way.nodes.indexOf(selectedIDs[0]);
+             var index = -1;
 
-         context.cleanRelationRole = function (val) {
-           return cleanOsmString(val, context.maxCharsForRelationRole());
-         };
-         /* History */
+             if (curr > 0) {
+               index = curr - 1;
+             } else if (way.isClosed()) {
+               index = length - 2;
+             }
 
+             if (index !== -1) {
+               context.enter(mode.selectedIDs([way.nodes[index]]).follow(true));
+             }
+           }
 
-         var _inIntro = false;
+           function nextVertex(d3_event) {
+             d3_event.preventDefault();
+             var parentId = parentWayIdForVertexNavigation();
+             _focusedParentWayId = parentId;
+             if (!parentId) return;
+             var way = context.entity(parentId);
+             var length = way.nodes.length;
+             var curr = way.nodes.indexOf(selectedIDs[0]);
+             var index = -1;
 
-         context.inIntro = function (val) {
-           if (!arguments.length) return _inIntro;
-           _inIntro = val;
-           return context;
-         }; // Immediately save the user's history to localstorage, if possible
-         // This is called someteimes, but also on the `window.onbeforeunload` handler
+             if (curr < length - 1) {
+               index = curr + 1;
+             } else if (way.isClosed()) {
+               index = 0;
+             }
 
+             if (index !== -1) {
+               context.enter(mode.selectedIDs([way.nodes[index]]).follow(true));
+             }
+           }
 
-         context.save = function () {
-           // no history save, no message onbeforeunload
-           if (_inIntro || context.container().select('.modal').size()) return;
-           var canSave;
+           function focusNextParent(d3_event) {
+             d3_event.preventDefault();
+             var parents = parentWaysIdsOfSelection(true);
+             if (!parents || parents.length < 2) return;
+             var index = parents.indexOf(_focusedParentWayId);
 
-           if (_mode && _mode.id === 'save') {
-             canSave = false; // Attempt to prevent user from creating duplicate changes - see #5200
+             if (index < 0 || index > parents.length - 2) {
+               _focusedParentWayId = parents[0];
+             } else {
+               _focusedParentWayId = parents[index + 1];
+             }
 
-             if (services.osm && services.osm.isChangesetInflight()) {
-               _history.clearSaved();
+             var surface = context.surface();
+             surface.selectAll('.related').classed('related', false);
 
-               return;
+             if (_focusedParentWayId) {
+               surface.selectAll(utilEntitySelector([_focusedParentWayId])).classed('related', true);
              }
-           } else {
-             canSave = context.selectedIDs().every(function (id) {
-               var entity = context.hasEntity(id);
-               return entity && !entity.isDegenerate();
-             });
            }
 
-           if (canSave) {
-             _history.save();
+           function selectParent(d3_event) {
+             d3_event.preventDefault();
+             var currentSelectedIds = mode.selectedIDs();
+             var parentIds = _focusedParentWayId ? [_focusedParentWayId] : parentWaysIdsOfSelection(false);
+             if (!parentIds.length) return;
+             context.enter(mode.selectedIDs(parentIds)); // set this after re-entering the selection since we normally want it cleared on exit
+
+             _focusedVertexIds = currentSelectedIds;
            }
 
-           if (_history.hasChanges()) {
-             return _t('save.unsaved_changes');
+           function selectChild(d3_event) {
+             d3_event.preventDefault();
+             var currentSelectedIds = mode.selectedIDs();
+             var childIds = _focusedVertexIds ? _focusedVertexIds.filter(function (id) {
+               return context.hasEntity(id);
+             }) : childNodeIdsOfSelection(true);
+             if (!childIds || !childIds.length) return;
+             if (currentSelectedIds.length === 1) _focusedParentWayId = currentSelectedIds[0];
+             context.enter(mode.selectedIDs(childIds));
            }
-         }; // Debounce save, since it's a synchronous localStorage write,
-         // and history changes can happen frequently (e.g. when dragging).
+         };
 
+         mode.exit = function () {
+           // we could enter the mode multiple times but it's only new the first time
+           _newFeature = false;
+           _focusedVertexIds = null;
 
-         context.debouncedSave = debounce(context.save, 350);
+           _operations.forEach(function (operation) {
+             if (operation.behavior) {
+               context.uninstall(operation.behavior);
+             }
+           });
 
-         function withDebouncedSave(fn) {
-           return function () {
-             var result = fn.apply(_history, arguments);
-             context.debouncedSave();
-             return result;
-           };
-         }
-         /* Graph */
+           _operations = [];
 
+           _behaviors.forEach(context.uninstall);
 
-         context.hasEntity = function (id) {
-           return _history.graph().hasEntity(id);
-         };
+           select(document).call(keybinding.unbind);
+           context.ui().closeEditMenu();
+           context.history().on('change.select', null).on('undone.select', null).on('redone.select', null);
+           var surface = context.surface();
+           surface.selectAll('.selected-member').classed('selected-member', false);
+           surface.selectAll('.selected').classed('selected', false);
+           surface.selectAll('.highlighted').classed('highlighted', false);
+           surface.selectAll('.related').classed('related', false);
+           context.map().on('drawn.select', null);
+           context.ui().sidebar.hide();
+           context.features().forceVisible([]);
+           var entity = singular();
 
-         context.entity = function (id) {
-           return _history.graph().entity(id);
+           if (_newFeature && entity && entity.type === 'relation' && // no tags
+           Object.keys(entity.tags).length === 0 && // no parent relations
+           context.graph().parentRelations(entity).length === 0 && ( // no members or one member with no role
+           entity.members.length === 0 || entity.members.length === 1 && !entity.members[0].role)) {
+             // the user added this relation but didn't edit it at all, so just delete it
+             var deleteAction = actionDeleteRelation(entity.id, true
+             /* don't delete untagged members */
+             );
+             context.perform(deleteAction, _t('operations.delete.annotation.relation'));
+           }
          };
-         /* Modes */
-
 
-         var _mode;
+         return mode;
+       }
 
-         context.mode = function () {
-           return _mode;
-         };
+       function uiLasso(context) {
+         var group, polygon;
+         lasso.coordinates = [];
 
-         context.enter = function (newMode) {
-           if (_mode) {
-             _mode.exit();
+         function lasso(selection) {
+           context.container().classed('lasso', true);
+           group = selection.append('g').attr('class', 'lasso hide');
+           polygon = group.append('path').attr('class', 'lasso-path');
+           group.call(uiToggle(true));
+         }
 
-             dispatch$1.call('exit', _this, _mode);
+         function draw() {
+           if (polygon) {
+             polygon.data([lasso.coordinates]).attr('d', function (d) {
+               return 'M' + d.join(' L') + ' Z';
+             });
            }
+         }
 
-           _mode = newMode;
-
-           _mode.enter();
-
-           dispatch$1.call('enter', _this, _mode);
+         lasso.extent = function () {
+           return lasso.coordinates.reduce(function (extent, point) {
+             return extent.extend(geoExtent(point));
+           }, geoExtent());
          };
 
-         context.selectedIDs = function () {
-           return _mode && _mode.selectedIDs && _mode.selectedIDs() || [];
+         lasso.p = function (_) {
+           if (!arguments.length) return lasso;
+           lasso.coordinates.push(_);
+           draw();
+           return lasso;
          };
 
-         context.activeID = function () {
-           return _mode && _mode.activeID && _mode.activeID();
-         };
+         lasso.close = function () {
+           if (group) {
+             group.call(uiToggle(false, function () {
+               select(this).remove();
+             }));
+           }
 
-         var _selectedNoteID;
+           context.container().classed('lasso', false);
+         };
 
-         context.selectedNoteID = function (noteID) {
-           if (!arguments.length) return _selectedNoteID;
-           _selectedNoteID = noteID;
-           return context;
-         }; // NOTE: Don't change the name of this until UI v3 is merged
+         return lasso;
+       }
 
+       function behaviorLasso(context) {
+         // use pointer events on supported platforms; fallback to mouse events
+         var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
 
-         var _selectedErrorID;
+         var behavior = function behavior(selection) {
+           var lasso;
 
-         context.selectedErrorID = function (errorID) {
-           if (!arguments.length) return _selectedErrorID;
-           _selectedErrorID = errorID;
-           return context;
-         };
-         /* Behaviors */
+           function pointerdown(d3_event) {
+             var button = 0; // left
 
+             if (d3_event.button === button && d3_event.shiftKey === true) {
+               lasso = null;
+               select(window).on(_pointerPrefix + 'move.lasso', pointermove).on(_pointerPrefix + 'up.lasso', pointerup);
+               d3_event.stopPropagation();
+             }
+           }
 
-         context.install = function (behavior) {
-           return context.surface().call(behavior);
-         };
+           function pointermove() {
+             if (!lasso) {
+               lasso = uiLasso(context);
+               context.surface().call(lasso);
+             }
 
-         context.uninstall = function (behavior) {
-           return context.surface().call(behavior.off);
-         };
-         /* Copy/Paste */
+             lasso.p(context.map().mouse());
+           }
 
+           function normalize(a, b) {
+             return [[Math.min(a[0], b[0]), Math.min(a[1], b[1])], [Math.max(a[0], b[0]), Math.max(a[1], b[1])]];
+           }
 
-         var _copyGraph;
+           function lassoed() {
+             if (!lasso) return [];
+             var graph = context.graph();
+             var limitToNodes;
 
-         context.copyGraph = function () {
-           return _copyGraph;
-         };
+             if (context.map().editableDataEnabled(true
+             /* skipZoomCheck */
+             ) && context.map().isInWideSelection()) {
+               // only select from the visible nodes
+               limitToNodes = new Set(utilGetAllNodes(context.selectedIDs(), graph));
+             } else if (!context.map().editableDataEnabled()) {
+               return [];
+             }
 
-         var _copyIDs = [];
+             var bounds = lasso.extent().map(context.projection.invert);
+             var extent = geoExtent(normalize(bounds[0], bounds[1]));
+             var intersects = context.history().intersects(extent).filter(function (entity) {
+               return entity.type === 'node' && (!limitToNodes || limitToNodes.has(entity)) && geoPointInPolygon(context.projection(entity.loc), lasso.coordinates) && !context.features().isHidden(entity, graph, entity.geometry(graph));
+             }); // sort the lassoed nodes as best we can
 
-         context.copyIDs = function (val) {
-           if (!arguments.length) return _copyIDs;
-           _copyIDs = val;
-           _copyGraph = _history.graph();
-           return context;
-         };
+             intersects.sort(function (node1, node2) {
+               var parents1 = graph.parentWays(node1);
+               var parents2 = graph.parentWays(node2);
 
-         var _copyLonLat;
+               if (parents1.length && parents2.length) {
+                 // both nodes are vertices
+                 var sharedParents = utilArrayIntersection(parents1, parents2);
 
-         context.copyLonLat = function (val) {
-           if (!arguments.length) return _copyLonLat;
-           _copyLonLat = val;
-           return context;
-         };
-         /* Background */
+                 if (sharedParents.length) {
+                   var sharedParentNodes = sharedParents[0].nodes; // vertices are members of the same way; sort them in their listed order
 
+                   return sharedParentNodes.indexOf(node1.id) - sharedParentNodes.indexOf(node2.id);
+                 } else {
+                   // vertices do not share a way; group them by their respective parent ways
+                   return parseFloat(parents1[0].id.slice(1)) - parseFloat(parents2[0].id.slice(1));
+                 }
+               } else if (parents1.length || parents2.length) {
+                 // only one node is a vertex; sort standalone points before vertices
+                 return parents1.length - parents2.length;
+               } // both nodes are standalone points; sort left to right
 
-         var _background;
 
-         context.background = function () {
-           return _background;
-         };
-         /* Features */
+               return node1.loc[0] - node2.loc[0];
+             });
+             return intersects.map(function (entity) {
+               return entity.id;
+             });
+           }
 
+           function pointerup() {
+             select(window).on(_pointerPrefix + 'move.lasso', null).on(_pointerPrefix + 'up.lasso', null);
+             if (!lasso) return;
+             var ids = lassoed();
+             lasso.close();
 
-         var _features;
+             if (ids.length) {
+               context.enter(modeSelect(context, ids));
+             }
+           }
 
-         context.features = function () {
-           return _features;
+           selection.on(_pointerPrefix + 'down.lasso', pointerdown);
          };
 
-         context.hasHiddenConnections = function (id) {
-           var graph = _history.graph();
-
-           var entity = graph.entity(id);
-           return _features.hasHiddenConnections(entity, graph);
+         behavior.off = function (selection) {
+           selection.on(_pointerPrefix + 'down.lasso', null);
          };
-         /* Photos */
-
 
-         var _photos;
+         return behavior;
+       }
 
-         context.photos = function () {
-           return _photos;
+       function modeBrowse(context) {
+         var mode = {
+           button: 'browse',
+           id: 'browse',
+           title: _t('modes.browse.title'),
+           description: _t('modes.browse.description')
          };
-         /* Map */
+         var sidebar;
 
+         var _selectBehavior;
 
-         var _map;
+         var _behaviors = [];
 
-         context.map = function () {
-           return _map;
+         mode.selectBehavior = function (val) {
+           if (!arguments.length) return _selectBehavior;
+           _selectBehavior = val;
+           return mode;
          };
 
-         context.layers = function () {
-           return _map.layers();
-         };
+         mode.enter = function () {
+           if (!_behaviors.length) {
+             if (!_selectBehavior) _selectBehavior = behaviorSelect(context);
+             _behaviors = [behaviorPaste(context), behaviorHover(context).on('hover', context.ui().sidebar.hover), _selectBehavior, behaviorLasso(context), modeDragNode(context).behavior, modeDragNote(context).behavior];
+           }
 
-         context.surface = function () {
-           return _map.surface;
-         };
+           _behaviors.forEach(context.install); // Get focus on the body.
 
-         context.editableDataEnabled = function () {
-           return _map.editableDataEnabled();
-         };
 
-         context.surfaceRect = function () {
-           return _map.surface.node().getBoundingClientRect();
-         };
+           if (document.activeElement && document.activeElement.blur) {
+             document.activeElement.blur();
+           }
 
-         context.editable = function () {
-           // don't allow editing during save
-           var mode = context.mode();
-           if (!mode || mode.id === 'save') return false;
-           return _map.editableDataEnabled();
+           if (sidebar) {
+             context.ui().sidebar.show(sidebar);
+           } else {
+             context.ui().sidebar.select(null);
+           }
          };
-         /* Debug */
 
+         mode.exit = function () {
+           context.ui().sidebar.hover.cancel();
 
-         var _debugFlags = {
-           tile: false,
-           // tile boundaries
-           collision: false,
-           // label collision bounding boxes
-           imagery: false,
-           // imagery bounding polygons
-           target: false,
-           // touch targets
-           downloaded: false // downloaded data from osm
-
-         };
+           _behaviors.forEach(context.uninstall);
 
-         context.debugFlags = function () {
-           return _debugFlags;
+           if (sidebar) {
+             context.ui().sidebar.hide();
+           }
          };
 
-         context.getDebug = function (flag) {
-           return flag && _debugFlags[flag];
+         mode.sidebar = function (_) {
+           if (!arguments.length) return sidebar;
+           sidebar = _;
+           return mode;
          };
 
-         context.setDebug = function (flag, val) {
-           if (arguments.length === 1) val = true;
-           _debugFlags[flag] = val;
-           dispatch$1.call('change');
-           return context;
+         mode.operations = function () {
+           return [operationPaste(context)];
          };
-         /* Container */
-
 
-         var _container = select(null);
-
-         context.container = function (val) {
-           if (!arguments.length) return _container;
-           _container = val;
+         return mode;
+       }
 
-           _container.classed('ideditor', true);
+       function behaviorAddWay(context) {
+         var dispatch = dispatch$8('start', 'startFromWay', 'startFromNode');
+         var draw = behaviorDraw(context);
 
-           return context;
-         };
+         function behavior(surface) {
+           draw.on('click', function () {
+             dispatch.apply('start', this, arguments);
+           }).on('clickWay', function () {
+             dispatch.apply('startFromWay', this, arguments);
+           }).on('clickNode', function () {
+             dispatch.apply('startFromNode', this, arguments);
+           }).on('cancel', behavior.cancel).on('finish', behavior.cancel);
+           context.map().dblclickZoomEnable(false);
+           surface.call(draw);
+         }
 
-         context.containerNode = function (val) {
-           if (!arguments.length) return context.container().node();
-           context.container(select(val));
-           return context;
+         behavior.off = function (surface) {
+           surface.call(draw.off);
          };
 
-         var _embed;
-
-         context.embed = function (val) {
-           if (!arguments.length) return _embed;
-           _embed = val;
-           return context;
+         behavior.cancel = function () {
+           window.setTimeout(function () {
+             context.map().dblclickZoomEnable(true);
+           }, 1000);
+           context.enter(modeBrowse(context));
          };
-         /* Assets */
 
+         return utilRebind(behavior, dispatch, 'on');
+       }
 
-         var _assetPath = '';
+       function behaviorHash(context) {
+         // cached window.location.hash
+         var _cachedHash = null; // allowable latitude range
 
-         context.assetPath = function (val) {
-           if (!arguments.length) return _assetPath;
-           _assetPath = val;
-           _mainFileFetcher.assetPath(val);
-           return context;
-         };
+         var _latitudeLimit = 90 - 1e-8;
 
-         var _assetMap = {};
+         function computedHashParameters() {
+           var map = context.map();
+           var center = map.center();
+           var zoom = map.zoom();
+           var precision = Math.max(0, Math.ceil(Math.log(zoom) / Math.LN2));
+           var oldParams = utilObjectOmit(utilStringQs(window.location.hash), ['comment', 'source', 'hashtags', 'walkthrough']);
+           var newParams = {};
+           delete oldParams.id;
+           var selected = context.selectedIDs().filter(function (id) {
+             return context.hasEntity(id);
+           });
 
-         context.assetMap = function (val) {
-           if (!arguments.length) return _assetMap;
-           _assetMap = val;
-           _mainFileFetcher.assetMap(val);
-           return context;
-         };
+           if (selected.length) {
+             newParams.id = selected.join(',');
+           }
 
-         context.asset = function (val) {
-           if (/^http(s)?:\/\//i.test(val)) return val;
-           var filename = _assetPath + val;
-           return _assetMap[filename] || filename;
-         };
+           newParams.map = zoom.toFixed(2) + '/' + center[1].toFixed(precision) + '/' + center[0].toFixed(precision);
+           return Object.assign(oldParams, newParams);
+         }
 
-         context.imagePath = function (val) {
-           return context.asset("img/".concat(val));
-         };
-         /* reset (aka flush) */
+         function computedHash() {
+           return '#' + utilQsString(computedHashParameters(), true);
+         }
 
+         function computedTitle(includeChangeCount) {
+           var baseTitle = context.documentTitleBase() || 'iD';
+           var contextual;
+           var changeCount;
+           var titleID;
+           var selected = context.selectedIDs().filter(function (id) {
+             return context.hasEntity(id);
+           });
 
-         context.reset = context.flush = function () {
-           context.debouncedSave.cancel();
-           Array.from(_deferred).forEach(function (handle) {
-             window.cancelIdleCallback(handle);
+           if (selected.length) {
+             var firstLabel = utilDisplayLabel(context.entity(selected[0]), context.graph());
 
-             _deferred["delete"](handle);
-           });
-           Object.values(services).forEach(function (service) {
-             if (service && typeof service.reset === 'function') {
-               service.reset(context);
+             if (selected.length > 1) {
+               contextual = _t('title.labeled_and_more', {
+                 labeled: firstLabel,
+                 count: selected.length - 1
+               });
+             } else {
+               contextual = firstLabel;
              }
-           });
-           context.changeset = null;
 
-           _validator.reset();
+             titleID = 'context';
+           }
 
-           _features.reset();
+           if (includeChangeCount) {
+             changeCount = context.history().difference().summary().length;
 
-           _history.reset();
+             if (changeCount > 0) {
+               titleID = contextual ? 'changes_context' : 'changes';
+             }
+           }
 
-           _uploader.reset(); // don't leave stale state in the inspector
+           if (titleID) {
+             return _t('title.format.' + titleID, {
+               changes: changeCount,
+               base: baseTitle,
+               context: contextual
+             });
+           }
 
+           return baseTitle;
+         }
 
-           context.container().select('.inspector-wrap *').remove();
-           return context;
-         };
-         /* Projections */
+         function updateTitle(includeChangeCount) {
+           if (!context.setsDocumentTitle()) return;
+           var newTitle = computedTitle(includeChangeCount);
 
+           if (document.title !== newTitle) {
+             document.title = newTitle;
+           }
+         }
 
-         context.projection = geoRawMercator();
-         context.curtainProjection = geoRawMercator();
-         /* Init */
+         function updateHashIfNeeded() {
+           if (context.inIntro()) return;
+           var latestHash = computedHash();
 
-         context.init = function () {
-           instantiateInternal();
-           initializeDependents();
-           return context; // Load variables and properties. No property of `context` should be accessed
-           // until this is complete since load statuses are indeterminate. The order
-           // of instantiation shouldn't matter.
+           if (_cachedHash !== latestHash) {
+             _cachedHash = latestHash; // Update the URL hash without affecting the browser navigation stack,
+             // though unavoidably creating a browser history entry
 
-           function instantiateInternal() {
-             _history = coreHistory(context);
-             context.graph = _history.graph;
-             context.pauseChangeDispatch = _history.pauseChangeDispatch;
-             context.resumeChangeDispatch = _history.resumeChangeDispatch;
-             context.perform = withDebouncedSave(_history.perform);
-             context.replace = withDebouncedSave(_history.replace);
-             context.pop = withDebouncedSave(_history.pop);
-             context.overwrite = withDebouncedSave(_history.overwrite);
-             context.undo = withDebouncedSave(_history.undo);
-             context.redo = withDebouncedSave(_history.redo);
-             _validator = coreValidator(context);
-             _uploader = coreUploader(context);
-             _background = rendererBackground(context);
-             _features = rendererFeatures(context);
-             _map = rendererMap(context);
-             _photos = rendererPhotos(context);
-             _ui = uiInit(context);
-           } // Set up objects that might need to access properties of `context`. The order
-           // might matter if dependents make calls to each other. Be wary of async calls.
+             window.history.replaceState(null, computedTitle(false
+             /* includeChangeCount */
+             ), latestHash); // set the title we want displayed for the browser tab/window
 
+             updateTitle(true
+             /* includeChangeCount */
+             );
+           }
+         }
 
-           function initializeDependents() {
-             if (context.initialHashParams.presets) {
-               _mainPresetIndex.addablePresetIDs(new Set(context.initialHashParams.presets.split(',')));
-             }
+         var _throttledUpdate = throttle(updateHashIfNeeded, 500);
 
-             if (context.initialHashParams.locale) {
-               _mainLocalizer.preferredLocaleCodes(context.initialHashParams.locale);
-             } // kick off some async work
+         var _throttledUpdateTitle = throttle(function () {
+           updateTitle(true
+           /* includeChangeCount */
+           );
+         }, 500);
 
+         function hashchange() {
+           // ignore spurious hashchange events
+           if (window.location.hash === _cachedHash) return;
+           _cachedHash = window.location.hash;
+           var q = utilStringQs(_cachedHash);
+           var mapArgs = (q.map || '').split('/').map(Number);
 
-             _mainLocalizer.ensureLoaded();
+           if (mapArgs.length < 3 || mapArgs.some(isNaN)) {
+             // replace bogus hash
+             updateHashIfNeeded();
+           } else {
+             // don't update if the new hash already reflects the state of iD
+             if (_cachedHash === computedHash()) return;
+             var mode = context.mode();
+             context.map().centerZoom([mapArgs[2], Math.min(_latitudeLimit, Math.max(-_latitudeLimit, mapArgs[1]))], mapArgs[0]);
 
-             _background.ensureLoaded();
+             if (q.id && mode) {
+               var ids = q.id.split(',').filter(function (id) {
+                 return context.hasEntity(id);
+               });
 
-             _mainPresetIndex.ensureLoaded();
-             Object.values(services).forEach(function (service) {
-               if (service && typeof service.init === 'function') {
-                 service.init();
+               if (ids.length && (mode.id === 'browse' || mode.id === 'select' && !utilArrayIdentical(mode.selectedIDs(), ids))) {
+                 context.enter(modeSelect(context, ids));
+                 return;
                }
-             });
+             }
 
-             _map.init();
+             var center = context.map().center();
+             var dist = geoSphericalDistance(center, [mapArgs[2], mapArgs[1]]);
+             var maxdist = 500; // Don't allow the hash location to change too much while drawing
+             // This can happen if the user accidentally hit the back button.  #3996
 
-             _validator.init();
+             if (mode && mode.id.match(/^draw/) !== null && dist > maxdist) {
+               context.enter(modeBrowse(context));
+               return;
+             }
+           }
+         }
 
-             _features.init();
+         function behavior() {
+           context.map().on('move.behaviorHash', _throttledUpdate);
+           context.history().on('change.behaviorHash', _throttledUpdateTitle);
+           context.on('enter.behaviorHash', _throttledUpdate);
+           select(window).on('hashchange.behaviorHash', hashchange);
 
-             if (services.maprules && context.initialHashParams.maprules) {
-               d3_json(context.initialHashParams.maprules).then(function (mapcss) {
-                 services.maprules.init();
-                 mapcss.forEach(function (mapcssSelector) {
-                   return services.maprules.addRule(mapcssSelector);
-                 });
-               })["catch"](function () {
-                 /* ignore */
-               });
-             } // if the container isn't available, e.g. when testing, don't load the UI
+           if (window.location.hash) {
+             var q = utilStringQs(window.location.hash);
 
+             if (q.id) {
+               //if (!context.history().hasRestorableChanges()) {
+               // targeting specific features: download, select, and zoom to them
+               context.zoomToEntity(q.id.split(',')[0], !q.map); //}
+             }
 
-             if (!context.container().empty()) {
-               _ui.ensureLoaded().then(function () {
-                 _photos.init();
-               });
+             if (q.walkthrough === 'true') {
+               behavior.startWalkthrough = true;
+             }
+
+             if (q.map) {
+               behavior.hadHash = true;
              }
+
+             hashchange();
+             updateTitle(false);
            }
+         }
+
+         behavior.off = function () {
+           _throttledUpdate.cancel();
+
+           _throttledUpdateTitle.cancel();
+
+           context.map().on('move.behaviorHash', null);
+           context.on('enter.behaviorHash', null);
+           select(window).on('hashchange.behaviorHash', null);
+           window.location.hash = '';
          };
 
-         return context;
+         return behavior;
        }
 
        // This is only done in testing because of the performance penalty.
 
        var debug = false; // Reexport just what our tests use, see #4379
        var d3 = {
-         dispatch: dispatch,
+         dispatch: dispatch$8,
          geoMercator: mercator,
          geoProjection: projection,
          polygonArea: d3_polygonArea,
                coreLocalizer: coreLocalizer,
                t: _t,
                localizer: _mainLocalizer,
+               coreLocations: coreLocations,
+               locationManager: _mainLocations,
                prefs: corePreferences,
                coreTree: coreTree,
                coreUploader: coreUploader,
                serviceMapillary: serviceMapillary,
                serviceMapRules: serviceMapRules,
                serviceNominatim: serviceNominatim,
+               serviceNsi: serviceNsi,
                serviceOpenstreetcam: serviceOpenstreetcam,
                serviceOsm: serviceOsm,
                serviceOsmWikibase: serviceOsmWikibase,
                uiFieldCycleway: uiFieldCycleway,
                uiFieldLanes: uiFieldLanes,
                uiFieldLocalized: uiFieldLocalized,
-               uiFieldMaxspeed: uiFieldMaxspeed,
+               uiFieldRoadspeed: uiFieldRoadspeed,
                uiFieldStructureRadio: uiFieldRadio,
                uiFieldRadio: uiFieldRadio,
                uiFieldRestrictions: uiFieldRestrictions,
                utilEntityOrMemberSelector: utilEntityOrMemberSelector,
                utilEntityOrDeepMemberSelector: utilEntityOrDeepMemberSelector,
                utilFastMouse: utilFastMouse,
+               utilFetchJson: utilFetchJson,
                utilFunctor: utilFunctor,
                utilGetAllNodes: utilGetAllNodes,
                utilGetSetValue: utilGetSetValue,